/[sudobot]/branches/8.x/src/services/LogServer.ts
ViewVC logotype

Contents of /branches/8.x/src/services/LogServer.ts

Parent Directory Parent Directory | Revision Log Revision Log


Revision 577 - (show annotations)
Mon Jul 29 18:52:37 2024 UTC (8 months ago) by rakinar2
File MIME type: application/typescript
File size: 3794 byte(s)
chore: add old version archive branches (2.x to 9.x-dev)
1 /**
2 * This file is part of SudoBot.
3 *
4 * Copyright (C) 2021-2023 OSN Developers.
5 *
6 * SudoBot is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU Affero General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * SudoBot is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU Affero General Public License for more details.
15 *
16 * You should have received a copy of the GNU Affero General Public License
17 * along with SudoBot. If not, see <https://www.gnu.org/licenses/>.
18 */
19
20 import bcrypt from "bcrypt";
21 import chalk from "chalk";
22 import { Server, Socket } from "socket.io";
23 import Client from "../core/Client";
24 import Service from "../core/Service";
25 import { logInfo, logWarn } from "../utils/Logger";
26
27 export const name = "logServer";
28
29 export default class LogServer extends Service {
30 public readonly MAX_CONNECTIONS = 5;
31 protected connections = 0;
32 private _io?: Server;
33 public readonly sockets: Array<Socket> = [];
34
35 constructor(client: Client) {
36 super(client);
37
38 if (!client.configManager.systemConfig.log_server?.auto_start) {
39 return;
40 }
41
42 this.initialize();
43 }
44
45 get io() {
46 return this._io;
47 }
48
49 initialize() {
50 this._io = new Server();
51 this.setupListeners();
52 }
53
54 start() {
55 this.initialize();
56 this.listen();
57 }
58
59 close() {
60 this.io?.close();
61 this._io = undefined;
62 }
63
64 isAuthorized(socket: Socket) {
65 const { authorization } = socket.request.headers;
66 const [type, password] = authorization?.split(/\s+/) ?? ["", ""];
67
68 if (type !== "Bearer") {
69 socket.send("[error] Only bearer tokens are supported!");
70 socket.disconnect(true);
71 return false;
72 }
73
74 if (!bcrypt.compareSync(password, process.env.LOG_SERVER_PASSWORD ?? Math.random().toString())) {
75 socket.send("[error] Authentication failed");
76 socket.disconnect(true);
77 return false;
78 }
79
80 return true;
81 }
82
83 setupListeners() {
84 this.io?.on("connection", socket => {
85 if (this.connections >= this.MAX_CONNECTIONS) {
86 socket.disconnect();
87 return;
88 }
89
90 this.connections++;
91 logInfo("New client connected to the log server", socket.id);
92
93 socket.on("disconnect", reason => {
94 this.connections--;
95 logInfo("Client Disconnected from log server: ", socket.id, reason);
96 const index = this.sockets.findIndex(s => s.id === socket.id);
97
98 if (index !== -1) {
99 this.sockets.splice(index, 1);
100 }
101 });
102
103 if (!this.isAuthorized(socket)) {
104 logWarn("Unauthorized client attempted to connect to the log server", socket.id);
105 return;
106 }
107
108 socket.write(`${chalk.cyan("[info]")} Connection Accepted`);
109
110 this.sockets.push(socket);
111 });
112 }
113
114 send(message: string) {
115 for (const socket of this.sockets) {
116 socket.send(message);
117 }
118 }
119
120 log(message: string) {
121 this.sockets.forEach(socket => {
122 socket.send(`${message}`);
123 });
124 }
125
126 listen() {
127 let port = process.env.LOG_SERVER_PORT ? parseInt(process.env.LOG_SERVER_PORT) : 3500;
128 port = isNaN(port) ? 3500 : port;
129
130 this.io?.listen(port);
131 logInfo("The log server is running at port", port);
132 }
133 }

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26