Implement Twitch API connector
parent
2ef965f749
commit
df68c990fc
|
@ -493,6 +493,11 @@
|
||||||
"integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==",
|
"integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"data-uri-to-buffer": {
|
||||||
|
"version": "3.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-3.0.1.tgz",
|
||||||
|
"integrity": "sha512-WboRycPNsVw3B3TL559F7kuBUM4d8CgMEvk6xEJlOp7OBPjt6G7z8WMWlD2rOFZLk6OYfFIUGsCOWzcQH9K2og=="
|
||||||
|
},
|
||||||
"debug": {
|
"debug": {
|
||||||
"version": "2.6.9",
|
"version": "2.6.9",
|
||||||
"resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
|
"resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
|
||||||
|
@ -658,6 +663,11 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"fetch-blob": {
|
||||||
|
"version": "2.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/fetch-blob/-/fetch-blob-2.0.1.tgz",
|
||||||
|
"integrity": "sha512-1jFpa68M4EzObtFa7XOKZoN1unsaeJ6hGSbxaWaVO+TkHmVvnyzRu1ktZAFbUvTZ9NC/qMKGKJ79dK4MzuSBiw=="
|
||||||
|
},
|
||||||
"fill-range": {
|
"fill-range": {
|
||||||
"version": "7.0.1",
|
"version": "7.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
|
||||||
|
@ -1024,6 +1034,15 @@
|
||||||
"resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
|
"resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
|
||||||
"integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
|
"integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
|
||||||
},
|
},
|
||||||
|
"node-fetch": {
|
||||||
|
"version": "3.0.0-beta.7",
|
||||||
|
"resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-3.0.0-beta.7.tgz",
|
||||||
|
"integrity": "sha512-UTmmxR2RCLiGL0q61p8DgMgw1UXd10+XVB77IHG55flJ/tHqQQXloNTm5dd/mB3RNXP3+CJPf++t0nb3whKNkw==",
|
||||||
|
"requires": {
|
||||||
|
"data-uri-to-buffer": "^3.0.1",
|
||||||
|
"fetch-blob": "^2.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"nodemon": {
|
"nodemon": {
|
||||||
"version": "2.0.4",
|
"version": "2.0.4",
|
||||||
"resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.4.tgz",
|
"resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.4.tgz",
|
||||||
|
|
|
@ -7,6 +7,7 @@
|
||||||
"connect-pg-simple": "^6.1.0",
|
"connect-pg-simple": "^6.1.0",
|
||||||
"express": "^4.17.1",
|
"express": "^4.17.1",
|
||||||
"express-session": "^1.17.1",
|
"express-session": "^1.17.1",
|
||||||
|
"node-fetch": "^3.0.0-beta.7",
|
||||||
"pg": "^8.2.1"
|
"pg": "^8.2.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
|
|
@ -9,7 +9,7 @@
|
||||||
<div id="topbar">
|
<div id="topbar">
|
||||||
<div id="logo">Learn Request Queue</div>
|
<div id="logo">Learn Request Queue</div>
|
||||||
<div id="nav-requests"><a href="/">Requests</a></div>
|
<div id="nav-requests"><a href="/">Requests</a></div>
|
||||||
<div id="nav-login"><a href="https://id.twitch.tv/oauth2/authorize?client_id=di37tc1dr9rhvmpvzgn8rkmi7bdhkk&redirect_uri=https://localhost/callback&response_type=token">Login</a></div>
|
<div id="nav-login"><a href="https://id.twitch.tv/oauth2/authorize?client_id=di37tc1dr9rhvmpvzgn8rkmi7bdhkk&redirect_uri=https://localhost/callback&response_type=code">Login</a></div>
|
||||||
</div>
|
</div>
|
||||||
<div id="main">
|
<div id="main">
|
||||||
<div id="requests"></div><br>
|
<div id="requests"></div><br>
|
||||||
|
|
28
src/app.ts
28
src/app.ts
|
@ -1,9 +1,12 @@
|
||||||
import * as config from "./config";
|
import * as config from "./config";
|
||||||
import * as requests from "./requests";
|
import * as requests from "./requests";
|
||||||
|
import * as twitch from "./twitch";
|
||||||
|
import { URLSearchParams } from "url";
|
||||||
import { QueryResult } from "pg";
|
import { QueryResult } from "pg";
|
||||||
import express from "express";
|
import express from "express";
|
||||||
import session from "express-session";
|
import session from "express-session";
|
||||||
import pgSessionStore from "connect-pg-simple";
|
import pgSessionStore from "connect-pg-simple";
|
||||||
|
import fetch, { Response as FetchResponse } from "node-fetch";
|
||||||
import db from "./db";
|
import db from "./db";
|
||||||
import errorHandler from "./errors";
|
import errorHandler from "./errors";
|
||||||
|
|
||||||
|
@ -14,9 +17,12 @@ app.use(session({
|
||||||
secret: config.sessionSecret,
|
secret: config.sessionSecret,
|
||||||
saveUninitialized: false,
|
saveUninitialized: false,
|
||||||
resave: false,
|
resave: false,
|
||||||
store: new (pgSessionStore(session))()
|
store: new (pgSessionStore(session))({
|
||||||
|
pool: db
|
||||||
|
})
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
// API
|
||||||
app.get("/api/getRequests", async (request, response) => {
|
app.get("/api/getRequests", async (request, response) => {
|
||||||
var requestCount = ( request.query.count ? parseInt(request.query.count as string, 10) : 5 );
|
var requestCount = ( request.query.count ? parseInt(request.query.count as string, 10) : 5 );
|
||||||
requests.getRequests(requestCount).then((val: QueryResult) => response.send(val))
|
requests.getRequests(requestCount).then((val: QueryResult) => response.send(val))
|
||||||
|
@ -101,6 +107,26 @@ app.post("/api/deleteRequest", async (request, response) => {
|
||||||
.catch((e: any) => errorHandler(request,response,e));
|
.catch((e: any) => errorHandler(request,response,e));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Twitch callback
|
||||||
|
app.get("/callback", async (request, response) => {
|
||||||
|
var authcode = request.query.code as string;
|
||||||
|
var tokenResponse = await fetch("https://id.twitch.tv/oauth2/token", { method: "POST", body: new URLSearchParams({
|
||||||
|
client_id: config.twitchClientId,
|
||||||
|
client_secret: config.twitchSecret,
|
||||||
|
code: authcode,
|
||||||
|
grant_type: "authorization_code",
|
||||||
|
redirect_uri: `${config.urlPrefix}/callback`
|
||||||
|
})}).then((res: FetchResponse) => res.json() as Promise<twitch.TokenResponse>)
|
||||||
|
.catch((e: any) => errorHandler(request,response,e));
|
||||||
|
if (typeof request.session == 'undefined') throw new Error('Session is undefined');
|
||||||
|
if (typeof tokenResponse == 'undefined') throw new Error('tokenResponse is undefined');
|
||||||
|
request.session.tokenpair = { access_token: tokenResponse.access_token, refresh_token: tokenResponse.refresh_token };
|
||||||
|
request.session.user = (await twitch.apiRequest(request.session.tokenpair,"GET","/users")).data[0];
|
||||||
|
response.redirect(307, '/');
|
||||||
|
});
|
||||||
|
|
||||||
|
//app.get("/session", (request, response) => { response.send(request.session); });
|
||||||
|
|
||||||
const server = app.listen(config.port, () => {
|
const server = app.listen(config.port, () => {
|
||||||
console.log(`Listening on port ${config.port}`);
|
console.log(`Listening on port ${config.port}`);
|
||||||
});
|
});
|
||||||
|
|
|
@ -4,6 +4,18 @@ if (!process.env.PORT) {
|
||||||
}
|
}
|
||||||
export const port: number = parseInt(process.env.PORT as string, 10);
|
export const port: number = parseInt(process.env.PORT as string, 10);
|
||||||
|
|
||||||
|
if (!process.env.URL_PREFIX) {
|
||||||
|
console.log("Missing environment variable URL_PREFIX");
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
export const urlPrefix: string = process.env.URL_PREFIX;
|
||||||
|
|
||||||
|
if (!process.env.TWITCH_CLIENTID) {
|
||||||
|
console.log("Missing environment variable TWITCH_CLIENTID");
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
export const twitchClientId: string = process.env.TWITCH_CLIENTID;
|
||||||
|
|
||||||
if (!process.env.TWITCH_SECRET) {
|
if (!process.env.TWITCH_SECRET) {
|
||||||
console.log("Missing environment variable TWITCH_SECRET");
|
console.log("Missing environment variable TWITCH_SECRET");
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
|
|
|
@ -0,0 +1,25 @@
|
||||||
|
import * as config from "./config";
|
||||||
|
import fetch, { Response as FetchResponse } from "node-fetch";
|
||||||
|
|
||||||
|
export interface TokenResponse {
|
||||||
|
access_token: string;
|
||||||
|
refresh_token: string;
|
||||||
|
token_type: string;
|
||||||
|
expires_in: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface TokenPair {
|
||||||
|
access_token: string;
|
||||||
|
refresh_token: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function apiRequest(tokens: TokenPair, method: string, endpoint: string): Promise<any>;
|
||||||
|
export async function apiRequest(tokens: TokenPair, method: string, endpoint: string, query: string): Promise<any>;
|
||||||
|
export async function apiRequest(tokens: TokenPair, method: string, endpoint: string, query?: string,) {
|
||||||
|
var headers = {
|
||||||
|
"Authorization": "Bearer " + tokens.access_token,
|
||||||
|
"Client-ID": config.twitchClientId
|
||||||
|
};
|
||||||
|
return fetch("https://api.twitch.tv/helix" + endpoint, { method: method, headers: headers})
|
||||||
|
.then(async (res: FetchResponse) => res.json());
|
||||||
|
}
|
Loading…
Reference in New Issue