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("node:child_process", () => ({ execFile: vi.fn( (_cmd: string, _args: string[], _opts: unknown, cb: Function) => { cb(null, { stdout: "abc1234\n", stderr: "" }); }, ), })); vi.mock("node:util", async () => { const actual = (await vi.importActual("node:util")) as Record< string, unknown >; return { ...actual, promisify: () => async () => ({ stdout: "abc1234\n", stderr: "" }), }; }); vi.mock("node:fs", () => ({ existsSync: vi.fn().mockReturnValue(true), mkdirSync: vi.fn(), writeFileSync: vi.fn(), readFileSync: vi .fn() .mockReturnValue('{"version":"0.4.0","sessions":[],"memories":[]}'), })); import { registerSnapshotFunction } from "../src/functions/snapshot.js"; import type { Session, Memory, SnapshotMeta } 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); }, }; } describe("Snapshot Functions", () => { let sdk: ReturnType; let kv: ReturnType; const snapshotDir = "/tmp/agentmemory-snapshots"; beforeEach(async () => { sdk = mockSdk(); kv = mockKV(); vi.clearAllMocks(); registerSnapshotFunction(sdk as never, kv as never, snapshotDir); const session: Session = { id: "ses_1", project: "test", cwd: "/tmp", startedAt: "2026-02-01T00:00:00Z", status: "completed", observationCount: 1, }; await kv.set("mem:sessions", "ses_1", session); const mem: Memory = { id: "mem_1", createdAt: "2026-02-01T00:00:00Z", updatedAt: "2026-02-01T00:00:00Z", type: "pattern", title: "Test pattern", content: "Always test", concepts: [], files: [], sessionIds: ["ses_1"], strength: 5, version: 1, isLatest: true, }; await kv.set("mem:memories", "mem_1", mem); }); it("snapshot-create serializes state and returns meta", async () => { const result = (await sdk.trigger("mem::snapshot-create", { message: "Test snapshot", })) as { success: boolean; snapshot: SnapshotMeta }; expect(result.success).toBe(true); expect(result.snapshot).toBeDefined(); expect(result.snapshot.commitHash).toBe("abc1234"); expect(result.snapshot.message).toBe("Test snapshot"); expect(result.snapshot.stats.sessions).toBe(1); expect(result.snapshot.stats.memories).toBe(1); }); it("snapshot-list returns snapshots from git log", async () => { const result = (await sdk.trigger("mem::snapshot-list", {})) as { snapshots: Array<{ commitHash: string; createdAt: string; message: string; }>; }; expect(result.snapshots).toBeDefined(); expect(Array.isArray(result.snapshots)).toBe(true); }); it("snapshot-restore requires commitHash", async () => { const result = (await sdk.trigger("mem::snapshot-restore", {})) as { success: boolean; error: string; }; expect(result.success).toBe(false); expect(result.error).toContain("commitHash"); }); it("snapshot-restore loads state from commit", async () => { const result = (await sdk.trigger("mem::snapshot-restore", { commitHash: "abc1234", })) as { success: boolean; commitHash: string }; expect(result.success).toBe(true); expect(result.commitHash).toBe("abc1234"); }); it("snapshot-create records an audit entry", async () => { await sdk.trigger("mem::snapshot-create", { message: "Audit test" }); const audits = await kv.list("mem:audit"); expect(audits.length).toBe(1); }); });