Files
codewhale/npm/runtime-sdk/test/fleet-client.test.mjs
T
Hunter B 26925ae644 feat(runtime-sdk): add fleet helper client
Refs #3163.

Adds the @codewhale/runtime-sdk workspace with typed fleet Runtime API helpers, protocol-shaped TypeScript declarations, JSON/SSE event fixture handling, and typed RuntimeCapabilityError failures for create/event-stream endpoints that the Rust API has not exposed yet.

Documents the SDK contract in docs/RUNTIME_API.md and wires npm workspace verification through npm test --workspace @codewhale/runtime-sdk.
2026-06-12 22:17:16 -07:00

175 lines
4.9 KiB
JavaScript

import assert from "node:assert/strict";
import test from "node:test";
import {
CodeWhaleRuntimeClient,
RuntimeApiError,
RuntimeCapabilityError,
createRuntimeClient,
} from "../index.js";
function jsonResponse(body, init = {}) {
return new Response(JSON.stringify(body), {
status: init.status ?? 200,
headers: { "content-type": "application/json", ...(init.headers ?? {}) },
});
}
function fakeFetch(responseFactory) {
const calls = [];
const fetch = async (url, init) => {
calls.push({ url: url.toString(), init });
return responseFactory(url, init, calls.length);
};
fetch.calls = calls;
return fetch;
}
test("listFleetRuns calls the Runtime API with bearer auth", async () => {
const fetch = fakeFetch(() =>
jsonResponse({
status: { runs: 1, workers: {} },
runs: [{ id: "run-1", name: "smoke", tasks: [], labels: {} }],
}),
);
const client = createRuntimeClient({
baseUrl: "http://127.0.0.1:7878",
token: "token-1",
fetch,
});
const response = await client.listFleetRuns();
assert.equal(response.runs[0].id, "run-1");
assert.equal(fetch.calls[0].url, "http://127.0.0.1:7878/v1/fleet/runs");
assert.equal(fetch.calls[0].init.method, "GET");
assert.equal(fetch.calls[0].init.headers.get("authorization"), "Bearer token-1");
});
test("worker and run actions use POST endpoints", async () => {
const fetch = fakeFetch((url) =>
jsonResponse(
url.pathname.endsWith("/stop")
? {
action: "stop",
run_id: "run-1",
stopped: 1,
status: { runs: 1, workers: {} },
}
: {
action: url.pathname.endsWith("/restart") ? "restart" : "interrupt",
worker: { worker_id: "w1", artifacts: [] },
},
),
);
const client = new CodeWhaleRuntimeClient({ fetch });
await client.interruptWorker("w1");
await client.restartWorker("w1");
await client.stopFleetRun("run-1");
assert.deepEqual(
fetch.calls.map((call) => [new URL(call.url).pathname, call.init.method]),
[
["/v1/fleet/workers/w1/interrupt", "POST"],
["/v1/fleet/workers/w1/restart", "POST"],
["/v1/fleet/runs/run-1/stop", "POST"],
],
);
});
test("unsupported fleet capabilities raise typed errors", async () => {
const fetch = fakeFetch(() => jsonResponse({ error: "not found" }, { status: 404 }));
const client = new CodeWhaleRuntimeClient({ fetch });
await assert.rejects(
() => client.createFleetRun({ name: "future" }),
(error) =>
error instanceof RuntimeCapabilityError &&
error.capability === "fleet_run_create" &&
error.status === 404,
);
await assert.rejects(
async () => {
for await (const _event of client.fleetEvents("run-1")) {
throw new Error("unexpected event");
}
},
(error) =>
error instanceof RuntimeCapabilityError &&
error.capability === "fleet_event_stream" &&
error.status === 404,
);
});
test("fleetEvents can replay JSON event fixtures when the API exposes them", async () => {
const fetch = fakeFetch(() =>
jsonResponse({
events: [
{
seq: 1,
run_id: "run-1",
worker_id: "w1",
task_id: "task-1",
timestamp: "2026-06-13T00:00:00Z",
label: "running",
payload: { state: "running" },
},
],
}),
);
const client = new CodeWhaleRuntimeClient({ fetch });
const events = [];
for await (const event of client.fleetEvents("run-1", { path: "/v1/fleet/runs/run-1/events" })) {
events.push(event);
}
assert.equal(events.length, 1);
assert.equal(events[0].payload.state, "running");
});
test("fleetEvents parses text/event-stream frames", async () => {
const encoder = new TextEncoder();
const body = new ReadableStream({
start(controller) {
controller.enqueue(
encoder.encode(
'data: {"seq":2,"run_id":"run-1","worker_id":"w1","task_id":"task-1","timestamp":"2026-06-13T00:00:01Z","label":"heartbeat","payload":{"state":"heartbeat","memory_mb":128}}\n\n',
),
);
controller.close();
},
});
const fetch = fakeFetch(
() =>
new Response(body, {
status: 200,
headers: { "content-type": "text/event-stream" },
}),
);
const client = new CodeWhaleRuntimeClient({ fetch });
const events = [];
for await (const event of client.fleetEvents("run-1")) {
events.push(event);
}
assert.equal(events.length, 1);
assert.equal(events[0].payload.state, "heartbeat");
assert.equal(events[0].payload.memory_mb, 128);
});
test("ordinary HTTP errors remain RuntimeApiError", async () => {
const fetch = fakeFetch(() => jsonResponse({ error: "bad" }, { status: 500 }));
const client = new CodeWhaleRuntimeClient({ fetch });
await assert.rejects(
() => client.getFleetRun("run-1"),
(error) =>
error instanceof RuntimeApiError &&
!(error instanceof RuntimeCapabilityError) &&
error.status === 500,
);
});