import { describe, it, expect, beforeEach, vi } from "vitest"; vi.mock("../src/logger.js", () => ({ logger: { info: vi.fn(), warn: vi.fn(), error: vi.fn() }, })); import { registerExportImportFunction } from "../src/functions/export-import.js"; import type { Session, CompressedObservation, Memory, SessionSummary, ExportData, } 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); }, }; } const testSession: Session = { id: "ses_1", project: "my-project", cwd: "/tmp", startedAt: "2026-02-01T00:00:00Z", status: "completed", observationCount: 1, }; const testObs: CompressedObservation = { id: "obs_1", sessionId: "ses_1", timestamp: "2026-02-01T10:00:00Z", type: "file_edit", title: "Edit auth", facts: ["Added check"], narrative: "Auth changes", concepts: ["auth"], files: ["src/auth.ts"], importance: 7, }; const testMemory: Memory = { id: "mem_1", createdAt: "2026-02-01T00:00:00Z", updatedAt: "2026-02-01T00:00:00Z", type: "pattern", title: "Auth pattern", content: "Always validate tokens", concepts: ["auth"], files: [], sessionIds: ["ses_1"], strength: 5, version: 1, isLatest: true, }; const testSummary: SessionSummary = { sessionId: "ses_1", project: "my-project", createdAt: "2026-02-01T00:00:00Z", title: "Auth work", narrative: "Worked on auth", keyDecisions: ["Use JWT"], filesModified: ["src/auth.ts"], concepts: ["auth"], observationCount: 1, }; describe("Export/Import Functions", () => { let sdk: ReturnType; let kv: ReturnType; beforeEach(async () => { sdk = mockSdk(); kv = mockKV(); registerExportImportFunction(sdk as never, kv as never); await kv.set("mem:sessions", "ses_1", testSession); await kv.set("mem:obs:ses_1", "obs_1", testObs); await kv.set("mem:memories", "mem_1", testMemory); await kv.set("mem:summaries", "ses_1", testSummary); }); it("export produces valid ExportData structure", async () => { const result = (await sdk.trigger("mem::export", {})) as ExportData; expect(result.version).toBe("0.9.27"); expect(result.exportedAt).toBeDefined(); expect(result.sessions.length).toBe(1); expect(result.sessions[0].id).toBe("ses_1"); expect(result.observations["ses_1"].length).toBe(1); expect(result.memories.length).toBe(1); expect(result.summaries.length).toBe(1); }); it("import with merge strategy adds data", async () => { const exportData: ExportData = { version: "0.3.0", exportedAt: new Date().toISOString(), sessions: [{ ...testSession, id: "ses_2", observationCount: 0 }], observations: {}, memories: [{ ...testMemory, id: "mem_2", title: "New pattern" }], summaries: [], }; const result = (await sdk.trigger("mem::import", { exportData, strategy: "merge", })) as { success: boolean; sessions: number; memories: number }; expect(result.success).toBe(true); expect(result.sessions).toBe(1); expect(result.memories).toBe(1); const allSessions = await kv.list("mem:sessions"); expect(allSessions.length).toBe(2); }); it("import with skip strategy does not overwrite existing", async () => { const exportData: ExportData = { version: "0.3.0", exportedAt: new Date().toISOString(), sessions: [testSession], observations: { ses_1: [testObs] }, memories: [testMemory], summaries: [testSummary], }; const result = (await sdk.trigger("mem::import", { exportData, strategy: "skip", })) as { success: boolean; skipped: number; sessions: number }; expect(result.success).toBe(true); expect(result.skipped).toBeGreaterThan(0); expect(result.sessions).toBe(0); }); it("import with replace strategy clears existing data first", async () => { const newSession: Session = { id: "ses_new", project: "new-project", cwd: "/tmp/new", startedAt: "2026-03-01T00:00:00Z", status: "active", observationCount: 0, }; const exportData: ExportData = { version: "0.3.0", exportedAt: new Date().toISOString(), sessions: [newSession], observations: {}, memories: [], summaries: [], }; const result = (await sdk.trigger("mem::import", { exportData, strategy: "replace", })) as { success: boolean; sessions: number }; expect(result.success).toBe(true); expect(result.sessions).toBe(1); const oldSession = await kv.get("mem:sessions", "ses_1"); expect(oldSession).toBeNull(); }); it("export then import round-trip preserves data", async () => { const exported = (await sdk.trigger("mem::export", {})) as ExportData; const freshKv = mockKV(); const freshSdk = mockSdk(); registerExportImportFunction(freshSdk as never, freshKv as never); const importResult = (await freshSdk.trigger("mem::import", { exportData: exported, strategy: "merge", })) as { success: boolean; sessions: number; observations: number; memories: number; }; expect(importResult.success).toBe(true); expect(importResult.sessions).toBe(1); expect(importResult.observations).toBe(1); expect(importResult.memories).toBe(1); const reExported = (await freshSdk.trigger( "mem::export", {}, )) as ExportData; expect(reExported.sessions.length).toBe(exported.sessions.length); expect(reExported.memories.length).toBe(exported.memories.length); }); it("import rejects unsupported version", async () => { const exportData = { version: "1.0.0", exportedAt: new Date().toISOString(), sessions: [], observations: {}, memories: [], summaries: [], } as unknown as ExportData; const result = (await sdk.trigger("mem::import", { exportData, strategy: "merge", })) as { success: boolean; error: string }; expect(result.success).toBe(false); expect(result.error).toContain("Unsupported export version"); }); });