119 lines
4.4 KiB
TypeScript
119 lines
4.4 KiB
TypeScript
import * as config from "./config";
|
|
import * as queries from "./queries";
|
|
import { log, LogLevel } from "./logging"
|
|
import fetch, { Response as FetchResponse } from "node-fetch";
|
|
import db from "./db";
|
|
|
|
export interface TokenPair {
|
|
access_token: string;
|
|
refresh_token: string;
|
|
}
|
|
|
|
export interface StreamerUserIdTokenPair {
|
|
userid: number
|
|
tokenpair: TokenPair
|
|
}
|
|
|
|
// Refresh the API token. Returns true on success and false on failure.
|
|
async function refreshApiToken(tokens: TokenPair): Promise<boolean> {
|
|
log(LogLevel.DEBUG,`Call: twitch.refreshApiToken(${JSON.stringify(tokens,null,2)})`);
|
|
return fetch("https://id.twitch.tv/oauth2/token", {
|
|
method: 'POST',
|
|
body: new URLSearchParams({
|
|
client_id: config.twitchClientId,
|
|
client_secret: config.twitchSecret,
|
|
grant_type: "refresh_token",
|
|
refresh_token: tokens.refresh_token
|
|
})
|
|
}).then(async (res: FetchResponse) => {
|
|
if (res.status == 200) {
|
|
log(LogLevel.INFO,"twitch.refreshApiToken: Refresh returned success.");
|
|
var data = await (res.json() as Promise<TokenPair>);
|
|
log(LogLevel.DEBUG, "Returned data:");
|
|
log(LogLevel.DEBUG, JSON.stringify(data,null,2));
|
|
tokens.access_token = data.access_token;
|
|
tokens.refresh_token = data.refresh_token;
|
|
return true;
|
|
} else {
|
|
log(LogLevel.ERROR,"twitch.refreshApiToken: Refresh returned failure. Response object:");
|
|
log(LogLevel.ERROR,JSON.stringify(await res.json(),null,2));
|
|
return false;
|
|
}
|
|
})
|
|
}
|
|
|
|
// Send an API request. On success, return the specified data. On failure,
|
|
// attempt to refresh the API token and retry
|
|
export async function apiRequest(tokens: TokenPair, endpoint: string): Promise <any> {
|
|
log(LogLevel.DEBUG,`Call: twitch.apiRequest(${JSON.stringify(tokens,null,2)},${endpoint})`);
|
|
var headers = {
|
|
"Authorization": "Bearer " + tokens.access_token,
|
|
"Client-ID": config.twitchClientId
|
|
};
|
|
return fetch("https://api.twitch.tv/helix" + endpoint, { headers: headers })
|
|
.then(async (res: FetchResponse) => {
|
|
if (res.status == 200) {
|
|
return res.json();
|
|
} else {
|
|
log(LogLevel.WARNING,"twitch.apiRequest: Failed API request (pre-refresh):");
|
|
log(LogLevel.WARNING,"Request URL: https://api.twitch.tv/helix" + endpoint);
|
|
log(LogLevel.WARNING,"Headers:");
|
|
log(LogLevel.WARNING,JSON.stringify(headers,null,2));
|
|
log(LogLevel.WARNING,"Response:");
|
|
log(LogLevel.WARNING,JSON.stringify(await res.json(),null,2));
|
|
log(LogLevel.WARNING,"Attempting refresh");
|
|
if (await refreshApiToken(tokens)) {
|
|
headers = {
|
|
"Authorization": "Bearer " + tokens.access_token,
|
|
"Client-ID": config.twitchClientId
|
|
};
|
|
return fetch("https://api.twitch.tv/helix" + endpoint, { headers: headers })
|
|
.then(async (res: FetchResponse) => {
|
|
if (res.status == 200) {
|
|
log(LogLevel.WARNING,"twitch.apiRequest: API call succeeded after token refresh.")
|
|
return res.json();
|
|
} else {
|
|
log(LogLevel.ERROR,"Failed API request:");
|
|
log(LogLevel.ERROR,"Request URL: https://api.twitch.tv/helix" + endpoint);
|
|
log(LogLevel.ERROR,"Headers:");
|
|
log(LogLevel.ERROR,JSON.stringify(headers,null,2));
|
|
log(LogLevel.ERROR,"Response:");
|
|
log(LogLevel.ERROR,JSON.stringify(await res.json(),null,2));
|
|
return false;
|
|
}
|
|
})
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
export async function streamerApiRequest(streamer: StreamerUserIdTokenPair, endpoint: string) {
|
|
log(LogLevel.DEBUG,`Call: twitch.streamerApiRequest(${endpoint})`);
|
|
var originaltoken = streamer.tokenpair.access_token;
|
|
log(LogLevel.DEBUG,"Original token: " + originaltoken);
|
|
var response = await apiRequest(streamer.tokenpair,endpoint);
|
|
log(LogLevel.DEBUG,"New token: " + streamer.tokenpair.access_token);
|
|
if (streamer.tokenpair.access_token != originaltoken)
|
|
await db.query(Object.assign(queries.updateStreamer,
|
|
{ values: [streamer.userid,streamer.tokenpair] }))
|
|
return response;
|
|
}
|
|
|
|
// Check if API token is valid. Checks Twitch's OAuth validation endpoint. If
|
|
// success, return true. If failure, return the result of attempting to refresh
|
|
// the API token.
|
|
export async function isApiTokenValid(tokens: TokenPair) {
|
|
log(LogLevel.DEBUG,`Call: twitch.isApiTokenValid(${JSON.stringify(tokens,null,2)})`);
|
|
return fetch("https://id.twitch.tv/oauth2/validate", {
|
|
headers: {'Authorization': `OAuth ${tokens.access_token}`}
|
|
}).then((res: FetchResponse) => {
|
|
if (res.status == 200) {
|
|
return true;
|
|
} else {
|
|
return refreshApiToken(tokens);
|
|
}
|
|
})
|
|
}
|