mirror of https://github.com/openclaw/openclaw.git
147 lines
4.2 KiB
TypeScript
147 lines
4.2 KiB
TypeScript
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
|
|
import { waitForTransportReady } from "./transport-ready.js";
|
|
|
|
// Perf: `sleepWithAbort` uses `node:timers/promises` which isn't controlled by fake timers.
|
|
// Route sleeps through global `setTimeout` so tests can advance time deterministically.
|
|
vi.mock("./backoff.js", () => ({
|
|
sleepWithAbort: async (ms: number, signal?: AbortSignal) => {
|
|
if (signal?.aborted) {
|
|
throw new Error("aborted");
|
|
}
|
|
if (ms <= 0) {
|
|
return;
|
|
}
|
|
await new Promise<void>((resolve, reject) => {
|
|
const timer = setTimeout(() => {
|
|
signal?.removeEventListener("abort", onAbort);
|
|
resolve();
|
|
}, ms);
|
|
const onAbort = () => {
|
|
clearTimeout(timer);
|
|
signal?.removeEventListener("abort", onAbort);
|
|
reject(new Error("aborted"));
|
|
};
|
|
signal?.addEventListener("abort", onAbort, { once: true });
|
|
});
|
|
},
|
|
}));
|
|
|
|
function createRuntime() {
|
|
return { log: vi.fn(), error: vi.fn(), exit: vi.fn() };
|
|
}
|
|
|
|
describe("waitForTransportReady", () => {
|
|
beforeEach(() => {
|
|
vi.useFakeTimers();
|
|
});
|
|
|
|
afterEach(() => {
|
|
vi.useRealTimers();
|
|
});
|
|
|
|
it("returns when the check succeeds and logs after the delay", async () => {
|
|
const runtime = createRuntime();
|
|
let attempts = 0;
|
|
const readyPromise = waitForTransportReady({
|
|
label: "test transport",
|
|
timeoutMs: 220,
|
|
// Deterministic: first attempt at t=0 won't log; second attempt at t=50 will.
|
|
logAfterMs: 1,
|
|
logIntervalMs: 1_000,
|
|
pollIntervalMs: 50,
|
|
runtime,
|
|
check: async () => {
|
|
attempts += 1;
|
|
if (attempts > 2) {
|
|
return { ok: true };
|
|
}
|
|
return { ok: false, error: "not ready" };
|
|
},
|
|
});
|
|
|
|
await vi.advanceTimersByTimeAsync(200);
|
|
|
|
await readyPromise;
|
|
expect(runtime.error).toHaveBeenCalled();
|
|
});
|
|
|
|
it("throws after the timeout", async () => {
|
|
const runtime = createRuntime();
|
|
const waitPromise = waitForTransportReady({
|
|
label: "test transport",
|
|
timeoutMs: 110,
|
|
logAfterMs: 0,
|
|
logIntervalMs: 1_000,
|
|
pollIntervalMs: 50,
|
|
runtime,
|
|
check: async () => ({ ok: false, error: "still down" }),
|
|
});
|
|
const asserted = expect(waitPromise).rejects.toThrow("test transport not ready");
|
|
await vi.advanceTimersByTimeAsync(200);
|
|
await asserted;
|
|
expect(runtime.error).toHaveBeenCalled();
|
|
});
|
|
|
|
it("returns early when aborted", async () => {
|
|
const runtime = createRuntime();
|
|
const controller = new AbortController();
|
|
controller.abort();
|
|
await waitForTransportReady({
|
|
label: "test transport",
|
|
timeoutMs: 200,
|
|
runtime,
|
|
abortSignal: controller.signal,
|
|
check: async () => ({ ok: false, error: "still down" }),
|
|
});
|
|
expect(runtime.error).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("stops polling when aborted during the sleep interval", async () => {
|
|
const runtime = createRuntime();
|
|
const controller = new AbortController();
|
|
let attempts = 0;
|
|
|
|
const waitPromise = waitForTransportReady({
|
|
label: "test transport",
|
|
timeoutMs: 500,
|
|
pollIntervalMs: 50,
|
|
runtime,
|
|
abortSignal: controller.signal,
|
|
check: async () => {
|
|
attempts += 1;
|
|
setTimeout(() => controller.abort(), 10);
|
|
return { ok: false, error: "still down" };
|
|
},
|
|
});
|
|
|
|
await vi.advanceTimersByTimeAsync(100);
|
|
await waitPromise;
|
|
|
|
expect(attempts).toBe(1);
|
|
expect(runtime.error).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("logs repeated unknown-error retries and the final timeout message", async () => {
|
|
const runtime = createRuntime();
|
|
const waitPromise = waitForTransportReady({
|
|
label: "test transport",
|
|
timeoutMs: 120,
|
|
logAfterMs: 0,
|
|
logIntervalMs: 50,
|
|
pollIntervalMs: 50,
|
|
runtime,
|
|
check: async () => ({ ok: false, error: null }),
|
|
});
|
|
|
|
const asserted = expect(waitPromise).rejects.toThrow(
|
|
"test transport not ready (unknown error)",
|
|
);
|
|
await vi.advanceTimersByTimeAsync(200);
|
|
await asserted;
|
|
|
|
expect(runtime.error).toHaveBeenCalledTimes(2);
|
|
expect(runtime.error.mock.calls.at(0)?.[0]).toContain("unknown error");
|
|
expect(runtime.error.mock.calls.at(-1)?.[0]).toContain("not ready after 120ms");
|
|
});
|
|
});
|