1
0

♻️ Extracted tokenData and twitch authentication

To be finished
This commit is contained in:
2021-06-16 18:46:05 +02:00
parent e913b0484c
commit 00f753b761
4 changed files with 177 additions and 72 deletions

View File

@@ -1,5 +1,5 @@
import { PubSubClient, PubSubRedemptionMessage } from "twitch-pubsub-client"; import { PubSubClient, PubSubRedemptionMessage } from "twitch-pubsub-client";
import { RefreshableAuthProvider, StaticAuthProvider } from "twitch-auth"; import { getApiClient, getAuthProvider } from "./src/backend/helpers/twitch";
import { AddressInfo } from "net"; import { AddressInfo } from "net";
import { ApiClient } from "twitch"; import { ApiClient } from "twitch";
@@ -9,7 +9,6 @@ import express from "express";
import { promises as fs } from "fs"; import { promises as fs } from "fs";
import path from "path"; import path from "path";
const TOKENS_FILE = "./tokens.json";
const SCHEDULED_FILE = "./scheduled.json"; const SCHEDULED_FILE = "./scheduled.json";
const DEV_MODE = process.env.NODE_ENV === "development"; const DEV_MODE = process.env.NODE_ENV === "development";
@@ -24,57 +23,9 @@ let chatClient: ChatClient;
//! Important: store users & channels by id, not by username //! Important: store users & channels by id, not by username
async function init() { async function init() {
let tokenData; const authProvider = await getAuthProvider();
try {
tokenData = JSON.parse((await fs.readFile(TOKENS_FILE)).toString());
} catch (error) {
console.error(`${TOKENS_FILE} not found, cannot init chatbot.`);
process.exit(1);
}
if ( apiClient = await getApiClient();
!tokenData.refresh_token ||
!tokenData.access_token
) {
console.error(`Missing parameters in ${TOKENS_FILE}, refresh_token or access_token.`);
process.exit(1);
}
if (
!process.env.TWITCH_CLIENT_ID ||
!process.env.TWITCH_CLIENT_SECRET
) {
console.error(
`Missing environment parameters TWITCH_CLIENT_ID or TWITCH_CLIENT_SECRET`
);
process.exit(1);
}
const authProvider = new RefreshableAuthProvider(
new StaticAuthProvider(
process.env.TWITCH_CLIENT_ID,
tokenData.access_token
),
{
clientSecret: process.env.TWITCH_CLIENT_SECRET,
refreshToken: tokenData.refresh_token,
expiry:
tokenData.expiryTimestamp === null
? null
: new Date(tokenData.expiryTimestamp),
onRefresh: async ({ accessToken, refreshToken, expiryDate }) => {
console.log("Tokens refreshed");
const newTokenData = {
access_token: accessToken,
refresh_token: refreshToken,
expiryTimestamp: expiryDate === null ? null : expiryDate.getTime()
};
await fs.writeFile(TOKENS_FILE, JSON.stringify(newTokenData));
}
}
);
apiClient = new ApiClient({ authProvider });
const pubSubClient = new PubSubClient(); const pubSubClient = new PubSubClient();
const userId = await pubSubClient.registerUserListener(apiClient, channel); const userId = await pubSubClient.registerUserListener(apiClient, channel);
@@ -84,7 +35,22 @@ async function init() {
chatClient = new ChatClient(authProvider, { channels: [channel] }); chatClient = new ChatClient(authProvider, { channels: [channel] });
chatClient.onConnect(async () => { chatClient.onConnect(onConnect);
chatClient.onDisconnect((e: any) => {
console.log(`[ChatClient] Disconnected ${e.message}`);
});
chatClient.onNoPermission((channel, message) => {
console.log(`[ChatClient] No permission on ${channel}: ${message}`);
});
chatClient.connect();
}
init();
async function onConnect() {
console.log("[ChatClient] Connected"); console.log("[ChatClient] Connected");
// *Check this, not working // *Check this, not working
@@ -103,21 +69,8 @@ async function init() {
setTimeout(checkScheduledActions, 1000 * 5); setTimeout(checkScheduledActions, 1000 * 5);
saInterval = setInterval(checkScheduledActions, 1000 * 60); saInterval = setInterval(checkScheduledActions, 1000 * 60);
} }
});
chatClient.onDisconnect((e: any) => {
console.log(`[ChatClient] Disconnected ${e.message}`);
});
chatClient.onNoPermission((channel, message) => {
console.log(`[ChatClient] No permission on ${channel}: ${message}`);
});
chatClient.connect();
} }
init();
async function onRedemption(message: PubSubRedemptionMessage) { async function onRedemption(message: PubSubRedemptionMessage) {
console.log( console.log(
`Reward: "${message.rewardName}" (${message.rewardId}) redeemed by ${message.userDisplayName}` `Reward: "${message.rewardName}" (${message.rewardId}) redeemed by ${message.userDisplayName}`

View File

@@ -0,0 +1,51 @@
import { TokenData } from "../../interfaces/TokenData";
import {promises as fs} from "fs";
import { resolve } from "path";
const TOKENS_FILE = "tokens.json";
const LOG_PREFIX = "[TokenData] ";
export {
getTokenData,
saveTokenData
}
function getTokenDataFilePath(): string {
return resolve(process.cwd(), TOKENS_FILE);
}
async function getTokenData(): Promise<TokenData> {
const tokenDataFilePath = getTokenDataFilePath();
let buffer: Buffer;
try {
buffer = await fs.readFile(tokenDataFilePath);
} catch (e) {
console.error(`${LOG_PREFIX}${TOKENS_FILE} not found on ${tokenDataFilePath}.`);
process.exit(1);
}
const tokenData = await JSON.parse(buffer.toString());
checkTokenData(tokenData);
return tokenData;
}
async function saveTokenData(tokenData: TokenData): Promise<void> {
const tokenDataFilePath = getTokenDataFilePath();
const jsonTokenData = JSON.stringify(tokenData);
await fs.writeFile(tokenDataFilePath, jsonTokenData);
console.log(`${LOG_PREFIX}Token data saved`);
}
function checkTokenData(tokenData: TokenData) {
if (
!tokenData.access_token ||
!tokenData.refresh_token
) {
console.error(`${LOG_PREFIX}Missing refresh_token or access_token in ${TOKENS_FILE}.`);
process.exit(1);
}
}

View File

@@ -0,0 +1,96 @@
import { AccessToken, RefreshableAuthProvider, StaticAuthProvider } from "twitch-auth";
import { getTokenData, saveTokenData } from "./tokenData";
import { ApiClient } from "twitch";
import { TokenData } from "../../interfaces/TokenData";
const LOG_PREFIX = "[Twitch] ";
let refreshAuthProvider: RefreshableAuthProvider;
export {
getAuthProvider,
getApiClient
}
interface ClientCredentials {
clientId: string;
clientSecret: string;
}
function getClientCredentials(): ClientCredentials {
if (
!process.env.TWITCH_CLIENT_ID ||
!process.env.TWITCH_CLIENT_SECRET
) {
console.error(
`${LOG_PREFIX}Missing environment parameters TWITCH_CLIENT_ID or TWITCH_CLIENT_SECRET`
);
process.exit(1);
}
return {
clientId: process.env.TWITCH_CLIENT_ID,
clientSecret: process.env.TWITCH_CLIENT_SECRET
};
}
async function createStaticAuthProvider(): Promise<StaticAuthProvider> {
let tokenData = await getTokenData();
const credentials = getClientCredentials();
return new StaticAuthProvider(
credentials.clientId,
tokenData.access_token
);
}
async function getAuthProvider(): Promise<RefreshableAuthProvider> {
if (refreshAuthProvider) {
return refreshAuthProvider;
}
let tokenData = await getTokenData();
const staticAuthProvider = await createStaticAuthProvider();
const credentials = getClientCredentials();
const expiry = tokenData.expiryTimestamp === null
? null
: new Date(tokenData.expiryTimestamp);
refreshAuthProvider = new RefreshableAuthProvider(
staticAuthProvider,
{
clientSecret: credentials.clientSecret,
refreshToken: tokenData.refresh_token,
expiry,
onRefresh: onRefresh
}
) as RefreshableAuthProvider;
return refreshAuthProvider;
}
async function onRefresh(refreshData: AccessToken): Promise<void> {
const { accessToken, refreshToken, expiryDate } = refreshData;
console.log(`${LOG_PREFIX}Tokens refreshed`);
const expiryTimestamp = expiryDate === null
? 0
: expiryDate.getTime()
const newTokenData: TokenData = {
access_token: accessToken,
refresh_token: refreshToken,
expiryTimestamp
};
await saveTokenData(newTokenData);
}
async function getApiClient() {
const authProvider = await getAuthProvider();
return new ApiClient({ authProvider });
}

View File

@@ -0,0 +1,5 @@
export interface TokenData {
access_token: string;
refresh_token: string;
expiryTimestamp: number;
}