1
0

♻️ Extracted web server and websocket

initializacion and added helper to get NODE_ENV value
This commit is contained in:
2021-06-16 23:29:37 +02:00
parent 00f753b761
commit 42bd583a43
3 changed files with 114 additions and 72 deletions

View File

@@ -1,16 +1,12 @@
import { PubSubClient, PubSubRedemptionMessage } from "twitch-pubsub-client"; import { PubSubClient, PubSubRedemptionMessage } from "twitch-pubsub-client";
import { getApiClient, getAuthProvider } from "./src/backend/helpers/twitch"; import { getApiClient, getAuthProvider } from "./src/backend/helpers/twitch";
import { listen, sockets } from "./src/backend/webServer";
import { AddressInfo } from "net";
import { ApiClient } from "twitch"; import { ApiClient } from "twitch";
import { ChatClient } from "twitch-chat-client"; import { ChatClient } from "twitch-chat-client";
import WebSocket from "ws";
import express from "express";
import { promises as fs } from "fs"; import { promises as fs } from "fs";
import path from "path";
const SCHEDULED_FILE = "./scheduled.json"; const SCHEDULED_FILE = "./scheduled.json";
const DEV_MODE = process.env.NODE_ENV === "development";
const scheduledActions: Array<any> = []; const scheduledActions: Array<any> = [];
let saInterval: NodeJS.Timeout; let saInterval: NodeJS.Timeout;
@@ -20,6 +16,12 @@ const channel = "alexbcberio";
let apiClient: ApiClient; let apiClient: ApiClient;
let chatClient: ChatClient; let chatClient: ChatClient;
export {
handleClientAction,
scheduledActions,
saveScheduledActions
}
//! Important: store users & channels by id, not by username //! Important: store users & channels by id, not by username
async function init() { async function init() {
@@ -46,6 +48,8 @@ async function init() {
}); });
chatClient.connect(); chatClient.connect();
listen();
} }
init(); init();
@@ -100,58 +104,12 @@ async function onRedemption(message: PubSubRedemptionMessage) {
} }
if (msg) { if (msg) {
console.log(msg);
broadcast(msg); broadcast(msg);
} }
} }
const app = express();
const wsServer = new WebSocket.Server({
noServer: true
});
let sockets: Array<WebSocket> = [];
wsServer.on("connection", (socket, req) => {
console.log(`[WS] ${req.socket.remoteAddress} New connection established`);
sockets.push(socket);
socket.send(
JSON.stringify({
env: DEV_MODE ? "dev" : "prod"
})
);
socket.on("message", async (msg: string) => {
const data = JSON.parse(msg);
// broadcast message
if (!data.actions || data.actions.length === 0) {
sockets
.filter(s => s !== socket)
.forEach(s => s.send(msg));
return;
}
for (const action of data.actions) {
if (!action.scheduledAt) {
await handleClientAction(action);
} else {
scheduledActions.push(action);
scheduledActions.sort((a, b) => a.scheduledAt - b.scheduledAt);
saveScheduledActions();
}
}
console.log(
`[WS] Received message with ${data.actions.length} actions:`,
data
);
});
socket.on("close", () => {
sockets = sockets.filter(s => s !== socket);
console.log("[WS] Connection closed");
});
});
async function handleClientAction(action: any) { async function handleClientAction(action: any) {
if (action.channel && !isNaN(action.channel)) { if (action.channel && !isNaN(action.channel)) {
action.channel = await getUsernameFromId(parseInt(action.channel)); action.channel = await getUsernameFromId(parseInt(action.channel));
@@ -325,21 +283,3 @@ async function stealVip(msg: {
return null; return null;
} }
/*
Webserver
*/
app.use(express.static(path.join(__dirname, "client")));
const server = app.listen(!DEV_MODE ? 8080 : 8081, "0.0.0.0");
server.on("listening", () => {
console.log(
`[Webserver] Listening on port ${(server.address() as AddressInfo).port}`
);
});
server.on("upgrade", (req, socket, head) => {
wsServer.handleUpgrade(req, socket, head, socket => {
wsServer.emit("connection", socket, req);
});
});

View File

@@ -0,0 +1,9 @@
const environment = process.env.NODE_ENV;
const isDevelopment = environment === "development";
const isProduction = environment === "production";
export {
environment,
isDevelopment,
isProduction
};

View File

@@ -0,0 +1,93 @@
import { IncomingMessage, Server } from "http";
import { handleClientAction, saveScheduledActions, scheduledActions } from "../../..";
import { AddressInfo } from "net";
import { Socket } from "net";
import WebSocket from "ws";
import express from "express";
import { isDevelopment } from "../helpers/util";
import { join } from "path";
const app = express();
const sockets: Array<WebSocket> = [];
const wsServer = new WebSocket.Server({
noServer: true
});
let server: Server;
export {
listen,
sockets
}
wsServer.on("connection", onConnection);
app.use(express.static(join(process.cwd(), "client")));
function listen() {
if (server) {
console.log("[Webserver] Server is already running");
return;
}
server = app.listen(!isDevelopment ? 8080 : 8081, "0.0.0.0");
server.on("listening", onListening);
server.on("upgrade", onUpgrade);
}
function onListening() {
console.log(
`[Webserver] Listening on port ${(server.address() as AddressInfo).port}`
);
}
function onUpgrade(req: IncomingMessage, socket: Socket, head: Buffer) {
wsServer.handleUpgrade(req, socket, head, socket => {
wsServer.emit("connection", socket, req);
});
}
function onConnection(socket: WebSocket, req: IncomingMessage) {
console.log(`[WS] ${req.socket.remoteAddress} New connection established`);
sockets.push(socket);
socket.send(
JSON.stringify({
env: isDevelopment ? "dev" : "prod"
})
);
socket.on("message", (msg: string) => onMessage(msg, socket));
socket.on("close", () => onClose(socket));
}
async function onMessage(msg: string, socket: WebSocket) {
const data = JSON.parse(msg);
// broadcast message
if (!data.actions || data.actions.length === 0) {
sockets.filter(s => s !== socket).forEach(s => s.send(msg));
return;
}
for (const action of data.actions) {
if (!action.scheduledAt) {
await handleClientAction(action);
} else {
scheduledActions.push(action);
scheduledActions.sort((a, b) => a.scheduledAt - b.scheduledAt);
saveScheduledActions();
}
}
console.log(`[WS] Received message with ${data.actions.length} actions:`, data);
}
function onClose(socket: WebSocket) {
const socketIdx = sockets.indexOf(socket);
sockets.splice(socketIdx, 1);
console.log("[WS] Connection closed");
}