mirror of
https://github.com/hensm/fx_cast.git
synced 2026-06-08 08:39:59 +00:00
148 lines
4.0 KiB
TypeScript
148 lines
4.0 KiB
TypeScript
import http from "http";
|
|
import https from "https";
|
|
import { ChildProcess, spawn } from "child_process";
|
|
import { Readable } from "stream";
|
|
|
|
import WebSocket from "ws";
|
|
|
|
import { DecodeTransform, EncodeTransform } from "./transforms.js";
|
|
|
|
const bridgeInstances = new Set<ChildProcess>();
|
|
|
|
// Ensure child processes are killed on exit
|
|
process.on("SIGTERM", async () => {
|
|
for (const bridge of bridgeInstances) {
|
|
bridge.kill();
|
|
}
|
|
process.exit(1);
|
|
});
|
|
|
|
export interface DaemonOpts {
|
|
host: string;
|
|
port: number;
|
|
password?: string;
|
|
secure?: boolean;
|
|
key?: Buffer;
|
|
cert?: Buffer;
|
|
}
|
|
|
|
export function init(opts: DaemonOpts) {
|
|
const server = !opts.secure
|
|
? http.createServer()
|
|
: https.createServer({
|
|
key: opts.key,
|
|
cert: opts.cert
|
|
});
|
|
|
|
const wss = new WebSocket.Server({ noServer: true });
|
|
|
|
wss.on("connection", socket => {
|
|
// Stream for incoming WebSocket messages
|
|
const messageStream = new Readable({ objectMode: true });
|
|
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
|
messageStream._read = () => {};
|
|
|
|
socket.on("message", (message: string) => {
|
|
try {
|
|
messageStream.push(JSON.parse(message));
|
|
} catch (err) {
|
|
// Catch parse errors and close socket
|
|
socket.close();
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Daemon and bridge are the same binary, so spawn a new
|
|
* version of self in bridge mode.
|
|
*/
|
|
const bridge = spawn(process.execPath, [process.argv[1]]);
|
|
bridgeInstances.add(bridge);
|
|
|
|
// socket -> bridge.stdin
|
|
messageStream.pipe(new EncodeTransform()).pipe(bridge.stdin);
|
|
|
|
// bridge.stdout -> socket
|
|
bridge.stdout.pipe(new DecodeTransform()).on("data", data => {
|
|
if (socket.readyState !== WebSocket.OPEN) {
|
|
return;
|
|
}
|
|
|
|
socket.send(JSON.stringify(data));
|
|
});
|
|
|
|
// Handle termination
|
|
socket.on("close", () => bridge.kill());
|
|
bridge.on("exit", () => {
|
|
socket.close();
|
|
bridgeInstances.delete(bridge);
|
|
});
|
|
});
|
|
|
|
/**
|
|
* Authenticates requests by checking password URL param against
|
|
* server password specified in launch options.
|
|
*/
|
|
function authenticate(req: http.IncomingMessage) {
|
|
if (!opts.password) return true;
|
|
|
|
const password = new URL(
|
|
req.url!,
|
|
`http://${req.headers.host}`
|
|
).searchParams.get("password");
|
|
|
|
return password === opts.password;
|
|
}
|
|
|
|
server.on("upgrade", (req, socket, head) => {
|
|
/**
|
|
* Only accept authenticated WebSocket requests from extension
|
|
* origins.
|
|
*/
|
|
if (
|
|
req.headers.origin?.startsWith("moz-extension://") &&
|
|
authenticate(req)
|
|
) {
|
|
wss.handleUpgrade(req, socket, head, ws => {
|
|
wss.emit("connection", ws, req);
|
|
});
|
|
|
|
return;
|
|
}
|
|
|
|
socket.write("HTTP/1.1 401 Unauthorized\r\n\r\n");
|
|
socket.destroy();
|
|
});
|
|
|
|
/**
|
|
* Browser WebSocket API does not allow access to connection errors,
|
|
* so provide an endpoint for feedback on invalid authentication.
|
|
*/
|
|
server.on("request", (req, res) => {
|
|
/**
|
|
* Requests from extensions have their origin header stripped,
|
|
* so block all requests with origin headers.
|
|
*/
|
|
if ("origin" in req.headers) {
|
|
req.destroy();
|
|
return;
|
|
}
|
|
|
|
res.writeHead(authenticate(req) ? 200 : 401);
|
|
res.end();
|
|
});
|
|
|
|
process.stdout.write(
|
|
`Starting WebSocket server at ${opts.secure ? "wss" : "ws"}://${
|
|
opts.host.includes(":") ? `[${opts.host}]` : opts.host
|
|
}:${opts.port}... `
|
|
);
|
|
server.listen({ port: opts.port, host: opts.host }, () => {
|
|
process.stdout.write("Done!\n");
|
|
});
|
|
|
|
server.on("error", err => {
|
|
console.error("Failed!");
|
|
console.error(err.message);
|
|
});
|
|
}
|