@stackmemoryai/stackmemory 0.6.0 → 0.6.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/src/cli/commands/daemon.js +27 -0
- package/dist/src/cli/commands/daemon.js.map +2 -2
- package/dist/src/core/context/frame-database.js +12 -3
- package/dist/src/core/context/frame-database.js.map +2 -2
- package/dist/src/core/database/embedding-provider.js +17 -0
- package/dist/src/core/database/embedding-provider.js.map +7 -0
- package/dist/src/core/database/sqlite-adapter.js +257 -11
- package/dist/src/core/database/sqlite-adapter.js.map +2 -2
- package/dist/src/daemon/daemon-config.js +17 -0
- package/dist/src/daemon/daemon-config.js.map +2 -2
- package/dist/src/daemon/services/maintenance-service.js +230 -0
- package/dist/src/daemon/services/maintenance-service.js.map +7 -0
- package/dist/src/daemon/unified-daemon.js +36 -3
- package/dist/src/daemon/unified-daemon.js.map +2 -2
- package/package.json +2 -2
- package/scripts/install-claude-hooks-auto.js +10 -10
- package/templates/claude-hooks/on-task-complete.js +3 -4
|
@@ -14,15 +14,20 @@ class SQLiteAdapter extends FeatureAwareDatabaseAdapter {
|
|
|
14
14
|
db = null;
|
|
15
15
|
dbPath;
|
|
16
16
|
inTransactionFlag = false;
|
|
17
|
+
ftsEnabled = false;
|
|
18
|
+
vecEnabled = false;
|
|
19
|
+
embeddingProvider;
|
|
20
|
+
embeddingDimension;
|
|
17
21
|
constructor(projectId, config) {
|
|
18
22
|
super(projectId, config);
|
|
19
23
|
this.dbPath = config.dbPath;
|
|
24
|
+
this.embeddingProvider = config.embeddingProvider;
|
|
25
|
+
this.embeddingDimension = config.embeddingDimension || 384;
|
|
20
26
|
}
|
|
21
27
|
getFeatures() {
|
|
22
28
|
return {
|
|
23
|
-
supportsFullTextSearch:
|
|
24
|
-
|
|
25
|
-
supportsVectorSearch: false,
|
|
29
|
+
supportsFullTextSearch: this.ftsEnabled,
|
|
30
|
+
supportsVectorSearch: this.vecEnabled,
|
|
26
31
|
supportsPartitioning: false,
|
|
27
32
|
supportsAnalytics: false,
|
|
28
33
|
supportsCompression: false,
|
|
@@ -149,6 +154,8 @@ class SQLiteAdapter extends FeatureAwareDatabaseAdapter {
|
|
|
149
154
|
} catch (e) {
|
|
150
155
|
logger.warn("Failed to ensure cascade constraints", e);
|
|
151
156
|
}
|
|
157
|
+
this.initializeFts();
|
|
158
|
+
this.initializeVec();
|
|
152
159
|
}
|
|
153
160
|
/**
|
|
154
161
|
* Ensure ON DELETE CASCADE exists for events/anchors referencing frames
|
|
@@ -208,6 +215,115 @@ class SQLiteAdapter extends FeatureAwareDatabaseAdapter {
|
|
|
208
215
|
if (needsCascade("events")) migrateTable("events");
|
|
209
216
|
if (needsCascade("anchors")) migrateTable("anchors");
|
|
210
217
|
}
|
|
218
|
+
/**
|
|
219
|
+
* Initialize FTS5 virtual table and sync triggers
|
|
220
|
+
*/
|
|
221
|
+
initializeFts() {
|
|
222
|
+
if (!this.db) return;
|
|
223
|
+
try {
|
|
224
|
+
this.db.exec(`
|
|
225
|
+
CREATE VIRTUAL TABLE IF NOT EXISTS frames_fts USING fts5(
|
|
226
|
+
name, digest_text, inputs, outputs,
|
|
227
|
+
content='frames', content_rowid='rowid'
|
|
228
|
+
);
|
|
229
|
+
`);
|
|
230
|
+
this.db.exec(`
|
|
231
|
+
CREATE TRIGGER IF NOT EXISTS frames_ai AFTER INSERT ON frames BEGIN
|
|
232
|
+
INSERT INTO frames_fts(rowid, name, digest_text, inputs, outputs)
|
|
233
|
+
VALUES (new.rowid, new.name, new.digest_text, new.inputs, new.outputs);
|
|
234
|
+
END;
|
|
235
|
+
`);
|
|
236
|
+
this.db.exec(`
|
|
237
|
+
CREATE TRIGGER IF NOT EXISTS frames_ad AFTER DELETE ON frames BEGIN
|
|
238
|
+
INSERT INTO frames_fts(frames_fts, rowid, name, digest_text, inputs, outputs)
|
|
239
|
+
VALUES ('delete', old.rowid, old.name, old.digest_text, old.inputs, old.outputs);
|
|
240
|
+
END;
|
|
241
|
+
`);
|
|
242
|
+
this.db.exec(`
|
|
243
|
+
CREATE TRIGGER IF NOT EXISTS frames_au AFTER UPDATE ON frames BEGIN
|
|
244
|
+
INSERT INTO frames_fts(frames_fts, rowid, name, digest_text, inputs, outputs)
|
|
245
|
+
VALUES ('delete', old.rowid, old.name, old.digest_text, old.inputs, old.outputs);
|
|
246
|
+
INSERT INTO frames_fts(rowid, name, digest_text, inputs, outputs)
|
|
247
|
+
VALUES (new.rowid, new.name, new.digest_text, new.inputs, new.outputs);
|
|
248
|
+
END;
|
|
249
|
+
`);
|
|
250
|
+
this.migrateToFts();
|
|
251
|
+
this.ftsEnabled = true;
|
|
252
|
+
logger.info("FTS5 full-text search initialized");
|
|
253
|
+
} catch (e) {
|
|
254
|
+
logger.warn(
|
|
255
|
+
"FTS5 initialization failed, falling back to LIKE search",
|
|
256
|
+
e
|
|
257
|
+
);
|
|
258
|
+
this.ftsEnabled = false;
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
/**
|
|
262
|
+
* One-time migration: populate FTS index from existing frames data
|
|
263
|
+
*/
|
|
264
|
+
migrateToFts() {
|
|
265
|
+
if (!this.db) return;
|
|
266
|
+
const version = this.db.prepare("SELECT MAX(version) as version FROM schema_version").get()?.version || 1;
|
|
267
|
+
if (version < 2) {
|
|
268
|
+
this.db.exec(`
|
|
269
|
+
INSERT OR IGNORE INTO frames_fts(rowid, name, digest_text, inputs, outputs)
|
|
270
|
+
SELECT rowid, name, digest_text, inputs, outputs FROM frames;
|
|
271
|
+
`);
|
|
272
|
+
this.db.prepare("INSERT OR REPLACE INTO schema_version (version) VALUES (?)").run(2);
|
|
273
|
+
logger.info(
|
|
274
|
+
"FTS5 index populated from existing frames (migration v1\u2192v2)"
|
|
275
|
+
);
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* Initialize sqlite-vec for vector search
|
|
280
|
+
*/
|
|
281
|
+
initializeVec() {
|
|
282
|
+
if (!this.db || !this.embeddingProvider) return;
|
|
283
|
+
try {
|
|
284
|
+
let sqliteVec;
|
|
285
|
+
try {
|
|
286
|
+
sqliteVec = require("sqlite-vec");
|
|
287
|
+
} catch {
|
|
288
|
+
logger.info("sqlite-vec not installed, vector search disabled");
|
|
289
|
+
return;
|
|
290
|
+
}
|
|
291
|
+
sqliteVec.load(this.db);
|
|
292
|
+
this.db.exec(`
|
|
293
|
+
CREATE VIRTUAL TABLE IF NOT EXISTS frame_embeddings USING vec0(
|
|
294
|
+
frame_id TEXT PRIMARY KEY,
|
|
295
|
+
embedding float[${this.embeddingDimension}]
|
|
296
|
+
);
|
|
297
|
+
`);
|
|
298
|
+
this.vecEnabled = true;
|
|
299
|
+
logger.info("sqlite-vec vector search initialized", {
|
|
300
|
+
dimension: this.embeddingDimension
|
|
301
|
+
});
|
|
302
|
+
} catch (e) {
|
|
303
|
+
logger.warn(
|
|
304
|
+
"sqlite-vec initialization failed, vector search disabled",
|
|
305
|
+
e
|
|
306
|
+
);
|
|
307
|
+
this.vecEnabled = false;
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
/**
|
|
311
|
+
* Rebuild the FTS5 index (for maintenance)
|
|
312
|
+
*/
|
|
313
|
+
async rebuildFtsIndex() {
|
|
314
|
+
if (!this.db) {
|
|
315
|
+
throw new DatabaseError(
|
|
316
|
+
"Database not connected",
|
|
317
|
+
ErrorCode.DB_CONNECTION_FAILED
|
|
318
|
+
);
|
|
319
|
+
}
|
|
320
|
+
if (!this.ftsEnabled) {
|
|
321
|
+
logger.warn("FTS not enabled, skipping rebuild");
|
|
322
|
+
return;
|
|
323
|
+
}
|
|
324
|
+
this.db.exec("INSERT INTO frames_fts(frames_fts) VALUES('rebuild')");
|
|
325
|
+
logger.info("FTS5 index rebuilt");
|
|
326
|
+
}
|
|
211
327
|
async migrateSchema(targetVersion) {
|
|
212
328
|
if (!this.db)
|
|
213
329
|
throw new DatabaseError(
|
|
@@ -462,16 +578,63 @@ class SQLiteAdapter extends FeatureAwareDatabaseAdapter {
|
|
|
462
578
|
);
|
|
463
579
|
this.db.prepare("DELETE FROM anchors WHERE frame_id = ?").run(frameId);
|
|
464
580
|
}
|
|
465
|
-
//
|
|
581
|
+
// Full-text search with FTS5 + BM25 ranking (fallback to LIKE)
|
|
466
582
|
async search(options) {
|
|
467
583
|
if (!this.db)
|
|
468
584
|
throw new DatabaseError(
|
|
469
585
|
"Database not connected",
|
|
470
586
|
ErrorCode.DB_CONNECTION_FAILED
|
|
471
587
|
);
|
|
588
|
+
if (this.ftsEnabled) {
|
|
589
|
+
try {
|
|
590
|
+
return this.searchFts(options);
|
|
591
|
+
} catch (e) {
|
|
592
|
+
logger.debug("FTS search failed, falling back to LIKE", {
|
|
593
|
+
error: e instanceof Error ? e.message : String(e),
|
|
594
|
+
query: options.query
|
|
595
|
+
});
|
|
596
|
+
}
|
|
597
|
+
}
|
|
598
|
+
return this.searchLike(options);
|
|
599
|
+
}
|
|
600
|
+
/**
|
|
601
|
+
* FTS5 MATCH search with BM25 ranking
|
|
602
|
+
*/
|
|
603
|
+
searchFts(options) {
|
|
604
|
+
const boost = options.boost || {};
|
|
605
|
+
const w0 = boost["name"] || 10;
|
|
606
|
+
const w1 = boost["digest_text"] || 5;
|
|
607
|
+
const w2 = boost["inputs"] || 2;
|
|
608
|
+
const w3 = boost["outputs"] || 1;
|
|
472
609
|
const sql = `
|
|
473
|
-
SELECT
|
|
474
|
-
|
|
610
|
+
SELECT f.*, -bm25(frames_fts, ${w0}, ${w1}, ${w2}, ${w3}) as score
|
|
611
|
+
FROM frames_fts fts
|
|
612
|
+
JOIN frames f ON f.rowid = fts.rowid
|
|
613
|
+
WHERE frames_fts MATCH ?
|
|
614
|
+
ORDER BY score DESC
|
|
615
|
+
LIMIT ? OFFSET ?
|
|
616
|
+
`;
|
|
617
|
+
const limit = options.limit || 50;
|
|
618
|
+
const offset = options.offset || 0;
|
|
619
|
+
const rows = this.db.prepare(sql).all(
|
|
620
|
+
options.query,
|
|
621
|
+
limit,
|
|
622
|
+
offset
|
|
623
|
+
);
|
|
624
|
+
return rows.map((row) => ({
|
|
625
|
+
...row,
|
|
626
|
+
inputs: JSON.parse(row.inputs || "{}"),
|
|
627
|
+
outputs: JSON.parse(row.outputs || "{}"),
|
|
628
|
+
digest_json: JSON.parse(row.digest_json || "{}")
|
|
629
|
+
}));
|
|
630
|
+
}
|
|
631
|
+
/**
|
|
632
|
+
* Fallback LIKE search for when FTS is unavailable
|
|
633
|
+
*/
|
|
634
|
+
searchLike(options) {
|
|
635
|
+
const sql = `
|
|
636
|
+
SELECT *,
|
|
637
|
+
CASE
|
|
475
638
|
WHEN name LIKE ? THEN 1.0
|
|
476
639
|
WHEN digest_text LIKE ? THEN 0.8
|
|
477
640
|
WHEN inputs LIKE ? THEN 0.6
|
|
@@ -498,12 +661,95 @@ class SQLiteAdapter extends FeatureAwareDatabaseAdapter {
|
|
|
498
661
|
digest_json: JSON.parse(row.digest_json || "{}")
|
|
499
662
|
}));
|
|
500
663
|
}
|
|
501
|
-
async searchByVector(
|
|
502
|
-
|
|
503
|
-
|
|
664
|
+
async searchByVector(embedding, options) {
|
|
665
|
+
if (!this.db)
|
|
666
|
+
throw new DatabaseError(
|
|
667
|
+
"Database not connected",
|
|
668
|
+
ErrorCode.DB_CONNECTION_FAILED
|
|
669
|
+
);
|
|
670
|
+
if (!this.vecEnabled) {
|
|
671
|
+
logger.warn("Vector search not available (sqlite-vec not loaded)");
|
|
672
|
+
return [];
|
|
673
|
+
}
|
|
674
|
+
const limit = options?.limit || 20;
|
|
675
|
+
const sql = `
|
|
676
|
+
SELECT f.*, ve.distance as similarity
|
|
677
|
+
FROM frame_embeddings ve
|
|
678
|
+
JOIN frames f ON f.frame_id = ve.frame_id
|
|
679
|
+
WHERE ve.embedding MATCH ?
|
|
680
|
+
ORDER BY ve.distance
|
|
681
|
+
LIMIT ?
|
|
682
|
+
`;
|
|
683
|
+
const rows = this.db.prepare(sql).all(JSON.stringify(embedding), limit);
|
|
684
|
+
return rows.map((row) => ({
|
|
685
|
+
...row,
|
|
686
|
+
inputs: JSON.parse(row.inputs || "{}"),
|
|
687
|
+
outputs: JSON.parse(row.outputs || "{}"),
|
|
688
|
+
digest_json: JSON.parse(row.digest_json || "{}")
|
|
689
|
+
}));
|
|
504
690
|
}
|
|
505
|
-
async searchHybrid(textQuery,
|
|
506
|
-
|
|
691
|
+
async searchHybrid(textQuery, embedding, weights) {
|
|
692
|
+
const textWeight = weights?.text ?? 0.6;
|
|
693
|
+
const vecWeight = weights?.vector ?? 0.4;
|
|
694
|
+
const textResults = await this.search({ query: textQuery, limit: 50 });
|
|
695
|
+
const vecResults = this.vecEnabled ? await this.searchByVector(embedding, { limit: 50 }) : [];
|
|
696
|
+
if (vecResults.length === 0) {
|
|
697
|
+
return textResults;
|
|
698
|
+
}
|
|
699
|
+
const scoreMap = /* @__PURE__ */ new Map();
|
|
700
|
+
const maxText = Math.max(...textResults.map((r) => r.score), 1);
|
|
701
|
+
for (const r of textResults) {
|
|
702
|
+
const normalizedScore = r.score / maxText * textWeight;
|
|
703
|
+
scoreMap.set(r.frame_id, { frame: r, score: normalizedScore });
|
|
704
|
+
}
|
|
705
|
+
const maxDist = Math.max(...vecResults.map((r) => r.similarity), 1);
|
|
706
|
+
for (const r of vecResults) {
|
|
707
|
+
const normalizedScore = (1 - r.similarity / maxDist) * vecWeight;
|
|
708
|
+
const existing = scoreMap.get(r.frame_id);
|
|
709
|
+
if (existing) {
|
|
710
|
+
existing.score += normalizedScore;
|
|
711
|
+
} else {
|
|
712
|
+
scoreMap.set(r.frame_id, { frame: r, score: normalizedScore });
|
|
713
|
+
}
|
|
714
|
+
}
|
|
715
|
+
return Array.from(scoreMap.values()).sort((a, b) => b.score - a.score).map(({ frame, score }) => ({ ...frame, score }));
|
|
716
|
+
}
|
|
717
|
+
/**
|
|
718
|
+
* Store an embedding for a frame
|
|
719
|
+
*/
|
|
720
|
+
async storeEmbedding(frameId, embedding) {
|
|
721
|
+
if (!this.db)
|
|
722
|
+
throw new DatabaseError(
|
|
723
|
+
"Database not connected",
|
|
724
|
+
ErrorCode.DB_CONNECTION_FAILED
|
|
725
|
+
);
|
|
726
|
+
if (!this.vecEnabled) return;
|
|
727
|
+
this.db.prepare(
|
|
728
|
+
"INSERT OR REPLACE INTO frame_embeddings (frame_id, embedding) VALUES (?, ?)"
|
|
729
|
+
).run(frameId, JSON.stringify(embedding));
|
|
730
|
+
}
|
|
731
|
+
/**
|
|
732
|
+
* Get frames that are missing embeddings
|
|
733
|
+
*/
|
|
734
|
+
async getFramesMissingEmbeddings(limit = 50) {
|
|
735
|
+
if (!this.db)
|
|
736
|
+
throw new DatabaseError(
|
|
737
|
+
"Database not connected",
|
|
738
|
+
ErrorCode.DB_CONNECTION_FAILED
|
|
739
|
+
);
|
|
740
|
+
const sql = `
|
|
741
|
+
SELECT f.* FROM frames f
|
|
742
|
+
LEFT JOIN frame_embeddings ve ON f.frame_id = ve.frame_id
|
|
743
|
+
WHERE ve.frame_id IS NULL
|
|
744
|
+
LIMIT ?
|
|
745
|
+
`;
|
|
746
|
+
const rows = this.db.prepare(sql).all(limit);
|
|
747
|
+
return rows.map((row) => ({
|
|
748
|
+
...row,
|
|
749
|
+
inputs: JSON.parse(row.inputs || "{}"),
|
|
750
|
+
outputs: JSON.parse(row.outputs || "{}"),
|
|
751
|
+
digest_json: JSON.parse(row.digest_json || "{}")
|
|
752
|
+
}));
|
|
507
753
|
}
|
|
508
754
|
// Basic aggregation
|
|
509
755
|
async aggregate(table, options) {
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/core/database/sqlite-adapter.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * SQLite Database Adapter\n * Maintains backward compatibility with existing SQLite implementation\n */\n\nimport Database from 'better-sqlite3';\nimport {\n FeatureAwareDatabaseAdapter,\n DatabaseFeatures,\n SearchOptions,\n QueryOptions,\n AggregationOptions,\n BulkOperation,\n DatabaseStats,\n CountResult,\n VersionResult,\n FrameRow,\n} from './database-adapter.js';\nimport type { Frame, Event, Anchor } from '../context/index.js';\nimport { logger } from '../monitoring/logger.js';\nimport { DatabaseError, ErrorCode, ValidationError } from '../errors/index.js';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport interface SQLiteConfig {\n dbPath: string;\n walMode?: boolean;\n busyTimeout?: number;\n cacheSize?: number;\n synchronous?: 'OFF' | 'NORMAL' | 'FULL' | 'EXTRA';\n}\n\nexport class SQLiteAdapter extends FeatureAwareDatabaseAdapter {\n private db: Database.Database | null = null;\n private readonly dbPath: string;\n private inTransactionFlag = false;\n\n constructor(projectId: string, config: SQLiteConfig) {\n super(projectId, config);\n this.dbPath = config.dbPath;\n }\n\n getFeatures(): DatabaseFeatures {\n return {\n supportsFullTextSearch: false, // Could enable with FTS5\n supportsVectorSearch: false,\n supportsPartitioning: false,\n supportsAnalytics: false,\n supportsCompression: false,\n supportsMaterializedViews: false,\n supportsParallelQueries: false,\n };\n }\n\n async connect(): Promise<void> {\n if (this.db) return;\n\n const config = this.config as SQLiteConfig;\n\n // Ensure directory exists\n const dir = path.dirname(this.dbPath);\n await fs.mkdir(dir, { recursive: true });\n\n this.db = new Database(this.dbPath);\n\n // Enforce referential integrity\n this.db.pragma('foreign_keys = ON');\n\n // Configure SQLite for better performance\n if (config.walMode !== false) {\n this.db.pragma('journal_mode = WAL');\n }\n\n if (config.busyTimeout) {\n this.db.pragma(`busy_timeout = ${config.busyTimeout}`);\n }\n\n if (config.cacheSize) {\n this.db.pragma(`cache_size = ${config.cacheSize}`);\n }\n\n if (config.synchronous) {\n this.db.pragma(`synchronous = ${config.synchronous}`);\n }\n\n logger.info('SQLite database connected', { dbPath: this.dbPath });\n }\n\n async disconnect(): Promise<void> {\n if (!this.db) return;\n\n this.db.close();\n this.db = null;\n logger.info('SQLite database disconnected');\n }\n\n /**\n * Get raw database handle for testing purposes\n * @internal\n */\n getRawDatabase(): Database.Database | null {\n return this.db;\n }\n\n isConnected(): boolean {\n return this.db !== null && this.db.open;\n }\n\n async ping(): Promise<boolean> {\n if (!this.db) return false;\n\n try {\n this.db.prepare('SELECT 1').get();\n return true;\n } catch (error: unknown) {\n // Database may be closed or corrupted\n logger.debug('Database ping failed', {\n error: error instanceof Error ? error.message : String(error),\n });\n return false;\n }\n }\n\n async initializeSchema(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS frames (\n frame_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id TEXT REFERENCES frames(frame_id),\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT DEFAULT 'active',\n inputs TEXT DEFAULT '{}',\n outputs TEXT DEFAULT '{}',\n digest_text TEXT,\n digest_json TEXT DEFAULT '{}',\n created_at INTEGER DEFAULT (unixepoch()),\n closed_at INTEGER\n );\n\n CREATE TABLE IF NOT EXISTS events (\n event_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n frame_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL,\n ts INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n created_at INTEGER DEFAULT (unixepoch()),\n metadata TEXT DEFAULT '{}',\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n\n CREATE TABLE IF NOT EXISTS schema_version (\n version INTEGER PRIMARY KEY,\n applied_at INTEGER DEFAULT (unixepoch())\n );\n\n -- Indexes for performance\n CREATE INDEX IF NOT EXISTS idx_frames_run ON frames(run_id);\n CREATE INDEX IF NOT EXISTS idx_frames_project ON frames(project_id);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_frames_state ON frames(state);\n CREATE INDEX IF NOT EXISTS idx_frames_created ON frames(created_at DESC);\n CREATE INDEX IF NOT EXISTS idx_events_frame ON events(frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_seq ON events(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors(frame_id);\n\n -- Set initial schema version if not exists\n INSERT OR IGNORE INTO schema_version (version) VALUES (1);\n `);\n\n // Ensure cascade constraints exist on dependent tables for existing DBs\n try {\n this.ensureCascadeConstraints();\n } catch (e) {\n logger.warn('Failed to ensure cascade constraints', e as Error);\n }\n }\n\n /**\n * Ensure ON DELETE CASCADE exists for events/anchors referencing frames\n * Migrates existing tables in-place if needed without data loss.\n */\n private ensureCascadeConstraints(): void {\n if (!this.db) return;\n\n const needsCascade = (table: string): boolean => {\n const rows = this.db!.prepare(\n `PRAGMA foreign_key_list(${table})`\n ).all() as any[];\n // If any FK points to frames without cascade, we need migration\n return rows.some(\n (r) =>\n r.table === 'frames' &&\n String(r.on_delete).toUpperCase() !== 'CASCADE'\n );\n };\n\n const migrateTable = (table: 'events' | 'anchors') => {\n const createSql =\n table === 'events'\n ? `CREATE TABLE events_new (\n event_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n frame_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL,\n ts INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );`\n : `CREATE TABLE anchors_new (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n created_at INTEGER DEFAULT (unixepoch()),\n metadata TEXT DEFAULT '{}',\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );`;\n\n const cols =\n table === 'events'\n ? 'event_id, run_id, frame_id, seq, event_type, payload, ts'\n : 'anchor_id, frame_id, project_id, type, text, priority, created_at, metadata';\n\n const idxSql =\n table === 'events'\n ? [\n 'CREATE INDEX IF NOT EXISTS idx_events_frame ON events(frame_id);',\n 'CREATE INDEX IF NOT EXISTS idx_events_seq ON events(frame_id, seq);',\n ]\n : [\n 'CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors(frame_id);',\n ];\n\n this.db!.exec('PRAGMA foreign_keys = OFF;');\n this.db!.exec('BEGIN;');\n this.db!.exec(createSql);\n this.db!.prepare(\n `INSERT INTO ${table === 'events' ? 'events_new' : 'anchors_new'} (${cols}) SELECT ${cols} FROM ${table}`\n ).run();\n this.db!.exec(`DROP TABLE ${table};`);\n this.db!.exec(`ALTER TABLE ${table}_new RENAME TO ${table};`);\n for (const stmt of idxSql) this.db!.exec(stmt);\n this.db!.exec('COMMIT;');\n this.db!.exec('PRAGMA foreign_keys = ON;');\n logger.info(`Migrated ${table} to include ON DELETE CASCADE`);\n };\n\n if (needsCascade('events')) migrateTable('events');\n if (needsCascade('anchors')) migrateTable('anchors');\n }\n\n async migrateSchema(targetVersion: number): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const currentVersion = await this.getSchemaVersion();\n\n if (currentVersion >= targetVersion) {\n logger.info('Schema already at target version', {\n currentVersion,\n targetVersion,\n });\n return;\n }\n\n // Apply migrations sequentially\n for (let v = currentVersion + 1; v <= targetVersion; v++) {\n logger.info(`Applying migration to version ${v}`);\n // Migration logic would go here\n this.db.prepare('UPDATE schema_version SET version = ?').run(v);\n }\n }\n\n async getSchemaVersion(): Promise<number> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n try {\n const result = this.db\n .prepare('SELECT MAX(version) as version FROM schema_version')\n .get() as VersionResult;\n return result?.version || 0;\n } catch (error: unknown) {\n // Table may not exist yet in a fresh database\n logger.debug('Schema version table not found, returning 0', {\n error: error instanceof Error ? error.message : String(error),\n });\n return 0;\n }\n }\n\n // Frame operations\n async createFrame(frame: Partial<Frame>): Promise<string> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const frameId = frame.frame_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO frames (\n frame_id, run_id, project_id, parent_frame_id, depth,\n type, name, state, inputs, outputs, digest_text, digest_json\n ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n frameId,\n frame.run_id,\n frame.project_id || this.projectId,\n frame.parent_frame_id || null,\n frame.depth || 0,\n frame.type,\n frame.name,\n frame.state || 'active',\n JSON.stringify(frame.inputs || {}),\n JSON.stringify(frame.outputs || {}),\n frame.digest_text || null,\n JSON.stringify(frame.digest_json || {})\n );\n\n return frameId;\n }\n\n async getFrame(frameId: string): Promise<Frame | null> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const row = this.db\n .prepare('SELECT * FROM frames WHERE frame_id = ?')\n .get(frameId) as FrameRow | undefined;\n\n if (!row) return null;\n\n return {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n };\n }\n\n async updateFrame(frameId: string, updates: Partial<Frame>): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const fields = [];\n const values = [];\n\n if (updates.state !== undefined) {\n fields.push('state = ?');\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n fields.push('outputs = ?');\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.digest_text !== undefined) {\n fields.push('digest_text = ?');\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n fields.push('digest_json = ?');\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n fields.push('closed_at = ?');\n values.push(updates.closed_at);\n }\n\n if (fields.length === 0) return;\n\n values.push(frameId);\n\n this.db\n .prepare(\n `\n UPDATE frames SET ${fields.join(', ')} WHERE frame_id = ?\n `\n )\n .run(...values);\n }\n\n async deleteFrame(frameId: string): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n // Delete in order due to foreign keys\n await this.deleteFrameAnchors(frameId);\n await this.deleteFrameEvents(frameId);\n\n this.db.prepare('DELETE FROM frames WHERE frame_id = ?').run(frameId);\n }\n\n async getActiveFrames(runId?: string): Promise<Frame[]> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n let query = \"SELECT * FROM frames WHERE state = 'active'\";\n const params = [];\n\n if (runId) {\n query += ' AND run_id = ?';\n params.push(runId);\n }\n\n query += ' ORDER BY depth ASC, created_at ASC';\n\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async closeFrame(frameId: string, outputs?: any): Promise<void> {\n await this.updateFrame(frameId, {\n state: 'closed',\n outputs,\n closed_at: Date.now(),\n });\n }\n\n // Event operations\n async createEvent(event: Partial<Event>): Promise<string> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const eventId = event.event_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO events (event_id, run_id, frame_id, seq, event_type, payload, ts)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n eventId,\n event.run_id,\n event.frame_id,\n event.seq || 0,\n event.event_type,\n JSON.stringify(event.payload || {}),\n event.ts || Date.now()\n );\n\n return eventId;\n }\n\n async getFrameEvents(\n frameId: string,\n options?: QueryOptions\n ): Promise<Event[]> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n let query = 'SELECT * FROM events WHERE frame_id = ?';\n query += this.buildOrderByClause(\n options?.orderBy || 'seq',\n options?.orderDirection\n );\n query += this.buildLimitClause(options?.limit, options?.offset);\n\n const rows = this.db.prepare(query).all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n payload: JSON.parse(row.payload || '{}'),\n }));\n }\n\n async deleteFrameEvents(frameId: string): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.prepare('DELETE FROM events WHERE frame_id = ?').run(frameId);\n }\n\n // Anchor operations\n async createAnchor(anchor: Partial<Anchor>): Promise<string> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const anchorId = anchor.anchor_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO anchors (anchor_id, frame_id, project_id, type, text, priority, metadata)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n anchorId,\n anchor.frame_id,\n anchor.project_id || this.projectId,\n anchor.type,\n anchor.text,\n anchor.priority || 0,\n JSON.stringify(anchor.metadata || {})\n );\n\n return anchorId;\n }\n\n async getFrameAnchors(frameId: string): Promise<Anchor[]> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const rows = this.db\n .prepare(\n `\n SELECT * FROM anchors WHERE frame_id = ? \n ORDER BY priority DESC, created_at ASC\n `\n )\n .all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n }\n\n async deleteFrameAnchors(frameId: string): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.prepare('DELETE FROM anchors WHERE frame_id = ?').run(frameId);\n }\n\n // Limited search (basic LIKE queries)\n async search(\n options: SearchOptions\n ): Promise<Array<Frame & { score: number }>> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n // SQLite doesn't support HAVING on non-aggregate queries, so we filter in application\n const sql = `\n SELECT *, \n CASE \n WHEN name LIKE ? THEN 1.0\n WHEN digest_text LIKE ? THEN 0.8\n WHEN inputs LIKE ? THEN 0.6\n ELSE 0.5\n END as score\n FROM frames\n WHERE name LIKE ? OR digest_text LIKE ? OR inputs LIKE ?\n ORDER BY score DESC\n `;\n\n const params = Array(6).fill(`%${options.query}%`);\n\n let rows = this.db.prepare(sql).all(...params) as any[];\n\n // Apply score threshold in application layer\n if (options.scoreThreshold) {\n rows = rows.filter((row) => row.score >= options.scoreThreshold);\n }\n\n // Apply limit and offset in application layer if threshold is used\n if (options.limit || options.offset) {\n const start = options.offset || 0;\n const end = options.limit ? start + options.limit : rows.length;\n rows = rows.slice(start, end);\n }\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async searchByVector(\n _embedding: number[],\n _options?: QueryOptions\n ): Promise<Array<Frame & { similarity: number }>> {\n // Not supported in SQLite\n logger.warn('Vector search not supported in SQLite adapter');\n return [];\n }\n\n async searchHybrid(\n textQuery: string,\n _embedding: number[],\n weights?: { text: number; vector: number }\n ): Promise<Array<Frame & { score: number }>> {\n // Fall back to text search only\n return this.search({ query: textQuery, ...weights });\n }\n\n // Basic aggregation\n async aggregate(\n table: string,\n options: AggregationOptions\n ): Promise<Record<string, any>[]> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const metrics = options.metrics\n .map(\n (m) =>\n `${m.operation}(${m.field}) AS ${m.alias || `${m.operation}_${m.field}`}`\n )\n .join(', ');\n\n let sql = `SELECT ${options.groupBy.join(', ')}, ${metrics} FROM ${table}`;\n sql += ` GROUP BY ${options.groupBy.join(', ')}`;\n\n if (options.having) {\n const havingClauses = Object.entries(options.having).map(\n ([key, value]) =>\n `${key} ${typeof value === 'object' ? value.op : '='} ?`\n );\n sql += ` HAVING ${havingClauses.join(' AND ')}`;\n }\n\n return this.db\n .prepare(sql)\n .all(...Object.values(options.having || {})) as any[];\n }\n\n // Pattern detection (basic)\n async detectPatterns(timeRange?: { start: Date; end: Date }): Promise<\n Array<{\n pattern: string;\n type: string;\n frequency: number;\n lastSeen: Date;\n }>\n > {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n let sql = `\n SELECT type as pattern, type, COUNT(*) as frequency, MAX(created_at) as last_seen\n FROM frames\n `;\n\n const params = [];\n if (timeRange) {\n sql += ' WHERE created_at >= ? AND created_at <= ?';\n params.push(\n Math.floor(timeRange.start.getTime() / 1000),\n Math.floor(timeRange.end.getTime() / 1000)\n );\n }\n\n sql += ' GROUP BY type HAVING COUNT(*) > 1 ORDER BY frequency DESC';\n\n const rows = this.db.prepare(sql).all(...params) as any[];\n\n return rows.map((row) => ({\n pattern: row.pattern,\n type: row.type,\n frequency: row.frequency,\n lastSeen: new Date(row.last_seen * 1000),\n }));\n }\n\n // Bulk operations\n async executeBulk(operations: BulkOperation[]): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n await this.inTransaction(async () => {\n for (const op of operations) {\n switch (op.type) {\n case 'insert':\n // Build insert dynamically based on data\n const insertCols = Object.keys(op.data);\n const insertPlaceholders = insertCols.map(() => '?').join(',');\n this.db!.prepare(\n `INSERT INTO ${op.table} (${insertCols.join(',')}) VALUES (${insertPlaceholders})`\n ).run(...Object.values(op.data));\n break;\n\n case 'update':\n const updateSets = Object.keys(op.data)\n .map((k) => `${k} = ?`)\n .join(',');\n const whereClause = this.buildWhereClause(op.where || {});\n this.db!.prepare(\n `UPDATE ${op.table} SET ${updateSets} ${whereClause}`\n ).run(...Object.values(op.data), ...Object.values(op.where || {}));\n break;\n\n case 'delete':\n const deleteWhere = this.buildWhereClause(op.where || {});\n this.db!.prepare(`DELETE FROM ${op.table} ${deleteWhere}`).run(\n ...Object.values(op.where || {})\n );\n break;\n }\n }\n });\n }\n\n async vacuum(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.pragma('vacuum');\n logger.info('SQLite database vacuumed');\n }\n\n async analyze(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.pragma('analyze');\n logger.info('SQLite database analyzed');\n }\n\n // Statistics\n async getStats(): Promise<DatabaseStats> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const stats = {\n totalFrames: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM frames')\n .get() as CountResult\n ).count,\n activeFrames: (\n this.db\n .prepare(\n \"SELECT COUNT(*) as count FROM frames WHERE state = 'active'\"\n )\n .get() as CountResult\n ).count,\n totalEvents: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM events')\n .get() as CountResult\n ).count,\n totalAnchors: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM anchors')\n .get() as CountResult\n ).count,\n diskUsage: 0,\n };\n\n // Get file size\n try {\n const fileStats = await fs.stat(this.dbPath);\n stats.diskUsage = fileStats.size;\n } catch (error: unknown) {\n // File may not exist yet or be inaccessible - disk usage remains 0\n logger.debug('Failed to get database file size', {\n dbPath: this.dbPath,\n error: error instanceof Error ? error.message : String(error),\n });\n }\n\n return stats;\n }\n\n async getQueryStats(): Promise<\n Array<{\n query: string;\n calls: number;\n meanTime: number;\n totalTime: number;\n }>\n > {\n // SQLite doesn't have built-in query stats\n logger.warn('Query stats not available for SQLite');\n return [];\n }\n\n // Transaction support\n async beginTransaction(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.prepare('BEGIN').run();\n this.inTransactionFlag = true;\n }\n\n async commitTransaction(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.prepare('COMMIT').run();\n this.inTransactionFlag = false;\n }\n\n async rollbackTransaction(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.prepare('ROLLBACK').run();\n this.inTransactionFlag = false;\n }\n\n async inTransaction(\n callback: (adapter: DatabaseAdapter) => Promise<void>\n ): Promise<void> {\n await this.beginTransaction();\n\n try {\n await callback(this);\n await this.commitTransaction();\n } catch (error: unknown) {\n await this.rollbackTransaction();\n throw error;\n }\n }\n\n // Export/Import\n async exportData(\n tables: string[],\n format: 'json' | 'parquet' | 'csv'\n ): Promise<Buffer> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n if (format !== 'json') {\n throw new ValidationError(\n `Format ${format} not supported for SQLite export`,\n ErrorCode.VALIDATION_FAILED,\n { format, supportedFormats: ['json'] }\n );\n }\n\n const data: Record<string, any[]> = {};\n\n for (const table of tables) {\n data[table] = this.db.prepare(`SELECT * FROM ${table}`).all();\n }\n\n return Buffer.from(JSON.stringify(data, null, 2));\n }\n\n async importData(\n data: Buffer,\n format: 'json' | 'parquet' | 'csv',\n options?: { truncate?: boolean; upsert?: boolean }\n ): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n if (format !== 'json') {\n throw new ValidationError(\n `Format ${format} not supported for SQLite import`,\n ErrorCode.VALIDATION_FAILED,\n { format, supportedFormats: ['json'] }\n );\n }\n\n const parsed = JSON.parse(data.toString());\n\n await this.inTransaction(async () => {\n for (const [table, rows] of Object.entries(parsed)) {\n if (options?.truncate) {\n this.db!.prepare(`DELETE FROM ${table}`).run();\n }\n\n for (const row of rows as any[]) {\n const cols = Object.keys(row);\n const placeholders = cols.map(() => '?').join(',');\n\n if (options?.upsert) {\n const updates = cols.map((c) => `${c} = excluded.${c}`).join(',');\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})\n ON CONFLICT DO UPDATE SET ${updates}`\n ).run(...Object.values(row));\n } else {\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})`\n ).run(...Object.values(row));\n }\n }\n }\n });\n }\n}\n"],
|
|
5
|
-
"mappings": "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",
|
|
4
|
+
"sourcesContent": ["/**\n * SQLite Database Adapter\n * Maintains backward compatibility with existing SQLite implementation\n */\n\nimport Database from 'better-sqlite3';\nimport {\n FeatureAwareDatabaseAdapter,\n DatabaseFeatures,\n SearchOptions,\n QueryOptions,\n AggregationOptions,\n BulkOperation,\n DatabaseStats,\n CountResult,\n VersionResult,\n FrameRow,\n} from './database-adapter.js';\nimport type { Frame, Event, Anchor } from '../context/index.js';\nimport { logger } from '../monitoring/logger.js';\nimport { DatabaseError, ErrorCode, ValidationError } from '../errors/index.js';\nimport type { EmbeddingProvider } from './embedding-provider.js';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport interface SQLiteConfig {\n dbPath: string;\n walMode?: boolean;\n busyTimeout?: number;\n cacheSize?: number;\n synchronous?: 'OFF' | 'NORMAL' | 'FULL' | 'EXTRA';\n embeddingProvider?: EmbeddingProvider;\n embeddingDimension?: number;\n}\n\nexport class SQLiteAdapter extends FeatureAwareDatabaseAdapter {\n private db: Database.Database | null = null;\n private readonly dbPath: string;\n private inTransactionFlag = false;\n private ftsEnabled = false;\n private vecEnabled = false;\n private embeddingProvider?: EmbeddingProvider;\n private embeddingDimension: number;\n\n constructor(projectId: string, config: SQLiteConfig) {\n super(projectId, config);\n this.dbPath = config.dbPath;\n this.embeddingProvider = config.embeddingProvider;\n this.embeddingDimension = config.embeddingDimension || 384;\n }\n\n getFeatures(): DatabaseFeatures {\n return {\n supportsFullTextSearch: this.ftsEnabled,\n supportsVectorSearch: this.vecEnabled,\n supportsPartitioning: false,\n supportsAnalytics: false,\n supportsCompression: false,\n supportsMaterializedViews: false,\n supportsParallelQueries: false,\n };\n }\n\n async connect(): Promise<void> {\n if (this.db) return;\n\n const config = this.config as SQLiteConfig;\n\n // Ensure directory exists\n const dir = path.dirname(this.dbPath);\n await fs.mkdir(dir, { recursive: true });\n\n this.db = new Database(this.dbPath);\n\n // Enforce referential integrity\n this.db.pragma('foreign_keys = ON');\n\n // Configure SQLite for better performance\n if (config.walMode !== false) {\n this.db.pragma('journal_mode = WAL');\n }\n\n if (config.busyTimeout) {\n this.db.pragma(`busy_timeout = ${config.busyTimeout}`);\n }\n\n if (config.cacheSize) {\n this.db.pragma(`cache_size = ${config.cacheSize}`);\n }\n\n if (config.synchronous) {\n this.db.pragma(`synchronous = ${config.synchronous}`);\n }\n\n logger.info('SQLite database connected', { dbPath: this.dbPath });\n }\n\n async disconnect(): Promise<void> {\n if (!this.db) return;\n\n this.db.close();\n this.db = null;\n logger.info('SQLite database disconnected');\n }\n\n /**\n * Get raw database handle for testing purposes\n * @internal\n */\n getRawDatabase(): Database.Database | null {\n return this.db;\n }\n\n isConnected(): boolean {\n return this.db !== null && this.db.open;\n }\n\n async ping(): Promise<boolean> {\n if (!this.db) return false;\n\n try {\n this.db.prepare('SELECT 1').get();\n return true;\n } catch (error: unknown) {\n // Database may be closed or corrupted\n logger.debug('Database ping failed', {\n error: error instanceof Error ? error.message : String(error),\n });\n return false;\n }\n }\n\n async initializeSchema(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS frames (\n frame_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id TEXT REFERENCES frames(frame_id),\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT DEFAULT 'active',\n inputs TEXT DEFAULT '{}',\n outputs TEXT DEFAULT '{}',\n digest_text TEXT,\n digest_json TEXT DEFAULT '{}',\n created_at INTEGER DEFAULT (unixepoch()),\n closed_at INTEGER\n );\n\n CREATE TABLE IF NOT EXISTS events (\n event_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n frame_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL,\n ts INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n created_at INTEGER DEFAULT (unixepoch()),\n metadata TEXT DEFAULT '{}',\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n\n CREATE TABLE IF NOT EXISTS schema_version (\n version INTEGER PRIMARY KEY,\n applied_at INTEGER DEFAULT (unixepoch())\n );\n\n -- Indexes for performance\n CREATE INDEX IF NOT EXISTS idx_frames_run ON frames(run_id);\n CREATE INDEX IF NOT EXISTS idx_frames_project ON frames(project_id);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_frames_state ON frames(state);\n CREATE INDEX IF NOT EXISTS idx_frames_created ON frames(created_at DESC);\n CREATE INDEX IF NOT EXISTS idx_events_frame ON events(frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_seq ON events(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors(frame_id);\n\n -- Set initial schema version if not exists\n INSERT OR IGNORE INTO schema_version (version) VALUES (1);\n `);\n\n // Ensure cascade constraints exist on dependent tables for existing DBs\n try {\n this.ensureCascadeConstraints();\n } catch (e) {\n logger.warn('Failed to ensure cascade constraints', e as Error);\n }\n\n // Initialize FTS5 full-text search\n this.initializeFts();\n\n // Initialize sqlite-vec if provider is configured\n this.initializeVec();\n }\n\n /**\n * Ensure ON DELETE CASCADE exists for events/anchors referencing frames\n * Migrates existing tables in-place if needed without data loss.\n */\n private ensureCascadeConstraints(): void {\n if (!this.db) return;\n\n const needsCascade = (table: string): boolean => {\n const rows = this.db!.prepare(\n `PRAGMA foreign_key_list(${table})`\n ).all() as any[];\n // If any FK points to frames without cascade, we need migration\n return rows.some(\n (r) =>\n r.table === 'frames' &&\n String(r.on_delete).toUpperCase() !== 'CASCADE'\n );\n };\n\n const migrateTable = (table: 'events' | 'anchors') => {\n const createSql =\n table === 'events'\n ? `CREATE TABLE events_new (\n event_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n frame_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL,\n ts INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );`\n : `CREATE TABLE anchors_new (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n created_at INTEGER DEFAULT (unixepoch()),\n metadata TEXT DEFAULT '{}',\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );`;\n\n const cols =\n table === 'events'\n ? 'event_id, run_id, frame_id, seq, event_type, payload, ts'\n : 'anchor_id, frame_id, project_id, type, text, priority, created_at, metadata';\n\n const idxSql =\n table === 'events'\n ? [\n 'CREATE INDEX IF NOT EXISTS idx_events_frame ON events(frame_id);',\n 'CREATE INDEX IF NOT EXISTS idx_events_seq ON events(frame_id, seq);',\n ]\n : [\n 'CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors(frame_id);',\n ];\n\n this.db!.exec('PRAGMA foreign_keys = OFF;');\n this.db!.exec('BEGIN;');\n this.db!.exec(createSql);\n this.db!.prepare(\n `INSERT INTO ${table === 'events' ? 'events_new' : 'anchors_new'} (${cols}) SELECT ${cols} FROM ${table}`\n ).run();\n this.db!.exec(`DROP TABLE ${table};`);\n this.db!.exec(`ALTER TABLE ${table}_new RENAME TO ${table};`);\n for (const stmt of idxSql) this.db!.exec(stmt);\n this.db!.exec('COMMIT;');\n this.db!.exec('PRAGMA foreign_keys = ON;');\n logger.info(`Migrated ${table} to include ON DELETE CASCADE`);\n };\n\n if (needsCascade('events')) migrateTable('events');\n if (needsCascade('anchors')) migrateTable('anchors');\n }\n\n /**\n * Initialize FTS5 virtual table and sync triggers\n */\n private initializeFts(): void {\n if (!this.db) return;\n\n try {\n // Create FTS5 virtual table (external content, references frames)\n this.db.exec(`\n CREATE VIRTUAL TABLE IF NOT EXISTS frames_fts USING fts5(\n name, digest_text, inputs, outputs,\n content='frames', content_rowid='rowid'\n );\n `);\n\n // Create triggers to keep FTS in sync\n this.db.exec(`\n CREATE TRIGGER IF NOT EXISTS frames_ai AFTER INSERT ON frames BEGIN\n INSERT INTO frames_fts(rowid, name, digest_text, inputs, outputs)\n VALUES (new.rowid, new.name, new.digest_text, new.inputs, new.outputs);\n END;\n `);\n\n this.db.exec(`\n CREATE TRIGGER IF NOT EXISTS frames_ad AFTER DELETE ON frames BEGIN\n INSERT INTO frames_fts(frames_fts, rowid, name, digest_text, inputs, outputs)\n VALUES ('delete', old.rowid, old.name, old.digest_text, old.inputs, old.outputs);\n END;\n `);\n\n this.db.exec(`\n CREATE TRIGGER IF NOT EXISTS frames_au AFTER UPDATE ON frames BEGIN\n INSERT INTO frames_fts(frames_fts, rowid, name, digest_text, inputs, outputs)\n VALUES ('delete', old.rowid, old.name, old.digest_text, old.inputs, old.outputs);\n INSERT INTO frames_fts(rowid, name, digest_text, inputs, outputs)\n VALUES (new.rowid, new.name, new.digest_text, new.inputs, new.outputs);\n END;\n `);\n\n // Populate FTS index for existing data (schema version migration 1\u21922)\n this.migrateToFts();\n\n this.ftsEnabled = true;\n logger.info('FTS5 full-text search initialized');\n } catch (e) {\n logger.warn(\n 'FTS5 initialization failed, falling back to LIKE search',\n e as Error\n );\n this.ftsEnabled = false;\n }\n }\n\n /**\n * One-time migration: populate FTS index from existing frames data\n */\n private migrateToFts(): void {\n if (!this.db) return;\n\n const version =\n (\n this.db\n .prepare('SELECT MAX(version) as version FROM schema_version')\n .get() as { version: number }\n )?.version || 1;\n\n if (version < 2) {\n // Populate FTS from existing data\n this.db.exec(`\n INSERT OR IGNORE INTO frames_fts(rowid, name, digest_text, inputs, outputs)\n SELECT rowid, name, digest_text, inputs, outputs FROM frames;\n `);\n this.db\n .prepare('INSERT OR REPLACE INTO schema_version (version) VALUES (?)')\n .run(2);\n logger.info(\n 'FTS5 index populated from existing frames (migration v1\u2192v2)'\n );\n }\n }\n\n /**\n * Initialize sqlite-vec for vector search\n */\n private initializeVec(): void {\n if (!this.db || !this.embeddingProvider) return;\n\n try {\n // Try to load sqlite-vec extension\n let sqliteVec;\n try {\n sqliteVec = require('sqlite-vec');\n } catch {\n logger.info('sqlite-vec not installed, vector search disabled');\n return;\n }\n\n sqliteVec.load(this.db);\n\n // Create vec0 virtual table for embeddings\n this.db.exec(`\n CREATE VIRTUAL TABLE IF NOT EXISTS frame_embeddings USING vec0(\n frame_id TEXT PRIMARY KEY,\n embedding float[${this.embeddingDimension}]\n );\n `);\n\n this.vecEnabled = true;\n logger.info('sqlite-vec vector search initialized', {\n dimension: this.embeddingDimension,\n });\n } catch (e) {\n logger.warn(\n 'sqlite-vec initialization failed, vector search disabled',\n e as Error\n );\n this.vecEnabled = false;\n }\n }\n\n /**\n * Rebuild the FTS5 index (for maintenance)\n */\n async rebuildFtsIndex(): Promise<void> {\n if (!this.db) {\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n }\n if (!this.ftsEnabled) {\n logger.warn('FTS not enabled, skipping rebuild');\n return;\n }\n this.db.exec(\"INSERT INTO frames_fts(frames_fts) VALUES('rebuild')\");\n logger.info('FTS5 index rebuilt');\n }\n\n async migrateSchema(targetVersion: number): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const currentVersion = await this.getSchemaVersion();\n\n if (currentVersion >= targetVersion) {\n logger.info('Schema already at target version', {\n currentVersion,\n targetVersion,\n });\n return;\n }\n\n // Apply migrations sequentially\n for (let v = currentVersion + 1; v <= targetVersion; v++) {\n logger.info(`Applying migration to version ${v}`);\n // Migration logic would go here\n this.db.prepare('UPDATE schema_version SET version = ?').run(v);\n }\n }\n\n async getSchemaVersion(): Promise<number> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n try {\n const result = this.db\n .prepare('SELECT MAX(version) as version FROM schema_version')\n .get() as VersionResult;\n return result?.version || 0;\n } catch (error: unknown) {\n // Table may not exist yet in a fresh database\n logger.debug('Schema version table not found, returning 0', {\n error: error instanceof Error ? error.message : String(error),\n });\n return 0;\n }\n }\n\n // Frame operations\n async createFrame(frame: Partial<Frame>): Promise<string> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const frameId = frame.frame_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO frames (\n frame_id, run_id, project_id, parent_frame_id, depth,\n type, name, state, inputs, outputs, digest_text, digest_json\n ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n frameId,\n frame.run_id,\n frame.project_id || this.projectId,\n frame.parent_frame_id || null,\n frame.depth || 0,\n frame.type,\n frame.name,\n frame.state || 'active',\n JSON.stringify(frame.inputs || {}),\n JSON.stringify(frame.outputs || {}),\n frame.digest_text || null,\n JSON.stringify(frame.digest_json || {})\n );\n\n return frameId;\n }\n\n async getFrame(frameId: string): Promise<Frame | null> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const row = this.db\n .prepare('SELECT * FROM frames WHERE frame_id = ?')\n .get(frameId) as FrameRow | undefined;\n\n if (!row) return null;\n\n return {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n };\n }\n\n async updateFrame(frameId: string, updates: Partial<Frame>): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const fields = [];\n const values = [];\n\n if (updates.state !== undefined) {\n fields.push('state = ?');\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n fields.push('outputs = ?');\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.digest_text !== undefined) {\n fields.push('digest_text = ?');\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n fields.push('digest_json = ?');\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n fields.push('closed_at = ?');\n values.push(updates.closed_at);\n }\n\n if (fields.length === 0) return;\n\n values.push(frameId);\n\n this.db\n .prepare(\n `\n UPDATE frames SET ${fields.join(', ')} WHERE frame_id = ?\n `\n )\n .run(...values);\n }\n\n async deleteFrame(frameId: string): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n // Delete in order due to foreign keys\n await this.deleteFrameAnchors(frameId);\n await this.deleteFrameEvents(frameId);\n\n this.db.prepare('DELETE FROM frames WHERE frame_id = ?').run(frameId);\n }\n\n async getActiveFrames(runId?: string): Promise<Frame[]> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n let query = \"SELECT * FROM frames WHERE state = 'active'\";\n const params = [];\n\n if (runId) {\n query += ' AND run_id = ?';\n params.push(runId);\n }\n\n query += ' ORDER BY depth ASC, created_at ASC';\n\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async closeFrame(frameId: string, outputs?: any): Promise<void> {\n await this.updateFrame(frameId, {\n state: 'closed',\n outputs,\n closed_at: Date.now(),\n });\n }\n\n // Event operations\n async createEvent(event: Partial<Event>): Promise<string> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const eventId = event.event_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO events (event_id, run_id, frame_id, seq, event_type, payload, ts)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n eventId,\n event.run_id,\n event.frame_id,\n event.seq || 0,\n event.event_type,\n JSON.stringify(event.payload || {}),\n event.ts || Date.now()\n );\n\n return eventId;\n }\n\n async getFrameEvents(\n frameId: string,\n options?: QueryOptions\n ): Promise<Event[]> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n let query = 'SELECT * FROM events WHERE frame_id = ?';\n query += this.buildOrderByClause(\n options?.orderBy || 'seq',\n options?.orderDirection\n );\n query += this.buildLimitClause(options?.limit, options?.offset);\n\n const rows = this.db.prepare(query).all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n payload: JSON.parse(row.payload || '{}'),\n }));\n }\n\n async deleteFrameEvents(frameId: string): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.prepare('DELETE FROM events WHERE frame_id = ?').run(frameId);\n }\n\n // Anchor operations\n async createAnchor(anchor: Partial<Anchor>): Promise<string> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const anchorId = anchor.anchor_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO anchors (anchor_id, frame_id, project_id, type, text, priority, metadata)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n anchorId,\n anchor.frame_id,\n anchor.project_id || this.projectId,\n anchor.type,\n anchor.text,\n anchor.priority || 0,\n JSON.stringify(anchor.metadata || {})\n );\n\n return anchorId;\n }\n\n async getFrameAnchors(frameId: string): Promise<Anchor[]> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const rows = this.db\n .prepare(\n `\n SELECT * FROM anchors WHERE frame_id = ? \n ORDER BY priority DESC, created_at ASC\n `\n )\n .all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n }\n\n async deleteFrameAnchors(frameId: string): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.prepare('DELETE FROM anchors WHERE frame_id = ?').run(frameId);\n }\n\n // Full-text search with FTS5 + BM25 ranking (fallback to LIKE)\n async search(\n options: SearchOptions\n ): Promise<Array<Frame & { score: number }>> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n if (this.ftsEnabled) {\n try {\n return this.searchFts(options);\n } catch (e) {\n // FTS MATCH can fail on bad syntax \u2014 fall back to LIKE\n logger.debug('FTS search failed, falling back to LIKE', {\n error: e instanceof Error ? e.message : String(e),\n query: options.query,\n });\n }\n }\n\n return this.searchLike(options);\n }\n\n /**\n * FTS5 MATCH search with BM25 ranking\n */\n private searchFts(options: SearchOptions): Array<Frame & { score: number }> {\n // BM25 weights: name=10, digest_text=5, inputs=2, outputs=1\n const boost = options.boost || {};\n const w0 = boost['name'] || 10.0;\n const w1 = boost['digest_text'] || 5.0;\n const w2 = boost['inputs'] || 2.0;\n const w3 = boost['outputs'] || 1.0;\n\n const sql = `\n SELECT f.*, -bm25(frames_fts, ${w0}, ${w1}, ${w2}, ${w3}) as score\n FROM frames_fts fts\n JOIN frames f ON f.rowid = fts.rowid\n WHERE frames_fts MATCH ?\n ORDER BY score DESC\n LIMIT ? OFFSET ?\n `;\n\n const limit = options.limit || 50;\n const offset = options.offset || 0;\n\n const rows = this.db!.prepare(sql).all(\n options.query,\n limit,\n offset\n ) as any[];\n\n // Note: scoreThreshold is not applied to FTS results because BM25 scores\n // are on a different scale than LIKE-based scores. FTS results are already\n // ranked by relevance via ORDER BY score DESC.\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n /**\n * Fallback LIKE search for when FTS is unavailable\n */\n private searchLike(options: SearchOptions): Array<Frame & { score: number }> {\n const sql = `\n SELECT *,\n CASE\n WHEN name LIKE ? THEN 1.0\n WHEN digest_text LIKE ? THEN 0.8\n WHEN inputs LIKE ? THEN 0.6\n ELSE 0.5\n END as score\n FROM frames\n WHERE name LIKE ? OR digest_text LIKE ? OR inputs LIKE ?\n ORDER BY score DESC\n `;\n\n const params = Array(6).fill(`%${options.query}%`);\n\n let rows = this.db!.prepare(sql).all(...params) as any[];\n\n if (options.scoreThreshold) {\n rows = rows.filter((row) => row.score >= options.scoreThreshold);\n }\n\n if (options.limit || options.offset) {\n const start = options.offset || 0;\n const end = options.limit ? start + options.limit : rows.length;\n rows = rows.slice(start, end);\n }\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async searchByVector(\n embedding: number[],\n options?: QueryOptions\n ): Promise<Array<Frame & { similarity: number }>> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n if (!this.vecEnabled) {\n logger.warn('Vector search not available (sqlite-vec not loaded)');\n return [];\n }\n\n const limit = options?.limit || 20;\n const sql = `\n SELECT f.*, ve.distance as similarity\n FROM frame_embeddings ve\n JOIN frames f ON f.frame_id = ve.frame_id\n WHERE ve.embedding MATCH ?\n ORDER BY ve.distance\n LIMIT ?\n `;\n\n const rows = this.db\n .prepare(sql)\n .all(JSON.stringify(embedding), limit) as any[];\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async searchHybrid(\n textQuery: string,\n embedding: number[],\n weights?: { text: number; vector: number }\n ): Promise<Array<Frame & { score: number }>> {\n const textWeight = weights?.text ?? 0.6;\n const vecWeight = weights?.vector ?? 0.4;\n\n // Get text results\n const textResults = await this.search({ query: textQuery, limit: 50 });\n\n // Get vector results if available\n const vecResults = this.vecEnabled\n ? await this.searchByVector(embedding, { limit: 50 })\n : [];\n\n if (vecResults.length === 0) {\n return textResults;\n }\n\n // Merge: build score map by frame_id\n const scoreMap = new Map<string, { frame: Frame; score: number }>();\n\n // Normalize text scores: max becomes 1.0\n const maxText = Math.max(...textResults.map((r) => r.score), 1);\n for (const r of textResults) {\n const normalizedScore = (r.score / maxText) * textWeight;\n scoreMap.set(r.frame_id, { frame: r, score: normalizedScore });\n }\n\n // Normalize vec scores: smaller distance = higher score\n const maxDist = Math.max(...vecResults.map((r) => r.similarity), 1);\n for (const r of vecResults) {\n const normalizedScore = (1 - r.similarity / maxDist) * vecWeight;\n const existing = scoreMap.get(r.frame_id);\n if (existing) {\n existing.score += normalizedScore;\n } else {\n scoreMap.set(r.frame_id, { frame: r, score: normalizedScore });\n }\n }\n\n return Array.from(scoreMap.values())\n .sort((a, b) => b.score - a.score)\n .map(({ frame, score }) => ({ ...frame, score }));\n }\n\n /**\n * Store an embedding for a frame\n */\n async storeEmbedding(frameId: string, embedding: number[]): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n if (!this.vecEnabled) return;\n\n this.db\n .prepare(\n 'INSERT OR REPLACE INTO frame_embeddings (frame_id, embedding) VALUES (?, ?)'\n )\n .run(frameId, JSON.stringify(embedding));\n }\n\n /**\n * Get frames that are missing embeddings\n */\n async getFramesMissingEmbeddings(limit: number = 50): Promise<Frame[]> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const sql = `\n SELECT f.* FROM frames f\n LEFT JOIN frame_embeddings ve ON f.frame_id = ve.frame_id\n WHERE ve.frame_id IS NULL\n LIMIT ?\n `;\n\n const rows = this.db.prepare(sql).all(limit) as any[];\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n // Basic aggregation\n async aggregate(\n table: string,\n options: AggregationOptions\n ): Promise<Record<string, any>[]> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const metrics = options.metrics\n .map(\n (m) =>\n `${m.operation}(${m.field}) AS ${m.alias || `${m.operation}_${m.field}`}`\n )\n .join(', ');\n\n let sql = `SELECT ${options.groupBy.join(', ')}, ${metrics} FROM ${table}`;\n sql += ` GROUP BY ${options.groupBy.join(', ')}`;\n\n if (options.having) {\n const havingClauses = Object.entries(options.having).map(\n ([key, value]) =>\n `${key} ${typeof value === 'object' ? value.op : '='} ?`\n );\n sql += ` HAVING ${havingClauses.join(' AND ')}`;\n }\n\n return this.db\n .prepare(sql)\n .all(...Object.values(options.having || {})) as any[];\n }\n\n // Pattern detection (basic)\n async detectPatterns(timeRange?: { start: Date; end: Date }): Promise<\n Array<{\n pattern: string;\n type: string;\n frequency: number;\n lastSeen: Date;\n }>\n > {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n let sql = `\n SELECT type as pattern, type, COUNT(*) as frequency, MAX(created_at) as last_seen\n FROM frames\n `;\n\n const params = [];\n if (timeRange) {\n sql += ' WHERE created_at >= ? AND created_at <= ?';\n params.push(\n Math.floor(timeRange.start.getTime() / 1000),\n Math.floor(timeRange.end.getTime() / 1000)\n );\n }\n\n sql += ' GROUP BY type HAVING COUNT(*) > 1 ORDER BY frequency DESC';\n\n const rows = this.db.prepare(sql).all(...params) as any[];\n\n return rows.map((row) => ({\n pattern: row.pattern,\n type: row.type,\n frequency: row.frequency,\n lastSeen: new Date(row.last_seen * 1000),\n }));\n }\n\n // Bulk operations\n async executeBulk(operations: BulkOperation[]): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n await this.inTransaction(async () => {\n for (const op of operations) {\n switch (op.type) {\n case 'insert':\n // Build insert dynamically based on data\n const insertCols = Object.keys(op.data);\n const insertPlaceholders = insertCols.map(() => '?').join(',');\n this.db!.prepare(\n `INSERT INTO ${op.table} (${insertCols.join(',')}) VALUES (${insertPlaceholders})`\n ).run(...Object.values(op.data));\n break;\n\n case 'update':\n const updateSets = Object.keys(op.data)\n .map((k) => `${k} = ?`)\n .join(',');\n const whereClause = this.buildWhereClause(op.where || {});\n this.db!.prepare(\n `UPDATE ${op.table} SET ${updateSets} ${whereClause}`\n ).run(...Object.values(op.data), ...Object.values(op.where || {}));\n break;\n\n case 'delete':\n const deleteWhere = this.buildWhereClause(op.where || {});\n this.db!.prepare(`DELETE FROM ${op.table} ${deleteWhere}`).run(\n ...Object.values(op.where || {})\n );\n break;\n }\n }\n });\n }\n\n async vacuum(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.pragma('vacuum');\n logger.info('SQLite database vacuumed');\n }\n\n async analyze(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.pragma('analyze');\n logger.info('SQLite database analyzed');\n }\n\n // Statistics\n async getStats(): Promise<DatabaseStats> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n const stats = {\n totalFrames: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM frames')\n .get() as CountResult\n ).count,\n activeFrames: (\n this.db\n .prepare(\n \"SELECT COUNT(*) as count FROM frames WHERE state = 'active'\"\n )\n .get() as CountResult\n ).count,\n totalEvents: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM events')\n .get() as CountResult\n ).count,\n totalAnchors: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM anchors')\n .get() as CountResult\n ).count,\n diskUsage: 0,\n };\n\n // Get file size\n try {\n const fileStats = await fs.stat(this.dbPath);\n stats.diskUsage = fileStats.size;\n } catch (error: unknown) {\n // File may not exist yet or be inaccessible - disk usage remains 0\n logger.debug('Failed to get database file size', {\n dbPath: this.dbPath,\n error: error instanceof Error ? error.message : String(error),\n });\n }\n\n return stats;\n }\n\n async getQueryStats(): Promise<\n Array<{\n query: string;\n calls: number;\n meanTime: number;\n totalTime: number;\n }>\n > {\n // SQLite doesn't have built-in query stats\n logger.warn('Query stats not available for SQLite');\n return [];\n }\n\n // Transaction support\n async beginTransaction(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.prepare('BEGIN').run();\n this.inTransactionFlag = true;\n }\n\n async commitTransaction(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.prepare('COMMIT').run();\n this.inTransactionFlag = false;\n }\n\n async rollbackTransaction(): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n this.db.prepare('ROLLBACK').run();\n this.inTransactionFlag = false;\n }\n\n async inTransaction(\n callback: (adapter: DatabaseAdapter) => Promise<void>\n ): Promise<void> {\n await this.beginTransaction();\n\n try {\n await callback(this);\n await this.commitTransaction();\n } catch (error: unknown) {\n await this.rollbackTransaction();\n throw error;\n }\n }\n\n // Export/Import\n async exportData(\n tables: string[],\n format: 'json' | 'parquet' | 'csv'\n ): Promise<Buffer> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n if (format !== 'json') {\n throw new ValidationError(\n `Format ${format} not supported for SQLite export`,\n ErrorCode.VALIDATION_FAILED,\n { format, supportedFormats: ['json'] }\n );\n }\n\n const data: Record<string, any[]> = {};\n\n for (const table of tables) {\n data[table] = this.db.prepare(`SELECT * FROM ${table}`).all();\n }\n\n return Buffer.from(JSON.stringify(data, null, 2));\n }\n\n async importData(\n data: Buffer,\n format: 'json' | 'parquet' | 'csv',\n options?: { truncate?: boolean; upsert?: boolean }\n ): Promise<void> {\n if (!this.db)\n throw new DatabaseError(\n 'Database not connected',\n ErrorCode.DB_CONNECTION_FAILED\n );\n\n if (format !== 'json') {\n throw new ValidationError(\n `Format ${format} not supported for SQLite import`,\n ErrorCode.VALIDATION_FAILED,\n { format, supportedFormats: ['json'] }\n );\n }\n\n const parsed = JSON.parse(data.toString());\n\n await this.inTransaction(async () => {\n for (const [table, rows] of Object.entries(parsed)) {\n if (options?.truncate) {\n this.db!.prepare(`DELETE FROM ${table}`).run();\n }\n\n for (const row of rows as any[]) {\n const cols = Object.keys(row);\n const placeholders = cols.map(() => '?').join(',');\n\n if (options?.upsert) {\n const updates = cols.map((c) => `${c} = excluded.${c}`).join(',');\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})\n ON CONFLICT DO UPDATE SET ${updates}`\n ).run(...Object.values(row));\n } else {\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})`\n ).run(...Object.values(row));\n }\n }\n }\n });\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -22,6 +22,17 @@ const DEFAULT_DAEMON_CONFIG = {
|
|
|
22
22
|
retryAttempts: 3,
|
|
23
23
|
retryDelay: 3e4
|
|
24
24
|
},
|
|
25
|
+
maintenance: {
|
|
26
|
+
enabled: true,
|
|
27
|
+
interval: 360,
|
|
28
|
+
// 6 hours
|
|
29
|
+
staleFrameThresholdDays: 30,
|
|
30
|
+
ftsRebuildInterval: 24,
|
|
31
|
+
// hours
|
|
32
|
+
embeddingBatchSize: 50,
|
|
33
|
+
vacuumInterval: 168
|
|
34
|
+
// weekly
|
|
35
|
+
},
|
|
25
36
|
fileWatch: {
|
|
26
37
|
enabled: false,
|
|
27
38
|
// Disabled by default
|
|
@@ -75,6 +86,10 @@ function loadDaemonConfig() {
|
|
|
75
86
|
...config,
|
|
76
87
|
context: { ...DEFAULT_DAEMON_CONFIG.context, ...config.context },
|
|
77
88
|
linear: { ...DEFAULT_DAEMON_CONFIG.linear, ...config.linear },
|
|
89
|
+
maintenance: {
|
|
90
|
+
...DEFAULT_DAEMON_CONFIG.maintenance,
|
|
91
|
+
...config.maintenance
|
|
92
|
+
},
|
|
78
93
|
fileWatch: { ...DEFAULT_DAEMON_CONFIG.fileWatch, ...config.fileWatch }
|
|
79
94
|
};
|
|
80
95
|
} catch {
|
|
@@ -89,6 +104,7 @@ function saveDaemonConfig(config) {
|
|
|
89
104
|
...config,
|
|
90
105
|
context: { ...currentConfig.context, ...config.context },
|
|
91
106
|
linear: { ...currentConfig.linear, ...config.linear },
|
|
107
|
+
maintenance: { ...currentConfig.maintenance, ...config.maintenance },
|
|
92
108
|
fileWatch: { ...currentConfig.fileWatch, ...config.fileWatch }
|
|
93
109
|
};
|
|
94
110
|
writeFileSync(configFile, JSON.stringify(newConfig, null, 2));
|
|
@@ -100,6 +116,7 @@ function readDaemonStatus() {
|
|
|
100
116
|
services: {
|
|
101
117
|
context: { enabled: false },
|
|
102
118
|
linear: { enabled: false },
|
|
119
|
+
maintenance: { enabled: false },
|
|
103
120
|
fileWatch: { enabled: false }
|
|
104
121
|
},
|
|
105
122
|
errors: []
|