// Gateway Bench Probes script supports OpenClaw repository automation. import { spawnSync } from "node:child_process"; import { request } from "node:http"; import { createServer } from "node:net"; export async function getFreePort(): Promise { return new Promise((resolve, reject) => { const server = createServer(); server.on("error", reject); server.listen(0, "127.0.0.1", () => { const address = server.address(); if (!address || typeof address === "string") { server.close(() => reject(new Error("failed to allocate port"))); return; } const { port } = address; server.close(() => resolve(port)); }); }); } export async function requestProbeStatus( port: number, pathname: string, ): Promise<{ errorKind: string | null; status: number | null }> { try { const status = await requestStatus(port, pathname); return { errorKind: status === 200 ? null : `http-${status}`, status, }; } catch (error) { return { errorKind: classifyProbeErrorKind(error), status: null, }; } } export function classifyProbeErrorKind(error: unknown): string { if (typeof error === "object" && error !== null) { const code = (error as { code?: unknown }).code; if (typeof code === "string" && code.trim()) { return code.trim().toLowerCase(); } const message = (error as { message?: unknown }).message; if (typeof message === "string" && message.toLowerCase().includes("probe timeout")) { return "timeout"; } const name = (error as { name?: unknown }).name; if (typeof name === "string" && name.trim()) { return name.trim().toLowerCase(); } } return "error"; } export function readProcessRssMb(pid: number | undefined): number | null { if (!pid || process.platform === "win32") { return null; } const result = spawnSync("ps", ["-o", "rss=", "-p", String(pid)], { encoding: "utf8", stdio: ["ignore", "pipe", "ignore"], }); if (result.status !== 0) { return null; } const rssKb = parseProcessRssKb(result.stdout); return rssKb === null ? null : rssKb / 1024; } export function parseProcessRssKb(raw: string): number | null { const value = raw.trim(); if (!/^[1-9][0-9]*$/u.test(value)) { return null; } const rssKb = Number(value); return Number.isSafeInteger(rssKb) ? rssKb : null; } export function readProcessTreeCpuMs(rootPid: number | undefined): number | null { if (!rootPid || process.platform === "win32") { return null; } const result = spawnSync("ps", ["-eo", "pid=,ppid=,time="], { encoding: "utf8", stdio: ["ignore", "pipe", "ignore"], }); if (result.status !== 0) { return null; } const childrenByParent = new Map(); const cpuByPid = new Map(); for (const line of result.stdout.split("\n")) { const match = line.trim().match(/^(\d+)\s+(\d+)\s+(\S+)$/u); if (!match) { continue; } const pid = Number(match[1]); const ppid = Number(match[2]); const cpuMs = parsePsCpuTimeMs(match[3]); if (!Number.isInteger(pid) || !Number.isInteger(ppid) || cpuMs === null) { continue; } cpuByPid.set(pid, cpuMs); const children = childrenByParent.get(ppid) ?? []; children.push(pid); childrenByParent.set(ppid, children); } if (!cpuByPid.has(rootPid)) { return null; } let totalCpuMs = 0; const seen = new Set(); const stack = [rootPid]; while (stack.length > 0) { const pid = stack.pop(); if (!pid || seen.has(pid)) { continue; } seen.add(pid); totalCpuMs += cpuByPid.get(pid) ?? 0; for (const childPid of childrenByParent.get(pid) ?? []) { stack.push(childPid); } } return totalCpuMs; } function requestStatus(port: number, pathname: string): Promise { return new Promise((resolve, reject) => { const req = request( { host: "127.0.0.1", method: "GET", path: pathname, port, timeout: 100 }, (res) => { res.resume(); res.on("end", () => resolve(res.statusCode ?? 0)); }, ); req.on("error", reject); req.on("timeout", () => { req.destroy(new Error("probe timeout")); }); req.end(); }); } function parsePsCpuTimeMs(raw: string): number | null { const parts = raw.trim().split(":").map(Number); if (parts.some((part) => !Number.isFinite(part) || part < 0)) { return null; } if (parts.length === 2) { return Math.round((parts[0] * 60 + parts[1]) * 1000); } if (parts.length === 3) { return Math.round((parts[0] * 60 * 60 + parts[1] * 60 + parts[2]) * 1000); } return null; }