excalidraw-backend/src/index.ts
2024-06-06 09:01:58 +02:00

123 lines
3.6 KiB
TypeScript

import debug from 'debug';
import dotenv from 'dotenv';
import express from 'express';
import http from 'http';
import { Socket, Server } from 'socket.io';
//import * as prometheus from 'socket.io-prometheus-metrics';
const serverDebug = debug('server');
dotenv.config(
process.env.NODE_ENV === 'development'
? { path: '.env.development' }
: { path: '.env.production' }
);
const app = express();
const port = process.env.PORT || 80; // default port to listen
const users: Socket[] = [];
const userLimit = Number(process.env.USER_LIMIT) || Infinity;
app.get('/', (req, res) => {
res.send('Excalidraw backend is up :)');
});
const server = http.createServer(app);
server.listen(port, () => {
serverDebug(`listening on port: ${port}`);
});
const corsOptions = {
origin: process.env.CORS_ORIGIN,
methods: [ 'GET', 'POST' ]
};
const io = new Server(server, {
allowEIO3: true,
cors: corsOptions,
maxHttpBufferSize: 20e6,
pingTimeout: 60000
});
// listens on host:9090/metrics
// prometheus.metrics(io, {
// collectDefaultMetrics: true
// });
io.on('connection', socket => {
serverDebug(`connection established! ${socket.conn.request.url}`);
io.to(`${socket.id}`).emit('init-room');
socket.on('join-room', roomID => {
serverDebug(`${socket.id} has joined ${roomID} for url ${socket.conn.request.url}`);
socket.join(roomID);
users.push(socket);
socket.on('close', () => {
users.splice(users.indexOf(socket), 1);
});
const clients = Object.keys(io.sockets.adapter.rooms.get(roomID)?.keys() ?? new Set<string>());
if (clients.length > userLimit) {
clients.forEach((clientKey: string) => {
const clientSocket = io.sockets.sockets.get(clientKey);
if (clientSocket !== undefined) {
serverDebug(`${clientSocket} has left the ${roomID} room because the user limit was reached.`);
clientSocket.leave(roomID);
}
});
return;
}
if (io.sockets.adapter.rooms.get(roomID)?.size ?? 0 <= 1) {
io.to(`${socket.id}`).emit('first-in-room');
} else {
socket.broadcast.to(roomID).emit('new-user', socket.id);
}
io.in(roomID).emit(
'room-user-change', Array.from(io.sockets.adapter.rooms.get(roomID) ?? [])
);
});
socket.on(
'server-broadcast',
(roomID: string, encryptedData: ArrayBuffer, iv: Uint8Array) => {
socket.broadcast.to(roomID).emit('client-broadcast', encryptedData, iv);
}
);
socket.on(
'server-volatile-broadcast',
(roomID: string, encryptedData: ArrayBuffer, iv: Uint8Array) => {
socket.volatile.broadcast
.to(roomID)
.emit('client-broadcast', encryptedData, iv);
}
);
socket.on('disconnecting', () => {
const rooms = io.sockets.adapter.rooms;
for (const roomID of Object.keys(socket.rooms)) {
const clients = Array.from(rooms.get(roomID) ?? []).filter(id => id !== socket.id);
if (roomID !== socket.id) {
socket.to(roomID).emit('user has left', socket.id);
}
if (clients.length > 0) {
socket.broadcast.to(roomID).emit('room-user-change', clients);
}
}
});
socket.on('disconnect', (reason, details) => {
serverDebug(
`${socket.id} was disconnected from url ${socket.conn.request.url} for the following reason: ${reason}
${JSON.stringify(details)}`
);
socket.removeAllListeners();
});
});