import assert from "node:assert/strict"; import { AUTHORITY_GRAPH_STORE_KIND, AUTHORITY_GRAPH_STORE_MODE, AuthorityGraphStore, AuthoritySqlHttpClient, } from "../sync/authority-graph-store.js"; import { BME_DB_SCHEMA_VERSION, BME_TOMBSTONE_RETENTION_MS, } from "../sync/bme-db.js"; const PREFIX = "[ST-BME][authority-graph-store]"; class MockAuthoritySqlClient { constructor() { this.meta = new Map(); this.nodes = new Map(); this.edges = new Map(); this.tombstones = new Map(); this.statements = []; } async transaction(statements = []) { for (const statement of statements) { await this.execute(statement.sql, statement.params || {}); } return { executed: statements.length }; } async execute(sql, params = {}) { this.statements.push({ sql, params }); const normalizedSql = String(sql || "").toLowerCase(); if (normalizedSql.startsWith("create table")) { return { ok: true }; } if (normalizedSql.includes("insert into st_bme_graph_meta")) { this.meta.set(this._key(params.chatId, params.key), { chat_id: params.chatId, meta_key: params.key, value_json: params.valueJson, updated_at: params.updatedAt, }); return { ok: true }; } if (normalizedSql.includes("insert into st_bme_graph_nodes")) { this.nodes.set(this._key(params.chatId, params.id), { chat_id: params.chatId, record_id: params.id, payload_json: params.payloadJson, node_type: params.type, source_floor: params.sourceFloor, archived: params.archived, updated_at: params.updatedAt, deleted_at: params.deletedAt, }); return { ok: true }; } if (normalizedSql.includes("insert into st_bme_graph_edges")) { this.edges.set(this._key(params.chatId, params.id), { chat_id: params.chatId, record_id: params.id, payload_json: params.payloadJson, from_id: params.fromId, to_id: params.toId, relation: params.relation, source_floor: params.sourceFloor, updated_at: params.updatedAt, deleted_at: params.deletedAt, }); return { ok: true }; } if (normalizedSql.includes("insert into st_bme_graph_tombstones")) { this.tombstones.set(this._key(params.chatId, params.id), { chat_id: params.chatId, record_id: params.id, payload_json: params.payloadJson, tombstone_kind: params.kind, target_id: params.targetId, deleted_at: params.deletedAt, source_device_id: params.sourceDeviceId, }); return { ok: true }; } if (normalizedSql.startsWith("delete from st_bme_graph_nodes")) { this._deleteRows(this.nodes, params); return { ok: true }; } if (normalizedSql.startsWith("delete from st_bme_graph_edges")) { this._deleteRows(this.edges, params); return { ok: true }; } if (normalizedSql.startsWith("delete from st_bme_graph_tombstones")) { this._deleteRows(this.tombstones, params); return { ok: true }; } if (normalizedSql.startsWith("delete from st_bme_graph_meta")) { this._deleteRows(this.meta, params); return { ok: true }; } throw new Error(`Unhandled SQL execute: ${sql}`); } async query(sql, params = {}) { const normalizedSql = String(sql || "").toLowerCase(); if (normalizedSql.includes("from st_bme_graph_meta")) { return this._readRows(this.meta, params).map((row) => ({ key: row.meta_key, valueJson: row.value_json, })); } if (normalizedSql.includes("from st_bme_graph_nodes")) { if (normalizedSql.includes("count(*)")) { return [{ count: this._readRows(this.nodes, params).length }]; } return this._readRows(this.nodes, params).map((row) => ({ payloadJson: row.payload_json, })); } if (normalizedSql.includes("from st_bme_graph_edges")) { if (normalizedSql.includes("count(*)")) { return [{ count: this._readRows(this.edges, params).length }]; } return this._readRows(this.edges, params).map((row) => ({ payloadJson: row.payload_json, })); } if (normalizedSql.includes("from st_bme_graph_tombstones")) { if (normalizedSql.includes("count(*)")) { return [{ count: this._readRows(this.tombstones, params).length }]; } if (normalizedSql.includes("deleted_at <")) { return this._readRows(this.tombstones, params) .filter((row) => Number(row.deleted_at) < Number(params.cutoffMs)) .map((row) => ({ id: row.record_id })); } return this._readRows(this.tombstones, params).map((row) => ({ payloadJson: row.payload_json, })); } throw new Error(`Unhandled SQL query: ${sql}`); } _key(chatId, id) { return `${String(chatId || "")}\u0000${String(id || "")}`; } _readRows(table, params = {}) { const chatId = String(params.chatId || ""); const id = params.id ?? params.key; return Array.from(table.values()).filter((row) => { if (String(row.chat_id || "") !== chatId) return false; if (id == null) return true; return String(row.record_id ?? row.meta_key ?? "") === String(id); }); } _deleteRows(table, params = {}) { const chatId = String(params.chatId || ""); const id = params.id ?? params.key; for (const [key, row] of table.entries()) { if (String(row.chat_id || "") !== chatId) continue; if (id != null && String(row.record_id ?? row.meta_key ?? "") !== String(id)) continue; table.delete(key); } } } async function testOpenSeedsAuthorityMeta() { const sqlClient = new MockAuthoritySqlClient(); const store = new AuthorityGraphStore("authority-chat-a", { sqlClient }); await store.open(); assert.equal(store.storeKind, AUTHORITY_GRAPH_STORE_KIND); assert.equal(store.storeMode, AUTHORITY_GRAPH_STORE_MODE); assert.equal(await store.getMeta("schemaVersion"), BME_DB_SCHEMA_VERSION); assert.equal(await store.getMeta("storagePrimary"), AUTHORITY_GRAPH_STORE_KIND); assert.equal(await store.getRevision(), 0); const diagnostics = store.getStorageDiagnosticsSync(); assert.equal(diagnostics.storageKind, AUTHORITY_GRAPH_STORE_KIND); assert.equal(diagnostics.browserCacheMode, "minimal"); } async function testImportCommitAndExportSnapshot() { const sqlClient = new MockAuthoritySqlClient(); const store = new AuthorityGraphStore("authority-chat-b", { sqlClient }); await store.open(); const importResult = await store.importSnapshot( { meta: { revision: 7, lastProcessedFloor: 3, extractionCount: 4, }, nodes: [ { id: "node-1", type: "event", sourceFloor: 1, updatedAt: 10 }, { id: "node-2", type: "event", archived: true, updatedAt: 20 }, { id: "node-3", type: "memory", deletedAt: 30, updatedAt: 30 }, ], edges: [ { id: "edge-1", fromId: "node-1", toId: "node-3", relation: "refers", updatedAt: 40, }, ], tombstones: [ { id: "tombstone-1", kind: "node", targetId: "node-old", deletedAt: 50, }, ], }, { preserveRevision: true }, ); assert.equal(importResult.revision, 7); assert.deepEqual(importResult.imported, { nodes: 3, edges: 1, tombstones: 1 }); assert.equal((await store.listNodes()).length, 3); assert.deepEqual( (await store.listNodes({ includeArchived: false, includeDeleted: false })).map((node) => node.id), ["node-1"], ); assert.deepEqual((await store.listEdges({ relation: "refers" })).map((edge) => edge.id), ["edge-1"]); const commitResult = await store.commitDelta( { upsertNodes: [{ id: "node-4", type: "event", updatedAt: 60 }], deleteNodeIds: ["node-2"], countDelta: { previous: { nodes: 3, edges: 1, tombstones: 1 }, delta: { nodes: 0, edges: 0, tombstones: 0 }, }, runtimeMetaPatch: { lastProcessedFloor: 8, revision: 999, }, }, { reason: "test-commit", requestedRevision: 9, }, ); assert.equal(commitResult.revision, 9); assert.deepEqual(commitResult.imported, { nodes: 3, edges: 1, tombstones: 1 }); assert.equal(await store.getMeta("lastProcessedFloor"), 8); assert.equal(await store.getRevision(), 9); assert.equal(await store.getMeta("lastMutationReason"), "test-commit"); assert.equal(await store.getMeta("syncDirty"), true); assert.deepEqual((await store.listNodes()).map((node) => node.id).sort(), ["node-1", "node-3", "node-4"]); const snapshot = await store.exportSnapshot(); assert.equal(snapshot.meta.revision, 9); assert.equal(snapshot.meta.storagePrimary, AUTHORITY_GRAPH_STORE_KIND); assert.equal(snapshot.meta.storageMode, AUTHORITY_GRAPH_STORE_MODE); assert.equal(snapshot.meta.nodeCount, 3); assert.equal(snapshot.nodes.length, 3); assert.equal(snapshot.edges.length, 1); assert.equal(snapshot.tombstones.length, 1); assert.equal(snapshot.state.lastProcessedFloor, 8); } async function testPruneAndClear() { const sqlClient = new MockAuthoritySqlClient(); const store = new AuthorityGraphStore("authority-chat-c", { sqlClient }); await store.importSnapshot({ nodes: [{ id: "node-1", type: "event", updatedAt: 1 }], tombstones: [ { id: "old-tombstone", kind: "node", targetId: "old", deletedAt: 1 }, { id: "new-tombstone", kind: "node", targetId: "new", deletedAt: BME_TOMBSTONE_RETENTION_MS, }, ], }); const pruneResult = await store.pruneExpiredTombstones(BME_TOMBSTONE_RETENTION_MS + 100); assert.equal(pruneResult.pruned, 1); assert.deepEqual((await store.listTombstones()).map((item) => item.id), ["new-tombstone"]); const clearResult = await store.clearAll(); assert.equal(clearResult.cleared, true); assert.equal((await store.isEmpty({ includeTombstones: true })).empty, true); assert.equal(await store.getMeta("storagePrimary"), AUTHORITY_GRAPH_STORE_KIND); } async function testHttpSqlClientBoundary() { const requests = []; const client = new AuthoritySqlHttpClient({ baseUrl: "https://authority.example.test/root/", headerProvider: () => ({ "X-Test": "1" }), fetchImpl: async (url, init) => { requests.push({ url, init }); return { ok: true, status: 200, async json() { return { rows: [{ value: 1 }] }; }, }; }, }); const result = await client.query("SELECT 1", { chatId: "chat" }); assert.deepEqual(result, { rows: [{ value: 1 }] }); assert.equal(requests[0].url, "https://authority.example.test/root/v1/sql"); assert.equal(requests[0].init.method, "POST"); assert.equal(requests[0].init.headers["X-Test"], "1"); assert.deepEqual(JSON.parse(requests[0].init.body), { action: "query", sql: "SELECT 1", params: { chatId: "chat" }, }); } await testOpenSeedsAuthorityMeta(); await testImportCommitAndExportSnapshot(); await testPruneAndClear(); await testHttpSqlClientBoundary(); console.log(`${PREFIX} all tests passed`);