import { describe, it, expect, beforeEach, vi } from "vitest"; vi.mock("../src/logger.js", () => ({ logger: { info: vi.fn(), warn: vi.fn(), error: vi.fn() }, })); vi.mock("../src/config.js", () => ({ getConsolidationDecayDays: () => 30, isConsolidationEnabled: vi.fn(() => true), })); import { registerConsolidationPipelineFunction } from "../src/functions/consolidation-pipeline.js"; import { isConsolidationEnabled } from "../src/config.js"; import type { SessionSummary, Memory, SemanticMemory, ProceduralMemory } from "../src/types.js"; function mockKV() { const store = new Map>(); return { get: async (scope: string, key: string): Promise => { return (store.get(scope)?.get(key) as T) ?? null; }, set: async (scope: string, key: string, data: T): Promise => { if (!store.has(scope)) store.set(scope, new Map()); store.get(scope)!.set(key, data); return data; }, delete: async (scope: string, key: string): Promise => { store.get(scope)?.delete(key); }, list: async (scope: string): Promise => { const entries = store.get(scope); return entries ? (Array.from(entries.values()) as T[]) : []; }, }; } function mockSdk() { const functions = new Map(); return { registerFunction: (idOrOpts: string | { id: string }, handler: Function) => { const id = typeof idOrOpts === "string" ? idOrOpts : idOrOpts.id; functions.set(id, handler); }, registerTrigger: () => {}, trigger: async (idOrInput: string | { function_id: string; payload: unknown }, data?: unknown) => { const id = typeof idOrInput === "string" ? idOrInput : idOrInput.function_id; const payload = typeof idOrInput === "string" ? data : idOrInput.payload; const fn = functions.get(id); if (!fn) throw new Error(`No function: ${id}`); return fn(payload); }, }; } function makeSummary(i: number): SessionSummary { return { sessionId: `ses_${i}`, project: "test-project", createdAt: new Date(Date.now() - i * 86400000).toISOString(), title: `Session ${i} summary`, narrative: `Worked on feature ${i}`, keyDecisions: [`Decision ${i}`], filesModified: [`src/file${i}.ts`], concepts: ["typescript", "testing"], observationCount: 5, }; } function makePattern(i: number): Memory { return { id: `mem_${i}`, createdAt: "2026-01-01T00:00:00Z", updatedAt: "2026-01-01T00:00:00Z", type: "pattern", title: `Pattern ${i}`, content: `Always do thing ${i}`, concepts: ["testing"], files: [], sessionIds: ["ses_1", "ses_2"], strength: 5, version: 1, isLatest: true, }; } describe("Consolidation Pipeline", () => { let sdk: ReturnType; let kv: ReturnType; beforeEach(() => { sdk = mockSdk(); kv = mockKV(); }); it("pipeline skips semantic when fewer than 5 summaries", async () => { const provider = { name: "test", compress: vi.fn(), summarize: vi.fn(), }; registerConsolidationPipelineFunction(sdk as never, kv as never, provider as never); for (let i = 0; i < 3; i++) { await kv.set("mem:summaries", `ses_${i}`, makeSummary(i)); } const result = (await sdk.trigger("mem::consolidate-pipeline", { tier: "semantic", })) as { success: boolean; results: Record }; expect(result.success).toBe(true); const semantic = result.results.semantic as { skipped: boolean; reason: string }; expect(semantic.skipped).toBe(true); expect(semantic.reason).toContain("fewer than 5"); expect(provider.summarize).not.toHaveBeenCalled(); }); it("pipeline skips procedural when fewer than 2 patterns", async () => { const provider = { name: "test", compress: vi.fn(), summarize: vi.fn(), }; registerConsolidationPipelineFunction(sdk as never, kv as never, provider as never); const mem: Memory = { ...makePattern(1), sessionIds: ["ses_1", "ses_2"], }; await kv.set("mem:memories", "mem_1", mem); const result = (await sdk.trigger("mem::consolidate-pipeline", { tier: "procedural", })) as { success: boolean; results: Record }; expect(result.success).toBe(true); const procedural = result.results.procedural as { skipped: boolean; reason: string }; expect(procedural.skipped).toBe(true); expect(procedural.reason).toContain("fewer than 2"); }); it("with enough summaries, creates semantic memories from provider response", async () => { const provider = { name: "test", compress: vi.fn(), summarize: vi.fn().mockResolvedValue( `TypeScript is the primary language`, ), }; registerConsolidationPipelineFunction(sdk as never, kv as never, provider as never); for (let i = 0; i < 6; i++) { await kv.set("mem:summaries", `ses_${i}`, makeSummary(i)); } const result = (await sdk.trigger("mem::consolidate-pipeline", { tier: "semantic", })) as { success: boolean; results: Record }; expect(result.success).toBe(true); const semantic = result.results.semantic as { newFacts: number }; expect(semantic.newFacts).toBe(1); const stored = await kv.list("mem:semantic"); expect(stored.length).toBe(1); expect(stored[0].fact).toBe("TypeScript is the primary language"); expect(stored[0].confidence).toBe(0.9); }); it("with enough patterns, creates procedural memories from provider response", async () => { const provider = { name: "test", compress: vi.fn(), summarize: vi.fn().mockResolvedValue( `Create test fileWrite assertions`, ), }; registerConsolidationPipelineFunction(sdk as never, kv as never, provider as never); for (let i = 0; i < 3; i++) { await kv.set("mem:memories", `mem_${i}`, makePattern(i)); } const result = (await sdk.trigger("mem::consolidate-pipeline", { tier: "procedural", })) as { success: boolean; results: Record }; expect(result.success).toBe(true); const procedural = result.results.procedural as { newProcedures: number }; expect(procedural.newProcedures).toBe(1); const stored = await kv.list("mem:procedural"); expect(stored.length).toBe(1); expect(stored[0].name).toBe("Test Workflow"); expect(stored[0].steps.length).toBe(2); expect(stored[0].triggerCondition).toBe("when writing tests"); }); it("consolidation records an audit entry", async () => { const provider = { name: "test", compress: vi.fn(), summarize: vi.fn(), }; registerConsolidationPipelineFunction(sdk as never, kv as never, provider as never); await sdk.trigger("mem::consolidate-pipeline", { tier: "semantic" }); const audits = await kv.list("mem:audit"); expect(audits.length).toBe(1); }); it("pipeline returns early when consolidation is disabled", async () => { vi.mocked(isConsolidationEnabled).mockReturnValue(false); const provider = { name: "test", compress: vi.fn(), summarize: vi.fn(), }; registerConsolidationPipelineFunction(sdk as never, kv as never, provider as never); const result = (await sdk.trigger("mem::consolidate-pipeline", {})) as { success: boolean; skipped?: boolean; reason?: string; }; expect(result.success).toBe(false); expect(result.skipped).toBe(true); expect(result.reason).toContain("CONSOLIDATION_ENABLED"); expect(provider.summarize).not.toHaveBeenCalled(); vi.mocked(isConsolidationEnabled).mockReturnValue(true); }); it("pipeline proceeds with force=true even when consolidation is disabled", async () => { vi.mocked(isConsolidationEnabled).mockReturnValue(false); const provider = { name: "test", compress: vi.fn(), summarize: vi.fn(), }; registerConsolidationPipelineFunction(sdk as never, kv as never, provider as never); const result = (await sdk.trigger("mem::consolidate-pipeline", { force: true, })) as { success: boolean; results: Record }; expect(result.success).toBe(true); expect(result.results).toBeDefined(); vi.mocked(isConsolidationEnabled).mockReturnValue(true); }); });