2023-11-26 21:33:46 +01:00
|
|
|
import type * as http from "node:http";
|
2024-04-19 20:09:18 +02:00
|
|
|
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";
|
2023-12-05 08:12:10 +01:00
|
|
|
import authenticate from "./authenticate.js";
|
2016-12-28 23:49:51 +01:00
|
|
|
|
2022-02-27 03:07:39 +01:00
|
|
|
export const initializeStreamingServer = (server: http.Server) => {
|
2018-10-07 04:06:17 +02:00
|
|
|
// 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) => {
|
2018-10-07 04:06:17 +02:00
|
|
|
const q = request.resourceURL.query as ParsedUrlQuery;
|
2023-02-11 00:41:19 +01:00
|
|
|
const headers = request.httpRequest.headers["sec-websocket-protocol"] || "";
|
2023-02-09 23:21:50 +01:00
|
|
|
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,
|
2023-02-09 23:21:50 +01:00
|
|
|
accessToken,
|
2023-01-13 05:40:33 +01:00
|
|
|
).catch((err) => {
|
|
|
|
request.reject(403, err.message);
|
|
|
|
return [];
|
|
|
|
});
|
|
|
|
if (typeof user === "undefined") {
|
2022-07-18 23:32:03 +02:00
|
|
|
return;
|
|
|
|
}
|
2017-06-08 18:03:54 +02:00
|
|
|
|
2021-07-18 12:57:53 +02:00
|
|
|
if (user?.isSuspended) {
|
|
|
|
request.reject(400);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-11-11 16:31:09 +01:00
|
|
|
const connection = request.accept();
|
|
|
|
|
2021-03-23 03:53:25 +01:00
|
|
|
const ev = new EventEmitter();
|
2018-10-11 11:09:41 +02:00
|
|
|
|
2021-03-23 03:53:25 +01:00
|
|
|
async function onRedisMessage(_: string, data: string) {
|
|
|
|
const parsed = JSON.parse(data);
|
|
|
|
ev.emit(parsed.channel, parsed.message);
|
|
|
|
}
|
2018-10-11 11:09:41 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
redisClient.on("message", onRedisMessage);
|
2023-02-09 23:21:50 +01:00
|
|
|
const host = `https://${request.host}`;
|
|
|
|
const prepareStream = q.stream?.toString();
|
2023-02-11 00:41:19 +01:00
|
|
|
console.log("start", q);
|
2018-10-11 11:09:41 +02:00
|
|
|
|
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,
|
2024-03-28 06:25:33 +01:00
|
|
|
)
|
2023-01-13 05:40:33 +01:00
|
|
|
: null;
|
2021-04-18 15:35:47 +02:00
|
|
|
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();
|
2018-10-07 04:06:17 +02:00
|
|
|
main.dispose();
|
2023-01-13 05:40:33 +01:00
|
|
|
redisClient.off("message", onRedisMessage);
|
2021-04-18 15:35:47 +02:00
|
|
|
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
|
|
|
});
|
|
|
|
};
|