brainbank 0.6.0 → 0.7.0
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/README.md +132 -61
- package/dist/{base-B_vJSAbj.d.ts → base-3SNc_CeY.d.ts} +4 -4
- package/dist/{chunk-YC4ZQLDN.js → chunk-DI3H6JVZ.js} +8 -7
- package/dist/chunk-DI3H6JVZ.js.map +1 -0
- package/dist/{chunk-PXK62M5W.js → chunk-FGL32LUJ.js} +4 -3
- package/dist/{chunk-PXK62M5W.js.map → chunk-FGL32LUJ.js.map} +1 -1
- package/dist/{chunk-HPNUMUIF.js → chunk-JRSKWF6K.js} +4 -3
- package/dist/{chunk-HPNUMUIF.js.map → chunk-JRSKWF6K.js.map} +1 -1
- package/dist/{chunk-C4KDZGRX.js → chunk-VQ27YUHH.js} +10 -6
- package/dist/{chunk-C4KDZGRX.js.map → chunk-VQ27YUHH.js.map} +1 -1
- package/dist/cli.js +90 -25
- package/dist/cli.js.map +1 -1
- package/dist/code.d.ts +3 -1
- package/dist/code.js +1 -1
- package/dist/docs.d.ts +7 -3
- package/dist/docs.js +1 -1
- package/dist/git.d.ts +3 -1
- package/dist/git.js +1 -1
- package/dist/index.d.ts +2 -2
- package/dist/index.js +4 -4
- package/dist/memory.d.ts +1 -1
- package/dist/notes.d.ts +1 -1
- package/package.json +2 -2
- package/dist/chunk-YC4ZQLDN.js.map +0 -1
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/config/defaults.ts","../src/domain/collection.ts","../src/providers/vector/hnsw-index.ts","../src/search/vector/mmr.ts","../src/search/vector/vector-search.ts","../src/search/keyword/keyword-search.ts","../src/search/context-builder.ts","../src/brainbank.ts","../src/bootstrap/registry.ts","../src/bootstrap/initializer.ts","../src/db/database.ts","../src/db/schema.ts","../src/services/embedding-meta.ts","../src/api/search-api.ts","../src/indexers/base.ts","../src/api/index-api.ts","../src/services/reembed.ts","../src/services/watch.ts"],"sourcesContent":["import type { BrainBankConfig, ResolvedConfig } from '@/types.ts';\nimport * as path from 'node:path';\n\n// ── Defaults ────────────────────────────────────────\n\nexport const DEFAULTS: ResolvedConfig = {\n repoPath: '.',\n dbPath: '.brainbank/brainbank.db',\n gitDepth: 500,\n maxFileSize: 512_000, // 500KB\n maxDiffBytes: 8192,\n hnswM: 16,\n hnswEfConstruction: 200,\n hnswEfSearch: 50,\n embeddingDims: 384,\n maxElements: 2_000_000,\n};\n\n// ── Resolver ────────────────────────────────────────\n\n/**\n * Merge partial config with defaults.\n * All fields become required.\n * Relative dbPath is resolved against repoPath.\n */\nexport function resolveConfig(partial: BrainBankConfig = {}): ResolvedConfig {\n const repoPath = path.resolve(partial.repoPath ?? DEFAULTS.repoPath);\n const rawDbPath = partial.dbPath ?? DEFAULTS.dbPath;\n // Resolve relative dbPath against repoPath so DB lives alongside the repo\n const dbPath = path.isAbsolute(rawDbPath) ? rawDbPath : path.join(repoPath, rawDbPath);\n\n return {\n repoPath,\n dbPath,\n gitDepth: partial.gitDepth ?? DEFAULTS.gitDepth,\n maxFileSize: partial.maxFileSize ?? DEFAULTS.maxFileSize,\n maxDiffBytes: partial.maxDiffBytes ?? DEFAULTS.maxDiffBytes,\n hnswM: partial.hnswM ?? DEFAULTS.hnswM,\n hnswEfConstruction: partial.hnswEfConstruction ?? DEFAULTS.hnswEfConstruction,\n hnswEfSearch: partial.hnswEfSearch ?? DEFAULTS.hnswEfSearch,\n embeddingDims: partial.embeddingDims ?? DEFAULTS.embeddingDims,\n maxElements: partial.maxElements ?? DEFAULTS.maxElements,\n embeddingProvider: partial.embeddingProvider,\n reranker: partial.reranker,\n };\n}\n\n","/**\n * BrainBank — Collection\n * \n * Universal key-value store with vector + BM25 hybrid search.\n * The foundation primitive — store anything, search semantically.\n * \n * const errors = brain.collection('debug_errors');\n * await errors.add('Fixed null check in api handler', { file: 'api.ts' });\n * const hits = await errors.search('null pointer');\n */\n\nimport type { Database } from '@/db/database.ts';\nimport type { KvDataRow, CountRow } from '@/db/rows.ts';\nimport { vecToBuffer } from '@/lib/math.ts';\nimport type { EmbeddingProvider, Reranker, SearchResult } from '@/types.ts';\nimport type { HNSWIndex } from '@/providers/vector/hnsw-index.ts';\nimport { reciprocalRankFusion } from '@/lib/rrf.ts';\nimport { rerank } from '@/search/vector/rerank.ts';\nimport { sanitizeFTS, normalizeBM25 } from '@/lib/fts.ts';\n\nexport interface CollectionItem {\n id: number;\n collection: string;\n content: string;\n metadata: Record<string, any>;\n tags: string[];\n createdAt: number;\n expiresAt?: number;\n score?: number;\n}\n\nexport interface CollectionSearchOptions {\n /** Max results. Default: 5 */\n k?: number;\n /** Search mode. Default: 'hybrid' */\n mode?: 'hybrid' | 'vector' | 'keyword';\n /** Minimum score threshold. Default: 0.15 */\n minScore?: number;\n /** Filter by tags (item must have ALL specified tags). */\n tags?: string[];\n}\n\nexport interface CollectionAddOptions {\n /** Metadata key-value pairs. */\n metadata?: Record<string, any>;\n /** Tags for filtering. */\n tags?: string[];\n /** Time-to-live duration string (e.g. '7d', '24h', '30m'). */\n ttl?: string;\n}\n\nexport class Collection {\n constructor(\n private _name: string,\n private _db: Database,\n private _embedding: EmbeddingProvider,\n private _hnsw: HNSWIndex,\n private _vecs: Map<number, Float32Array>,\n private _reranker?: Reranker,\n ) {}\n\n /** Collection name. */\n get name(): string { return this._name; }\n\n /** Add an item. Returns its ID. */\n async add(content: string, options: CollectionAddOptions | Record<string, any> = {}): Promise<number> {\n // Support both signatures: add(content, { metadata, tags, ttl }) and add(content, metadata)\n const opts = 'tags' in options || 'ttl' in options || 'metadata' in options\n ? options as CollectionAddOptions\n : { metadata: options as Record<string, any> };\n\n const metadata = opts.metadata ?? {};\n const tags = opts.tags ?? [];\n const expiresAt = opts.ttl ? Math.floor(Date.now() / 1000) + parseDuration(opts.ttl) : null;\n\n // Embed FIRST — if this throws, no orphaned rows are left in kv_data\n const vec = await this._embedding.embed(content);\n\n const result = this._db.prepare(\n 'INSERT INTO kv_data (collection, content, meta_json, tags_json, expires_at) VALUES (?, ?, ?, ?, ?)'\n ).run(this._name, content, JSON.stringify(metadata), JSON.stringify(tags), expiresAt);\n\n const id = Number(result.lastInsertRowid);\n this._db.prepare(\n 'INSERT INTO kv_vectors (data_id, embedding) VALUES (?, ?)'\n ).run(id, vecToBuffer(vec));\n\n this._hnsw.add(vec, id);\n this._vecs.set(id, vec);\n\n return id;\n }\n\n /** Add multiple items. Returns their IDs. */\n async addMany(items: { content: string; metadata?: Record<string, any>; tags?: string[]; ttl?: string }[]): Promise<number[]> {\n if (items.length === 0) return [];\n\n // Batch embed all texts at once\n const texts = items.map(i => i.content);\n const vecs = await this._embedding.embedBatch(texts);\n\n // Commit DB rows atomically. HNSW is updated ONLY after this succeeds.\n // If the transaction throws, execution never reaches the HNSW loop below.\n const ids: number[] = [];\n const insertData = this._db.prepare(\n 'INSERT INTO kv_data (collection, content, meta_json, tags_json, expires_at) VALUES (?, ?, ?, ?, ?)'\n );\n const insertVec = this._db.prepare(\n 'INSERT INTO kv_vectors (data_id, embedding) VALUES (?, ?)'\n );\n\n this._db.transaction(() => {\n for (let i = 0; i < items.length; i++) {\n const item = items[i];\n const expiresAt = item.ttl ? Math.floor(Date.now() / 1000) + parseDuration(item.ttl) : null;\n\n const result = insertData.run(\n this._name,\n item.content,\n JSON.stringify(item.metadata ?? {}),\n JSON.stringify(item.tags ?? []),\n expiresAt,\n );\n\n const id = Number(result.lastInsertRowid);\n insertVec.run(id, vecToBuffer(vecs[i]));\n ids.push(id);\n }\n });\n\n // HNSW + cache updated after successful commit — no orphan risk on rollback.\n for (let i = 0; i < ids.length; i++) {\n this._hnsw.add(vecs[i], ids[i]);\n this._vecs.set(ids[i], vecs[i]);\n }\n\n return ids;\n }\n\n /** Search this collection. */\n async search(query: string, options: CollectionSearchOptions = {}): Promise<CollectionItem[]> {\n const { k = 5, mode = 'hybrid', minScore = 0.15, tags } = options;\n\n // Auto-prune expired items before search\n this._pruneExpired();\n\n if (mode === 'keyword') return this._filterByTags(this._searchBM25(query, k, minScore), tags);\n if (mode === 'vector') return this._filterByTags(await this._searchVector(query, k, minScore), tags);\n\n // Hybrid: vector + BM25 → RRF\n const [vectorHits, bm25Hits] = await Promise.all([\n this._searchVector(query, k, 0),\n Promise.resolve(this._searchBM25(query, k, 0)),\n ]);\n\n const fused = reciprocalRankFusion([\n vectorHits.map(h => ({ type: 'collection' as const, score: h.score ?? 0, content: h.content, metadata: { id: h.id } })),\n bm25Hits.map(h => ({ type: 'collection' as const, score: h.score ?? 0, content: h.content, metadata: { id: h.id } })),\n ]);\n\n const allById = new Map<number, CollectionItem>();\n for (const h of [...vectorHits, ...bm25Hits]) allById.set(h.id, h);\n\n const results: CollectionItem[] = [];\n for (const r of fused) {\n const meta = r.metadata as Record<string, unknown> | undefined;\n const item = allById.get(meta?.id as number);\n if (!item) continue;\n const scored = { ...item, score: r.score };\n if (scored.score >= minScore) results.push(scored);\n if (results.length >= k) break;\n }\n\n // Apply re-ranking if available\n if (this._reranker && results.length > 1) {\n const asSearchResults: SearchResult[] = results.map(r => ({\n type: 'collection' as const,\n score: r.score ?? 0,\n content: r.content,\n metadata: { id: r.id },\n }));\n const reranked = await rerank(query, asSearchResults, this._reranker);\n const rerankedById = new Map(reranked.map(r => [(r.metadata as any)?.id as number, r.score]));\n const blended = results.map(r => ({ ...r, score: rerankedById.get(r.id) ?? r.score ?? 0 }));\n return this._filterByTags(\n blended.sort((a, b) => (b.score ?? 0) - (a.score ?? 0)),\n tags,\n );\n }\n\n return this._filterByTags(results, tags);\n }\n\n /** List items (newest first). */\n list(options: { limit?: number; offset?: number; tags?: string[] } = {}): CollectionItem[] {\n const { limit = 20, offset = 0, tags } = options;\n\n // Auto-prune expired items\n this._pruneExpired();\n\n const rows = this._db.prepare(\n 'SELECT * FROM kv_data WHERE collection = ? AND (expires_at IS NULL OR expires_at > ?) ORDER BY created_at DESC, id DESC LIMIT ? OFFSET ?'\n ).all(this._name, Math.floor(Date.now() / 1000), limit, offset) as KvDataRow[];\n return this._filterByTags(rows.map(r => this._rowToItem(r)), tags);\n }\n\n /** Count items in this collection. */\n count(): number {\n return (this._db.prepare(\n 'SELECT COUNT(*) as c FROM kv_data WHERE collection = ? AND (expires_at IS NULL OR expires_at > ?)'\n ).get(this._name, Math.floor(Date.now() / 1000)) as CountRow).c;\n }\n\n /** Keep only the N most recent items, remove the rest. */\n async trim(options: { keep: number }): Promise<{ removed: number }> {\n const before = this.count();\n if (before <= options.keep) return { removed: 0 };\n\n // Get IDs to remove (oldest first, beyond the keep window)\n const toRemove = this._db.prepare(`\n SELECT id FROM kv_data \n WHERE collection = ? \n ORDER BY created_at DESC, id DESC \n LIMIT -1 OFFSET ?\n `).all(this._name, options.keep) as Pick<KvDataRow, 'id'>[];\n\n for (const row of toRemove) {\n this._removeById(row.id);\n }\n\n return { removed: toRemove.length };\n }\n\n /** Remove items older than a duration string (e.g. '30d', '12h'). */\n async prune(options: { olderThan: string }): Promise<{ removed: number }> {\n const seconds = parseDuration(options.olderThan);\n const cutoff = Math.floor(Date.now() / 1000) - seconds;\n\n const toRemove = this._db.prepare(\n 'SELECT id FROM kv_data WHERE collection = ? AND created_at < ?'\n ).all(this._name, cutoff) as Pick<KvDataRow, 'id'>[];\n\n for (const row of toRemove) {\n this._removeById(row.id);\n }\n\n return { removed: toRemove.length };\n }\n\n /** Remove a specific item by ID. */\n remove(id: number): void {\n this._removeById(id);\n }\n\n /** Clear all items in this collection. */\n clear(): void {\n const rows = this._db.prepare(\n 'SELECT id FROM kv_data WHERE collection = ?'\n ).all(this._name) as Pick<KvDataRow, 'id'>[];\n\n for (const row of rows) {\n this._removeById(row.id);\n }\n }\n\n // ── Private ──────────────────────────────────────\n\n private _removeById(id: number): void {\n // DB first — can fail (disk full, lock). If it throws, HNSW+cache stay consistent.\n this._db.prepare('DELETE FROM kv_data WHERE id = ?').run(id);\n // HNSW + cache after — these always succeed\n this._hnsw.remove(id);\n this._vecs.delete(id);\n }\n\n private async _searchVector(query: string, k: number, minScore: number): Promise<CollectionItem[]> {\n if (this._hnsw.size === 0) return [];\n\n const queryVec = await this._embedding.embed(query);\n // Over-fetch from shared HNSW to compensate for cross-collection filtering\n const searchK = Math.min(k * 10, this._hnsw.size);\n const hits = this._hnsw.search(queryVec, searchK);\n\n const ids = hits.map(h => h.id);\n if (ids.length === 0) return [];\n\n const scoreMap = new Map(hits.map(h => [h.id, h.score]));\n const placeholders = ids.map(() => '?').join(',');\n\n const rows = this._db.prepare(\n `SELECT * FROM kv_data WHERE id IN (${placeholders}) AND collection = ?`\n ).all(...ids, this._name) as KvDataRow[];\n\n return rows\n .map(r => ({ ...this._rowToItem(r), score: scoreMap.get(r.id) ?? 0 }))\n .filter(r => r.score >= minScore)\n .sort((a, b) => (b.score ?? 0) - (a.score ?? 0))\n .slice(0, k);\n }\n\n private _searchBM25(query: string, k: number, minScore: number): CollectionItem[] {\n const ftsQuery = sanitizeFTS(query);\n if (!ftsQuery) return [];\n\n try {\n const rows = this._db.prepare(`\n SELECT d.*, bm25(fts_kv, 5.0, 1.0) AS score\n FROM fts_kv f\n JOIN kv_data d ON d.id = f.rowid\n WHERE fts_kv MATCH ? AND d.collection = ?\n ORDER BY score ASC\n LIMIT ?\n `).all(ftsQuery, this._name, k) as (KvDataRow & { score: number })[];\n\n return rows\n .map(r => ({\n ...this._rowToItem(r),\n score: normalizeBM25(r.score),\n }))\n .filter(r => (r.score ?? 0) >= minScore);\n } catch {\n return [];\n }\n }\n\n private _rowToItem(r: KvDataRow): CollectionItem {\n return {\n id: r.id,\n collection: r.collection,\n content: r.content,\n metadata: JSON.parse(r.meta_json || '{}'),\n tags: JSON.parse(r.tags_json || '[]'),\n createdAt: r.created_at,\n expiresAt: r.expires_at ?? undefined,\n };\n }\n\n /** Filter results by tags (item must have ALL specified tags). */\n private _filterByTags(items: CollectionItem[], tags?: string[]): CollectionItem[] {\n if (!tags || tags.length === 0) return items;\n return items.filter(item =>\n tags.every(t => item.tags.includes(t))\n );\n }\n\n /** Remove expired items (TTL). Called automatically on search/list. */\n private _pruneExpired(): void {\n const now = Math.floor(Date.now() / 1000);\n const expired = this._db.prepare(\n 'SELECT id FROM kv_data WHERE collection = ? AND expires_at IS NOT NULL AND expires_at <= ?'\n ).all(this._name, now) as Pick<KvDataRow, 'id'>[];\n\n for (const row of expired) {\n this._removeById(row.id);\n }\n }\n}\n\n/** Parse a duration string like '30d', '12h', '5m' to seconds. */\nfunction parseDuration(s: string): number {\n const match = s.match(/^(\\d+)([dhms])$/);\n if (!match) throw new Error(`Invalid duration: \"${s}\". Use format like '30d', '12h', '5m'.`);\n\n const n = parseInt(match[1], 10);\n switch (match[2]) {\n case 'd': return n * 86400;\n case 'h': return n * 3600;\n case 'm': return n * 60;\n case 's': return n;\n default: return n;\n }\n}\n","/**\n * BrainBank — HNSW Vector Index\n * \n * Wraps hnswlib-node for O(log n) approximate nearest neighbor search.\n * M=16 connections, ef=200 construction, ef=50 search by default.\n * 150x faster than brute force at 1M vectors.\n *\n * Supports disk persistence: save(path) / tryLoad(path, count)\n * to skip costly vector-by-vector rebuild on startup.\n */\n\nimport { existsSync } from 'node:fs';\nimport type { VectorIndex, SearchHit } from '@/types.ts';\n\nexport class HNSWIndex implements VectorIndex {\n private _index: any = null;\n private _lib: any = null;\n private _ids = new Set<number>();\n\n constructor(\n private _dims: number,\n private _maxElements: number = 2_000_000,\n private _M: number = 16,\n private _efConstruction: number = 200,\n private _efSearch: number = 50,\n ) {}\n\n /**\n * Initialize the HNSW index.\n * Must be called before add/search.\n */\n async init(): Promise<this> {\n this._lib = await import('hnswlib-node');\n this._createIndex();\n return this;\n }\n\n /**\n * Reinitialize the index in-place, clearing all vectors.\n * Required after reembed or full re-index to avoid duplicate IDs.\n * init() must have been called first.\n */\n reinit(): void {\n if (!this._lib) throw new Error('HNSW not initialized — call init() first');\n this._createIndex();\n }\n\n private _createIndex(): void {\n const HNSW = this._lib.default?.HierarchicalNSW ?? this._lib.HierarchicalNSW;\n this._index = new HNSW('cosine', this._dims);\n this._index.initIndex(this._maxElements, this._M, this._efConstruction);\n this._index.setEf(this._efSearch);\n this._ids = new Set();\n }\n\n /** Maximum capacity of this index. */\n get maxElements(): number { return this._maxElements; }\n\n /**\n * Add a vector with an integer ID.\n * The vector should be pre-normalized for cosine distance.\n */\n add(vector: Float32Array, id: number): void {\n if (!this._index) throw new Error('HNSW index not initialized — call init() first');\n if (this._ids.has(id)) return; // idempotent: skip duplicates\n if (this._ids.size >= this._maxElements) {\n throw new Error(\n `HNSW index full (${this._maxElements} elements). ` +\n `Increase maxElements in config or prune old data.`\n );\n }\n this._index.addPoint(Array.from(vector), id);\n this._ids.add(id);\n }\n\n /**\n * Mark a vector as deleted so it no longer appears in searches.\n * Uses hnswlib-node markDelete under the hood.\n * Safe to call with an ID that doesn't exist.\n */\n remove(id: number): void {\n if (!this._index || this._ids.size === 0) return;\n if (!this._ids.has(id)) return;\n try {\n this._index.markDelete(id);\n this._ids.delete(id);\n } catch {\n // ID not found — ignore silently\n }\n }\n\n /**\n * Search for the k nearest neighbors.\n * Returns results sorted by score (highest first).\n * Score is 1 - cosine_distance (1.0 = identical).\n */\n search(query: Float32Array, k: number): SearchHit[] {\n if (!this._index || this._ids.size === 0) return [];\n\n const actualK = Math.min(k, this._ids.size);\n const result = this._index.searchKnn(Array.from(query), actualK);\n\n return result.neighbors.map((id: number, i: number) => ({\n id,\n score: 1 - result.distances[i],\n }));\n }\n\n /** Number of vectors in the index. */\n get size(): number {\n return this._ids.size;\n }\n\n /**\n * Save the HNSW graph to disk.\n * The file can be loaded later with tryLoad() to skip vector-by-vector insertion.\n */\n save(path: string): void {\n if (!this._index || this._ids.size === 0) return;\n this._index.writeIndexSync(path);\n }\n\n /**\n * Try to load a previously saved HNSW index from disk.\n * Returns true if loaded successfully, false if stale or missing.\n * @param path File path to the saved index\n * @param expectedCount Expected number of vectors (from SQLite) — used to detect staleness\n */\n tryLoad(path: string, expectedCount: number): boolean {\n if (!this._index || !existsSync(path)) return false;\n\n try {\n this._index.readIndexSync(path);\n const loadedCount = this._index.getCurrentCount();\n\n // Stale: vector count in DB differs from saved index\n if (loadedCount !== expectedCount) {\n this.reinit();\n return false;\n }\n\n // Rebuild _ids set from the loaded index\n const ids = this._index.getIdsList();\n this._ids = new Set(ids);\n this._index.setEf(this._efSearch);\n return true;\n } catch {\n this.reinit();\n return false;\n }\n }\n}\n","/**\n * BrainBank — Maximum Marginal Relevance (MMR)\n * \n * Diversifies vector search results to avoid returning redundant items.\n * λ=1.0 → pure relevance, λ=0.0 → pure diversity.\n * Default λ=0.7 balances both.\n */\n\nimport type { VectorIndex, SearchHit } from '@/types.ts';\nimport { cosineSimilarity } from '@/lib/math.ts';\n\n/**\n * Search with Maximum Marginal Relevance for diversified results.\n * \n * Algorithm:\n * 1. Get 3x candidates from HNSW\n * 2. Greedily select items that maximize: λ * relevance - (1-λ) * max_sim_to_selected\n */\nexport function searchMMR(\n index: VectorIndex,\n query: Float32Array,\n vectorCache: Map<number, Float32Array>,\n k: number,\n lambda: number = 0.7,\n): SearchHit[] {\n // Get more candidates than needed\n const candidates = index.search(query, k * 3);\n if (candidates.length <= k) return candidates;\n\n const selected: SearchHit[] = [];\n const remaining = [...candidates];\n\n while (selected.length < k && remaining.length > 0) {\n let bestScore = -Infinity;\n let bestIdx = 0;\n\n for (let i = 0; i < remaining.length; i++) {\n const relevance = remaining[i].score;\n\n // Max similarity to any already-selected item\n let maxSim = 0;\n for (const sel of selected) {\n const candidateVec = vectorCache.get(remaining[i].id);\n const selectedVec = vectorCache.get(sel.id);\n if (candidateVec && selectedVec) {\n maxSim = Math.max(maxSim, cosineSimilarity(candidateVec, selectedVec));\n }\n }\n\n // MMR score: balance relevance vs diversity\n const mmrScore = lambda * relevance - (1 - lambda) * maxSim;\n\n if (mmrScore > bestScore) {\n bestScore = mmrScore;\n bestIdx = i;\n }\n }\n\n selected.push(remaining[bestIdx]);\n remaining.splice(bestIdx, 1);\n }\n\n return selected;\n}\n","/**\n * BrainBank — Vector Search Strategy\n * \n * Searches across code, git, and memory pattern HNSW indices.\n * Returns typed results sorted by relevance.\n */\n\nimport type { Database } from '@/db/database.ts';\nimport type { EmbeddingProvider, Reranker, SearchResult } from '@/types.ts';\nimport type { HNSWIndex } from '@/providers/vector/hnsw-index.ts';\nimport type { SearchStrategy, SearchOptions } from '@/search/types.ts';\nimport { searchMMR } from './mmr.ts';\nimport { rerank } from './rerank.ts';\n\nexport interface VectorSearchConfig {\n db: Database;\n codeHnsw?: HNSWIndex;\n gitHnsw?: HNSWIndex;\n patternHnsw?: HNSWIndex;\n codeVecs: Map<number, Float32Array>;\n gitVecs: Map<number, Float32Array>;\n patternVecs: Map<number, Float32Array>;\n embedding: EmbeddingProvider;\n reranker?: Reranker;\n}\n\nexport class VectorSearch implements SearchStrategy {\n private _config: VectorSearchConfig;\n\n constructor(config: VectorSearchConfig) {\n this._config = config;\n }\n\n /** Search across all indices. Returns combined results sorted by score. */\n async search(query: string, options: SearchOptions = {}): Promise<SearchResult[]> {\n const {\n codeK = 6, gitK = 5, patternK = 4,\n minScore = 0.25, useMMR = true, mmrLambda = 0.7,\n } = options;\n\n const queryVec = await this._config.embedding.embed(query);\n const results: SearchResult[] = [];\n\n this._searchCode(queryVec, codeK, minScore, useMMR, mmrLambda, results);\n this._searchGit(queryVec, gitK, minScore, results);\n this._searchPatterns(queryVec, patternK, minScore, useMMR, mmrLambda, results);\n\n results.sort((a, b) => b.score - a.score);\n\n if (this._config.reranker && results.length > 1) {\n return rerank(query, results, this._config.reranker);\n }\n return results;\n }\n\n /** Vector search across code chunks. */\n private _searchCode(\n queryVec: Float32Array, k: number, minScore: number,\n useMMR: boolean, mmrLambda: number, results: SearchResult[],\n ): void {\n const { codeHnsw, codeVecs, db } = this._config;\n if (!codeHnsw || codeHnsw.size === 0) return;\n\n const hits = useMMR\n ? searchMMR(codeHnsw, queryVec, codeVecs, k, mmrLambda)\n : codeHnsw.search(queryVec, k);\n if (hits.length === 0) return;\n\n const ids = hits.map(h => h.id);\n const scoreMap = new Map(hits.map(h => [h.id, h.score]));\n const placeholders = ids.map(() => '?').join(',');\n const rows = db.prepare(\n `SELECT * FROM code_chunks WHERE id IN (${placeholders})`\n ).all(...ids) as any[];\n\n for (const r of rows) {\n const score = scoreMap.get(r.id) ?? 0;\n if (score >= minScore) {\n results.push({\n type: 'code', score, filePath: r.file_path, content: r.content,\n metadata: {\n chunkType: r.chunk_type, name: r.name,\n startLine: r.start_line, endLine: r.end_line, language: r.language,\n },\n });\n }\n }\n }\n\n /** Vector search across git commits. */\n private _searchGit(\n queryVec: Float32Array, k: number, minScore: number, results: SearchResult[],\n ): void {\n const { gitHnsw, db } = this._config;\n if (!gitHnsw || gitHnsw.size === 0) return;\n\n const hits = gitHnsw.search(queryVec, k * 2);\n if (hits.length === 0) return;\n\n const ids = hits.map(h => h.id);\n const scoreMap = new Map(hits.map(h => [h.id, h.score]));\n const placeholders = ids.map(() => '?').join(',');\n const rows = db.prepare(\n `SELECT * FROM git_commits WHERE id IN (${placeholders}) AND is_merge = 0`\n ).all(...ids) as any[];\n\n for (const r of rows) {\n const score = scoreMap.get(r.id) ?? 0;\n if (score >= minScore) {\n results.push({\n type: 'commit', score, content: r.message,\n metadata: {\n hash: r.hash, shortHash: r.short_hash,\n author: r.author, date: r.date,\n files: JSON.parse(r.files_json ?? '[]'),\n additions: r.additions, deletions: r.deletions, diff: r.diff,\n },\n });\n }\n }\n }\n\n /** Vector search across memory patterns. */\n private _searchPatterns(\n queryVec: Float32Array, k: number, minScore: number,\n useMMR: boolean, mmrLambda: number, results: SearchResult[],\n ): void {\n const { patternHnsw, patternVecs, db } = this._config;\n if (!patternHnsw || patternHnsw.size === 0) return;\n\n const hits = useMMR\n ? searchMMR(patternHnsw, queryVec, patternVecs, k, mmrLambda)\n : patternHnsw.search(queryVec, k);\n if (hits.length === 0) return;\n\n const ids = hits.map(h => h.id);\n const scoreMap = new Map(hits.map(h => [h.id, h.score]));\n const placeholders = ids.map(() => '?').join(',');\n const rows = db.prepare(\n `SELECT * FROM memory_patterns WHERE id IN (${placeholders}) AND success_rate >= 0.5`\n ).all(...ids) as any[];\n\n for (const r of rows) {\n const score = scoreMap.get(r.id) ?? 0;\n if (score >= minScore) {\n results.push({\n type: 'pattern', score, content: r.approach,\n metadata: {\n taskType: r.task_type, task: r.task,\n outcome: r.outcome, successRate: r.success_rate, critique: r.critique,\n },\n });\n }\n }\n }\n}\n","/**\n * BrainBank — Keyword Search Strategy\n * \n * Keyword search via SQLite FTS5 with BM25 ranking.\n * Searches across code chunks, git commits, and memory patterns.\n * Uses Porter stemming + unicode61 tokenizer.\n */\n\nimport type { Database } from '@/db/database.ts';\nimport type { SearchResult } from '@/types.ts';\nimport type { SearchStrategy, SearchOptions } from '@/search/types.ts';\nimport { sanitizeFTS, normalizeBM25 } from '@/lib/fts.ts';\n\n/** Check if an error is an FTS5 query syntax error (expected, safe to ignore). */\nfunction isFTSError(e: unknown): boolean {\n return e instanceof Error && /fts5|syntax error|parse error/i.test(e.message);\n}\n\nexport class KeywordSearch implements SearchStrategy {\n constructor(private _db: Database) {}\n\n /**\n * Full-text keyword search across all FTS5 indices.\n * Uses BM25 scoring — lower scores = better matches.\n */\n async search(query: string, options: SearchOptions = {}): Promise<SearchResult[]> {\n const { codeK = 8, gitK = 5, patternK = 4 } = options;\n\n const ftsQuery = sanitizeFTS(query);\n if (!ftsQuery) return [];\n\n const results: SearchResult[] = [];\n\n if (codeK > 0) this._searchCode(ftsQuery, query, codeK, results);\n if (gitK > 0) this._searchGit(ftsQuery, gitK, results);\n if (patternK > 0) this._searchPatterns(ftsQuery, patternK, results);\n\n return results.sort((a, b) => b.score - a.score);\n }\n\n /** FTS5 search across code chunks + file-path fallback. */\n private _searchCode(ftsQuery: string, rawQuery: string, k: number, results: SearchResult[]): void {\n const seenIds = new Set<number>();\n\n try {\n const rows = this._db.prepare(`\n SELECT c.id, c.file_path, c.chunk_type, c.name, c.start_line, c.end_line,\n c.content, c.language, bm25(fts_code, 5.0, 3.0, 1.0) AS score\n FROM fts_code f\n JOIN code_chunks c ON c.id = f.rowid\n WHERE fts_code MATCH ?\n ORDER BY score ASC\n LIMIT ?\n `).all(ftsQuery, k) as any[];\n\n for (const r of rows) {\n seenIds.add(r.id);\n results.push(this._toCodeResult(r, normalizeBM25(r.score), 'bm25'));\n }\n } catch (e) { if (!isFTSError(e)) throw e; }\n\n this._searchCodeByPath(rawQuery, seenIds, results);\n }\n\n /** File-path fallback: match filenames via LIKE. */\n private _searchCodeByPath(rawQuery: string, seenIds: Set<number>, results: SearchResult[]): void {\n try {\n const words = rawQuery.replace(/[^a-zA-Z0-9]/g, ' ').split(/\\s+/).filter(w => w.length > 2);\n for (const word of words.slice(0, 3)) {\n const pathRows = this._db.prepare(`\n SELECT id, file_path, chunk_type, name, start_line, end_line, content, language\n FROM code_chunks\n WHERE file_path LIKE ? AND chunk_type = 'file'\n LIMIT 3\n `).all(`%${word}%`) as any[];\n\n for (const r of pathRows) {\n if (seenIds.has(r.id)) continue;\n seenIds.add(r.id);\n results.push(this._toCodeResult(r, 0.6, 'bm25-path'));\n }\n }\n } catch (e) { if (!isFTSError(e)) throw e; }\n }\n\n /** FTS5 search across git commits. */\n private _searchGit(ftsQuery: string, k: number, results: SearchResult[]): void {\n try {\n const rows = this._db.prepare(`\n SELECT c.id, c.hash, c.short_hash, c.message, c.author, c.date,\n c.files_json, c.diff, c.additions, c.deletions,\n bm25(fts_commits, 5.0, 2.0, 1.0) AS score\n FROM fts_commits f\n JOIN git_commits c ON c.id = f.rowid\n WHERE fts_commits MATCH ? AND c.is_merge = 0\n ORDER BY score ASC\n LIMIT ?\n `).all(ftsQuery, k) as any[];\n\n for (const r of rows) {\n results.push({\n type: 'commit',\n score: normalizeBM25(r.score),\n content: r.message,\n metadata: {\n hash: r.hash,\n shortHash: r.short_hash,\n author: r.author,\n date: r.date,\n files: JSON.parse(r.files_json ?? '[]'),\n additions: r.additions,\n deletions: r.deletions,\n diff: r.diff,\n searchType: 'bm25',\n },\n });\n }\n } catch (e) { if (!isFTSError(e)) throw e; }\n }\n\n /** FTS5 search across memory patterns. */\n private _searchPatterns(ftsQuery: string, k: number, results: SearchResult[]): void {\n try {\n const rows = this._db.prepare(`\n SELECT p.id, p.task_type, p.task, p.approach, p.outcome,\n p.success_rate, p.critique,\n bm25(fts_patterns, 3.0, 5.0, 5.0, 1.0) AS score\n FROM fts_patterns f\n JOIN memory_patterns p ON p.id = f.rowid\n WHERE fts_patterns MATCH ? AND p.success_rate >= 0.5\n ORDER BY score ASC\n LIMIT ?\n `).all(ftsQuery, k) as any[];\n\n for (const r of rows) {\n results.push({\n type: 'pattern',\n score: normalizeBM25(r.score),\n content: r.approach,\n metadata: {\n taskType: r.task_type,\n task: r.task,\n outcome: r.outcome,\n successRate: r.success_rate,\n critique: r.critique,\n searchType: 'bm25',\n },\n });\n }\n } catch (e) { if (!isFTSError(e)) throw e; }\n }\n\n /** Map a code_chunks row to a CodeResult. */\n private _toCodeResult(r: any, score: number, searchType: string): SearchResult {\n return {\n type: 'code',\n score,\n filePath: r.file_path,\n content: r.content,\n metadata: {\n chunkType: r.chunk_type,\n name: r.name,\n startLine: r.start_line,\n endLine: r.end_line,\n language: r.language,\n searchType,\n },\n };\n }\n\n /** Rebuild the FTS index from scratch. */\n rebuild(): void {\n try {\n this._db.prepare(\"INSERT INTO fts_code(fts_code) VALUES('rebuild')\").run();\n this._db.prepare(\"INSERT INTO fts_commits(fts_commits) VALUES('rebuild')\").run();\n this._db.prepare(\"INSERT INTO fts_patterns(fts_patterns) VALUES('rebuild')\").run();\n } catch {}\n }\n}\n","/**\n * BrainBank — Context Builder\n * \n * Builds a formatted markdown context block from search results.\n * Ready for injection into an LLM system prompt.\n * Groups code by file, includes git history and learned patterns.\n */\n\nimport type { SearchResult, ContextOptions } from '@/types.ts';\nimport type { SearchStrategy } from '@/search/types.ts';\nimport type { CoEditAnalyzer } from '@/indexers/git/co-edit-analyzer.ts';\n\nexport class ContextBuilder {\n constructor(\n private _search: SearchStrategy,\n private _coEdits?: CoEditAnalyzer,\n ) {}\n\n /** Build a full context block for a task. Returns markdown for system prompt. */\n async build(task: string, options: ContextOptions = {}): Promise<string> {\n const {\n codeResults = 6, gitResults = 5, patternResults = 4,\n affectedFiles = [], minScore = 0.25,\n useMMR = true, mmrLambda = 0.7,\n } = options;\n\n const results = await this._search.search(task, {\n codeK: codeResults, gitK: gitResults, patternK: patternResults,\n minScore, useMMR, mmrLambda,\n });\n\n const parts: string[] = [`# Context for: \"${task}\"\\n`];\n\n this._formatCodeResults(results, codeResults, parts);\n this._formatGitResults(results, gitResults, parts);\n this._formatCoEdits(affectedFiles, parts);\n this._formatPatternResults(results, patternResults, parts);\n\n return parts.join('\\n');\n }\n\n /** Format code search results grouped by file. */\n private _formatCodeResults(results: SearchResult[], limit: number, parts: string[]): void {\n const codeHits = results.filter(r => r.type === 'code').slice(0, limit);\n if (codeHits.length === 0) return;\n\n parts.push('## Relevant Code\\n');\n\n const byFile = new Map<string, typeof codeHits>();\n for (const r of codeHits) {\n const key = r.filePath ?? 'unknown';\n if (!byFile.has(key)) byFile.set(key, []);\n byFile.get(key)!.push(r);\n }\n\n for (const [file, chunks] of byFile) {\n parts.push(`### ${file}`);\n for (const c of chunks) {\n const m = c.metadata;\n const label = m.name\n ? `${m.chunkType} \\`${m.name}\\` (L${m.startLine}-${m.endLine})`\n : `L${m.startLine}-${m.endLine}`;\n parts.push(`**${label}** — ${Math.round(c.score * 100)}% match`);\n parts.push('```' + (m.language || ''));\n parts.push(c.content);\n parts.push('```\\n');\n }\n }\n }\n\n /** Format git commit results with diff snippets. */\n private _formatGitResults(results: SearchResult[], limit: number, parts: string[]): void {\n const gitHits = results.filter(r => r.type === 'commit').slice(0, limit);\n if (gitHits.length === 0) return;\n\n parts.push('## Related Git History\\n');\n for (const c of gitHits) {\n const m = c.metadata;\n const score = Math.round(c.score * 100);\n const files = (m.files ?? []).slice(0, 4).join(', ');\n parts.push(`**[${m.shortHash}]** ${c.content} *(${m.author}, ${m.date?.slice(0, 10)}, ${score}%)*`);\n if (files) parts.push(` Files: ${files}`);\n if (m.diff) {\n const snippet = m.diff\n .split('\\n')\n .filter((l: string) => l.startsWith('+') || l.startsWith('-') || l.startsWith('@@'))\n .slice(0, 10)\n .join('\\n');\n if (snippet) {\n parts.push('```diff');\n parts.push(snippet);\n parts.push('```');\n }\n }\n parts.push('');\n }\n }\n\n /** Format co-edit suggestions for affected files. */\n private _formatCoEdits(affectedFiles: string[], parts: string[]): void {\n if (affectedFiles.length === 0 || !this._coEdits) return;\n\n const coEditLines: string[] = [];\n for (const file of affectedFiles.slice(0, 3)) {\n const suggestions = this._coEdits.suggest(file, 4);\n if (suggestions.length > 0) {\n coEditLines.push(\n `- **${file}** → also tends to change: ${suggestions.map(s => `${s.file} (${s.count}x)`).join(', ')}`\n );\n }\n }\n if (coEditLines.length > 0) {\n parts.push('## Co-Edit Patterns\\n');\n parts.push(...coEditLines);\n parts.push('');\n }\n }\n\n /** Format memory pattern results. */\n private _formatPatternResults(results: SearchResult[], limit: number, parts: string[]): void {\n const memHits = results.filter(r => r.type === 'pattern').slice(0, limit);\n if (memHits.length === 0) return;\n\n parts.push('## Learned Patterns\\n');\n for (const p of memHits) {\n const m = p.metadata;\n const score = Math.round(p.score * 100);\n const success = Math.round((m.successRate ?? 0) * 100);\n parts.push(`**${m.taskType}** — ${success}% success, ${score}% match`);\n parts.push(`Task: ${m.task}`);\n parts.push(`Approach: ${p.content}`);\n if (m.critique) parts.push(`Lesson: ${m.critique}`);\n parts.push('');\n }\n }\n}\n","/**\n * BrainBank — Main Orchestrator\n *\n * Thin facade that composes four services:\n * PluginRegistry — registration + lookup\n * Initializer — two-phase startup (earlyInit / lateInit)\n * SearchAPI — all search + context logic\n * IndexAPI — code / git / docs indexing orchestration\n *\n * All heavy logic lives in those modules; BrainBank owns state,\n * guards (requireInit / initialize()), and public API shape.\n */\n\nimport { EventEmitter } from 'node:events';\nimport { resolveConfig } from '@/config/defaults.ts';\nimport { Database } from '@/db/database.ts';\nimport { HNSWIndex } from '@/providers/vector/hnsw-index.ts';\nimport { Collection } from '@/domain/collection.ts';\nimport { PluginRegistry } from '@/bootstrap/registry.ts';\nimport { Initializer } from '@/bootstrap/initializer.ts';\nimport { SearchAPI } from '@/api/search-api.ts';\nimport { IndexAPI } from '@/api/index-api.ts';\nimport { reembedAll } from '@/services/reembed.ts';\nimport { createWatcher, type WatchOptions, type Watcher } from '@/services/watch.ts';\nimport type { ReembedResult, ReembedOptions } from '@/services/reembed.ts';\nimport type { Plugin, CollectionPlugin } from '@/indexers/base.ts';\nimport { isCollectionPlugin } from '@/indexers/base.ts';\nimport type {\n BrainBankConfig, ResolvedConfig, EmbeddingProvider,\n IndexResult, IndexStats, SearchResult,\n ContextOptions, CoEditSuggestion, ProgressCallback, StageProgressCallback,\n DocumentCollection,\n} from '@/types.ts';\n\nexport class BrainBank extends EventEmitter {\n // ── State ───────────────────────────────────────\n private _config: ResolvedConfig;\n private _db!: Database;\n private _embedding!: EmbeddingProvider;\n private _registry = new PluginRegistry();\n private _searchAPI?: SearchAPI;\n private _indexAPI?: IndexAPI;\n private _initialized = false;\n private _initPromise: Promise<void> | null = null;\n private _watcher?: Watcher;\n\n // Collections (KV store)\n private _collections = new Map<string, Collection>();\n private _kvHnsw?: HNSWIndex;\n private _kvVecs = new Map<number, Float32Array>();\n\n // Shared HNSW pool — code:frontend + code:backend share one index\n private _sharedHnsw = new Map<string, { hnsw: HNSWIndex; vecCache: Map<number, Float32Array> }>();\n\n constructor(config: BrainBankConfig = {}) {\n super();\n this._config = resolveConfig(config);\n }\n\n // ── Plugin registration ──────────────────────────\n\n /**\n * Register a plugin. Chainable.\n *\n * brain.use(code({ repoPath: '.' })).use(docs());\n */\n use(plugin: Plugin): this {\n if (this._initialized)\n throw new Error(`BrainBank: Cannot add plugin '${plugin.name}' after initialization. Call .use() before any operations.`);\n this._registry.register(plugin);\n return this;\n }\n\n /** Get the list of registered plugin names. */\n get plugins(): string[] { return this._registry.names; }\n\n /** Check if a plugin is loaded. Also matches type prefix (e.g. 'code' matches 'code:frontend'). */\n has(name: string): boolean { return this._registry.has(name); }\n\n /** Get a plugin instance. Throws if not loaded. */\n plugin<T extends Plugin = Plugin>(n: string): T { return this._registry.get<T>(n); }\n\n // ── Initialization ───────────────────────────────\n\n /**\n * Initialize database, HNSW indices, and load existing vectors.\n * Only initializes registered modules.\n * Automatically called by index/search methods if not yet initialized.\n */\n async initialize(options: { force?: boolean } = {}): Promise<void> {\n if (this._initialized) return;\n if (this._initPromise) return this._initPromise;\n\n this._initPromise = this._runInitialize(options)\n .catch(err => {\n // Reset shared state so a retry starts clean\n for (const { hnsw } of this._sharedHnsw.values()) try { hnsw.reinit(); } catch {}\n this._kvVecs.clear();\n if (this._kvHnsw) try { this._kvHnsw.reinit(); } catch {}\n try { this._db?.close(); } catch {}\n this._db = undefined!;\n this._kvHnsw = undefined!;\n this._searchAPI = undefined;\n this._indexAPI = undefined;\n throw err;\n })\n .finally(() => { this._initPromise = null; });\n\n return this._initPromise;\n }\n\n private async _runInitialize(options: { force?: boolean } = {}): Promise<void> {\n if (this._initialized) return;\n\n const initializer = new Initializer(this._config, (e, d) => this.emit(e, d));\n\n // Phase 1: set this._kvHnsw BEFORE phase 2 so collection() works\n // when indexers call ctx.collection() during their initialize()\n const early = await initializer.early(options);\n this._db = early.db;\n this._embedding = early.embedding;\n this._kvHnsw = early.kvHnsw;\n\n // Phase 2: load vectors, run indexers, build search services\n const late = await initializer.late(\n early,\n this._registry,\n this._sharedHnsw,\n this._kvVecs,\n (name) => this.collection(name),\n );\n\n this._searchAPI = new SearchAPI({\n ...late,\n registry: this._registry,\n config: this._config,\n getDocsPlugin: () => {\n const docs = this._registry.get('docs');\n return docs && isCollectionPlugin(docs) ? docs : undefined;\n },\n collection: (n) => this.collection(n),\n });\n\n this._indexAPI = new IndexAPI({\n registry: this._registry,\n gitDepth: this._config.gitDepth,\n emit: (e, d) => this.emit(e, d),\n });\n\n this._initialized = true;\n this.emit('initialized', { plugins: this.plugins });\n }\n\n // ── Collections (KV) ────────────────────────────\n\n /**\n * Get or create a dynamic collection.\n * Collections are the universal data primitive — store anything, search semantically.\n *\n * const errors = brain.collection('debug_errors');\n * await errors.add('Fixed null check', { file: 'api.ts' });\n * const hits = await errors.search('null pointer');\n */\n collection(name: string): Collection {\n if (this._collections.has(name)) return this._collections.get(name)!;\n if (!this._kvHnsw)\n throw new Error('BrainBank: Collections not ready. Call await brain.initialize() first.');\n const coll = new Collection(name, this._db, this._embedding, this._kvHnsw, this._kvVecs, this._config.reranker);\n this._collections.set(name, coll);\n return coll;\n }\n\n /** List all collection names that have data. */\n listCollectionNames(): string[] {\n this._requireInit('listCollectionNames');\n return (this._db.prepare('SELECT DISTINCT collection FROM kv_data ORDER BY collection').all() as any[])\n .map(r => r.collection);\n }\n\n /** Delete a collection's data and evict from cache. */\n deleteCollection(name: string): void {\n this._requireInit('deleteCollection');\n this._db.prepare('DELETE FROM kv_data WHERE collection = ?').run(name);\n this._collections.delete(name);\n }\n\n // ── Indexing (delegated to IndexAPI) ─────────────\n\n async index(options: {\n modules?: ('code' | 'git' | 'docs')[];\n gitDepth?: number; forceReindex?: boolean; onProgress?: StageProgressCallback;\n } = {}): Promise<{ code?: IndexResult; git?: IndexResult; docs?: Record<string, { indexed: number; skipped: number; chunks: number }> }> {\n await this.initialize();\n return this._indexAPI!.index(options);\n }\n\n /** Index only code files (all repos in multi-repo mode). */\n async indexCode(options: { forceReindex?: boolean; onProgress?: ProgressCallback } = {}): Promise<IndexResult> {\n await this.initialize();\n return this._indexAPI!.indexCode(options);\n }\n\n /** Index only git history (all repos in multi-repo mode). */\n async indexGit(options: { depth?: number; onProgress?: ProgressCallback } = {}): Promise<IndexResult> {\n await this.initialize();\n return this._indexAPI!.indexGit(options);\n }\n\n // ── Document collections ─────────────────────────\n\n /** Register a document collection. */\n async addCollection(collection: DocumentCollection): Promise<void> {\n await this.initialize();\n this._docsPlugin('addCollection').addCollection(collection);\n }\n\n /** Remove a collection and all its indexed data. */\n async removeCollection(name: string): Promise<void> {\n await this.initialize();\n this._docsPlugin('removeCollection').removeCollection(name);\n }\n\n /** List all registered collections. */\n listCollections(): DocumentCollection[] {\n return this._docsPlugin('listCollections').listCollections();\n }\n\n /** Index all (or specific) document collections. */\n async indexDocs(options: {\n collections?: string[];\n onProgress?: (collection: string, file: string, current: number, total: number) => void;\n } = {}): Promise<Record<string, { indexed: number; skipped: number; chunks: number }>> {\n await this.initialize();\n const results = await this._docsPlugin('indexDocs').indexCollections(options);\n this.emit('docsIndexed', results);\n return results;\n }\n\n /** Search documents only. */\n async searchDocs(query: string, options?: { collection?: string; k?: number; minScore?: number }): Promise<SearchResult[]> {\n await this.initialize();\n if (!this.has('docs')) return [];\n return this._docsPlugin('searchDocs').search(query, options);\n }\n\n // ── Context metadata ─────────────────────────────\n\n /** Add context description for a collection path. */\n addContext(collection: string, path: string, context: string): void {\n const docs = this._docsPlugin('addContext');\n if (docs.addContext) docs.addContext(collection, path, context);\n }\n\n /** Remove context for a collection path. */\n removeContext(collection: string, path: string): void {\n const docs = this._docsPlugin('removeContext');\n if (docs.removeContext) docs.removeContext(collection, path);\n }\n\n /** List all context entries. */\n listContexts(): { collection: string; path: string; context: string }[] {\n const docs = this._docsPlugin('listContexts');\n return docs.listContexts?.() ?? [];\n }\n\n // ── Search (delegated to SearchAPI) ─────────────\n\n /**\n * Get formatted context for a task.\n * Returns markdown ready for system prompt injection.\n */\n async getContext(task: string, options: ContextOptions = {}): Promise<string> {\n await this.initialize();\n return this._searchAPI!.getContext(task, options);\n }\n\n /** Semantic search across all loaded modules. */\n async search(query: string, options?: {\n codeK?: number; gitK?: number; patternK?: number;\n minScore?: number; useMMR?: boolean;\n }): Promise<SearchResult[]> {\n await this.initialize();\n return this._searchAPI!.search(query, options);\n }\n\n /** Semantic search over code only. */\n async searchCode(query: string, k = 8): Promise<SearchResult[]> {\n await this.initialize();\n return this._searchAPI!.searchCode(query, k);\n }\n\n /** Semantic search over commits only. */\n async searchCommits(query: string, k = 8): Promise<SearchResult[]> {\n await this.initialize();\n return this._searchAPI!.searchCommits(query, k);\n }\n\n /**\n * Hybrid search: vector + BM25 fused with Reciprocal Rank Fusion.\n * Best quality — catches both exact keyword matches and conceptual similarities.\n */\n async hybridSearch(query: string, options?: {\n codeK?: number; gitK?: number; patternK?: number;\n minScore?: number; useMMR?: boolean;\n collections?: Record<string, number>;\n }): Promise<SearchResult[]> {\n await this.initialize();\n return this._searchAPI!.hybridSearch(query, options);\n }\n\n /** BM25 keyword search only (no embeddings needed). */\n async searchBM25(query: string, options?: { codeK?: number; gitK?: number; patternK?: number }): Promise<SearchResult[]> {\n this._requireInit('searchBM25');\n return this._searchAPI!.searchBM25(query, options);\n }\n\n /** Rebuild FTS5 indices. */\n rebuildFTS(): void {\n this._requireInit('rebuildFTS');\n this._searchAPI!.rebuildFTS();\n }\n\n // ── Queries ──────────────────────────────────────\n\n /** Get git history for a specific file. */\n async fileHistory(filePath: string, limit = 20): Promise<Record<string, unknown>[]> {\n await this.initialize();\n const gitPlugin = this.plugin('git') as Plugin & { fileHistory(f: string, l: number): Record<string, unknown>[] };\n return gitPlugin.fileHistory(filePath, limit);\n }\n\n /** Get co-edit suggestions for a file. */\n coEdits(filePath: string, limit = 5): CoEditSuggestion[] {\n this._requireInit('coEdits');\n const gitPlugin = this.plugin('git') as Plugin & { suggestCoEdits(f: string, l: number): CoEditSuggestion[] };\n return gitPlugin.suggestCoEdits(filePath, limit);\n }\n\n // ── Stats ────────────────────────────────────────\n\n /** Get statistics for all loaded modules. */\n stats(): IndexStats {\n this._requireInit('stats');\n const result: IndexStats = {};\n\n if (this.has('code')) {\n result.code = this._registry.firstByType('code')!.stats!() as IndexStats['code'];\n }\n if (this.has('git')) {\n result.git = this._registry.firstByType('git')!.stats!() as IndexStats['git'];\n }\n if (this.has('docs')) {\n result.documents = this._registry.firstByType('docs')!.stats!() as IndexStats['documents'];\n }\n\n return result;\n }\n\n // ── Watch ────────────────────────────────────────\n\n /**\n * Start watching for file changes and auto-re-index.\n * Works with built-in and custom indexers.\n */\n watch(options: WatchOptions = {}): Watcher {\n this._requireInit('watch');\n this._watcher?.close();\n this._watcher = createWatcher(\n async () => { await this.index(); },\n this._registry.raw,\n this._config.repoPath,\n options,\n );\n return this._watcher;\n }\n\n // ── Re-embed ─────────────────────────────────────\n\n /**\n * Re-embed all existing text with the current embedding provider.\n * Use this when switching providers (e.g. Local → OpenAI).\n */\n async reembed(options: ReembedOptions = {}): Promise<ReembedResult> {\n this._requireInit('reembed');\n\n const hnswMap = new Map<string, { hnsw: HNSWIndex; vecs: Map<number, Float32Array> }>();\n\n if (this._kvHnsw) hnswMap.set('kv', { hnsw: this._kvHnsw, vecs: this._kvVecs });\n\n for (const [type, shared] of this._sharedHnsw) {\n hnswMap.set(type, { hnsw: shared.hnsw, vecs: shared.vecCache });\n }\n\n for (const type of ['memory', 'notes', 'docs'] as const) {\n const mod = this._registry.firstByType(type) as any;\n if (mod?.hnsw) hnswMap.set(type, { hnsw: mod.hnsw, vecs: mod.vecCache });\n }\n\n const result = await reembedAll(this._db, this._embedding, hnswMap, options);\n this.emit('reembedded', result);\n return result;\n }\n\n // ── Lifecycle ────────────────────────────────────\n\n /** Close database and release resources. */\n close(): void {\n this._watcher?.close();\n for (const indexer of this._registry.all) indexer.close?.();\n this._embedding?.close().catch(() => {});\n this._db?.close();\n this._initialized = false;\n this._collections.clear();\n this._sharedHnsw.clear();\n this._kvVecs.clear();\n this._kvHnsw = undefined!;\n this._searchAPI = undefined;\n this._indexAPI = undefined;\n this._registry.clear();\n }\n\n /** Whether the brainbank has been initialized. */\n get isInitialized(): boolean { return this._initialized; }\n\n /** The resolved configuration. */\n get config(): Readonly<ResolvedConfig> { return this._config; }\n\n // ── Internal guard ───────────────────────────────\n\n private _requireInit(method: string): void {\n if (!this._initialized)\n throw new Error(`BrainBank: Not initialized. Call await brain.initialize() before ${method}().`);\n }\n\n /** Get the docs indexer as CollectionPlugin with init + type check. */\n private _docsPlugin(method: string): CollectionPlugin {\n this._requireInit(method);\n const docs = this._registry.get('docs');\n if (!docs || !isCollectionPlugin(docs))\n throw new Error(`BrainBank: Docs indexer not loaded. Add .use(docs()) before calling ${method}().`);\n return docs;\n }\n}\n","/**\n * BrainBank — Plugin Registry\n *\n * Manages registration and lookup of plugins.\n * Extracted from BrainBank so the facade stays focused on orchestration.\n *\n * Responsibilities:\n * - Store plugins by name\n * - Type-prefix matching ('code' finds 'code:frontend', 'code:backend')\n * - Alias resolution (currently none; add here if needed)\n * - Consistent error messages on missing plugins\n */\n\nimport type { Plugin } from '@/indexers/base.ts';\n\n/** Shorthand aliases that map public names to canonical plugin names. */\nconst ALIASES: Readonly<Record<string, string>> = {\n};\n\nexport class PluginRegistry {\n private _map = new Map<string, Plugin>();\n\n // ── Registration ────────────────────────────────\n\n /** Store a plugin. Duplicate names silently overwrite. */\n register(plugin: Plugin): void {\n this._map.set(plugin.name, plugin);\n }\n\n // ── Lookup ──────────────────────────────────────\n\n /**\n * Check whether a plugin is registered.\n * Supports type-prefix matching: `has('code')` returns true if\n * 'code', 'code:frontend', or 'code:backend' is registered.\n */\n has(name: string): boolean {\n if (this._map.has(name)) return true;\n for (const key of this._map.keys()) {\n if (key.startsWith(name + ':')) return true;\n }\n return false;\n }\n\n /**\n * Get a plugin by name. Throws a descriptive error if not found.\n *\n * Resolution order:\n * 1. Alias map (currently empty)\n * 2. Exact match\n * 3. First type-prefix match ('code' → 'code:frontend')\n */\n get<T extends Plugin = Plugin>(name: string): T {\n const resolved = ALIASES[name] ?? name;\n\n const exact = this._map.get(resolved);\n if (exact) return exact as T;\n\n const prefixed = this.firstByType(name);\n if (prefixed) return prefixed as T;\n\n throw new Error(\n `BrainBank: Plugin '${name}' is not loaded. ` +\n `Add .use(${name}()) to your BrainBank instance.`,\n );\n }\n\n /**\n * Return every plugin whose name equals `type` or starts with `type + ':'`.\n * Example: allByType('code') → [code, code:frontend, code:backend]\n */\n allByType(type: string): Plugin[] {\n return [...this._map.values()].filter(\n m => m.name === type || m.name.startsWith(type + ':'),\n );\n }\n\n /** Return the first plugin that matches the type prefix, or undefined. */\n firstByType(type: string): Plugin | undefined {\n for (const m of this._map.values()) {\n if (m.name === type || m.name.startsWith(type + ':')) return m;\n }\n return undefined;\n }\n\n // ── Accessors ───────────────────────────────────\n\n /** All registered plugin names (insertion order). */\n get names(): string[] {\n return [...this._map.keys()];\n }\n\n /** All registered plugin instances (insertion order). */\n get all(): Plugin[] {\n return [...this._map.values()];\n }\n\n /**\n * Underlying Map.\n * Prefer `all`, `allByType`, or `firstByType` everywhere else.\n */\n get raw(): Map<string, Plugin> {\n return this._map;\n }\n\n // ── Lifecycle ───────────────────────────────────\n\n /** Remove all registered plugins. Called by BrainBank.close(). */\n clear(): void {\n this._map.clear();\n }\n}\n\n","/**\n * BrainBank — Initializer\n *\n * Two-phase initialization keeps the dependency ordering correct:\n * Phase 1 (early) — db, embedding, kvHnsw. Must be assigned to\n * `this` on BrainBank before phase 2, so that collection()\n * works when indexers call ctx.collection() during initialize().\n * Phase 2 (late) — loads vectors, runs indexers, builds search.\n */\n\nimport { dirname, join } from 'node:path';\nimport { Database } from '@/db/database.ts';\nimport { HNSWIndex } from '@/providers/vector/hnsw-index.ts';\nimport { resolveEmbedding } from '@/providers/embeddings/resolve.ts';\nimport { VectorSearch } from '@/search/vector/vector-search.ts';\nimport { KeywordSearch } from '@/search/keyword/keyword-search.ts';\nimport { ContextBuilder } from '@/search/context-builder.ts';\nimport { setEmbeddingMeta, getEmbeddingMeta, detectProviderMismatch } from '@/services/embedding-meta.ts';\nimport type { PluginRegistry } from './registry.ts';\nimport type { Collection } from '@/domain/collection.ts';\nimport type { ResolvedConfig, EmbeddingProvider } from '@/types.ts';\nimport type { PluginContext } from '@/indexers/base.ts';\n\n// ── Result types ─────────────────────────────────────\n\n/** Available after phase 1 — before indexers run. */\nexport interface EarlyInit {\n db: Database;\n embedding: EmbeddingProvider;\n kvHnsw: HNSWIndex;\n /** True when force-init with mismatched dims — vectors skipped until reembed. */\n skipVectorLoad: boolean;\n}\n\n/** Available after phase 2 — once indexers have initialized. */\nexport interface LateInit {\n search?: VectorSearch;\n bm25?: KeywordSearch;\n contextBuilder?: ContextBuilder;\n}\n\n// ── Initializer class ────────────────────────────────\n\n/** Encapsulates BrainBank's two-phase initialization sequence. */\nexport class Initializer {\n private readonly _config: ResolvedConfig;\n private readonly _emit: (event: string, data: unknown) => void;\n\n constructor(config: ResolvedConfig, emit: (event: string, data: unknown) => void) {\n this._config = config;\n this._emit = emit;\n }\n\n /** Phase 1: database, embedding provider, KV HNSW index. */\n async early(options: { force?: boolean } = {}): Promise<EarlyInit> {\n const { _config: config } = this;\n const db = new Database(config.dbPath);\n const embedding = await this._resolveEmbedding(db);\n\n const mismatch = detectProviderMismatch(db, embedding);\n\n if (mismatch?.mismatch && !options.force) {\n db.close();\n throw new Error(\n `BrainBank: Embedding dimension mismatch (stored: ${mismatch.stored}, current: ${mismatch.current}). ` +\n `Run brain.reembed() to re-index with the new provider, or switch back to the original provider.`\n );\n }\n\n setEmbeddingMeta(db, embedding);\n\n const kvHnsw = new HNSWIndex(\n config.embeddingDims,\n config.maxElements ?? 500_000,\n config.hnswM,\n config.hnswEfConstruction,\n config.hnswEfSearch,\n );\n await kvHnsw.init();\n\n const skipVectorLoad = !!(options.force && mismatch?.mismatch);\n\n return { db, embedding, kvHnsw, skipVectorLoad };\n }\n\n /** Phase 2: load vectors, run indexers, build search layer. */\n async late(\n earlyResult: EarlyInit,\n registry: PluginRegistry,\n sharedHnsw: Map<string, { hnsw: HNSWIndex; vecCache: Map<number, Float32Array> }>,\n kvVecs: Map<number, Float32Array>,\n getCollection: (name: string) => Collection,\n ): Promise<LateInit> {\n const { _config: config } = this;\n const { db, embedding, kvHnsw, skipVectorLoad } = earlyResult;\n\n if (!skipVectorLoad) {\n const kvIndexPath = hnswPath(config.dbPath, 'kv');\n const kvCount = countRows(db, 'kv_vectors');\n if (kvHnsw.tryLoad(kvIndexPath, kvCount)) {\n loadVecCache(db, 'kv_vectors', 'data_id', kvVecs);\n } else {\n loadVectors(db, 'kv_vectors', 'data_id', kvHnsw, kvVecs);\n }\n }\n\n const ctx = this._buildPluginContext(db, embedding, sharedHnsw, skipVectorLoad, getCollection);\n\n for (const mod of registry.all) {\n await mod.initialize(ctx);\n }\n\n saveAllHnsw(config.dbPath, kvHnsw, sharedHnsw);\n\n return this._buildSearchLayer(db, embedding, registry, sharedHnsw);\n }\n\n /** Build the PluginContext passed to each plugin's initialize(). */\n private _buildPluginContext(\n db: Database,\n embedding: EmbeddingProvider,\n sharedHnsw: Map<string, { hnsw: HNSWIndex; vecCache: Map<number, Float32Array> }>,\n skipVectorLoad: boolean,\n getCollection: (name: string) => Collection,\n ): PluginContext {\n const { _config: config } = this;\n return {\n db,\n embedding,\n config,\n\n createHnsw: (maxElements?: number) =>\n new HNSWIndex(\n config.embeddingDims,\n maxElements ?? config.maxElements,\n config.hnswM,\n config.hnswEfConstruction,\n config.hnswEfSearch,\n ).init(),\n\n loadVectors: (table, idCol, hnsw, cache) => {\n if (skipVectorLoad) return;\n const indexName = table.replace('_vectors', '').replace('_chunks', '');\n const indexPath = hnswPath(config.dbPath, indexName);\n const rowCount = countRows(db, table);\n if (hnsw.tryLoad(indexPath, rowCount)) {\n loadVecCache(db, table, idCol, cache);\n } else {\n loadVectors(db, table, idCol, hnsw, cache);\n }\n },\n\n getOrCreateSharedHnsw: async (type, maxElements) => {\n const existing = sharedHnsw.get(type);\n if (existing) return { ...existing, isNew: false };\n\n const hnsw = await new HNSWIndex(\n config.embeddingDims,\n maxElements ?? config.maxElements,\n config.hnswM,\n config.hnswEfConstruction,\n config.hnswEfSearch,\n ).init();\n\n const vecCache = new Map<number, Float32Array>();\n sharedHnsw.set(type, { hnsw, vecCache });\n return { hnsw, vecCache, isNew: true };\n },\n\n collection: getCollection,\n };\n }\n\n /** Build VectorSearch + KeywordSearch + ContextBuilder from initialized plugins. */\n private _buildSearchLayer(\n db: Database,\n embedding: EmbeddingProvider,\n registry: PluginRegistry,\n sharedHnsw: Map<string, { hnsw: HNSWIndex; vecCache: Map<number, Float32Array> }>,\n ): LateInit {\n const { _config: config } = this;\n const codeMod = sharedHnsw.get('code');\n const gitMod = sharedHnsw.get('git');\n const memMod = registry.firstByType('memory') as any;\n\n if (!codeMod && !gitMod && !memMod) return {};\n\n const search = new VectorSearch({\n db,\n codeHnsw: codeMod?.hnsw,\n gitHnsw: gitMod?.hnsw,\n patternHnsw: memMod?.hnsw,\n codeVecs: codeMod?.vecCache ?? new Map(),\n gitVecs: gitMod?.vecCache ?? new Map(),\n patternVecs: memMod?.vecCache ?? new Map(),\n embedding,\n reranker: config.reranker,\n });\n const bm25 = new KeywordSearch(db);\n\n const firstGit = registry.firstByType('git') as any;\n const contextBuilder = new ContextBuilder(search, firstGit?.coEdits);\n\n return { search, bm25, contextBuilder };\n }\n\n /** Resolve embedding: explicit config > stored DB key > local default. */\n private async _resolveEmbedding(db: Database): Promise<EmbeddingProvider> {\n if (this._config.embeddingProvider) return this._config.embeddingProvider;\n const meta = getEmbeddingMeta(db);\n if (meta?.providerKey && meta.providerKey !== 'local') {\n this._emit('progress', `Embedding: auto-resolved '${meta.providerKey}' from DB`);\n return resolveEmbedding(meta.providerKey);\n }\n return resolveEmbedding('local');\n }\n}\n\n// ── Shared helpers ────────────────────────────────────\n\n/** Derive the HNSW index file path from the DB path. */\nfunction hnswPath(dbPath: string, name: string): string {\n return join(dirname(dbPath), `hnsw-${name}.index`);\n}\n\n/** Count rows in a vector table (fast, no data transfer). */\nfunction countRows(db: Database, table: string): number {\n const row = db.prepare(`SELECT COUNT(*) as c FROM ${table}`).get() as any;\n return row?.c ?? 0;\n}\n\n/** Save all HNSW indexes to disk for fast startup next time. */\nfunction saveAllHnsw(\n dbPath: string,\n kvHnsw: HNSWIndex,\n sharedHnsw: Map<string, { hnsw: HNSWIndex; vecCache: Map<number, Float32Array> }>,\n): void {\n try {\n kvHnsw.save(hnswPath(dbPath, 'kv'));\n for (const [name, { hnsw }] of sharedHnsw) {\n hnsw.save(hnswPath(dbPath, name));\n }\n } catch {\n // Non-fatal: next startup will just rebuild from SQLite\n }\n}\n\nfunction loadVectors(\n db: Database,\n table: string,\n idCol: string,\n hnsw: HNSWIndex,\n cache: Map<number, Float32Array>,\n): void {\n const iter = db.prepare(`SELECT ${idCol}, embedding FROM ${table}`).iterate() as IterableIterator<any>;\n for (const row of iter) {\n const vec = new Float32Array(\n row.embedding.buffer.slice(\n row.embedding.byteOffset,\n row.embedding.byteOffset + row.embedding.byteLength,\n ),\n );\n hnsw.add(vec, row[idCol]);\n cache.set(row[idCol], vec);\n }\n}\n\n/** Populate only the vecCache from SQLite (HNSW already loaded from file). */\nfunction loadVecCache(\n db: Database,\n table: string,\n idCol: string,\n cache: Map<number, Float32Array>,\n): void {\n const iter = db.prepare(`SELECT ${idCol}, embedding FROM ${table}`).iterate() as IterableIterator<any>;\n for (const row of iter) {\n const vec = new Float32Array(\n row.embedding.buffer.slice(\n row.embedding.byteOffset,\n row.embedding.byteOffset + row.embedding.byteLength,\n ),\n );\n cache.set(row[idCol], vec);\n }\n}\n","/**\n * BrainBank — Database\n * \n * Thin wrapper over better-sqlite3.\n * Handles WAL mode, directory creation, schema init, and transactions.\n */\n\nimport BetterSqlite3 from 'better-sqlite3';\nimport * as fs from 'node:fs';\nimport * as path from 'node:path';\nimport { createSchema } from './schema.ts';\n\nexport class Database {\n readonly db: BetterSqlite3.Database;\n\n constructor(dbPath: string) {\n // Ensure parent directory exists\n const dir = path.dirname(dbPath);\n if (!fs.existsSync(dir)) {\n fs.mkdirSync(dir, { recursive: true });\n }\n\n this.db = new BetterSqlite3(dbPath);\n this.db.pragma('journal_mode = WAL');\n this.db.pragma('busy_timeout = 5000');\n this.db.pragma('synchronous = NORMAL');\n this.db.pragma('foreign_keys = ON');\n\n // Initialize schema\n createSchema(this.db);\n }\n\n /**\n * Run a function inside a transaction.\n * Auto-commits on success, auto-rollbacks on error.\n */\n transaction<T>(fn: () => T): T {\n const tx = this.db.transaction(fn);\n return tx();\n }\n\n /**\n * Run a prepared statement on multiple rows.\n * Wraps in a single transaction for performance.\n */\n batch<T extends any[]>(sql: string, rows: T[]): void {\n const stmt = this.db.prepare(sql);\n const tx = this.db.transaction(() => {\n for (const row of rows) {\n stmt.run(...row);\n }\n });\n tx();\n }\n\n /** Prepare a reusable statement. */\n prepare(sql: string): BetterSqlite3.Statement {\n return this.db.prepare(sql);\n }\n\n /** Execute raw SQL (no results). */\n exec(sql: string): void {\n this.db.exec(sql);\n }\n\n /** Close the database. */\n close(): void {\n this.db.close();\n }\n}\n","/**\n * BrainBank — SQLite Schema\n * \n * Idempotent schema creation for the knowledge database.\n * Uses better-sqlite3 directly.\n */\n\nimport type Database from 'better-sqlite3';\n\nexport const SCHEMA_VERSION = 4;\n\n/**\n * Create all tables and indices.\n * Safe to call multiple times — uses IF NOT EXISTS.\n */\nexport function createSchema(db: Database.Database): void {\n db.exec(`\n -- ── Schema versioning ──────────────────────────\n CREATE TABLE IF NOT EXISTS schema_version (\n version INTEGER PRIMARY KEY,\n applied_at INTEGER NOT NULL DEFAULT (unixepoch())\n );\n INSERT OR IGNORE INTO schema_version (version) VALUES (${SCHEMA_VERSION});\n\n -- ── Code chunks ────────────────────────────────\n CREATE TABLE IF NOT EXISTS code_chunks (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n file_path TEXT NOT NULL,\n chunk_type TEXT NOT NULL,\n name TEXT,\n start_line INTEGER NOT NULL,\n end_line INTEGER NOT NULL,\n content TEXT NOT NULL,\n language TEXT NOT NULL,\n file_hash TEXT,\n indexed_at INTEGER NOT NULL DEFAULT (unixepoch())\n );\n\n CREATE TABLE IF NOT EXISTS code_vectors (\n chunk_id INTEGER PRIMARY KEY REFERENCES code_chunks(id) ON DELETE CASCADE,\n embedding BLOB NOT NULL\n );\n\n CREATE TABLE IF NOT EXISTS indexed_files (\n file_path TEXT PRIMARY KEY,\n file_hash TEXT NOT NULL,\n indexed_at INTEGER NOT NULL DEFAULT (unixepoch())\n );\n\n -- ── Git history ────────────────────────────────\n CREATE TABLE IF NOT EXISTS git_commits (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n hash TEXT UNIQUE NOT NULL,\n short_hash TEXT NOT NULL,\n message TEXT NOT NULL,\n author TEXT NOT NULL,\n date TEXT NOT NULL,\n timestamp INTEGER NOT NULL,\n files_json TEXT NOT NULL,\n diff TEXT,\n additions INTEGER DEFAULT 0,\n deletions INTEGER DEFAULT 0,\n is_merge INTEGER DEFAULT 0\n );\n\n CREATE TABLE IF NOT EXISTS commit_files (\n commit_id INTEGER NOT NULL REFERENCES git_commits(id),\n file_path TEXT NOT NULL\n );\n\n CREATE TABLE IF NOT EXISTS co_edits (\n file_a TEXT NOT NULL,\n file_b TEXT NOT NULL,\n count INTEGER NOT NULL DEFAULT 1,\n PRIMARY KEY (file_a, file_b)\n );\n\n CREATE TABLE IF NOT EXISTS git_vectors (\n commit_id INTEGER PRIMARY KEY REFERENCES git_commits(id) ON DELETE CASCADE,\n embedding BLOB NOT NULL\n );\n\n -- ── Agent memory ───────────────────────────────\n CREATE TABLE IF NOT EXISTS memory_patterns (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n task_type TEXT NOT NULL,\n task TEXT NOT NULL,\n approach TEXT NOT NULL,\n outcome TEXT,\n success_rate REAL NOT NULL DEFAULT 0.5,\n critique TEXT,\n tokens_used INTEGER,\n latency_ms INTEGER,\n created_at INTEGER NOT NULL DEFAULT (unixepoch())\n );\n\n CREATE TABLE IF NOT EXISTS memory_vectors (\n pattern_id INTEGER PRIMARY KEY REFERENCES memory_patterns(id) ON DELETE CASCADE,\n embedding BLOB NOT NULL\n );\n\n CREATE TABLE IF NOT EXISTS distilled_strategies (\n task_type TEXT PRIMARY KEY,\n strategy TEXT NOT NULL,\n confidence REAL NOT NULL DEFAULT 0.8,\n updated_at INTEGER NOT NULL DEFAULT (unixepoch())\n );\n\n -- ── Indices ────────────────────────────────────\n CREATE INDEX IF NOT EXISTS idx_cc_file ON code_chunks(file_path);\n CREATE INDEX IF NOT EXISTS idx_cf_path ON commit_files(file_path);\n CREATE INDEX IF NOT EXISTS idx_gc_ts ON git_commits(timestamp DESC);\n CREATE INDEX IF NOT EXISTS idx_gc_hash ON git_commits(hash);\n CREATE INDEX IF NOT EXISTS idx_mp_type ON memory_patterns(task_type);\n CREATE INDEX IF NOT EXISTS idx_mp_success ON memory_patterns(success_rate);\n CREATE INDEX IF NOT EXISTS idx_mp_created ON memory_patterns(created_at);\n\n -- ── FTS5 Full-Text Search ─────────────────────\n -- Code chunks: search by file path, name, and content\n CREATE VIRTUAL TABLE IF NOT EXISTS fts_code USING fts5(\n file_path,\n name,\n content,\n content='code_chunks',\n content_rowid='id',\n tokenize='porter unicode61'\n );\n\n -- Git commits: search by message, author, and diff\n CREATE VIRTUAL TABLE IF NOT EXISTS fts_commits USING fts5(\n message,\n author,\n diff,\n content='git_commits',\n content_rowid='id',\n tokenize='porter unicode61'\n );\n\n -- Memory patterns: search by task type, task, approach, and critique\n CREATE VIRTUAL TABLE IF NOT EXISTS fts_patterns USING fts5(\n task_type,\n task,\n approach,\n critique,\n content='memory_patterns',\n content_rowid='id',\n tokenize='porter unicode61'\n );\n\n -- ── FTS5 Sync Triggers ────────────────────────\n -- Auto-sync FTS indices on INSERT/UPDATE/DELETE\n\n CREATE TRIGGER IF NOT EXISTS trg_fts_code_insert AFTER INSERT ON code_chunks BEGIN\n INSERT INTO fts_code(rowid, file_path, name, content)\n VALUES (new.id, new.file_path, COALESCE(new.name, ''), new.content);\n END;\n CREATE TRIGGER IF NOT EXISTS trg_fts_code_delete AFTER DELETE ON code_chunks BEGIN\n INSERT INTO fts_code(fts_code, rowid, file_path, name, content)\n VALUES ('delete', old.id, old.file_path, COALESCE(old.name, ''), old.content);\n END;\n\n CREATE TRIGGER IF NOT EXISTS trg_fts_commits_insert AFTER INSERT ON git_commits BEGIN\n INSERT INTO fts_commits(rowid, message, author, diff)\n VALUES (new.id, new.message, new.author, COALESCE(new.diff, ''));\n END;\n CREATE TRIGGER IF NOT EXISTS trg_fts_commits_delete AFTER DELETE ON git_commits BEGIN\n INSERT INTO fts_commits(fts_commits, rowid, message, author, diff)\n VALUES ('delete', old.id, old.message, old.author, COALESCE(old.diff, ''));\n END;\n\n CREATE TRIGGER IF NOT EXISTS trg_fts_patterns_insert AFTER INSERT ON memory_patterns BEGIN\n INSERT INTO fts_patterns(rowid, task_type, task, approach, critique)\n VALUES (new.id, new.task_type, new.task, new.approach, COALESCE(new.critique, ''));\n END;\n CREATE TRIGGER IF NOT EXISTS trg_fts_patterns_delete AFTER DELETE ON memory_patterns BEGIN\n INSERT INTO fts_patterns(fts_patterns, rowid, task_type, task, approach, critique)\n VALUES ('delete', old.id, old.task_type, old.task, old.approach, COALESCE(old.critique, ''));\n END;\n\n -- ── Note Memory ───────────────────────────────\n CREATE TABLE IF NOT EXISTS note_memories (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n title TEXT NOT NULL,\n summary TEXT NOT NULL,\n decisions_json TEXT NOT NULL DEFAULT '[]',\n files_json TEXT NOT NULL DEFAULT '[]',\n patterns_json TEXT NOT NULL DEFAULT '[]',\n open_json TEXT NOT NULL DEFAULT '[]',\n tags_json TEXT NOT NULL DEFAULT '[]',\n tier TEXT NOT NULL DEFAULT 'short',\n created_at INTEGER NOT NULL DEFAULT (unixepoch())\n );\n\n CREATE TABLE IF NOT EXISTS note_vectors (\n note_id INTEGER PRIMARY KEY REFERENCES note_memories(id) ON DELETE CASCADE,\n embedding BLOB NOT NULL\n );\n\n CREATE VIRTUAL TABLE IF NOT EXISTS fts_notes USING fts5(\n title,\n summary,\n decisions,\n patterns,\n tags,\n content='note_memories',\n content_rowid='id',\n tokenize='porter unicode61'\n );\n\n CREATE TRIGGER IF NOT EXISTS trg_fts_notes_insert AFTER INSERT ON note_memories BEGIN\n INSERT INTO fts_notes(rowid, title, summary, decisions, patterns, tags)\n VALUES (new.id, new.title, new.summary, new.decisions_json, new.patterns_json, new.tags_json);\n END;\n CREATE TRIGGER IF NOT EXISTS trg_fts_notes_delete AFTER DELETE ON note_memories BEGIN\n INSERT INTO fts_notes(fts_notes, rowid, title, summary, decisions, patterns, tags)\n VALUES ('delete', old.id, old.title, old.summary, old.decisions_json, old.patterns_json, old.tags_json);\n END;\n\n CREATE INDEX IF NOT EXISTS idx_nm_tier ON note_memories(tier);\n CREATE INDEX IF NOT EXISTS idx_nm_created ON note_memories(created_at DESC);\n\n -- ── Document Collections ──────────────────────\n CREATE TABLE IF NOT EXISTS collections (\n name TEXT PRIMARY KEY,\n path TEXT NOT NULL,\n pattern TEXT NOT NULL DEFAULT '**/*.md',\n ignore_json TEXT NOT NULL DEFAULT '[]',\n context TEXT,\n created_at INTEGER NOT NULL DEFAULT (unixepoch())\n );\n\n CREATE TABLE IF NOT EXISTS doc_chunks (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n collection TEXT NOT NULL REFERENCES collections(name) ON DELETE CASCADE,\n file_path TEXT NOT NULL,\n title TEXT NOT NULL,\n content TEXT NOT NULL,\n seq INTEGER NOT NULL DEFAULT 0,\n pos INTEGER NOT NULL DEFAULT 0,\n content_hash TEXT NOT NULL,\n indexed_at INTEGER NOT NULL DEFAULT (unixepoch())\n );\n\n CREATE TABLE IF NOT EXISTS doc_vectors (\n chunk_id INTEGER PRIMARY KEY REFERENCES doc_chunks(id) ON DELETE CASCADE,\n embedding BLOB NOT NULL\n );\n\n CREATE INDEX IF NOT EXISTS idx_dc_collection ON doc_chunks(collection);\n CREATE INDEX IF NOT EXISTS idx_dc_file ON doc_chunks(file_path);\n CREATE INDEX IF NOT EXISTS idx_dc_hash ON doc_chunks(content_hash);\n\n CREATE VIRTUAL TABLE IF NOT EXISTS fts_docs USING fts5(\n title,\n content,\n file_path,\n collection,\n content='doc_chunks',\n content_rowid='id',\n tokenize='porter unicode61'\n );\n\n CREATE TRIGGER IF NOT EXISTS trg_fts_docs_insert AFTER INSERT ON doc_chunks BEGIN\n INSERT INTO fts_docs(rowid, title, content, file_path, collection)\n VALUES (new.id, new.title, new.content, new.file_path, new.collection);\n END;\n CREATE TRIGGER IF NOT EXISTS trg_fts_docs_delete AFTER DELETE ON doc_chunks BEGIN\n INSERT INTO fts_docs(fts_docs, rowid, title, content, file_path, collection)\n VALUES ('delete', old.id, old.title, old.content, old.file_path, old.collection);\n END;\n\n -- ── Path Contexts ─────────────────────────────\n CREATE TABLE IF NOT EXISTS path_contexts (\n collection TEXT NOT NULL,\n path TEXT NOT NULL,\n context TEXT NOT NULL,\n PRIMARY KEY (collection, path)\n );\n\n -- ── Dynamic Collections (KV Store) ───────────\n CREATE TABLE IF NOT EXISTS kv_data (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n collection TEXT NOT NULL,\n content TEXT NOT NULL,\n meta_json TEXT NOT NULL DEFAULT '{}',\n tags_json TEXT NOT NULL DEFAULT '[]',\n expires_at INTEGER,\n created_at INTEGER NOT NULL DEFAULT (unixepoch())\n );\n\n CREATE TABLE IF NOT EXISTS kv_vectors (\n data_id INTEGER PRIMARY KEY REFERENCES kv_data(id) ON DELETE CASCADE,\n embedding BLOB NOT NULL\n );\n\n CREATE VIRTUAL TABLE IF NOT EXISTS fts_kv USING fts5(\n content,\n collection,\n content='kv_data',\n content_rowid='id',\n tokenize='porter unicode61'\n );\n\n CREATE TRIGGER IF NOT EXISTS trg_fts_kv_insert AFTER INSERT ON kv_data BEGIN\n INSERT INTO fts_kv(rowid, content, collection)\n VALUES (new.id, new.content, new.collection);\n END;\n CREATE TRIGGER IF NOT EXISTS trg_fts_kv_delete AFTER DELETE ON kv_data BEGIN\n INSERT INTO fts_kv(fts_kv, rowid, content, collection)\n VALUES ('delete', old.id, old.content, old.collection);\n END;\n\n CREATE INDEX IF NOT EXISTS idx_kv_collection ON kv_data(collection);\n CREATE INDEX IF NOT EXISTS idx_kv_created ON kv_data(created_at DESC);\n\n -- ── Embedding Metadata ───────────────────────\n CREATE TABLE IF NOT EXISTS embedding_meta (\n key TEXT PRIMARY KEY,\n value TEXT NOT NULL\n );\n `);\n}\n\n/**\n * Get the current schema version from the database.\n */\nexport function getSchemaVersion(db: Database.Database): number {\n try {\n const row = db.prepare('SELECT MAX(version) as v FROM schema_version').get() as any;\n return row?.v ?? 0;\n } catch {\n return 0;\n }\n}\n","/**\n * BrainBank — Embedding Metadata\n *\n * Tracks which embedding provider is stored in the database.\n * Used at startup to detect dimension mismatches and during reembed\n * to update the metadata after re-embedding.\n */\n\nimport type { Database } from '@/db/database.ts';\nimport type { EmbeddingProvider } from '@/types.ts';\nimport { providerKey } from '@/providers/embeddings/resolve.ts';\n\n/** Stored embedding metadata shape. */\nexport interface EmbeddingMeta {\n provider: string;\n dims: number;\n /** Stable key for auto-resolving provider on startup (e.g. 'openai', 'local'). */\n providerKey: string;\n}\n\n/** Get stored embedding metadata. Returns null if not set. */\nexport function getEmbeddingMeta(db: Database): EmbeddingMeta | null {\n try {\n const provider = db.prepare(\n \"SELECT value FROM embedding_meta WHERE key = 'provider'\"\n ).get() as any;\n const dims = db.prepare(\n \"SELECT value FROM embedding_meta WHERE key = 'dims'\"\n ).get() as any;\n const key = db.prepare(\n \"SELECT value FROM embedding_meta WHERE key = 'provider_key'\"\n ).get() as any;\n\n if (!provider || !dims) return null;\n return {\n provider: provider.value,\n dims: Number(dims.value),\n providerKey: key?.value ?? 'local',\n };\n } catch {\n return null;\n }\n}\n\n/** Store current provider info. */\nexport function setEmbeddingMeta(db: Database, embedding: EmbeddingProvider): void {\n const upsert = db.prepare(\n 'INSERT OR REPLACE INTO embedding_meta (key, value) VALUES (?, ?)'\n );\n upsert.run('provider', embedding.constructor?.name ?? 'unknown');\n upsert.run('dims', String(embedding.dims));\n upsert.run('provider_key', providerKey(embedding));\n upsert.run('indexed_at', new Date().toISOString());\n}\n\n/** Check if the configured provider differs from what's stored. */\nexport function detectProviderMismatch(\n db: Database,\n embedding: EmbeddingProvider,\n): { mismatch: boolean; stored: string; current: string } | null {\n const meta = getEmbeddingMeta(db);\n if (!meta) return null; // First time, no mismatch\n\n const currentName = embedding.constructor?.name ?? 'unknown';\n const mismatch = meta.dims !== embedding.dims || meta.provider !== currentName;\n\n return {\n mismatch,\n stored: `${meta.provider}/${meta.dims}`,\n current: `${currentName}/${embedding.dims}`,\n };\n}\n","/**\n * BrainBank — Search API\n *\n * All search and context operations in one place.\n * Composed from VectorSearch, KeywordSearch, ContextBuilder, and KV collections.\n * Always created after initialization (even when search services are absent),\n * so BrainBank can unconditionally delegate to it.\n */\n\nimport type { SearchStrategy } from '@/search/types.ts';\nimport type { ContextBuilder } from '@/search/context-builder.ts';\nimport type { Collection } from '@/domain/collection.ts';\nimport type { PluginRegistry } from '@/bootstrap/registry.ts';\nimport type { CollectionPlugin } from '@/indexers/base.ts';\nimport type { ResolvedConfig, SearchResult, ContextOptions } from '@/types.ts';\nimport { reciprocalRankFusion } from '@/lib/rrf.ts';\nimport { rerank } from '@/search/vector/rerank.ts';\n\nexport interface SearchAPIDeps {\n search?: SearchStrategy;\n bm25?: SearchStrategy;\n contextBuilder?: ContextBuilder;\n registry: PluginRegistry;\n config: ResolvedConfig;\n getDocsPlugin(): CollectionPlugin | undefined;\n collection(name: string): Collection;\n}\n\nexport class SearchAPI {\n constructor(private _d: SearchAPIDeps) {}\n\n // ── Vector ──────────────────────────────────────\n\n async search(query: string, options?: {\n codeK?: number; gitK?: number; patternK?: number;\n minScore?: number; useMMR?: boolean;\n }): Promise<SearchResult[]> {\n if (!this._d.search) {\n return this._d.registry.has('docs')\n ? await this._searchDocs(query, { k: 8 })\n : [];\n }\n return this._d.search.search(query, options);\n }\n\n async searchCode(query: string, k = 8): Promise<SearchResult[]> {\n if (!this._d.registry.firstByType('code'))\n throw new Error(\"BrainBank: Indexer 'code' is not loaded. Add .use(code()) to your BrainBank instance.\");\n if (!this._d.search)\n throw new Error('BrainBank: MultiIndexSearch not available. Ensure code indexer is loaded.');\n return this._d.search.search(query, { codeK: k, gitK: 0, patternK: 0 });\n }\n\n async searchCommits(query: string, k = 8): Promise<SearchResult[]> {\n if (!this._d.registry.firstByType('git'))\n throw new Error(\"BrainBank: Indexer 'git' is not loaded. Add .use(git()) to your BrainBank instance.\");\n if (!this._d.search)\n throw new Error('BrainBank: MultiIndexSearch not available. Ensure git indexer is loaded.');\n return this._d.search.search(query, { codeK: 0, gitK: k, patternK: 0 });\n }\n\n // ── Hybrid ──────────────────────────────────────\n\n async hybridSearch(query: string, options?: {\n codeK?: number; gitK?: number; patternK?: number;\n minScore?: number; useMMR?: boolean;\n collections?: Record<string, number>;\n }): Promise<SearchResult[]> {\n const cols = options?.collections ?? {};\n const codeK = cols.code ?? options?.codeK ?? 6;\n const gitK = cols.git ?? options?.gitK ?? 5;\n const docsK = cols.docs ?? 8;\n\n const resultLists: SearchResult[][] = [];\n\n if (this._d.search) {\n const [vec, kw] = await Promise.all([\n this._d.search.search(query, { ...options, codeK, gitK }),\n Promise.resolve(this._d.bm25?.search(query, { codeK, gitK }) ?? []),\n ]);\n resultLists.push(vec, kw);\n }\n\n if (this._d.registry.has('docs')) {\n const docs = await this._searchDocs(query, { k: docsK });\n if (docs.length > 0) resultLists.push(docs);\n }\n\n await this._searchKvCollections(query, cols, resultLists);\n if (resultLists.length === 0) return [];\n\n const fused = reciprocalRankFusion(resultLists);\n return this._rerankResults(query, fused);\n }\n\n /** Search non-reserved KV collections and push results. */\n private async _searchKvCollections(\n query: string, cols: Record<string, number>, resultLists: SearchResult[][],\n ): Promise<void> {\n const reserved = new Set(['code', 'git', 'docs']);\n for (const [name, k] of Object.entries(cols)) {\n if (reserved.has(name)) continue;\n const hits = await this._d.collection(name).search(query, { k });\n if (hits.length > 0) {\n resultLists.push(hits.map(h => ({\n type: 'collection' as const,\n score: h.score ?? 0,\n content: h.content,\n metadata: { collection: name, id: h.id, ...h.metadata },\n })));\n }\n }\n }\n\n /** Apply reranking if a reranker is configured. */\n private async _rerankResults(query: string, fused: SearchResult[]): Promise<SearchResult[]> {\n if (!this._d.config.reranker || fused.length <= 1) return fused;\n return rerank(query, fused, this._d.config.reranker);\n }\n\n /** Search docs directly via the plugin — no circular callback. */\n private async _searchDocs(\n query: string, options?: { collection?: string; k?: number; minScore?: number },\n ): Promise<SearchResult[]> {\n const plugin = this._d.getDocsPlugin();\n if (!plugin) return [];\n return plugin.search(query, options);\n }\n\n // ── Keyword ─────────────────────────────────────\n\n async searchBM25(query: string, options?: { codeK?: number; gitK?: number; patternK?: number }): Promise<SearchResult[]> {\n return this._d.bm25?.search(query, options) ?? [];\n }\n\n rebuildFTS(): void { this._d.bm25?.rebuild?.(); }\n\n // ── Context ─────────────────────────────────────\n\n async getContext(task: string, options: ContextOptions = {}): Promise<string> {\n const sections: string[] = [];\n\n if (this._d.contextBuilder) {\n const core = await this._d.contextBuilder.build(task, options);\n if (core) sections.push(core);\n }\n\n if (this._d.registry.has('docs')) {\n const docs = await this._searchDocs(task, { k: options.codeResults ?? 4 });\n if (docs.length > 0) {\n const body = docs.map(r => {\n const m = r.metadata as Record<string, any>;\n const h = r.context\n ? `**[${m.collection}]** ${m.title} — _${r.context}_`\n : `**[${m.collection}]** ${m.title}`;\n return `${h}\\n\\n${r.content}`;\n }).join('\\n\\n---\\n\\n');\n sections.push(`## Relevant Documents\\n\\n${body}`);\n }\n }\n\n return sections.join('\\n\\n');\n }\n}\n","/**\n * BrainBank — Plugin System\n * \n * Plugins are pluggable strategies that scan external data sources\n * and push content into BrainBank. Built-in plugins handle code,\n * git, and docs. Third-party frameworks (LangChain, etc.)\n * can implement custom plugins.\n * \n * import { BrainBank } from 'brainbank';\n * import { code } from 'brainbank/indexers/code';\n * \n * const brain = new BrainBank()\n * .use(code({ repoPath: '.' }));\n */\n\nimport type { Database } from '@/db/database.ts';\nimport type { EmbeddingProvider, SearchResult } from '@/types.ts';\nimport type { HNSWIndex } from '@/providers/vector/hnsw-index.ts';\nimport type { ResolvedConfig, DocumentCollection } from '@/types.ts';\nimport type { Collection } from '@/domain/collection.ts';\n\n// ── Plugin Context ────────────────────────────────\n// Provided to each plugin during initialization.\n\nexport interface PluginContext {\n /** SQLite database (shared across all plugins). */\n db: Database;\n /** Embedding provider (shared). */\n embedding: EmbeddingProvider;\n /** Resolved BrainBank config. */\n config: ResolvedConfig;\n /** Create and initialize an HNSW index. */\n createHnsw(maxElements?: number): Promise<HNSWIndex>;\n /** Load existing vectors from a SQLite vectors table into an HNSW index + cache. */\n loadVectors(table: string, idCol: string, hnsw: HNSWIndex, cache: Map<number, Float32Array>): void;\n /** Get or create a shared HNSW index by type (e.g. 'code', 'git'). For multi-repo support. */\n getOrCreateSharedHnsw(type: string, maxElements?: number): Promise<{ hnsw: HNSWIndex; vecCache: Map<number, Float32Array>; isNew: boolean }>;\n /** Get or create a dynamic collection. */\n collection(name: string): Collection;\n}\n\n// ── Core Plugin Interface ─────────────────────────\n// Minimal contract: name + initialize. All capabilities are expressed\n// via composed interfaces below.\n\nexport interface Plugin {\n /** Unique plugin name (e.g. 'code', 'git', 'docs'). */\n readonly name: string;\n /** Initialize the plugin (create HNSW, load vectors, etc.). */\n initialize(ctx: PluginContext): Promise<void>;\n /** Return stats for this plugin. */\n stats?(): Record<string, any>;\n /** Clean up resources. */\n close?(): void;\n}\n\n// ── Capability Interfaces ──────────────────────────\n// Implemented by plugins that support specific capabilities.\n// Use type guards below to check at runtime.\n\n/** Plugins that can scan and index content (code, git). */\nexport interface IndexablePlugin extends Plugin {\n index(options?: any): Promise<any>;\n}\n\n/** Plugins that can search indexed content (docs). */\nexport interface SearchablePlugin extends Plugin {\n search(query: string, options?: any): Promise<SearchResult[]>;\n}\n\n/** Plugins that support file watch mode. */\nexport interface WatchablePlugin extends Plugin {\n onFileChange(filePath: string, event: 'create' | 'update' | 'delete'): Promise<boolean>;\n watchPatterns(): string[];\n}\n\n/** Plugins that manage document collections. */\nexport interface CollectionPlugin extends Plugin {\n addCollection(collection: DocumentCollection): void;\n removeCollection(name: string): void;\n listCollections(): DocumentCollection[];\n indexCollections(options?: any): Promise<any>;\n search(query: string, options?: any): Promise<SearchResult[]>;\n addContext?(collection: string, path: string, context: string): void;\n removeContext?(collection: string, path: string): void;\n listContexts?(): any[];\n}\n\n// ── Type Guards ────────────────────────────────────\n\n/** Check if a plugin can scan/index content. */\nexport function isIndexable(i: Plugin): i is IndexablePlugin {\n return typeof (i as IndexablePlugin).index === 'function';\n}\n\n/** Check if a plugin can search content. */\nexport function isSearchable(i: Plugin): i is SearchablePlugin {\n return typeof (i as SearchablePlugin).search === 'function';\n}\n\n/** Check if a plugin supports file watch mode. */\nexport function isWatchable(i: Plugin): i is WatchablePlugin {\n return typeof (i as WatchablePlugin).onFileChange === 'function'\n && typeof (i as WatchablePlugin).watchPatterns === 'function';\n}\n\n/** Check if a plugin manages document collections. */\nexport function isCollectionPlugin(i: Plugin): i is CollectionPlugin {\n return typeof (i as CollectionPlugin).addCollection === 'function'\n && typeof (i as CollectionPlugin).listCollections === 'function';\n}\n\n","/**\n * BrainBank — Index API\n *\n * Orchestrates indexing across code, git, and document indexers.\n * BrainBank delegates here after auto-initialization.\n */\n\nimport type { PluginRegistry } from '@/bootstrap/registry.ts';\nimport type { IndexResult, StageProgressCallback, ProgressCallback } from '@/types.ts';\nimport { isIndexable, isCollectionPlugin } from '@/indexers/base.ts';\n\nexport interface IndexAPIDeps {\n registry: PluginRegistry;\n gitDepth: number;\n emit: (event: string, data: any) => void;\n}\n\nexport class IndexAPI {\n constructor(private _d: IndexAPIDeps) {}\n\n async index(options: {\n modules?: ('code' | 'git' | 'docs')[];\n gitDepth?: number;\n forceReindex?: boolean;\n onProgress?: StageProgressCallback;\n } = {}): Promise<{ code?: IndexResult; git?: IndexResult; docs?: Record<string, { indexed: number; skipped: number; chunks: number }> }> {\n const want = new Set(options.modules ?? ['code', 'git', 'docs']);\n const result: { code?: IndexResult; git?: IndexResult; docs?: Record<string, any> } = {};\n\n if (want.has('code')) {\n for (const mod of this._d.registry.allByType('code')) {\n if (!isIndexable(mod)) continue;\n const label = mod.name === 'code' ? 'code' : mod.name;\n options.onProgress?.(label, 'Starting...');\n const r = await mod.index({\n forceReindex: options.forceReindex,\n onProgress: (f: string, i: number, t: number) => options.onProgress?.(label, `[${i}/${t}] ${f}`),\n });\n if (result.code) {\n result.code.indexed += r.indexed;\n result.code.skipped += r.skipped;\n result.code.chunks = (result.code.chunks ?? 0) + (r.chunks ?? 0);\n } else {\n result.code = r;\n }\n }\n }\n\n if (want.has('git')) {\n for (const mod of this._d.registry.allByType('git')) {\n if (!isIndexable(mod)) continue;\n const label = mod.name === 'git' ? 'git' : mod.name;\n options.onProgress?.(label, 'Starting...');\n const r = await mod.index({\n depth: options.gitDepth ?? this._d.gitDepth,\n onProgress: (f: string, i: number, t: number) => options.onProgress?.(label, `[${i}/${t}] ${f}`),\n });\n if (result.git) {\n result.git.indexed += r.indexed;\n result.git.skipped += r.skipped;\n } else {\n result.git = r;\n }\n }\n }\n\n if (want.has('docs') && this._d.registry.has('docs')) {\n const docsPlugin = this._d.registry.get('docs');\n if (isCollectionPlugin(docsPlugin)) {\n options.onProgress?.('docs', 'Starting...');\n result.docs = await docsPlugin.indexCollections({\n onProgress: (coll: string, file: string, cur: number, total: number) =>\n options.onProgress?.('docs', `[${coll}] ${cur}/${total}: ${file}`),\n });\n }\n }\n\n this._d.emit('indexed', result);\n return result;\n }\n\n async indexCode(options: { forceReindex?: boolean; onProgress?: ProgressCallback } = {}): Promise<IndexResult> {\n const mods = this._d.registry.allByType('code').filter(isIndexable);\n if (!mods.length) throw new Error(\"BrainBank: Indexer 'code' is not loaded. Add .use(code()) to your BrainBank instance.\");\n\n const acc: IndexResult = { indexed: 0, skipped: 0, chunks: 0 };\n for (const mod of mods) {\n const r = await mod.index(options);\n acc.indexed += r.indexed;\n acc.skipped += r.skipped;\n acc.chunks = (acc.chunks ?? 0) + (r.chunks ?? 0);\n }\n return acc;\n }\n\n async indexGit(options: { depth?: number; onProgress?: ProgressCallback } = {}): Promise<IndexResult> {\n const mods = this._d.registry.allByType('git').filter(isIndexable);\n if (!mods.length) throw new Error(\"BrainBank: Indexer 'git' is not loaded. Add .use(git()) to your BrainBank instance.\");\n\n const acc: IndexResult = { indexed: 0, skipped: 0 };\n for (const mod of mods) {\n const r = await mod.index(options);\n acc.indexed += r.indexed;\n acc.skipped += r.skipped;\n }\n return acc;\n }\n}\n","/**\n * BrainBank — Re-embedding Engine\n * \n * Regenerates all vectors without re-indexing.\n * Reads existing text from SQLite, embeds with the current provider,\n * and replaces vector BLOBs. No file I/O, no git parsing, no re-chunking.\n * \n * Usage:\n * const result = await brain.reembed({ onProgress });\n * // → { code: 1200, git: 500, docs: 80, kv: 45, notes: 12, total: 1837 }\n */\n\nimport type { Database } from '@/db/database.ts';\nimport { vecToBuffer } from '@/lib/math.ts';\nimport type { EmbeddingProvider, ProgressCallback } from '@/types.ts';\nimport type { HNSWIndex } from '@/providers/vector/hnsw-index.ts';\n\n// ── Table Definitions ───────────────────────────────\n\ninterface ReembedTable {\n /** Human-readable name (for progress) */\n name: string;\n /** Table with text content */\n textTable: string;\n /** Table with vector BLOBs */\n vectorTable: string;\n /** PK column in text table */\n idColumn: string;\n /** FK column in vector table */\n fkColumn: string;\n /** Build the embedding text from a row (same logic as each indexer) */\n textBuilder: (row: any) => string;\n}\n\nconst TABLES: ReembedTable[] = [\n {\n name: 'code',\n textTable: 'code_chunks',\n vectorTable: 'code_vectors',\n idColumn: 'id',\n fkColumn: 'chunk_id',\n textBuilder: (r) => [\n `File: ${r.file_path}`,\n r.name ? `${r.chunk_type}: ${r.name}` : r.chunk_type,\n r.content,\n ].join('\\n'),\n },\n {\n name: 'git',\n textTable: 'git_commits',\n vectorTable: 'git_vectors',\n idColumn: 'id',\n fkColumn: 'commit_id',\n // Must match git-engine.ts:119-125 exactly\n textBuilder: (r) => [\n `Commit: ${r.message}`,\n `Author: ${r.author}`,\n `Date: ${r.date}`,\n r.files_json && r.files_json !== '[]'\n ? `Files: ${JSON.parse(r.files_json).join(', ')}`\n : '',\n r.diff ? `Changes:\\n${r.diff.slice(0, 2000)}` : '',\n ].filter(Boolean).join('\\n'),\n },\n {\n name: 'memory',\n textTable: 'memory_patterns',\n vectorTable: 'memory_vectors',\n idColumn: 'id',\n fkColumn: 'pattern_id',\n // Must match memory/pattern-store.ts:49 exactly\n textBuilder: (r) => `${r.task_type} ${r.task} ${r.approach}`,\n },\n {\n name: 'notes',\n textTable: 'note_memories',\n vectorTable: 'note_vectors',\n idColumn: 'id',\n fkColumn: 'note_id',\n // Must match notes/engine.ts:90 exactly\n textBuilder: (r) => {\n const decisions = (JSON.parse(r.decisions_json || '[]') as string[]).join('. ');\n const patterns = (JSON.parse(r.patterns_json || '[]') as string[]).join('. ');\n return `${r.title}\\n${r.summary}\\n${decisions}\\n${patterns}`;\n },\n },\n {\n name: 'docs',\n textTable: 'doc_chunks',\n vectorTable: 'doc_vectors',\n idColumn: 'id',\n fkColumn: 'chunk_id',\n // Must match docs-engine.ts:160 exactly\n textBuilder: (r) => `title: ${r.title ?? ''} | text: ${r.content}`,\n },\n {\n name: 'kv',\n textTable: 'kv_data',\n vectorTable: 'kv_vectors',\n idColumn: 'id',\n fkColumn: 'data_id',\n textBuilder: (r) => r.content,\n },\n];\n\n// ── Result ──────────────────────────────────────────\n\nexport interface ReembedResult {\n code: number;\n git: number;\n memory: number;\n notes: number;\n docs: number;\n kv: number;\n total: number;\n}\n\nexport interface ReembedOptions {\n /** Progress callback: (tableName, current, total) */\n onProgress?: ProgressCallback;\n /** Batch size for embedBatch. Default: 50 */\n batchSize?: number;\n}\n\n// ── Engine ──────────────────────────────────────────\n\n/**\n * Re-embed all existing text with the current embedding provider.\n * Does NOT re-parse files, git, or documents — only replaces vectors.\n */\nexport async function reembedAll(\n db: Database,\n embedding: EmbeddingProvider,\n hnswMap: Map<string, { hnsw: HNSWIndex; vecs: Map<number, Float32Array> }>,\n options: ReembedOptions = {},\n): Promise<ReembedResult> {\n const { batchSize = 50, onProgress } = options;\n const result: Record<string, number> = {};\n let total = 0;\n\n for (const table of TABLES) {\n const count = await reembedTable(db, embedding, table, batchSize, onProgress);\n result[table.name] = count;\n total += count;\n\n // Rebuild HNSW if available\n const entry = hnswMap.get(table.name);\n if (entry && count > 0) {\n await rebuildHnsw(db, table, entry.hnsw, entry.vecs);\n }\n }\n\n // Update embedding metadata\n const meta = {\n provider: embedding.constructor?.name ?? 'unknown',\n dims: String(embedding.dims),\n reembedded_at: new Date().toISOString(),\n };\n const upsert = db.prepare(\n 'INSERT OR REPLACE INTO embedding_meta (key, value) VALUES (?, ?)'\n );\n for (const [k, v] of Object.entries(meta)) {\n upsert.run(k, v);\n }\n\n return {\n code: result.code ?? 0,\n git: result.git ?? 0,\n memory: result.memory ?? 0,\n notes: result.notes ?? 0,\n docs: result.docs ?? 0,\n kv: result.kv ?? 0,\n total,\n };\n}\n\n/**\n * Re-embed a single table. Returns count of vectors regenerated.\n * \n * Streams per-batch to avoid OOM on large tables — memory stays O(batchSize).\n * Tradeoff: if embedBatch fails mid-way, partial vectors exist. Reembed is\n * a destructive operation by design — re-run to completion if interrupted.\n */\nasync function reembedTable(\n db: Database,\n embedding: EmbeddingProvider,\n table: ReembedTable,\n batchSize: number,\n onProgress?: ProgressCallback,\n): Promise<number> {\n const totalCount = (db.prepare(\n `SELECT COUNT(*) as c FROM ${table.textTable}`\n ).get() as any).c;\n\n if (totalCount === 0) return 0;\n\n // Phase 1: Build new vectors in a temp table (safe — old data untouched)\n const tempTable = `_reembed_${table.vectorTable}`;\n db.exec(`DROP TABLE IF EXISTS ${tempTable}`);\n db.exec(`CREATE TABLE ${tempTable} AS SELECT * FROM ${table.vectorTable} WHERE 0`);\n\n const insertTemp = db.prepare(\n `INSERT INTO ${tempTable} (${table.fkColumn}, embedding) VALUES (?, ?)`\n );\n\n let processed = 0;\n try {\n for (let offset = 0; offset < totalCount; offset += batchSize) {\n const batch = db.prepare(\n `SELECT * FROM ${table.textTable} LIMIT ? OFFSET ?`\n ).all(batchSize, offset) as any[];\n const texts = batch.map((r: any) => table.textBuilder(r));\n const vectors = await embedding.embedBatch(texts);\n\n db.transaction(() => {\n for (let j = 0; j < batch.length; j++) {\n insertTemp.run(batch[j][table.idColumn], vecToBuffer(vectors[j]));\n }\n });\n\n processed += batch.length;\n onProgress?.(table.name, processed, totalCount);\n }\n\n // Phase 2: Atomic swap — all or nothing\n db.transaction(() => {\n db.exec(`DELETE FROM ${table.vectorTable}`);\n db.exec(`INSERT INTO ${table.vectorTable} SELECT * FROM ${tempTable}`);\n });\n } finally {\n // Always clean up temp table — even if embedBatch fails mid-batch\n db.exec(`DROP TABLE IF EXISTS ${tempTable}`);\n }\n\n return processed;\n}\n\n/** Rebuild HNSW index from vector table. */\nasync function rebuildHnsw(\n db: Database,\n table: ReembedTable,\n hnsw: HNSWIndex,\n vecs: Map<number, Float32Array>,\n): Promise<void> {\n // Wipe stale vectors before repopulating\n vecs.clear();\n hnsw.reinit();\n\n const rows = db.prepare(\n `SELECT ${table.fkColumn} as id, embedding FROM ${table.vectorTable}`\n ).all() as any[];\n\n for (const row of rows) {\n const buf = Buffer.from(row.embedding);\n const vec = new Float32Array(buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength));\n hnsw.add(vec, row.id);\n vecs.set(row.id, vec);\n }\n}\n\n// ── Re-exports (backward compat) ────────────────────\nexport { setEmbeddingMeta, getEmbeddingMeta, detectProviderMismatch } from './embedding-meta.ts';\n","/**\n * BrainBank — Watch Mode\n * \n * Auto-indexes on file changes using fs.watch.\n * Works with built-in indexers (code, git, docs) and custom indexers.\n * \n * Built-in behavior:\n * - Code files → re-indexes changed file\n * - Doc files → re-indexes changed collection\n * \n * Custom indexers:\n * - Implement `onFileChange(path, event)` to handle changes\n * - Implement `watchPatterns()` to specify which files to watch\n * \n * Usage:\n * const watcher = brain.watch({ paths: ['.'] });\n * watcher.close(); // stop watching\n */\n\nimport * as fs from 'node:fs';\nimport * as path from 'node:path';\nimport { isSupported, isIgnoredDir, isIgnoredFile } from '@/indexers/languages.ts';\nimport type { Plugin } from '@/indexers/base.ts';\nimport { isWatchable } from '@/indexers/base.ts';\n\n// ── Types ───────────────────────────────────────────\n\nexport interface WatchOptions {\n /** Paths to watch. Default: [config.repoPath] */\n paths?: string[];\n /** Debounce interval in ms. Default: 2000 */\n debounceMs?: number;\n /** Called when a file is re-indexed. */\n onIndex?: (filePath: string, indexer: string) => void;\n /** Called on errors. */\n onError?: (error: Error) => void;\n}\n\nexport interface Watcher {\n /** Stop watching. */\n close(): void;\n /** Whether the watcher is active. */\n readonly active: boolean;\n}\n\n// ── Watch Engine ────────────────────────────────────\n\n/**\n * Create a file watcher that auto-re-indexes on changes.\n * \n * @param reindexFn — called to re-index code+git (brain.index())\n * @param indexers — registered indexers (for custom onFileChange hooks)\n * @param repoPath — base repo path (for resolving relative paths)\n * @param options — watch configuration\n */\nexport function createWatcher(\n reindexFn: () => Promise<void>,\n indexers: Map<string, Plugin>,\n repoPath: string,\n options: WatchOptions = {},\n): Watcher {\n const {\n paths = [repoPath],\n debounceMs = 2000,\n onIndex,\n onError,\n } = options;\n\n let active = true;\n const watchers: fs.FSWatcher[] = [];\n const pending = new Set<string>();\n let timer: ReturnType<typeof setTimeout> | null = null;\n\n // Collect custom watch patterns from indexers\n const customPatterns: { indexer: Plugin; patterns: string[] }[] = [];\n for (const indexer of indexers.values()) {\n if (isWatchable(indexer)) {\n customPatterns.push({ indexer, patterns: indexer.watchPatterns() });\n }\n }\n\n // Check if a file matches any custom indexer pattern\n function matchCustomPlugin(filePath: string): Plugin | null {\n const rel = path.relative(repoPath, filePath);\n for (const { indexer, patterns } of customPatterns) {\n for (const pattern of patterns) {\n if (matchGlob(rel, pattern)) return indexer;\n }\n }\n return null;\n }\n\n // Simple glob matching (supports **, *, and extension matching)\n function matchGlob(filePath: string, pattern: string): boolean {\n // **/*.ext → match any file with that extension\n if (pattern.startsWith('**/')) {\n const suffix = pattern.slice(3); // e.g. \"*.csv\"\n const ext = suffix.startsWith('*.') ? suffix.slice(1) : null; // e.g. \".csv\"\n if (ext) return filePath.endsWith(ext);\n return path.basename(filePath) === suffix;\n }\n // *.ext → match extension in any directory\n if (pattern.startsWith('*.')) {\n return filePath.endsWith(pattern.slice(1));\n }\n return filePath === pattern;\n }\n\n // Process pending file changes (serialized — no concurrent flushes)\n let flushing = false;\n\n async function processPending() {\n if (flushing || pending.size === 0) return;\n flushing = true;\n\n try {\n const files = [...pending];\n pending.clear();\n\n // Group by handler\n let needsReindex = false;\n\n for (const filePath of files) {\n const absPath = path.resolve(repoPath, filePath);\n\n // Try custom indexers first\n const customIndexer = matchCustomPlugin(absPath);\n if (customIndexer && isWatchable(customIndexer)) {\n try {\n const handled = await customIndexer.onFileChange(absPath, detectEvent(absPath));\n if (handled) {\n onIndex?.(filePath, customIndexer.name);\n continue;\n }\n } catch (err) {\n onError?.(err instanceof Error ? err : new Error(String(err)));\n }\n }\n\n // Fall back to built-in re-index for supported code files\n if (isSupported(filePath)) {\n needsReindex = true;\n onIndex?.(filePath, 'code');\n }\n }\n\n // Batch re-index if any code files changed\n if (needsReindex) {\n try {\n await reindexFn();\n } catch (err) {\n onError?.(err instanceof Error ? err : new Error(String(err)));\n }\n }\n } finally {\n flushing = false;\n // Process any changes that arrived during this flush\n if (pending.size > 0) {\n timer = setTimeout(() => processPending(), debounceMs);\n }\n }\n }\n\n function detectEvent(filePath: string): 'create' | 'update' | 'delete' {\n try {\n fs.accessSync(filePath);\n return 'update';\n } catch {\n return 'delete';\n }\n }\n\n // Should we watch this file?\n function shouldWatch(filename: string): boolean {\n if (!filename) return false;\n const parts = filename.split(path.sep);\n\n // Skip ignored directories\n for (const part of parts) {\n if (isIgnoredDir(part)) return false;\n }\n\n // Skip ignored files\n if (isIgnoredFile(path.basename(filename))) return false;\n\n // Accept supported code files\n if (isSupported(filename)) return true;\n\n // Accept files matching custom indexer patterns\n if (matchCustomPlugin(path.resolve(repoPath, filename))) return true;\n\n return false;\n }\n\n // Set up watchers\n for (const watchPath of paths) {\n const resolved = path.resolve(watchPath);\n try {\n // { recursive: true } only works on macOS + Windows; Linux needs chokidar or per-dir watchers\n const supportsRecursive = process.platform === 'darwin' || process.platform === 'win32';\n const watcher = fs.watch(resolved, { recursive: supportsRecursive }, (_event, filename) => {\n if (!active || !filename) return;\n if (!shouldWatch(filename)) return;\n\n pending.add(filename);\n\n // Debounce\n if (timer) clearTimeout(timer);\n timer = setTimeout(() => processPending(), debounceMs);\n });\n\n watcher.on('error', (err) => {\n onError?.(err instanceof Error ? err : new Error(String(err)));\n });\n\n watchers.push(watcher);\n } catch (err) {\n onError?.(err instanceof Error ? err : new Error(String(err)));\n }\n }\n\n return {\n close() {\n active = false;\n if (timer) clearTimeout(timer);\n for (const w of watchers) w.close();\n watchers.length = 0;\n },\n get active() { return active; },\n };\n}\n"],"mappings":"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","names":["path","dirname","join","path","join","dirname","fs","path","path"]}
|