perf(test): speed up memory index suite

This commit is contained in:
Peter Steinberger
2026-02-14 22:36:13 +00:00
parent a0ff9d9bbb
commit 110cc5d791

View File

@@ -1,7 +1,7 @@
import fs from "node:fs/promises"; import fs from "node:fs/promises";
import os from "node:os"; import os from "node:os";
import path from "node:path"; import path from "node:path";
import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, it, vi } from "vitest"; import { afterAll, beforeAll, beforeEach, describe, expect, it, vi } from "vitest";
import { getMemorySearchManager, type MemoryIndexManager } from "./index.js"; import { getMemorySearchManager, type MemoryIndexManager } from "./index.js";
let embedBatchCalls = 0; let embedBatchCalls = 0;
@@ -43,16 +43,33 @@ vi.mock("./embeddings.js", () => {
describe("memory index", () => { describe("memory index", () => {
let fixtureRoot = ""; let fixtureRoot = "";
let fixtureCount = 0; let workspaceDir = "";
let workspaceDir: string; let memoryDir = "";
let indexPath: string; let extraDir = "";
let manager: MemoryIndexManager | null = null; let indexBasicPath = "";
let indexCachePath = "";
let indexHybridPath = "";
let indexVectorPath = "";
let indexExtraPath = "";
const persistentManagers = new Set<MemoryIndexManager>();
beforeAll(async () => { beforeAll(async () => {
fixtureRoot = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-mem-fixtures-")); fixtureRoot = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-mem-fixtures-"));
workspaceDir = path.join(fixtureRoot, "workspace");
memoryDir = path.join(workspaceDir, "memory");
extraDir = path.join(workspaceDir, "extra");
indexBasicPath = path.join(workspaceDir, "index-basic.sqlite");
indexCachePath = path.join(workspaceDir, "index-cache.sqlite");
indexHybridPath = path.join(workspaceDir, "index-hybrid.sqlite");
indexVectorPath = path.join(workspaceDir, "index-vector.sqlite");
indexExtraPath = path.join(workspaceDir, "index-extra.sqlite");
await fs.mkdir(memoryDir, { recursive: true });
}); });
afterAll(async () => { afterAll(async () => {
await Promise.all(Array.from(persistentManagers).map((manager) => manager.close()));
await fs.rm(fixtureRoot, { recursive: true, force: true }); await fs.rm(fixtureRoot, { recursive: true, force: true });
}); });
@@ -61,23 +78,46 @@ describe("memory index", () => {
// Keep atomic reindex tests on the safe path. // Keep atomic reindex tests on the safe path.
vi.stubEnv("OPENCLAW_TEST_MEMORY_UNSAFE_REINDEX", "1"); vi.stubEnv("OPENCLAW_TEST_MEMORY_UNSAFE_REINDEX", "1");
embedBatchCalls = 0; embedBatchCalls = 0;
workspaceDir = path.join(fixtureRoot, `case-${fixtureCount++}`);
await fs.mkdir(workspaceDir, { recursive: true }); // Keep the workspace stable to allow manager reuse across tests.
indexPath = path.join(workspaceDir, "index.sqlite"); await fs.mkdir(memoryDir, { recursive: true });
await fs.mkdir(path.join(workspaceDir, "memory"));
await fs.writeFile( await fs.writeFile(
path.join(workspaceDir, "memory", "2026-01-12.md"), path.join(memoryDir, "2026-01-12.md"),
"# Log\nAlpha memory line.\nZebra memory line.", "# Log\nAlpha memory line.\nZebra memory line.",
); );
});
afterEach(async () => { // Clean additional paths that may have been created by earlier cases.
if (manager) { await fs.rm(extraDir, { recursive: true, force: true });
await manager.close();
manager = null; for (const manager of persistentManagers) {
resetManagerForTest(manager);
} }
}); });
function resetManagerForTest(manager: MemoryIndexManager) {
// These tests reuse managers for performance. Clear the index + embedding
// cache to keep each test fully isolated.
(manager as unknown as { resetIndex: () => void }).resetIndex();
(manager as unknown as { db: { exec: (sql: string) => void } }).db.exec(
"DELETE FROM embedding_cache",
);
(manager as unknown as { dirty: boolean }).dirty = true;
(manager as unknown as { sessionsDirty: boolean }).sessionsDirty = false;
}
type TestCfg = Parameters<typeof getMemorySearchManager>[0]["cfg"];
async function getPersistentManager(cfg: TestCfg): Promise<MemoryIndexManager> {
const result = await getMemorySearchManager({ cfg, agentId: "main" });
expect(result.manager).not.toBeNull();
if (!result.manager) {
throw new Error("manager missing");
}
const manager = result.manager as MemoryIndexManager;
persistentManagers.add(manager);
return manager;
}
it("indexes memory files and searches by vector", async () => { it("indexes memory files and searches by vector", async () => {
const cfg = { const cfg = {
agents: { agents: {
@@ -86,7 +126,7 @@ describe("memory index", () => {
memorySearch: { memorySearch: {
provider: "openai", provider: "openai",
model: "mock-embed", model: "mock-embed",
store: { path: indexPath, vector: { enabled: false } }, store: { path: indexBasicPath, vector: { enabled: false } },
sync: { watch: false, onSessionStart: false, onSearch: true }, sync: { watch: false, onSessionStart: false, onSearch: true },
query: { minScore: 0, hybrid: { enabled: false } }, query: { minScore: 0, hybrid: { enabled: false } },
}, },
@@ -94,17 +134,12 @@ describe("memory index", () => {
list: [{ id: "main", default: true }], list: [{ id: "main", default: true }],
}, },
}; };
const result = await getMemorySearchManager({ cfg, agentId: "main" }); const manager = await getPersistentManager(cfg);
expect(result.manager).not.toBeNull(); await manager.sync({ reason: "test" });
if (!result.manager) { const results = await manager.search("alpha");
throw new Error("manager missing");
}
manager = result.manager;
await result.manager.sync({ reason: "test" });
const results = await result.manager.search("alpha");
expect(results.length).toBeGreaterThan(0); expect(results.length).toBeGreaterThan(0);
expect(results[0]?.path).toContain("memory/2026-01-12.md"); expect(results[0]?.path).toContain("memory/2026-01-12.md");
const status = result.manager.status(); const status = manager.status();
expect(status.sourceCounts).toEqual( expect(status.sourceCounts).toEqual(
expect.arrayContaining([ expect.arrayContaining([
expect.objectContaining({ expect.objectContaining({
@@ -117,13 +152,18 @@ describe("memory index", () => {
}); });
it("reindexes when the embedding model changes", async () => { it("reindexes when the embedding model changes", async () => {
const indexModelPath = path.join(workspaceDir, "index-model-change.sqlite");
await fs.rm(indexModelPath, { force: true });
await fs.rm(`${indexModelPath}-shm`, { force: true });
await fs.rm(`${indexModelPath}-wal`, { force: true });
const base = { const base = {
agents: { agents: {
defaults: { defaults: {
workspace: workspaceDir, workspace: workspaceDir,
memorySearch: { memorySearch: {
provider: "openai", provider: "openai",
store: { path: indexPath }, store: { path: indexModelPath },
sync: { watch: false, onSessionStart: false, onSearch: true }, sync: { watch: false, onSessionStart: false, onSearch: true },
query: { minScore: 0, hybrid: { enabled: false } }, query: { minScore: 0, hybrid: { enabled: false } },
}, },
@@ -176,11 +216,11 @@ describe("memory index", () => {
if (!second.manager) { if (!second.manager) {
throw new Error("manager missing"); throw new Error("manager missing");
} }
manager = second.manager;
await second.manager.sync({ reason: "test" }); await second.manager.sync({ reason: "test" });
expect(embedBatchCalls).toBeGreaterThan(callsAfterFirstSync); expect(embedBatchCalls).toBeGreaterThan(callsAfterFirstSync);
const status = second.manager.status(); const status = second.manager.status();
expect(status.files).toBeGreaterThan(0); expect(status.files).toBeGreaterThan(0);
await second.manager.close();
}); });
it("reuses cached embeddings on forced reindex", async () => { it("reuses cached embeddings on forced reindex", async () => {
@@ -191,7 +231,7 @@ describe("memory index", () => {
memorySearch: { memorySearch: {
provider: "openai", provider: "openai",
model: "mock-embed", model: "mock-embed",
store: { path: indexPath, vector: { enabled: false } }, store: { path: indexCachePath, vector: { enabled: false } },
sync: { watch: false, onSessionStart: false, onSearch: false }, sync: { watch: false, onSessionStart: false, onSearch: false },
query: { minScore: 0, hybrid: { enabled: false } }, query: { minScore: 0, hybrid: { enabled: false } },
cache: { enabled: true }, cache: { enabled: true },
@@ -200,12 +240,7 @@ describe("memory index", () => {
list: [{ id: "main", default: true }], list: [{ id: "main", default: true }],
}, },
}; };
const result = await getMemorySearchManager({ cfg, agentId: "main" }); const manager = await getPersistentManager(cfg);
expect(result.manager).not.toBeNull();
if (!result.manager) {
throw new Error("manager missing");
}
manager = result.manager;
await manager.sync({ force: true }); await manager.sync({ force: true });
const afterFirst = embedBatchCalls; const afterFirst = embedBatchCalls;
expect(afterFirst).toBeGreaterThan(0); expect(afterFirst).toBeGreaterThan(0);
@@ -222,7 +257,7 @@ describe("memory index", () => {
memorySearch: { memorySearch: {
provider: "openai", provider: "openai",
model: "mock-embed", model: "mock-embed",
store: { path: indexPath, vector: { enabled: false } }, store: { path: indexHybridPath, vector: { enabled: false } },
sync: { watch: false, onSessionStart: false, onSearch: true }, sync: { watch: false, onSessionStart: false, onSearch: true },
query: { query: {
minScore: 0, minScore: 0,
@@ -233,12 +268,7 @@ describe("memory index", () => {
list: [{ id: "main", default: true }], list: [{ id: "main", default: true }],
}, },
}; };
const result = await getMemorySearchManager({ cfg, agentId: "main" }); const manager = await getPersistentManager(cfg);
expect(result.manager).not.toBeNull();
if (!result.manager) {
throw new Error("manager missing");
}
manager = result.manager;
const status = manager.status(); const status = manager.status();
if (!status.fts?.available) { if (!status.fts?.available) {
@@ -259,21 +289,16 @@ describe("memory index", () => {
memorySearch: { memorySearch: {
provider: "openai", provider: "openai",
model: "mock-embed", model: "mock-embed",
store: { path: indexPath }, store: { path: indexVectorPath, vector: { enabled: true } },
sync: { watch: false, onSessionStart: false, onSearch: false }, sync: { watch: false, onSessionStart: false, onSearch: false },
}, },
}, },
list: [{ id: "main", default: true }], list: [{ id: "main", default: true }],
}, },
}; };
const result = await getMemorySearchManager({ cfg, agentId: "main" }); const manager = await getPersistentManager(cfg);
expect(result.manager).not.toBeNull(); const available = await manager.probeVectorAvailability();
if (!result.manager) { const status = manager.status();
throw new Error("manager missing");
}
manager = result.manager;
const available = await result.manager.probeVectorAvailability();
const status = result.manager.status();
expect(status.vector?.enabled).toBe(true); expect(status.vector?.enabled).toBe(true);
expect(typeof status.vector?.available).toBe("boolean"); expect(typeof status.vector?.available).toBe("boolean");
expect(status.vector?.available).toBe(available); expect(status.vector?.available).toBe(available);
@@ -287,7 +312,7 @@ describe("memory index", () => {
memorySearch: { memorySearch: {
provider: "openai", provider: "openai",
model: "mock-embed", model: "mock-embed",
store: { path: indexPath, vector: { enabled: false } }, store: { path: indexBasicPath, vector: { enabled: false } },
sync: { watch: false, onSessionStart: false, onSearch: true }, sync: { watch: false, onSessionStart: false, onSearch: true },
query: { minScore: 0, hybrid: { enabled: false } }, query: { minScore: 0, hybrid: { enabled: false } },
}, },
@@ -295,17 +320,11 @@ describe("memory index", () => {
list: [{ id: "main", default: true }], list: [{ id: "main", default: true }],
}, },
}; };
const result = await getMemorySearchManager({ cfg, agentId: "main" }); const manager = await getPersistentManager(cfg);
expect(result.manager).not.toBeNull(); await expect(manager.readFile({ relPath: "NOTES.md" })).rejects.toThrow("path required");
if (!result.manager) {
throw new Error("manager missing");
}
manager = result.manager;
await expect(result.manager.readFile({ relPath: "NOTES.md" })).rejects.toThrow("path required");
}); });
it("allows reading from additional memory paths and blocks symlinks", async () => { it("allows reading from additional memory paths and blocks symlinks", async () => {
const extraDir = path.join(workspaceDir, "extra");
await fs.mkdir(extraDir, { recursive: true }); await fs.mkdir(extraDir, { recursive: true });
await fs.writeFile(path.join(extraDir, "extra.md"), "Extra content."); await fs.writeFile(path.join(extraDir, "extra.md"), "Extra content.");
@@ -316,7 +335,7 @@ describe("memory index", () => {
memorySearch: { memorySearch: {
provider: "openai", provider: "openai",
model: "mock-embed", model: "mock-embed",
store: { path: indexPath, vector: { enabled: false } }, store: { path: indexExtraPath, vector: { enabled: false } },
sync: { watch: false, onSessionStart: false, onSearch: true }, sync: { watch: false, onSessionStart: false, onSearch: true },
query: { minScore: 0, hybrid: { enabled: false } }, query: { minScore: 0, hybrid: { enabled: false } },
extraPaths: [extraDir], extraPaths: [extraDir],
@@ -325,13 +344,8 @@ describe("memory index", () => {
list: [{ id: "main", default: true }], list: [{ id: "main", default: true }],
}, },
}; };
const result = await getMemorySearchManager({ cfg, agentId: "main" }); const manager = await getPersistentManager(cfg);
expect(result.manager).not.toBeNull(); await expect(manager.readFile({ relPath: "extra/extra.md" })).resolves.toEqual({
if (!result.manager) {
throw new Error("manager missing");
}
manager = result.manager;
await expect(result.manager.readFile({ relPath: "extra/extra.md" })).resolves.toEqual({
path: "extra/extra.md", path: "extra/extra.md",
text: "Extra content.", text: "Extra content.",
}); });
@@ -349,7 +363,7 @@ describe("memory index", () => {
} }
} }
if (symlinkOk) { if (symlinkOk) {
await expect(result.manager.readFile({ relPath: "extra/linked.md" })).rejects.toThrow( await expect(manager.readFile({ relPath: "extra/linked.md" })).rejects.toThrow(
"path required", "path required",
); );
} }