mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-07 02:50:42 +00:00
102 lines
2.8 KiB
TypeScript
102 lines
2.8 KiB
TypeScript
import { writeFile } from "node:fs/promises";
|
|
import path from "node:path";
|
|
import { say, warn } from "./host-command.ts";
|
|
|
|
export class PhaseRunner {
|
|
private logText = "";
|
|
private deadlineMs = 0;
|
|
private timings: Array<{
|
|
durationMs: number;
|
|
logPath: string;
|
|
name: string;
|
|
status: "pass" | "fail";
|
|
timeoutSeconds: number;
|
|
}> = [];
|
|
|
|
constructor(private runDir: string) {}
|
|
|
|
async phase(name: string, timeoutSeconds: number, fn: () => Promise<void> | void): Promise<void> {
|
|
const logPath = path.join(this.runDir, `${name}.log`);
|
|
say(name);
|
|
this.logText = "";
|
|
this.deadlineMs = Date.now() + timeoutSeconds * 1000;
|
|
const startedAt = Date.now();
|
|
let status: "pass" | "fail" = "fail";
|
|
let timer: NodeJS.Timeout | undefined;
|
|
const timeout = new Promise<never>((_, reject) => {
|
|
timer = setTimeout(
|
|
() => reject(new Error(`${name} timed out after ${timeoutSeconds}s`)),
|
|
timeoutSeconds * 1000,
|
|
);
|
|
});
|
|
try {
|
|
await Promise.race([Promise.resolve(fn()), timeout]);
|
|
await writeFile(logPath, this.logText, "utf8");
|
|
status = "pass";
|
|
} catch (error) {
|
|
await writeFile(logPath, this.logText, "utf8").catch(() => undefined);
|
|
warn(`${name} failed`);
|
|
warn(`log tail: ${logPath}`);
|
|
process.stderr.write(this.logText.split("\n").slice(-80).join("\n"));
|
|
process.stderr.write("\n");
|
|
throw error;
|
|
} finally {
|
|
this.timings.push({
|
|
durationMs: Date.now() - startedAt,
|
|
logPath,
|
|
name,
|
|
status,
|
|
timeoutSeconds,
|
|
});
|
|
await this.writeTimings().catch(() => undefined);
|
|
if (timer) {
|
|
clearTimeout(timer);
|
|
}
|
|
this.deadlineMs = 0;
|
|
}
|
|
}
|
|
|
|
async phaseReturns(
|
|
name: string,
|
|
timeoutSeconds: number,
|
|
fn: () => Promise<void> | void,
|
|
): Promise<boolean> {
|
|
try {
|
|
await this.phase(name, timeoutSeconds, fn);
|
|
return true;
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
remainingTimeoutMs(fallbackMs?: number): number | undefined {
|
|
if (this.deadlineMs === 0) {
|
|
return fallbackMs;
|
|
}
|
|
const remaining = this.deadlineMs - Date.now();
|
|
if (remaining <= 0) {
|
|
throw new Error("phase deadline exceeded before starting guest command");
|
|
}
|
|
return Math.max(1_000, fallbackMs == null ? remaining : Math.min(remaining, fallbackMs));
|
|
}
|
|
|
|
append(text: string): void {
|
|
if (!text) {
|
|
return;
|
|
}
|
|
this.logText += text;
|
|
if (!text.endsWith("\n")) {
|
|
this.logText += "\n";
|
|
}
|
|
}
|
|
|
|
private async writeTimings(): Promise<void> {
|
|
const slowest = this.timings.toSorted((a, b) => b.durationMs - a.durationMs)[0] ?? null;
|
|
await writeFile(
|
|
path.join(this.runDir, "phase-timings.json"),
|
|
`${JSON.stringify({ phases: this.timings, slowest }, null, 2)}\n`,
|
|
"utf8",
|
|
);
|
|
}
|
|
}
|