summary refs log tree commit diff
path: root/webrtc/src/Server.ts
diff options
context:
space:
mode:
Diffstat (limited to 'webrtc/src/Server.ts')
-rw-r--r--webrtc/src/Server.ts213
1 files changed, 191 insertions, 22 deletions
diff --git a/webrtc/src/Server.ts b/webrtc/src/Server.ts
index 6591691c..7a1070b9 100644
--- a/webrtc/src/Server.ts
+++ b/webrtc/src/Server.ts
@@ -1,46 +1,215 @@
 import { Server as WebSocketServer } from "ws";
-import { Config, db } from "@fosscord/util";
-import mediasoup from "mediasoup";
+import { WebSocket, CLOSECODES } from "@fosscord/gateway";
+import { Config, initDatabase } from "@fosscord/util";
+import OPCodeHandlers, { Payload } from "./opcodes";
+import { setHeartbeat } from "./util";
+import * as mediasoup from "mediasoup";
+import { types as MediasoupTypes } from "mediasoup";
+import udp from "dgram";
+import sodium from "libsodium-wrappers";
+import { assert } from "console";
 
 var port = Number(process.env.PORT);
 if (isNaN(port)) port = 3004;
 
 export class Server {
 	public ws: WebSocketServer;
-	public turn: any;
+	public mediasoupWorkers: MediasoupTypes.Worker[] = [];
+	public mediasoupRouters: MediasoupTypes.Router[] = [];
+	public mediasoupTransports: MediasoupTypes.WebRtcTransport[] = [];
+	public mediasoupProducers: MediasoupTypes.Producer[] = [];
+	public mediasoupConsumers: MediasoupTypes.Consumer[] = [];
+
+	public decryptKey: Uint8Array;
+	public testUdp = udp.createSocket("udp6");
 
 	constructor() {
 		this.ws = new WebSocketServer({
 			port,
 			maxPayload: 4096,
 		});
-		this.ws.on("connection", (socket) => {
-			socket.on("message", (message) => {
-				socket.emit(
-					JSON.stringify({
-						op: 2,
-						d: {
-							ssrc: 1,
-							ip: "127.0.0.1",
-							port: 3004,
-							modes: [
-								"xsalsa20_poly1305",
-								"xsalsa20_poly1305_suffix",
-								"xsalsa20_poly1305_lite",
-							],
-							heartbeat_interval: 1,
-						},
-					})
-				);
+		this.ws.on("connection", async (socket: WebSocket) => {
+			await setHeartbeat(socket);
+
+			socket.on("message", async (message: string) => {
+				const payload: Payload = JSON.parse(message);
+
+				if (OPCodeHandlers[payload.op])
+					try {
+						await OPCodeHandlers[payload.op].call(this, socket, payload);
+					}
+					catch (e) {
+						console.error(e);
+						socket.close(CLOSECODES.Unknown_error);
+					}
+				else {
+					console.error(`Unimplemented`, payload);
+					socket.close(CLOSECODES.Unknown_opcode);
+				}
 			});
+
+			socket.on("close", (code: number, reason: string) => {
+				console.log(`client closed ${code} ${reason}`);
+				for (var consumer of this.mediasoupConsumers)  consumer.close();
+				for (var producer of this.mediasoupProducers) producer.close();
+				for (var transport of this.mediasoupTransports) transport.close();
+
+				this.mediasoupConsumers = [];
+				this.mediasoupProducers = [];
+				this.mediasoupTransports = [];
+			})
+		});
+
+		this.testUdp.bind(60000);
+		this.testUdp.on("message", (msg, rinfo) => {
+			//random key from like, the libsodium examples on npm lol
+
+			//give me my remote port?
+			if (sodium.to_hex(msg) == "0001004600000001000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000") {
+				this.testUdp.send(Buffer.from([rinfo.port, 0]), rinfo.port, rinfo.address);
+				console.log(`got magic packet to send remote port? ${rinfo.address}:${rinfo.port}`);
+				return;
+			}
+
+			//Hello
+			if (sodium.to_hex(msg) == "0100000000000000") {
+				console.log(`[UDP] client helloed`);
+				return;
+			}
+
+			const nonce = Buffer.concat([msg.slice(-4), Buffer.from("\x00".repeat(20))]);
+			console.log(`[UDP] nonce for this message: ${nonce.toString("hex")}`);
+
+			console.log(`[UDP] message: ${sodium.to_hex(msg)}`);
+
+			// let encrypted;
+			// if (Buffer.from(msg).indexOf("\x81\xc9") == 0) {
+			// 	encrypted = msg.slice(0x18, -4);
+			// }
+			// else if (Buffer.from(msg).indexOf("\x90\x78") == 0) {
+			// 	encrypted = msg.slice(0x1C, -4);
+			// }
+			// else {
+			// 	encrypted = msg.slice(0x18, -4);
+			// 	console.log(`wtf header received: ${encrypted.toString("hex")}`);
+			// }
+
+			let encrypted = msg;
+
+			if (sodium.to_hex(msg).indexOf("80c8000600000001") == 0) {
+				//call status
+
+				encrypted = encrypted.slice(8, -4);
+				assert(encrypted.length == 40);
+
+				try {
+					const decrypted = sodium.crypto_secretbox_open_easy(encrypted, nonce, Buffer.from(this.decryptKey));
+					console.log("[UDP] [ call status ]" + decrypted);
+				}
+				catch (e) {
+					console.error(`[UDP] decrypt failure\n${e}\n${encrypted.toString("base64")}`);
+				}
+				return;
+			}
+
+			// try {
+			// 	const decrypted = sodium.crypto_secretbox_open_easy(encrypted, nonce, Buffer.from(this.decryptKey.map(x => String.fromCharCode(x)).join("")));
+			// 	console.log("[UDP] " + decrypted);
+			// }
+			// catch (e) {
+			// 	console.error(`[UDP] decrypt failure\n${e}\n${msg.toString("base64")}`);
+			// }
 		});
 	}
 
 	async listen(): Promise<void> {
 		// @ts-ignore
-		await (db as Promise<Connection>);
+		await initDatabase();
 		await Config.init();
+		await this.createWorkers();
 		console.log("[DB] connected");
 		console.log(`[WebRTC] online on 0.0.0.0:${port}`);
 	}
+
+	async createWorkers(): Promise<void> {
+		const numWorkers = 1;
+		for (let i = 0; i < numWorkers; i++) {
+			const worker = await mediasoup.createWorker({ logLevel: "debug", logTags: ["dtls", "ice", "info", "message", "bwe"] });
+			if (!worker) return;
+
+			worker.on("died", () => {
+				console.error("mediasoup worker died");
+			});
+
+			worker.observer.on("newrouter", async (router: MediasoupTypes.Router) => {
+				console.log("new router created [id:%s]", router.id);
+
+				this.mediasoupRouters.push(router);
+
+				router.observer.on("newtransport", async (transport: MediasoupTypes.WebRtcTransport) => {
+					console.log("new transport created [id:%s]", transport.id);
+
+					await transport.enableTraceEvent();
+
+					transport.on('dtlsstatechange', (dtlsstate) => {
+						console.log(dtlsstate);
+					});
+
+					transport.on("sctpstatechange", (sctpstate) => {
+						console.log(sctpstate);
+					});
+
+					router.observer.on("newrtpobserver", (rtpObserver: MediasoupTypes.RtpObserver) => {
+						console.log("new RTP observer created [id:%s]", rtpObserver.id);
+
+						// rtpObserver.observer.on("")
+					});
+
+					transport.on("connect", () => {
+						console.log("transport connect");
+					});
+
+					transport.observer.on("newproducer", (producer: MediasoupTypes.Producer) => {
+						console.log("new producer created [id:%s]", producer.id);
+
+						this.mediasoupProducers.push(producer);
+					});
+
+					transport.observer.on("newconsumer", (consumer: MediasoupTypes.Consumer) => {
+						console.log("new consumer created [id:%s]", consumer.id);
+
+						this.mediasoupConsumers.push(consumer);
+
+						consumer.on("rtp", (rtpPacket) => {
+							console.log(rtpPacket);
+						});
+					});
+
+					transport.observer.on("newdataproducer", (dataProducer) => {
+						console.log("new data producer created [id:%s]", dataProducer.id);
+					});
+
+					transport.on("trace", (trace) => {
+						console.log(trace);
+					});
+
+					this.mediasoupTransports.push(transport);
+				});
+			});
+
+			await worker.createRouter({
+				mediaCodecs: [
+					{
+						kind: "audio",
+						mimeType: "audio/opus",
+						clockRate: 48000,
+						channels: 2,
+						preferredPayloadType: 111,
+					},
+				],
+			});
+
+			this.mediasoupWorkers.push(worker);
+		}
+	}
 }