mirror of
https://github.com/different-ai/openwork
synced 2026-05-05 06:42:03 +02:00
Bind Vite to an explicit host/port and fail fast when the server binary is missing to avoid localhost-only binds and silent startup failures.
185 lines
5.4 KiB
TypeScript
185 lines
5.4 KiB
TypeScript
import { spawn } from "node:child_process";
|
|
import { openSync } from "node:fs";
|
|
import { access, mkdir } from "node:fs/promises";
|
|
import { createServer } from "node:net";
|
|
import { randomUUID } from "node:crypto";
|
|
import path from "node:path";
|
|
|
|
const cwd = process.cwd();
|
|
const tmpDir = path.join(cwd, "tmp");
|
|
|
|
const ensureTmp = async () => {
|
|
await mkdir(tmpDir, { recursive: true });
|
|
};
|
|
|
|
const isPortFree = (port: number, host: string) =>
|
|
new Promise<boolean>((resolve) => {
|
|
const server = createServer();
|
|
server.once("error", () => resolve(false));
|
|
server.listen(port, host, () => {
|
|
server.close(() => resolve(true));
|
|
});
|
|
});
|
|
|
|
const getFreePort = (host: string) =>
|
|
new Promise<number>((resolve, reject) => {
|
|
const server = createServer();
|
|
server.once("error", reject);
|
|
server.listen(0, host, () => {
|
|
const address = server.address();
|
|
if (!address || typeof address === "string") {
|
|
server.close(() => reject(new Error("Unable to resolve free port")));
|
|
return;
|
|
}
|
|
const port = address.port;
|
|
server.close(() => resolve(port));
|
|
});
|
|
});
|
|
|
|
const resolvePort = async (value: string | undefined, host: string) => {
|
|
if (value) {
|
|
const parsed = Number(value);
|
|
if (Number.isFinite(parsed) && parsed > 0) {
|
|
const free = await isPortFree(parsed, host);
|
|
if (free) return parsed;
|
|
}
|
|
}
|
|
return await getFreePort(host);
|
|
};
|
|
|
|
const logLine = (message: string) => {
|
|
process.stdout.write(`${message}\n`);
|
|
};
|
|
|
|
const spawnLogged = (command: string, args: string[], logPath: string, env: NodeJS.ProcessEnv) => {
|
|
const logFd = openSync(logPath, "w");
|
|
return spawn(command, args, {
|
|
cwd,
|
|
env,
|
|
stdio: ["ignore", logFd, logFd],
|
|
});
|
|
};
|
|
|
|
const shutdown = (label: string, code: number | null, signal: NodeJS.Signals | null) => {
|
|
const reason = code !== null ? `code ${code}` : signal ? `signal ${signal}` : "unknown";
|
|
logLine(`[dev:headless-web] ${label} exited (${reason})`);
|
|
process.exit(code ?? 1);
|
|
};
|
|
|
|
await ensureTmp();
|
|
|
|
const host = process.env.OPENWORK_HOST ?? "0.0.0.0";
|
|
const viteHost = process.env.VITE_HOST ?? process.env.HOST ?? host;
|
|
const publicHost = process.env.OPENWORK_PUBLIC_HOST ?? null;
|
|
const clientHost = publicHost ?? (host === "0.0.0.0" ? "127.0.0.1" : host);
|
|
const workspace = process.env.OPENWORK_WORKSPACE ?? cwd;
|
|
const openworkPort = await resolvePort(process.env.OPENWORK_PORT, "127.0.0.1");
|
|
const webPort = await resolvePort(process.env.OPENWORK_WEB_PORT, "127.0.0.1");
|
|
const openworkToken = process.env.OPENWORK_TOKEN ?? randomUUID();
|
|
const openworkHostToken = process.env.OPENWORK_HOST_TOKEN ?? randomUUID();
|
|
const openworkServerBin = path.join(cwd, "packages/server/dist/bin/openwork-server");
|
|
|
|
const ensureOpenworkServer = async () => {
|
|
try {
|
|
await access(openworkServerBin);
|
|
} catch {
|
|
logLine(`[dev:headless-web] Missing OpenWork server binary at ${openworkServerBin}`);
|
|
logLine("[dev:headless-web] Run: pnpm --filter openwork-server build:bin");
|
|
process.exit(1);
|
|
}
|
|
};
|
|
|
|
const openworkUrl = `http://${clientHost}:${openworkPort}`;
|
|
const webUrl = `http://${clientHost}:${webPort}`;
|
|
const viteEnv = {
|
|
...process.env,
|
|
HOST: viteHost,
|
|
PORT: String(webPort),
|
|
VITE_OPENWORK_URL: process.env.VITE_OPENWORK_URL ?? openworkUrl,
|
|
VITE_OPENWORK_PORT: process.env.VITE_OPENWORK_PORT ?? String(openworkPort),
|
|
VITE_OPENWORK_TOKEN: process.env.VITE_OPENWORK_TOKEN ?? openworkToken,
|
|
};
|
|
const headlessEnv = {
|
|
...process.env,
|
|
OPENWORK_WORKSPACE: workspace,
|
|
OPENWORK_HOST: host,
|
|
OPENWORK_PORT: String(openworkPort),
|
|
OPENWORK_TOKEN: openworkToken,
|
|
OPENWORK_HOST_TOKEN: openworkHostToken,
|
|
OPENWORK_SERVER_BIN: openworkServerBin,
|
|
};
|
|
|
|
await ensureOpenworkServer();
|
|
|
|
logLine("[dev:headless-web] Starting services");
|
|
logLine(`[dev:headless-web] Workspace: ${workspace}`);
|
|
logLine(`[dev:headless-web] OpenWork server: ${openworkUrl}`);
|
|
logLine(`[dev:headless-web] Web host: ${viteHost}`);
|
|
logLine(`[dev:headless-web] Web port: ${webPort}`);
|
|
logLine(`[dev:headless-web] Web URL: ${webUrl}`);
|
|
logLine(`[dev:headless-web] OPENWORK_TOKEN: ${openworkToken}`);
|
|
logLine(`[dev:headless-web] OPENWORK_HOST_TOKEN: ${openworkHostToken}`);
|
|
logLine(`[dev:headless-web] Web logs: ${path.relative(cwd, path.join(tmpDir, "dev-web.log"))}`);
|
|
logLine(`[dev:headless-web] Headless logs: ${path.relative(cwd, path.join(tmpDir, "dev-headless.log"))}`);
|
|
|
|
const webProcess = spawnLogged(
|
|
"pnpm",
|
|
[
|
|
"--filter",
|
|
"@different-ai/openwork-ui",
|
|
"exec",
|
|
"vite",
|
|
"--host",
|
|
viteHost,
|
|
"--port",
|
|
String(webPort),
|
|
"--strictPort",
|
|
],
|
|
path.join(tmpDir, "dev-web.log"),
|
|
viteEnv,
|
|
);
|
|
|
|
const headlessProcess = spawnLogged(
|
|
"pnpm",
|
|
[
|
|
"--filter",
|
|
"openwrk",
|
|
"dev",
|
|
"--",
|
|
"start",
|
|
"--workspace",
|
|
workspace,
|
|
"--approval",
|
|
"auto",
|
|
"--allow-external",
|
|
"--no-opencode-auth",
|
|
"--owpenbot",
|
|
"false",
|
|
"--openwork-host",
|
|
host,
|
|
"--openwork-port",
|
|
String(openworkPort),
|
|
"--openwork-token",
|
|
openworkToken,
|
|
"--openwork-host-token",
|
|
openworkHostToken,
|
|
],
|
|
path.join(tmpDir, "dev-headless.log"),
|
|
headlessEnv,
|
|
);
|
|
|
|
const stopAll = (signal: NodeJS.Signals) => {
|
|
webProcess.kill(signal);
|
|
headlessProcess.kill(signal);
|
|
};
|
|
|
|
process.on("SIGINT", () => {
|
|
stopAll("SIGINT");
|
|
});
|
|
process.on("SIGTERM", () => {
|
|
stopAll("SIGTERM");
|
|
});
|
|
|
|
webProcess.on("exit", (code, signal) => shutdown("web", code, signal));
|
|
headlessProcess.on("exit", (code, signal) => shutdown("openwrk", code, signal));
|