hippofish/packages/backend/src/server/api/streaming.ts

93 lines
2.2 KiB
TypeScript
Raw Normal View History

2023-11-26 21:33:46 +01:00
import type * as http from "node:http";
import { EventEmitter } from "node:events";
import type { ParsedUrlQuery } from "node:querystring";
2023-01-13 05:40:33 +01:00
import * as websocket from "websocket";
2016-12-28 23:49:51 +01:00
2023-01-13 05:40:33 +01:00
import { subscriber as redisClient } from "@/db/redis.js";
import { Users } from "@/models/index.js";
2023-11-26 21:33:46 +01:00
import MainStreamConnection from "./stream/index.js";
import authenticate from "./authenticate.js";
2016-12-28 23:49:51 +01:00
export const initializeStreamingServer = (server: http.Server) => {
// Init websocket server
2016-12-28 23:49:51 +01:00
const ws = new websocket.server({
2021-12-09 15:58:30 +01:00
httpServer: server,
2016-12-28 23:49:51 +01:00
});
2023-01-13 05:40:33 +01:00
ws.on("request", async (request) => {
const q = request.resourceURL.query as ParsedUrlQuery;
2023-02-11 00:41:19 +01:00
const headers = request.httpRequest.headers["sec-websocket-protocol"] || "";
const cred = q.i || q.access_token || headers;
const accessToken = cred.toString();
2020-04-26 04:19:57 +02:00
2023-01-13 05:40:33 +01:00
const [user, app] = await authenticate(
request.httpRequest.headers.authorization,
accessToken,
2023-01-13 05:40:33 +01:00
).catch((err) => {
request.reject(403, err.message);
return [];
});
if (typeof user === "undefined") {
return;
}
2017-06-08 18:03:54 +02:00
if (user?.isSuspended) {
request.reject(400);
return;
}
2018-11-11 16:31:09 +01:00
const connection = request.accept();
const ev = new EventEmitter();
async function onRedisMessage(_: string, data: string) {
const parsed = JSON.parse(data);
ev.emit(parsed.channel, parsed.message);
}
2023-01-13 05:40:33 +01:00
redisClient.on("message", onRedisMessage);
const host = `https://${request.host}`;
const prepareStream = q.stream?.toString();
2023-02-11 00:41:19 +01:00
console.log("start", q);
2023-02-11 00:41:19 +01:00
const main = new MainStreamConnection(
connection,
ev,
user,
app,
host,
accessToken,
prepareStream,
);
2016-12-28 23:49:51 +01:00
2023-01-13 05:40:33 +01:00
const intervalId = user
2024-02-11 18:50:57 +01:00
? setInterval(
() => {
Users.update(user.id, {
lastActiveDate: new Date(),
});
},
1000 * 60 * 5,
)
2023-01-13 05:40:33 +01:00
: null;
if (user) {
Users.update(user.id, {
lastActiveDate: new Date(),
});
}
2023-01-13 05:40:33 +01:00
connection.once("close", () => {
2018-07-30 00:20:27 +02:00
ev.removeAllListeners();
main.dispose();
2023-01-13 05:40:33 +01:00
redisClient.off("message", onRedisMessage);
if (intervalId) clearInterval(intervalId);
2016-12-28 23:49:51 +01:00
});
2023-01-13 05:40:33 +01:00
connection.on("message", async (data) => {
if (data.type === "utf8" && data.utf8Data === "ping") {
connection.send("pong");
2018-09-17 02:07:46 +02:00
}
});
2016-12-28 23:49:51 +01:00
});
};