working poc

This commit is contained in:
2025-11-28 13:47:17 -05:00
parent 4dea48c1e2
commit 7b6b278e90
25 changed files with 1018 additions and 168 deletions

View File

@@ -0,0 +1,89 @@
import type { Server, ServerWebSocket } from "bun";
import {LoggerFactory, type ILogger} from '@techniker-me/logger';
import { MessageKindMapping } from "./messaging/MessageKind";
export default class SignalingServer {
private readonly _logger: ILogger = LoggerFactory.getLogger('SignalingServer');
private readonly _port: number;
private readonly _hostname: string;
private readonly _development: boolean;
private readonly _clients: Set<ServerWebSocket<undefined>> = new Set();
constructor(port: number, hostname: string = '0.0.0.0', development: boolean = false) {
this._port = port;
this._hostname = hostname;
this._development = development;
}
get port(): number {
return this._port;
}
get hostname(): string {
return this._hostname;
}
get development(): boolean {
return this._development;
}
get websocket() {
return {
open: this.handleWebSocketOpen.bind(this),
message: this.handleWebSocketMessage.bind(this),
close: this.handleWebSocketClose.bind(this),
drain: this.handleWebSocketDrain.bind(this),
error: this.handleWebSocketError.bind(this),
perMessageDeflate: true,
maxPayloadLength: 10 * 1024
};
}
get fetch() {
return (req: Request, server: Server<undefined>) => {
this._logger.info(`Fetch request received [${req.url}] from [${server.requestIP(req)?.address}:${server.requestIP(req)?.port}]`);
const url = new URL(req.url);
if (url.pathname.endsWith('/ws')) {
this._logger.info('Upgrading to WebSocket');
server.upgrade(req)
return;
}
return new Response('Hello World');
};
}
private handleWebSocketOpen(ws: ServerWebSocket<undefined>): void {
this._logger.info('WebSocket opened');
this._clients.add(ws);
}
private handleWebSocketMessage(ws: ServerWebSocket<undefined>, message: string | Buffer): void {
const messageString = typeof message === 'string' ? message : message.toString();
const jsonMessage = JSON.parse(messageString);
this._logger.info(`WebSocket message received [${MessageKindMapping.convertMessageKindToMessageType(jsonMessage.type)}]`);
// Forward message to all other clients (following sequence diagram)
// This allows the signaling server to relay offers/answers between caller and callee
this._clients.forEach(client => {
if (client !== ws && client.readyState === 1) { // 1 = OPEN
client.send(messageString);
}
});
}
private handleWebSocketClose(ws: ServerWebSocket<undefined>): void {
this._logger.info('WebSocket closed');
this._clients.delete(ws);
}
private handleWebSocketError(ws: ServerWebSocket<undefined>, error: Error): void {
this._logger.error('WebSocket error', error);
}
private handleWebSocketDrain(ws: ServerWebSocket<undefined>): void {
this._logger.info('WebSocket drained');
}
}

7
signaling/src/index.ts Normal file
View File

@@ -0,0 +1,7 @@
import SignalingServer from "./SignalingServer";
const signalingServer = new SignalingServer(3000, '0.0.0.0', true);
Bun.serve(signalingServer);
console.log(`Signaling server started on [${signalingServer.hostname}:${signalingServer.port}]`);

1
signaling/src/messaging Symbolic link
View File

@@ -0,0 +1 @@
../../frontend-web-vanilla/src/messaging