226 lines
7.6 KiB
TypeScript
226 lines
7.6 KiB
TypeScript
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
import type { OpenClawConfig } from "../config/config.js";
|
|
|
|
const mockPrimary = {
|
|
search: vi.fn(async () => []),
|
|
readFile: vi.fn(async () => ({ text: "", path: "MEMORY.md" })),
|
|
status: vi.fn(() => ({
|
|
backend: "qmd" as const,
|
|
provider: "qmd",
|
|
model: "qmd",
|
|
requestedProvider: "qmd",
|
|
files: 0,
|
|
chunks: 0,
|
|
dirty: false,
|
|
workspaceDir: "/tmp",
|
|
dbPath: "/tmp/index.sqlite",
|
|
sources: ["memory" as const],
|
|
sourceCounts: [{ source: "memory" as const, files: 0, chunks: 0 }],
|
|
})),
|
|
sync: vi.fn(async () => {}),
|
|
probeEmbeddingAvailability: vi.fn(async () => ({ ok: true })),
|
|
probeVectorAvailability: vi.fn(async () => true),
|
|
close: vi.fn(async () => {}),
|
|
};
|
|
|
|
const fallbackSearch = vi.fn(async () => [
|
|
{
|
|
path: "MEMORY.md",
|
|
startLine: 1,
|
|
endLine: 1,
|
|
score: 1,
|
|
snippet: "fallback",
|
|
source: "memory" as const,
|
|
},
|
|
]);
|
|
|
|
const fallbackManager = {
|
|
search: fallbackSearch,
|
|
readFile: vi.fn(async () => ({ text: "", path: "MEMORY.md" })),
|
|
status: vi.fn(() => ({
|
|
backend: "builtin" as const,
|
|
provider: "openai",
|
|
model: "text-embedding-3-small",
|
|
requestedProvider: "openai",
|
|
files: 0,
|
|
chunks: 0,
|
|
dirty: false,
|
|
workspaceDir: "/tmp",
|
|
dbPath: "/tmp/index.sqlite",
|
|
})),
|
|
sync: vi.fn(async () => {}),
|
|
probeEmbeddingAvailability: vi.fn(async () => ({ ok: true })),
|
|
probeVectorAvailability: vi.fn(async () => true),
|
|
close: vi.fn(async () => {}),
|
|
};
|
|
|
|
const mockMemoryIndexGet = vi.fn(async () => fallbackManager);
|
|
|
|
vi.mock("./qmd-manager.js", () => ({
|
|
QmdMemoryManager: {
|
|
create: vi.fn(async () => mockPrimary),
|
|
},
|
|
}));
|
|
|
|
vi.mock("./manager.js", () => ({
|
|
MemoryIndexManager: {
|
|
get: mockMemoryIndexGet,
|
|
},
|
|
}));
|
|
|
|
import { QmdMemoryManager } from "./qmd-manager.js";
|
|
import { getMemorySearchManager } from "./search-manager.js";
|
|
// eslint-disable-next-line @typescript-eslint/unbound-method -- mocked static function
|
|
const createQmdManagerMock = vi.mocked(QmdMemoryManager.create);
|
|
|
|
type SearchManagerResult = Awaited<ReturnType<typeof getMemorySearchManager>>;
|
|
type SearchManager = NonNullable<SearchManagerResult["manager"]>;
|
|
|
|
function createQmdCfg(agentId: string): OpenClawConfig {
|
|
return {
|
|
memory: { backend: "qmd", qmd: {} },
|
|
agents: { list: [{ id: agentId, default: true, workspace: "/tmp/workspace" }] },
|
|
};
|
|
}
|
|
|
|
function requireManager(result: SearchManagerResult): SearchManager {
|
|
expect(result.manager).toBeTruthy();
|
|
if (!result.manager) {
|
|
throw new Error("manager missing");
|
|
}
|
|
return result.manager;
|
|
}
|
|
|
|
async function createFailedQmdSearchHarness(params: { agentId: string; errorMessage: string }) {
|
|
const cfg = createQmdCfg(params.agentId);
|
|
mockPrimary.search.mockRejectedValueOnce(new Error(params.errorMessage));
|
|
const first = await getMemorySearchManager({ cfg, agentId: params.agentId });
|
|
return { cfg, manager: requireManager(first), firstResult: first };
|
|
}
|
|
|
|
beforeEach(() => {
|
|
mockPrimary.search.mockClear();
|
|
mockPrimary.readFile.mockClear();
|
|
mockPrimary.status.mockClear();
|
|
mockPrimary.sync.mockClear();
|
|
mockPrimary.probeEmbeddingAvailability.mockClear();
|
|
mockPrimary.probeVectorAvailability.mockClear();
|
|
mockPrimary.close.mockClear();
|
|
fallbackSearch.mockClear();
|
|
fallbackManager.readFile.mockClear();
|
|
fallbackManager.status.mockClear();
|
|
fallbackManager.sync.mockClear();
|
|
fallbackManager.probeEmbeddingAvailability.mockClear();
|
|
fallbackManager.probeVectorAvailability.mockClear();
|
|
fallbackManager.close.mockClear();
|
|
mockMemoryIndexGet.mockClear();
|
|
mockMemoryIndexGet.mockResolvedValue(fallbackManager);
|
|
createQmdManagerMock.mockClear();
|
|
});
|
|
|
|
describe("getMemorySearchManager caching", () => {
|
|
it("reuses the same QMD manager instance for repeated calls", async () => {
|
|
const cfg = createQmdCfg("main");
|
|
|
|
const first = await getMemorySearchManager({ cfg, agentId: "main" });
|
|
const second = await getMemorySearchManager({ cfg, agentId: "main" });
|
|
|
|
expect(first.manager).toBe(second.manager);
|
|
// eslint-disable-next-line @typescript-eslint/unbound-method
|
|
expect(createQmdManagerMock).toHaveBeenCalledTimes(1);
|
|
});
|
|
|
|
it("evicts failed qmd wrapper so next call retries qmd", async () => {
|
|
const retryAgentId = "retry-agent";
|
|
const {
|
|
cfg,
|
|
manager: firstManager,
|
|
firstResult: first,
|
|
} = await createFailedQmdSearchHarness({
|
|
agentId: retryAgentId,
|
|
errorMessage: "qmd query failed",
|
|
});
|
|
|
|
const fallbackResults = await firstManager.search("hello");
|
|
expect(fallbackResults).toHaveLength(1);
|
|
expect(fallbackResults[0]?.path).toBe("MEMORY.md");
|
|
|
|
const second = await getMemorySearchManager({ cfg, agentId: retryAgentId });
|
|
requireManager(second);
|
|
expect(second.manager).not.toBe(first.manager);
|
|
// eslint-disable-next-line @typescript-eslint/unbound-method
|
|
expect(createQmdManagerMock).toHaveBeenCalledTimes(2);
|
|
});
|
|
|
|
it("does not cache status-only qmd managers", async () => {
|
|
const agentId = "status-agent";
|
|
const cfg = createQmdCfg(agentId);
|
|
|
|
const first = await getMemorySearchManager({ cfg, agentId, purpose: "status" });
|
|
const second = await getMemorySearchManager({ cfg, agentId, purpose: "status" });
|
|
|
|
requireManager(first);
|
|
requireManager(second);
|
|
// eslint-disable-next-line @typescript-eslint/unbound-method
|
|
expect(createQmdManagerMock).toHaveBeenCalledTimes(2);
|
|
// eslint-disable-next-line @typescript-eslint/unbound-method
|
|
expect(createQmdManagerMock).toHaveBeenNthCalledWith(
|
|
1,
|
|
expect.objectContaining({ agentId, mode: "status" }),
|
|
);
|
|
// eslint-disable-next-line @typescript-eslint/unbound-method
|
|
expect(createQmdManagerMock).toHaveBeenNthCalledWith(
|
|
2,
|
|
expect.objectContaining({ agentId, mode: "status" }),
|
|
);
|
|
});
|
|
|
|
it("does not evict a newer cached wrapper when closing an older failed wrapper", async () => {
|
|
const retryAgentId = "retry-agent-close";
|
|
const {
|
|
cfg,
|
|
manager: firstManager,
|
|
firstResult: first,
|
|
} = await createFailedQmdSearchHarness({
|
|
agentId: retryAgentId,
|
|
errorMessage: "qmd query failed",
|
|
});
|
|
await firstManager.search("hello");
|
|
|
|
const second = await getMemorySearchManager({ cfg, agentId: retryAgentId });
|
|
const secondManager = requireManager(second);
|
|
expect(second.manager).not.toBe(first.manager);
|
|
|
|
await firstManager.close?.();
|
|
|
|
const third = await getMemorySearchManager({ cfg, agentId: retryAgentId });
|
|
expect(third.manager).toBe(secondManager);
|
|
// eslint-disable-next-line @typescript-eslint/unbound-method
|
|
expect(createQmdManagerMock).toHaveBeenCalledTimes(2);
|
|
});
|
|
|
|
it("falls back to builtin search when qmd fails with sqlite busy", async () => {
|
|
const retryAgentId = "retry-agent-busy";
|
|
const { manager: firstManager } = await createFailedQmdSearchHarness({
|
|
agentId: retryAgentId,
|
|
errorMessage: "qmd index busy while reading results: SQLITE_BUSY: database is locked",
|
|
});
|
|
|
|
const results = await firstManager.search("hello");
|
|
expect(results).toHaveLength(1);
|
|
expect(results[0]?.path).toBe("MEMORY.md");
|
|
expect(fallbackSearch).toHaveBeenCalledTimes(1);
|
|
});
|
|
|
|
it("keeps original qmd error when fallback manager initialization fails", async () => {
|
|
const retryAgentId = "retry-agent-no-fallback-auth";
|
|
const { manager: firstManager } = await createFailedQmdSearchHarness({
|
|
agentId: retryAgentId,
|
|
errorMessage: "qmd query failed",
|
|
});
|
|
mockMemoryIndexGet.mockRejectedValueOnce(new Error("No API key found for provider openai"));
|
|
|
|
await expect(firstManager.search("hello")).rejects.toThrow("qmd query failed");
|
|
});
|
|
});
|