aiwcli 0.12.6 → 0.12.8

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.
Files changed (163) hide show
  1. package/bin/dev.cmd +3 -3
  2. package/bin/dev.js +16 -16
  3. package/bin/run.cmd +3 -3
  4. package/bin/run.js +21 -21
  5. package/dist/commands/branch.js +7 -2
  6. package/dist/lib/bmad-installer.js +37 -37
  7. package/dist/lib/terminal.d.ts +2 -0
  8. package/dist/lib/terminal.js +57 -7
  9. package/dist/templates/CLAUDE.md +232 -205
  10. package/dist/templates/_shared/.claude/settings.json +65 -65
  11. package/dist/templates/_shared/.claude/{commands/handoff.md → skills/handoff/SKILL.md} +13 -12
  12. package/dist/templates/_shared/.claude/{commands/handoff-resume.md → skills/handoff-resume/SKILL.md} +13 -12
  13. package/dist/templates/_shared/.codex/workflows/handoff.md +226 -226
  14. package/dist/templates/_shared/.windsurf/workflows/handoff.md +226 -226
  15. package/dist/templates/_shared/handoff-system/CLAUDE.md +15 -3
  16. package/dist/templates/_shared/handoff-system/lib/document-generator.ts +215 -215
  17. package/dist/templates/_shared/handoff-system/lib/handoff-reader.ts +158 -158
  18. package/dist/templates/_shared/handoff-system/scripts/resume_handoff.ts +373 -373
  19. package/dist/templates/_shared/handoff-system/scripts/save_handoff.ts +469 -469
  20. package/dist/templates/_shared/handoff-system/workflows/handoff-resume.md +66 -66
  21. package/dist/templates/_shared/handoff-system/workflows/handoff.md +254 -254
  22. package/dist/templates/_shared/hooks-ts/_utils/git-state.ts +2 -2
  23. package/dist/templates/_shared/hooks-ts/archive_plan.ts +159 -159
  24. package/dist/templates/_shared/hooks-ts/context_monitor.ts +147 -147
  25. package/dist/templates/_shared/hooks-ts/file-suggestion.ts +128 -128
  26. package/dist/templates/_shared/hooks-ts/pre_compact.ts +49 -49
  27. package/dist/templates/_shared/hooks-ts/session_end.ts +196 -196
  28. package/dist/templates/_shared/hooks-ts/session_start.ts +163 -163
  29. package/dist/templates/_shared/hooks-ts/task_create_capture.ts +48 -48
  30. package/dist/templates/_shared/hooks-ts/task_update_capture.ts +74 -74
  31. package/dist/templates/_shared/hooks-ts/user_prompt_submit.ts +93 -93
  32. package/dist/templates/_shared/lib-ts/CLAUDE.md +367 -367
  33. package/dist/templates/_shared/lib-ts/base/atomic-write.ts +138 -138
  34. package/dist/templates/_shared/lib-ts/base/constants.ts +24 -6
  35. package/dist/templates/_shared/lib-ts/base/git-state.ts +58 -58
  36. package/dist/templates/_shared/lib-ts/base/hook-utils.ts +582 -582
  37. package/dist/templates/_shared/lib-ts/base/inference.ts +301 -301
  38. package/dist/templates/_shared/lib-ts/base/logger.ts +247 -247
  39. package/dist/templates/_shared/lib-ts/base/state-io.ts +202 -202
  40. package/dist/templates/_shared/lib-ts/base/stop-words.ts +184 -184
  41. package/dist/templates/_shared/lib-ts/base/utils.ts +184 -184
  42. package/dist/templates/_shared/lib-ts/context/CLAUDE.md +134 -0
  43. package/dist/templates/_shared/lib-ts/context/context-formatter.ts +566 -566
  44. package/dist/templates/_shared/lib-ts/context/context-selector.ts +524 -524
  45. package/dist/templates/_shared/lib-ts/context/context-store.ts +712 -712
  46. package/dist/templates/_shared/lib-ts/context/plan-manager.ts +312 -312
  47. package/dist/templates/_shared/lib-ts/context/task-tracker.ts +185 -185
  48. package/dist/templates/_shared/lib-ts/package.json +20 -20
  49. package/dist/templates/_shared/lib-ts/templates/formatters.ts +102 -102
  50. package/dist/templates/_shared/lib-ts/templates/plan-context.ts +58 -58
  51. package/dist/templates/_shared/lib-ts/tsconfig.json +13 -13
  52. package/dist/templates/_shared/lib-ts/types.ts +186 -186
  53. package/dist/templates/_shared/scripts/resolve_context.ts +33 -33
  54. package/dist/templates/_shared/scripts/status_line.ts +687 -690
  55. package/dist/templates/cc-native/.claude/commands/cc-native/rlm/ask.md +136 -136
  56. package/dist/templates/cc-native/.claude/commands/cc-native/rlm/index.md +21 -21
  57. package/dist/templates/cc-native/.claude/commands/cc-native/rlm/overview.md +56 -56
  58. package/dist/templates/cc-native/.claude/commands/cc-native/specdev.md +10 -10
  59. package/dist/templates/cc-native/.claude/settings.json +3 -2
  60. package/dist/templates/cc-native/.windsurf/workflows/cc-native/fix.md +8 -8
  61. package/dist/templates/cc-native/.windsurf/workflows/cc-native/implement.md +8 -8
  62. package/dist/templates/cc-native/.windsurf/workflows/cc-native/research.md +8 -8
  63. package/dist/templates/cc-native/CC-NATIVE-README.md +189 -189
  64. package/dist/templates/cc-native/TEMPLATE-SCHEMA.md +304 -304
  65. package/dist/templates/cc-native/_cc-native/agents/CLAUDE.md +143 -143
  66. package/dist/templates/cc-native/_cc-native/agents/PLAN-ORCHESTRATOR.md +213 -213
  67. package/dist/templates/cc-native/_cc-native/agents/plan-questions/PLAN-QUESTIONER.md +70 -70
  68. package/dist/templates/cc-native/_cc-native/artifacts/CLAUDE.md +64 -0
  69. package/dist/templates/cc-native/_cc-native/{lib-ts/artifacts → artifacts/lib}/format.ts +1 -1
  70. package/dist/templates/cc-native/_cc-native/{lib-ts/artifacts → artifacts/lib}/write.ts +2 -2
  71. package/dist/templates/cc-native/_cc-native/cc-native.config.json +96 -96
  72. package/dist/templates/cc-native/_cc-native/hooks/CLAUDE.md +14 -24
  73. package/dist/templates/cc-native/_cc-native/hooks/cc-native-plan-review.ts +1 -1
  74. package/dist/templates/cc-native/_cc-native/hooks/enhance_plan_post_subagent.ts +54 -54
  75. package/dist/templates/cc-native/_cc-native/hooks/enhance_plan_post_write.ts +51 -51
  76. package/dist/templates/cc-native/_cc-native/hooks/mark_questions_asked.ts +53 -53
  77. package/dist/templates/cc-native/_cc-native/hooks/plan_questions_early.ts +61 -61
  78. package/dist/templates/cc-native/_cc-native/hooks/validate_task_prompt.ts +76 -0
  79. package/dist/templates/cc-native/_cc-native/lib-ts/aggregate-agents.ts +9 -2
  80. package/dist/templates/cc-native/_cc-native/lib-ts/cc-native-state.ts +319 -319
  81. package/dist/templates/cc-native/_cc-native/lib-ts/cli-output-parser.ts +144 -144
  82. package/dist/templates/cc-native/_cc-native/lib-ts/config.ts +57 -57
  83. package/dist/templates/cc-native/_cc-native/lib-ts/constants.ts +83 -83
  84. package/dist/templates/cc-native/_cc-native/lib-ts/debug.ts +79 -79
  85. package/dist/templates/cc-native/_cc-native/lib-ts/index.ts +4 -4
  86. package/dist/templates/cc-native/_cc-native/lib-ts/json-parser.ts +168 -168
  87. package/dist/templates/cc-native/_cc-native/lib-ts/plan-discovery.ts +80 -80
  88. package/dist/templates/cc-native/_cc-native/lib-ts/plan-enhancement.ts +41 -41
  89. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/CLAUDE.md +480 -480
  90. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/embedding-indexer.ts +287 -287
  91. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/hyde.ts +148 -148
  92. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/index.ts +54 -54
  93. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/logger.ts +58 -58
  94. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/ollama-client.ts +208 -208
  95. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/retrieval-pipeline.ts +460 -460
  96. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-indexer.ts +446 -446
  97. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-loader.ts +280 -280
  98. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-searcher.ts +274 -274
  99. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/types.ts +201 -201
  100. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/vector-store.ts +278 -278
  101. package/dist/templates/cc-native/_cc-native/lib-ts/settings.ts +184 -184
  102. package/dist/templates/cc-native/_cc-native/lib-ts/state.ts +275 -275
  103. package/dist/templates/cc-native/_cc-native/lib-ts/tsconfig.json +18 -18
  104. package/dist/templates/cc-native/_cc-native/lib-ts/types.ts +1 -1
  105. package/dist/templates/cc-native/_cc-native/plan-review/CLAUDE.md +149 -0
  106. package/dist/templates/cc-native/_cc-native/plan-review/agents/CLAUDE.md +143 -0
  107. package/dist/templates/cc-native/_cc-native/plan-review/agents/PLAN-ORCHESTRATOR.md +213 -0
  108. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-questions/PLAN-QUESTIONER.md +70 -0
  109. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ARCH-EVOLUTION.md +62 -0
  110. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ARCH-PATTERNS.md +61 -0
  111. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ARCH-STRUCTURE.md +62 -0
  112. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ASSUMPTION-TRACER.md +56 -0
  113. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/CLARITY-AUDITOR.md +53 -0
  114. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/COMPLETENESS-FEASIBILITY.md +66 -0
  115. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/COMPLETENESS-GAPS.md +70 -0
  116. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/COMPLETENESS-ORDERING.md +62 -0
  117. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/CONSTRAINT-VALIDATOR.md +72 -0
  118. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DESIGN-ADR-VALIDATOR.md +61 -0
  119. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DESIGN-SCALE-MATCHER.md +64 -0
  120. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DEVILS-ADVOCATE.md +56 -0
  121. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DOCUMENTATION-PHILOSOPHY.md +86 -0
  122. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/HANDOFF-READINESS.md +59 -0
  123. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/HIDDEN-COMPLEXITY.md +58 -0
  124. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/INCREMENTAL-DELIVERY.md +66 -0
  125. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-DEPENDENCY.md +62 -0
  126. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-FMEA.md +66 -0
  127. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-PREMORTEM.md +71 -0
  128. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-REVERSIBILITY.md +74 -0
  129. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/SCOPE-BOUNDARY.md +77 -0
  130. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/SIMPLICITY-GUARDIAN.md +62 -0
  131. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/SKEPTIC.md +68 -0
  132. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-BEHAVIOR-AUDITOR.md +61 -0
  133. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-CHARACTERIZATION.md +71 -0
  134. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-FIRST-VALIDATOR.md +61 -0
  135. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-PYRAMID-ANALYZER.md +61 -0
  136. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TRADEOFF-COSTS.md +67 -0
  137. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TRADEOFF-STAKEHOLDERS.md +65 -0
  138. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/VERIFY-COVERAGE.md +74 -0
  139. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/VERIFY-STRENGTH.md +69 -0
  140. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/agent-selection.ts +3 -3
  141. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/corroboration.ts +1 -1
  142. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/graduation.ts +1 -1
  143. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/orchestrator.ts +2 -2
  144. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/output-builder.ts +3 -3
  145. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/plan-questions.ts +6 -6
  146. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/review-pipeline.ts +15 -15
  147. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/agent.ts +5 -5
  148. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/base/base-agent.ts +4 -4
  149. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/providers/claude-agent.ts +4 -4
  150. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/providers/codex-agent.ts +6 -6
  151. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/providers/gemini-agent.ts +1 -1
  152. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/providers/orchestrator-claude-agent.ts +4 -4
  153. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/types.ts +3 -3
  154. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/verdict.ts +1 -1
  155. package/oclif.manifest.json +1 -1
  156. package/package.json +108 -108
  157. package/dist/templates/cc-native/_cc-native/lib-ts/artifacts.ts +0 -21
  158. package/dist/templates/cc-native/_cc-native/lib-ts/nul +0 -3
  159. /package/dist/templates/cc-native/_cc-native/{lib-ts/artifacts → artifacts/lib}/index.ts +0 -0
  160. /package/dist/templates/cc-native/_cc-native/{lib-ts/artifacts → artifacts/lib}/tracker.ts +0 -0
  161. /package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/index.ts +0 -0
  162. /package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/schemas.ts +0 -0
  163. /package/dist/templates/cc-native/_cc-native/{workflows → plan-review/workflows}/specdev.md +0 -0
@@ -1,278 +1,278 @@
1
- /**
2
- * Vector store using SQLite + sqlite-vec for KNN embedding search.
3
- *
4
- * Single-file DB at ~/.claude/rlm-vectors.db.
5
- * Uses bun:sqlite with the sqlite-vec extension for vector similarity search.
6
- */
7
-
8
- import { Database } from "bun:sqlite";
9
- import * as sqliteVec from "sqlite-vec";
10
- import { RLM_VECTOR_DB_PATH, EMBED_DIMENSIONS, type VectorSearchResult } from "./types.js";
11
- import { logDebug, logInfo } from "./logger.js";
12
-
13
- const HOOK_NAME = "rlm_vector";
14
-
15
- export interface ChunkRow {
16
- session_id: string;
17
- project: string;
18
- date: string;
19
- segment_index: number;
20
- line_start: number;
21
- line_end: number;
22
- topic: string;
23
- chunk_text: string;
24
- source_path: string;
25
- embedding: Float32Array;
26
- }
27
-
28
- export interface VectorStats {
29
- session_count: number;
30
- chunk_count: number;
31
- }
32
-
33
- /**
34
- * Open the vector DB, load sqlite-vec extension, create schema, set WAL mode.
35
- */
36
- export function openVectorDb(path?: string): Database {
37
- const dbPath = path ?? RLM_VECTOR_DB_PATH;
38
- const db = new Database(dbPath);
39
- sqliteVec.load(db);
40
- db.run("PRAGMA journal_mode=WAL");
41
-
42
- db.run(`
43
- CREATE TABLE IF NOT EXISTS embedded_sessions (
44
- session_id TEXT NOT NULL,
45
- project TEXT NOT NULL,
46
- source_mtime INTEGER NOT NULL,
47
- chunk_count INTEGER NOT NULL,
48
- embedded_at TEXT NOT NULL,
49
- PRIMARY KEY (session_id, project)
50
- )
51
- `);
52
-
53
- // vec0 virtual table for KNN search
54
- db.run(`
55
- CREATE VIRTUAL TABLE IF NOT EXISTS chunks USING vec0(
56
- embedding float[${EMBED_DIMENSIONS}],
57
- project text,
58
- date text,
59
- +session_id text,
60
- +segment_index integer,
61
- +line_start integer,
62
- +line_end integer,
63
- +topic text,
64
- +chunk_text text,
65
- +source_path text
66
- )
67
- `);
68
-
69
- logDebug(HOOK_NAME, `Opened vector DB at ${dbPath}`);
70
- return db;
71
- }
72
-
73
- /**
74
- * Insert chunks in a single transaction.
75
- */
76
- export function insertChunks(db: Database, chunks: ChunkRow[]): void {
77
- if (chunks.length === 0) return;
78
-
79
- const stmt = db.prepare(`
80
- INSERT INTO chunks (
81
- embedding, project, date,
82
- session_id, segment_index, line_start, line_end,
83
- topic, chunk_text, source_path
84
- ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
85
- `);
86
-
87
- const tx = db.transaction(() => {
88
- for (const chunk of chunks) {
89
- stmt.run(
90
- new Uint8Array(chunk.embedding.buffer),
91
- chunk.project,
92
- chunk.date,
93
- chunk.session_id,
94
- chunk.segment_index,
95
- chunk.line_start,
96
- chunk.line_end,
97
- chunk.topic,
98
- chunk.chunk_text,
99
- chunk.source_path,
100
- );
101
- }
102
- });
103
-
104
- tx();
105
- logDebug(HOOK_NAME, `Inserted ${chunks.length} chunks`);
106
- }
107
-
108
- /**
109
- * Mark a session as embedded (upsert).
110
- */
111
- export function markSessionEmbedded(
112
- db: Database,
113
- sessionId: string,
114
- project: string,
115
- mtime: number,
116
- count: number,
117
- ): void {
118
- db.run(
119
- `INSERT OR REPLACE INTO embedded_sessions (session_id, project, source_mtime, chunk_count, embedded_at)
120
- VALUES (?, ?, ?, ?, ?)`,
121
- [sessionId, project, mtime, count, new Date().toISOString()],
122
- );
123
- }
124
-
125
- /**
126
- * Check if a session is already embedded at the given mtime.
127
- */
128
- export function isSessionEmbedded(
129
- db: Database,
130
- sessionId: string,
131
- project: string,
132
- mtime: number,
133
- ): boolean {
134
- const row = db.query(
135
- `SELECT 1 FROM embedded_sessions WHERE session_id = ? AND project = ? AND source_mtime = ?`,
136
- ).get(sessionId, project, mtime);
137
- return row !== null && row !== undefined;
138
- }
139
-
140
- /**
141
- * Delete all chunks for a session (for re-indexing).
142
- */
143
- function isRowidResult(obj: unknown): obj is { rowid: number } {
144
- return typeof obj === "object" && obj !== null && "rowid" in obj && typeof (obj as { rowid: unknown }).rowid === "number";
145
- }
146
-
147
- export function deleteSessionChunks(
148
- db: Database,
149
- sessionId: string,
150
- project: string,
151
- ): void {
152
- // vec0 tables support DELETE with rowid ranges, but we need to find matching rowids first
153
- const rawRows = db.query(
154
- `SELECT rowid FROM chunks WHERE session_id = ? AND project = ?`,
155
- ).all(sessionId, project);
156
-
157
- const rows = (Array.isArray(rawRows) ? rawRows : []).filter(isRowidResult);
158
-
159
- if (rows.length > 0) {
160
- const tx = db.transaction(() => {
161
- for (const row of rows) {
162
- db.run(`DELETE FROM chunks WHERE rowid = ?`, [row.rowid]);
163
- }
164
- });
165
- tx();
166
- logDebug(HOOK_NAME, `Deleted ${rows.length} chunks for ${sessionId}`);
167
- }
168
-
169
- db.run(
170
- `DELETE FROM embedded_sessions WHERE session_id = ? AND project = ?`,
171
- [sessionId, project],
172
- );
173
- }
174
-
175
- function isSearchResultRow(obj: unknown): obj is {
176
- rowid: number;
177
- distance: number;
178
- project: string;
179
- date: string;
180
- session_id: string;
181
- segment_index: number;
182
- line_start: number;
183
- line_end: number;
184
- topic: string;
185
- source_path: string;
186
- } {
187
- if (typeof obj !== "object" || obj === null) return false;
188
- const r = obj as Record<string, unknown>;
189
- return (
190
- typeof r.rowid === "number" &&
191
- typeof r.distance === "number" &&
192
- typeof r.project === "string" &&
193
- typeof r.date === "string" &&
194
- typeof r.session_id === "string" &&
195
- typeof r.segment_index === "number" &&
196
- typeof r.line_start === "number" &&
197
- typeof r.line_end === "number" &&
198
- typeof r.topic === "string" &&
199
- typeof r.source_path === "string"
200
- );
201
- }
202
-
203
- /**
204
- * KNN search for the closest chunks to a query embedding.
205
- */
206
- export function searchKnn(
207
- db: Database,
208
- queryEmbedding: Float32Array,
209
- topK: number,
210
- projectFilter?: string,
211
- ): VectorSearchResult[] {
212
- const queryBytes = new Uint8Array(queryEmbedding.buffer);
213
-
214
- let sql: string;
215
- let params: unknown[];
216
-
217
- if (projectFilter) {
218
- sql = `
219
- SELECT rowid, distance, project, date,
220
- session_id, segment_index, line_start, line_end,
221
- topic, source_path
222
- FROM chunks
223
- WHERE embedding MATCH ? AND k = ? AND project = ?
224
- ORDER BY distance
225
- `;
226
- params = [queryBytes, topK, projectFilter];
227
- } else {
228
- sql = `
229
- SELECT rowid, distance, project, date,
230
- session_id, segment_index, line_start, line_end,
231
- topic, source_path
232
- FROM chunks
233
- WHERE embedding MATCH ? AND k = ?
234
- ORDER BY distance
235
- `;
236
- params = [queryBytes, topK];
237
- }
238
-
239
- const rawRows = db.query(sql).all(...params);
240
- const rows = (Array.isArray(rawRows) ? rawRows : []).filter(isSearchResultRow);
241
-
242
- return rows.map((r) => ({
243
- chunk_id: r.rowid,
244
- session_id: r.session_id,
245
- project: r.project,
246
- segment_index: r.segment_index,
247
- line_start: r.line_start,
248
- line_end: r.line_end,
249
- topic: r.topic,
250
- date: r.date,
251
- source_path: r.source_path,
252
- distance: r.distance,
253
- }));
254
- }
255
-
256
- function isCountResult(obj: unknown): obj is { cnt: number } {
257
- return typeof obj === "object" && obj !== null && "cnt" in obj && typeof (obj as { cnt: unknown }).cnt === "number";
258
- }
259
-
260
- /**
261
- * Get counts of embedded sessions and chunks.
262
- */
263
- export function getStats(db: Database): VectorStats {
264
- const sessionsRaw = db.query(
265
- `SELECT COUNT(*) as cnt FROM embedded_sessions`,
266
- ).get();
267
- const chunksRaw = db.query(
268
- `SELECT COUNT(*) as cnt FROM chunks`,
269
- ).get();
270
-
271
- const sessionCount = isCountResult(sessionsRaw) ? sessionsRaw.cnt : 0;
272
- const chunkCount = isCountResult(chunksRaw) ? chunksRaw.cnt : 0;
273
-
274
- return {
275
- session_count: sessionCount,
276
- chunk_count: chunkCount,
277
- };
278
- }
1
+ /**
2
+ * Vector store using SQLite + sqlite-vec for KNN embedding search.
3
+ *
4
+ * Single-file DB at ~/.claude/rlm-vectors.db.
5
+ * Uses bun:sqlite with the sqlite-vec extension for vector similarity search.
6
+ */
7
+
8
+ import { Database } from "bun:sqlite";
9
+ import * as sqliteVec from "sqlite-vec";
10
+ import { RLM_VECTOR_DB_PATH, EMBED_DIMENSIONS, type VectorSearchResult } from "./types.js";
11
+ import { logDebug, logInfo } from "./logger.js";
12
+
13
+ const HOOK_NAME = "rlm_vector";
14
+
15
+ export interface ChunkRow {
16
+ session_id: string;
17
+ project: string;
18
+ date: string;
19
+ segment_index: number;
20
+ line_start: number;
21
+ line_end: number;
22
+ topic: string;
23
+ chunk_text: string;
24
+ source_path: string;
25
+ embedding: Float32Array;
26
+ }
27
+
28
+ export interface VectorStats {
29
+ session_count: number;
30
+ chunk_count: number;
31
+ }
32
+
33
+ /**
34
+ * Open the vector DB, load sqlite-vec extension, create schema, set WAL mode.
35
+ */
36
+ export function openVectorDb(path?: string): Database {
37
+ const dbPath = path ?? RLM_VECTOR_DB_PATH;
38
+ const db = new Database(dbPath);
39
+ sqliteVec.load(db);
40
+ db.run("PRAGMA journal_mode=WAL");
41
+
42
+ db.run(`
43
+ CREATE TABLE IF NOT EXISTS embedded_sessions (
44
+ session_id TEXT NOT NULL,
45
+ project TEXT NOT NULL,
46
+ source_mtime INTEGER NOT NULL,
47
+ chunk_count INTEGER NOT NULL,
48
+ embedded_at TEXT NOT NULL,
49
+ PRIMARY KEY (session_id, project)
50
+ )
51
+ `);
52
+
53
+ // vec0 virtual table for KNN search
54
+ db.run(`
55
+ CREATE VIRTUAL TABLE IF NOT EXISTS chunks USING vec0(
56
+ embedding float[${EMBED_DIMENSIONS}],
57
+ project text,
58
+ date text,
59
+ +session_id text,
60
+ +segment_index integer,
61
+ +line_start integer,
62
+ +line_end integer,
63
+ +topic text,
64
+ +chunk_text text,
65
+ +source_path text
66
+ )
67
+ `);
68
+
69
+ logDebug(HOOK_NAME, `Opened vector DB at ${dbPath}`);
70
+ return db;
71
+ }
72
+
73
+ /**
74
+ * Insert chunks in a single transaction.
75
+ */
76
+ export function insertChunks(db: Database, chunks: ChunkRow[]): void {
77
+ if (chunks.length === 0) return;
78
+
79
+ const stmt = db.prepare(`
80
+ INSERT INTO chunks (
81
+ embedding, project, date,
82
+ session_id, segment_index, line_start, line_end,
83
+ topic, chunk_text, source_path
84
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
85
+ `);
86
+
87
+ const tx = db.transaction(() => {
88
+ for (const chunk of chunks) {
89
+ stmt.run(
90
+ new Uint8Array(chunk.embedding.buffer),
91
+ chunk.project,
92
+ chunk.date,
93
+ chunk.session_id,
94
+ chunk.segment_index,
95
+ chunk.line_start,
96
+ chunk.line_end,
97
+ chunk.topic,
98
+ chunk.chunk_text,
99
+ chunk.source_path,
100
+ );
101
+ }
102
+ });
103
+
104
+ tx();
105
+ logDebug(HOOK_NAME, `Inserted ${chunks.length} chunks`);
106
+ }
107
+
108
+ /**
109
+ * Mark a session as embedded (upsert).
110
+ */
111
+ export function markSessionEmbedded(
112
+ db: Database,
113
+ sessionId: string,
114
+ project: string,
115
+ mtime: number,
116
+ count: number,
117
+ ): void {
118
+ db.run(
119
+ `INSERT OR REPLACE INTO embedded_sessions (session_id, project, source_mtime, chunk_count, embedded_at)
120
+ VALUES (?, ?, ?, ?, ?)`,
121
+ [sessionId, project, mtime, count, new Date().toISOString()],
122
+ );
123
+ }
124
+
125
+ /**
126
+ * Check if a session is already embedded at the given mtime.
127
+ */
128
+ export function isSessionEmbedded(
129
+ db: Database,
130
+ sessionId: string,
131
+ project: string,
132
+ mtime: number,
133
+ ): boolean {
134
+ const row = db.query(
135
+ `SELECT 1 FROM embedded_sessions WHERE session_id = ? AND project = ? AND source_mtime = ?`,
136
+ ).get(sessionId, project, mtime);
137
+ return row !== null && row !== undefined;
138
+ }
139
+
140
+ /**
141
+ * Delete all chunks for a session (for re-indexing).
142
+ */
143
+ function isRowidResult(obj: unknown): obj is { rowid: number } {
144
+ return typeof obj === "object" && obj !== null && "rowid" in obj && typeof (obj as { rowid: unknown }).rowid === "number";
145
+ }
146
+
147
+ export function deleteSessionChunks(
148
+ db: Database,
149
+ sessionId: string,
150
+ project: string,
151
+ ): void {
152
+ // vec0 tables support DELETE with rowid ranges, but we need to find matching rowids first
153
+ const rawRows = db.query(
154
+ `SELECT rowid FROM chunks WHERE session_id = ? AND project = ?`,
155
+ ).all(sessionId, project);
156
+
157
+ const rows = (Array.isArray(rawRows) ? rawRows : []).filter(isRowidResult);
158
+
159
+ if (rows.length > 0) {
160
+ const tx = db.transaction(() => {
161
+ for (const row of rows) {
162
+ db.run(`DELETE FROM chunks WHERE rowid = ?`, [row.rowid]);
163
+ }
164
+ });
165
+ tx();
166
+ logDebug(HOOK_NAME, `Deleted ${rows.length} chunks for ${sessionId}`);
167
+ }
168
+
169
+ db.run(
170
+ `DELETE FROM embedded_sessions WHERE session_id = ? AND project = ?`,
171
+ [sessionId, project],
172
+ );
173
+ }
174
+
175
+ function isSearchResultRow(obj: unknown): obj is {
176
+ rowid: number;
177
+ distance: number;
178
+ project: string;
179
+ date: string;
180
+ session_id: string;
181
+ segment_index: number;
182
+ line_start: number;
183
+ line_end: number;
184
+ topic: string;
185
+ source_path: string;
186
+ } {
187
+ if (typeof obj !== "object" || obj === null) return false;
188
+ const r = obj as Record<string, unknown>;
189
+ return (
190
+ typeof r.rowid === "number" &&
191
+ typeof r.distance === "number" &&
192
+ typeof r.project === "string" &&
193
+ typeof r.date === "string" &&
194
+ typeof r.session_id === "string" &&
195
+ typeof r.segment_index === "number" &&
196
+ typeof r.line_start === "number" &&
197
+ typeof r.line_end === "number" &&
198
+ typeof r.topic === "string" &&
199
+ typeof r.source_path === "string"
200
+ );
201
+ }
202
+
203
+ /**
204
+ * KNN search for the closest chunks to a query embedding.
205
+ */
206
+ export function searchKnn(
207
+ db: Database,
208
+ queryEmbedding: Float32Array,
209
+ topK: number,
210
+ projectFilter?: string,
211
+ ): VectorSearchResult[] {
212
+ const queryBytes = new Uint8Array(queryEmbedding.buffer);
213
+
214
+ let sql: string;
215
+ let params: unknown[];
216
+
217
+ if (projectFilter) {
218
+ sql = `
219
+ SELECT rowid, distance, project, date,
220
+ session_id, segment_index, line_start, line_end,
221
+ topic, source_path
222
+ FROM chunks
223
+ WHERE embedding MATCH ? AND k = ? AND project = ?
224
+ ORDER BY distance
225
+ `;
226
+ params = [queryBytes, topK, projectFilter];
227
+ } else {
228
+ sql = `
229
+ SELECT rowid, distance, project, date,
230
+ session_id, segment_index, line_start, line_end,
231
+ topic, source_path
232
+ FROM chunks
233
+ WHERE embedding MATCH ? AND k = ?
234
+ ORDER BY distance
235
+ `;
236
+ params = [queryBytes, topK];
237
+ }
238
+
239
+ const rawRows = db.query(sql).all(...params);
240
+ const rows = (Array.isArray(rawRows) ? rawRows : []).filter(isSearchResultRow);
241
+
242
+ return rows.map((r) => ({
243
+ chunk_id: r.rowid,
244
+ session_id: r.session_id,
245
+ project: r.project,
246
+ segment_index: r.segment_index,
247
+ line_start: r.line_start,
248
+ line_end: r.line_end,
249
+ topic: r.topic,
250
+ date: r.date,
251
+ source_path: r.source_path,
252
+ distance: r.distance,
253
+ }));
254
+ }
255
+
256
+ function isCountResult(obj: unknown): obj is { cnt: number } {
257
+ return typeof obj === "object" && obj !== null && "cnt" in obj && typeof (obj as { cnt: unknown }).cnt === "number";
258
+ }
259
+
260
+ /**
261
+ * Get counts of embedded sessions and chunks.
262
+ */
263
+ export function getStats(db: Database): VectorStats {
264
+ const sessionsRaw = db.query(
265
+ `SELECT COUNT(*) as cnt FROM embedded_sessions`,
266
+ ).get();
267
+ const chunksRaw = db.query(
268
+ `SELECT COUNT(*) as cnt FROM chunks`,
269
+ ).get();
270
+
271
+ const sessionCount = isCountResult(sessionsRaw) ? sessionsRaw.cnt : 0;
272
+ const chunkCount = isCountResult(chunksRaw) ? chunksRaw.cnt : 0;
273
+
274
+ return {
275
+ session_count: sessionCount,
276
+ chunk_count: chunkCount,
277
+ };
278
+ }