- Add modular DB layer (adapters, migrations, repos, helpers) - Replace localDb/usageDb/requestDetailsDb monoliths with repos - Add Tailscale tunnel integration & status check API - Add /api/cli-tools/all-statuses aggregated endpoint - Add settingsStore (Zustand) and mitm/dbReader - Add DB unit tests (benchmark, concurrent, migration, vs-lowdb)
274 lines
11 KiB
JavaScript
274 lines
11 KiB
JavaScript
// Compare new SQLite-backed DB layer vs legacy lowdb behavior.
|
|
// Verifies: same public API signatures + equivalent results for core operations.
|
|
import fs from "node:fs";
|
|
import os from "node:os";
|
|
import path from "node:path";
|
|
import { describe, it, expect, beforeAll, afterAll, vi } from "vitest";
|
|
|
|
const originalDataDir = process.env.DATA_DIR;
|
|
let tempDir;
|
|
let sqliteDb;
|
|
|
|
beforeAll(async () => {
|
|
tempDir = fs.mkdtempSync(path.join(os.tmpdir(), "9router-db-compare-"));
|
|
process.env.DATA_DIR = tempDir;
|
|
vi.resetModules();
|
|
sqliteDb = await import("@/lib/db/index.js");
|
|
await sqliteDb.initDb();
|
|
});
|
|
|
|
afterAll(() => {
|
|
if (tempDir) fs.rmSync(tempDir, { recursive: true, force: true });
|
|
if (originalDataDir === undefined) delete process.env.DATA_DIR;
|
|
else process.env.DATA_DIR = originalDataDir;
|
|
});
|
|
|
|
describe("DB SQLite layer — public API parity", () => {
|
|
it("settings: get → defaults; update → merge", async () => {
|
|
const s = await sqliteDb.getSettings();
|
|
expect(s).toBeDefined();
|
|
expect(s.cloudEnabled).toBe(false);
|
|
expect(s.requireLogin).toBe(true);
|
|
|
|
const updated = await sqliteDb.updateSettings({ cloudEnabled: true, customField: "x" });
|
|
expect(updated.cloudEnabled).toBe(true);
|
|
expect(updated.customField).toBe("x");
|
|
expect(updated.requireLogin).toBe(true); // default preserved
|
|
|
|
const re = await sqliteDb.getSettings();
|
|
expect(re.cloudEnabled).toBe(true);
|
|
expect(re.customField).toBe("x");
|
|
});
|
|
|
|
it("isCloudEnabled reflects settings", async () => {
|
|
await sqliteDb.updateSettings({ cloudEnabled: true });
|
|
expect(await sqliteDb.isCloudEnabled()).toBe(true);
|
|
await sqliteDb.updateSettings({ cloudEnabled: false });
|
|
expect(await sqliteDb.isCloudEnabled()).toBe(false);
|
|
});
|
|
|
|
it("apiKeys: create/get/validate/delete", async () => {
|
|
const k = await sqliteDb.createApiKey("test-key", "machine-abc");
|
|
expect(k.id).toBeDefined();
|
|
expect(k.key).toMatch(/^sk-/);
|
|
expect(k.machineId).toBe("machine-abc");
|
|
expect(k.isActive).toBe(true);
|
|
|
|
const all = await sqliteDb.getApiKeys();
|
|
expect(all.find((x) => x.id === k.id)).toBeDefined();
|
|
|
|
expect(await sqliteDb.validateApiKey(k.key)).toBeTruthy();
|
|
expect(await sqliteDb.validateApiKey("invalid")).toBeFalsy();
|
|
|
|
const deleted = await sqliteDb.deleteApiKey(k.id);
|
|
expect(deleted).toBe(true);
|
|
expect(await sqliteDb.getApiKeyById(k.id)).toBeNull();
|
|
});
|
|
|
|
it("providerConnections: CRUD + reorder by priority", async () => {
|
|
const c1 = await sqliteDb.createProviderConnection({ provider: "test", authType: "apikey", name: "a", apiKey: "k1" });
|
|
const c2 = await sqliteDb.createProviderConnection({ provider: "test", authType: "apikey", name: "b", apiKey: "k2" });
|
|
const c3 = await sqliteDb.createProviderConnection({ provider: "test", authType: "apikey", name: "c", apiKey: "k3" });
|
|
|
|
const list = await sqliteDb.getProviderConnections({ provider: "test" });
|
|
expect(list).toHaveLength(3);
|
|
expect(list[0].priority).toBe(1);
|
|
expect(list[1].priority).toBe(2);
|
|
expect(list[2].priority).toBe(3);
|
|
|
|
// Update priority and reorder
|
|
await sqliteDb.updateProviderConnection(c3.id, { priority: 1 });
|
|
const reordered = await sqliteDb.getProviderConnections({ provider: "test" });
|
|
expect(reordered[0].name).toBe("c");
|
|
|
|
// Delete reorders remaining
|
|
await sqliteDb.deleteProviderConnection(c1.id);
|
|
const after = await sqliteDb.getProviderConnections({ provider: "test" });
|
|
expect(after).toHaveLength(2);
|
|
expect(after.every((c) => [1, 2].includes(c.priority))).toBe(true);
|
|
});
|
|
|
|
it("providerConnections: optional fields persisted via JSON column", async () => {
|
|
const c = await sqliteDb.createProviderConnection({
|
|
provider: "p2", authType: "oauth", email: "x@y.com",
|
|
accessToken: "tok", refreshToken: "rtok", expiresAt: 12345,
|
|
providerSpecificData: { foo: "bar" },
|
|
});
|
|
const back = await sqliteDb.getProviderConnectionById(c.id);
|
|
expect(back.accessToken).toBe("tok");
|
|
expect(back.refreshToken).toBe("rtok");
|
|
expect(back.expiresAt).toBe(12345);
|
|
expect(back.providerSpecificData).toEqual({ foo: "bar" });
|
|
});
|
|
|
|
it("providerNodes: CRUD", async () => {
|
|
const n = await sqliteDb.createProviderNode({ type: "openai", name: "Test", baseUrl: "https://api.test", apiType: "openai" });
|
|
expect(n.id).toBeDefined();
|
|
expect(n.baseUrl).toBe("https://api.test");
|
|
|
|
const all = await sqliteDb.getProviderNodes({ type: "openai" });
|
|
expect(all.find((x) => x.id === n.id)).toBeDefined();
|
|
|
|
await sqliteDb.updateProviderNode(n.id, { name: "Test2" });
|
|
const updated = await sqliteDb.getProviderNodeById(n.id);
|
|
expect(updated.name).toBe("Test2");
|
|
|
|
await sqliteDb.deleteProviderNode(n.id);
|
|
expect(await sqliteDb.getProviderNodeById(n.id)).toBeNull();
|
|
});
|
|
|
|
it("proxyPools: CRUD with sort by updatedAt desc", async () => {
|
|
const p1 = await sqliteDb.createProxyPool({ name: "p1", proxyUrl: "http://a", type: "http" });
|
|
await new Promise((r) => setTimeout(r, 10));
|
|
const p2 = await sqliteDb.createProxyPool({ name: "p2", proxyUrl: "http://b", type: "http" });
|
|
const list = await sqliteDb.getProxyPools();
|
|
expect(list[0].id).toBe(p2.id); // newest first
|
|
await sqliteDb.deleteProxyPool(p1.id);
|
|
await sqliteDb.deleteProxyPool(p2.id);
|
|
});
|
|
|
|
it("combos: CRUD", async () => {
|
|
const c = await sqliteDb.createCombo({ name: "combo1", models: ["m1", "m2"], kind: "fallback" });
|
|
expect(c.id).toBeDefined();
|
|
expect(c.models).toEqual(["m1", "m2"]);
|
|
const byName = await sqliteDb.getComboByName("combo1");
|
|
expect(byName.id).toBe(c.id);
|
|
await sqliteDb.updateCombo(c.id, { models: ["m3"] });
|
|
const updated = await sqliteDb.getComboById(c.id);
|
|
expect(updated.models).toEqual(["m3"]);
|
|
expect(await sqliteDb.deleteCombo(c.id)).toBe(true);
|
|
});
|
|
|
|
it("modelAliases: KV ops", async () => {
|
|
await sqliteDb.setModelAlias("alias1", "real-model-1");
|
|
await sqliteDb.setModelAlias("alias2", "real-model-2");
|
|
const all = await sqliteDb.getModelAliases();
|
|
expect(all.alias1).toBe("real-model-1");
|
|
expect(all.alias2).toBe("real-model-2");
|
|
await sqliteDb.deleteModelAlias("alias1");
|
|
expect((await sqliteDb.getModelAliases()).alias1).toBeUndefined();
|
|
});
|
|
|
|
it("customModels: add/list/delete with dedupe", async () => {
|
|
const ok1 = await sqliteDb.addCustomModel({ providerAlias: "p1", id: "m1", type: "llm", name: "Model 1" });
|
|
const dup = await sqliteDb.addCustomModel({ providerAlias: "p1", id: "m1", type: "llm" });
|
|
expect(ok1).toBe(true);
|
|
expect(dup).toBe(false);
|
|
const list = await sqliteDb.getCustomModels();
|
|
expect(list.find((m) => m.id === "m1")).toBeDefined();
|
|
await sqliteDb.deleteCustomModel({ providerAlias: "p1", id: "m1" });
|
|
const after = await sqliteDb.getCustomModels();
|
|
expect(after.find((m) => m.id === "m1")).toBeUndefined();
|
|
});
|
|
|
|
it("mitmAlias: get/set per tool", async () => {
|
|
await sqliteDb.setMitmAliasAll("cursor", { "gpt-5": "claude-3" });
|
|
const a = await sqliteDb.getMitmAlias("cursor");
|
|
expect(a["gpt-5"]).toBe("claude-3");
|
|
const all = await sqliteDb.getMitmAlias();
|
|
expect(all.cursor).toEqual({ "gpt-5": "claude-3" });
|
|
});
|
|
|
|
it("disabledModels: add/remove per provider", async () => {
|
|
await sqliteDb.disableModels("openai", ["gpt-3", "gpt-4"]);
|
|
expect(await sqliteDb.getDisabledByProvider("openai")).toEqual(expect.arrayContaining(["gpt-3", "gpt-4"]));
|
|
await sqliteDb.enableModels("openai", ["gpt-3"]);
|
|
expect(await sqliteDb.getDisabledByProvider("openai")).toEqual(["gpt-4"]);
|
|
await sqliteDb.enableModels("openai", []);
|
|
expect(await sqliteDb.getDisabledByProvider("openai")).toEqual([]);
|
|
});
|
|
|
|
it("usage: saveRequestUsage + getUsageHistory + getUsageStats", async () => {
|
|
await sqliteDb.saveRequestUsage({
|
|
provider: "openai", model: "gpt-4", connectionId: "c1",
|
|
tokens: { prompt_tokens: 100, completion_tokens: 50 },
|
|
endpoint: "/v1/chat/completions", status: "ok",
|
|
});
|
|
await sqliteDb.saveRequestUsage({
|
|
provider: "openai", model: "gpt-4", connectionId: "c1",
|
|
tokens: { prompt_tokens: 200, completion_tokens: 100 },
|
|
endpoint: "/v1/chat/completions", status: "ok",
|
|
});
|
|
|
|
const hist = await sqliteDb.getUsageHistory({ provider: "openai" });
|
|
expect(hist.length).toBeGreaterThanOrEqual(2);
|
|
expect(hist[0].tokens.prompt_tokens).toBeDefined();
|
|
|
|
const stats = await sqliteDb.getUsageStats("24h");
|
|
expect(stats.totalRequests).toBeGreaterThanOrEqual(2);
|
|
expect(stats.byProvider.openai).toBeDefined();
|
|
expect(stats.byProvider.openai.requests).toBeGreaterThanOrEqual(2);
|
|
expect(stats.byProvider.openai.promptTokens).toBeGreaterThanOrEqual(300);
|
|
});
|
|
|
|
it("usage: pending tracking in-memory", () => {
|
|
sqliteDb.trackPendingRequest("gpt-4", "openai", "c1", true);
|
|
expect(global._pendingRequests.byModel["gpt-4 (openai)"]).toBe(1);
|
|
sqliteDb.trackPendingRequest("gpt-4", "openai", "c1", false);
|
|
expect(global._pendingRequests.byModel["gpt-4 (openai)"]).toBeUndefined();
|
|
});
|
|
|
|
it("requestDetails: save → query with paging", async () => {
|
|
// Enable observability first
|
|
await sqliteDb.updateSettings({ enableObservability: true, observabilityBatchSize: 1 });
|
|
|
|
await sqliteDb.saveRequestDetail({
|
|
id: "d1", provider: "openai", model: "gpt-4", connectionId: "c1",
|
|
status: "ok", tokens: { prompt_tokens: 10 },
|
|
request: { method: "POST" }, response: { status: 200 },
|
|
});
|
|
|
|
// Wait for buffer flush
|
|
await new Promise((r) => setTimeout(r, 200));
|
|
|
|
const got = await sqliteDb.getRequestDetailById("d1");
|
|
expect(got).toBeDefined();
|
|
expect(got.id).toBe("d1");
|
|
|
|
const list = await sqliteDb.getRequestDetails({ provider: "openai" });
|
|
expect(list.details.length).toBeGreaterThanOrEqual(1);
|
|
expect(list.pagination.totalItems).toBeGreaterThanOrEqual(1);
|
|
});
|
|
|
|
it("exportDb / importDb roundtrip", async () => {
|
|
const exported = await sqliteDb.exportDb();
|
|
expect(exported.settings).toBeDefined();
|
|
expect(Array.isArray(exported.providerConnections)).toBe(true);
|
|
expect(typeof exported.modelAliases).toBe("object");
|
|
|
|
// Add marker, export, import a different payload, verify reset
|
|
await sqliteDb.setModelAlias("marker", "before");
|
|
const snap = await sqliteDb.exportDb();
|
|
|
|
await sqliteDb.setModelAlias("marker", "after");
|
|
expect((await sqliteDb.getModelAliases()).marker).toBe("after");
|
|
|
|
await sqliteDb.importDb(snap);
|
|
expect((await sqliteDb.getModelAliases()).marker).toBe("before");
|
|
});
|
|
|
|
it("pricing: user pricing merged with constants", async () => {
|
|
await sqliteDb.updatePricing({ openai: { "gpt-test": { input: 1, output: 2 } } });
|
|
const p = await sqliteDb.getPricing();
|
|
expect(p.openai["gpt-test"]).toEqual({ input: 1, output: 2 });
|
|
|
|
const single = await sqliteDb.getPricingForModel("openai", "gpt-test");
|
|
expect(single).toEqual({ input: 1, output: 2 });
|
|
|
|
await sqliteDb.resetPricing("openai", "gpt-test");
|
|
expect((await sqliteDb.getPricing()).openai?.["gpt-test"]).toBeUndefined();
|
|
});
|
|
|
|
it("getChartData: 24h buckets", async () => {
|
|
const data = await sqliteDb.getChartData("24h");
|
|
expect(data).toHaveLength(24);
|
|
expect(data[0]).toHaveProperty("label");
|
|
expect(data[0]).toHaveProperty("tokens");
|
|
expect(data[0]).toHaveProperty("cost");
|
|
});
|
|
|
|
it("getChartData: 7d buckets", async () => {
|
|
const data = await sqliteDb.getChartData("7d");
|
|
expect(data).toHaveLength(7);
|
|
});
|
|
});
|