import fs from "node:fs/promises"; import { tmpdir } from "node:os"; import path from "node:path"; import { describe, expect, it, vi } from "vitest"; import type { SessionEntry } from "../../config/sessions.js"; import type { FollowupRun } from "./queue.js"; import { createMockTypingController } from "./test-helpers.js"; const runEmbeddedPiAgentMock = vi.fn(); vi.mock("../../agents/model-fallback.js", () => ({ runWithModelFallback: async ({ provider, model, run, }: { provider: string; model: string; run: (provider: string, model: string) => Promise; }) => ({ result: await run(provider, model), provider, model, }), })); vi.mock("../../agents/pi-embedded.js", () => ({ runEmbeddedPiAgent: (params: unknown) => runEmbeddedPiAgentMock(params), })); import { createFollowupRunner } from "./followup-runner.js"; describe("createFollowupRunner compaction", () => { it("adds verbose auto-compaction notice and tracks count", async () => { const storePath = path.join( await fs.mkdtemp(path.join(tmpdir(), "clawdbot-compaction-")), "sessions.json", ); const sessionEntry: SessionEntry = { sessionId: "session", updatedAt: Date.now(), }; const sessionStore: Record = { main: sessionEntry, }; const onBlockReply = vi.fn(async () => {}); runEmbeddedPiAgentMock.mockImplementationOnce( async (params: { onAgentEvent?: (evt: { stream: string; data: Record; }) => void; }) => { params.onAgentEvent?.({ stream: "compaction", data: { phase: "end", willRetry: false }, }); return { payloads: [{ text: "final" }], meta: {} }; }, ); const runner = createFollowupRunner({ opts: { onBlockReply }, typing: createMockTypingController(), typingMode: "instant", sessionEntry, sessionStore, sessionKey: "main", storePath, defaultModel: "anthropic/claude-opus-4-5", }); const queued = { prompt: "hello", summaryLine: "hello", enqueuedAt: Date.now(), run: { sessionId: "session", sessionKey: "main", messageProvider: "whatsapp", sessionFile: "/tmp/session.jsonl", workspaceDir: "/tmp", config: {}, skillsSnapshot: {}, provider: "anthropic", model: "claude", thinkLevel: "low", verboseLevel: "on", elevatedLevel: "off", bashElevated: { enabled: false, allowed: false, defaultLevel: "off", }, timeoutMs: 1_000, blockReplyBreak: "message_end", }, } as FollowupRun; await runner(queued); expect(onBlockReply).toHaveBeenCalled(); expect(onBlockReply.mock.calls[0][0].text).toContain( "Auto-compaction complete", ); expect(sessionStore.main.compactionCount).toBe(1); }); });