clanker-agent/packages/coding-agent/test/gateway-steer.test.ts
2026-03-12 15:21:04 -04:00

160 lines
4.1 KiB
TypeScript

import { describe, expect, it, vi } from "vitest";
import type { ManagedGatewaySession } from "../src/core/gateway/internal-types.js";
import { GatewayRuntime } from "../src/core/gateway/runtime.js";
function createMockSession() {
return {
sessionId: "session-1",
sessionName: undefined,
messages: [],
prompt: vi.fn().mockResolvedValue(undefined),
steer: vi.fn().mockResolvedValue(undefined),
abort: vi.fn().mockResolvedValue(undefined),
dispose: vi.fn(),
subscribe: vi.fn(() => () => {}),
sessionManager: {
getSessionDir: () => "/tmp/companion-gateway-test",
},
};
}
function createRuntime(session = createMockSession()) {
return new GatewayRuntime({
config: {
bind: "127.0.0.1",
port: 0,
session: {
idleMinutes: 5,
maxQueuePerSession: 4,
},
webhook: {
enabled: false,
basePath: "/webhooks",
},
},
primarySessionKey: "primary",
primarySession: session as never,
createSession: async () => session as never,
});
}
function addManagedSession(
runtime: GatewayRuntime,
sessionKey: string,
session: ReturnType<typeof createMockSession>,
processing: boolean,
) {
const managedSession: ManagedGatewaySession = {
sessionKey,
session: session as never,
queue: [],
processing,
activeAssistantMessage: null,
pendingToolResults: [],
createdAt: Date.now(),
lastActiveAt: Date.now(),
listeners: new Set(),
unsubscribe: () => {},
};
(runtime as unknown as { sessions: Map<string, unknown> }).sessions.set(
sessionKey,
managedSession,
);
return managedSession;
}
describe("GatewayRuntime steer handling", () => {
it("steers the active session instead of queueing a second prompt", async () => {
const session = createMockSession();
const runtime = createRuntime(session);
addManagedSession(runtime, "chat", session, true);
const result = await (
runtime as unknown as {
handleSteer: (
sessionKey: string,
text: string,
) => Promise<{
ok: true;
mode: "steer" | "queued";
sessionKey: string;
}>;
}
).handleSteer("chat", "keep going");
expect(result).toEqual({
ok: true,
mode: "steer",
sessionKey: "chat",
});
expect(session.steer).toHaveBeenCalledWith("keep going");
expect(session.prompt).not.toHaveBeenCalled();
});
it("queues a prompt immediately when steer races an idle session", async () => {
const session = createMockSession();
const runtime = createRuntime(session);
addManagedSession(runtime, "chat", session, false);
const result = await (
runtime as unknown as {
handleSteer: (
sessionKey: string,
text: string,
) => Promise<{
ok: true;
mode: "steer" | "queued";
sessionKey: string;
}>;
}
).handleSteer("chat", "pick this up next");
expect(result).toEqual({
ok: true,
mode: "queued",
sessionKey: "chat",
});
expect(session.steer).not.toHaveBeenCalled();
await vi.waitFor(() => {
expect(session.prompt).toHaveBeenCalledWith("pick this up next", {
images: undefined,
source: "extension",
});
});
});
it("abort clears queued follow-ups before aborting the active session", () => {
const session = createMockSession();
const runtime = createRuntime(session);
const managedSession = addManagedSession(runtime, "chat", session, true);
const resolve = vi.fn();
managedSession.queue.push({
request: {
sessionKey: "chat",
text: "stale follow-up",
source: "extension",
},
resolve,
});
const result = (
runtime as unknown as {
abortSession: (sessionKey: string) => boolean;
}
).abortSession("chat");
expect(result).toBe(true);
expect(managedSession.queue).toHaveLength(0);
expect(resolve).toHaveBeenCalledWith({
ok: false,
response: "",
error: "Session aborted",
sessionKey: "chat",
});
expect(session.abort).toHaveBeenCalledTimes(1);
});
});