mindlore 0.1.0 → 0.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (98) hide show
  1. package/README.md +13 -5
  2. package/SCHEMA.md +60 -4
  3. package/dist/scripts/init.d.ts +10 -0
  4. package/dist/scripts/init.d.ts.map +1 -0
  5. package/dist/scripts/init.js +375 -0
  6. package/dist/scripts/init.js.map +1 -0
  7. package/dist/scripts/lib/constants.d.ts +29 -0
  8. package/dist/scripts/lib/constants.d.ts.map +1 -0
  9. package/dist/scripts/lib/constants.js +58 -0
  10. package/dist/scripts/lib/constants.js.map +1 -0
  11. package/dist/scripts/mindlore-fts5-index.d.ts +9 -0
  12. package/dist/scripts/mindlore-fts5-index.d.ts.map +1 -0
  13. package/dist/scripts/mindlore-fts5-index.js +89 -0
  14. package/dist/scripts/mindlore-fts5-index.js.map +1 -0
  15. package/dist/scripts/mindlore-fts5-search.d.ts +10 -0
  16. package/dist/scripts/mindlore-fts5-search.d.ts.map +1 -0
  17. package/dist/scripts/mindlore-fts5-search.js +108 -0
  18. package/dist/scripts/mindlore-fts5-search.js.map +1 -0
  19. package/dist/scripts/mindlore-health-check.d.ts +10 -0
  20. package/dist/scripts/mindlore-health-check.d.ts.map +1 -0
  21. package/dist/scripts/mindlore-health-check.js +337 -0
  22. package/dist/scripts/mindlore-health-check.js.map +1 -0
  23. package/dist/scripts/uninstall.d.ts +10 -0
  24. package/dist/scripts/uninstall.d.ts.map +1 -0
  25. package/dist/scripts/uninstall.js +143 -0
  26. package/dist/scripts/uninstall.js.map +1 -0
  27. package/dist/tests/compounding.test.d.ts +8 -0
  28. package/dist/tests/compounding.test.d.ts.map +1 -0
  29. package/dist/tests/compounding.test.js +51 -0
  30. package/dist/tests/compounding.test.js.map +1 -0
  31. package/dist/tests/decision.test.d.ts +2 -0
  32. package/dist/tests/decision.test.d.ts.map +1 -0
  33. package/dist/tests/decision.test.js +61 -0
  34. package/dist/tests/decision.test.js.map +1 -0
  35. package/dist/tests/dedup.test.d.ts +2 -0
  36. package/dist/tests/dedup.test.d.ts.map +1 -0
  37. package/dist/tests/dedup.test.js +74 -0
  38. package/dist/tests/dedup.test.js.map +1 -0
  39. package/dist/tests/frontmatter.test.d.ts +2 -0
  40. package/dist/tests/frontmatter.test.d.ts.map +1 -0
  41. package/dist/tests/frontmatter.test.js +90 -0
  42. package/dist/tests/frontmatter.test.js.map +1 -0
  43. package/dist/tests/fts5.test.d.ts +2 -0
  44. package/dist/tests/fts5.test.d.ts.map +1 -0
  45. package/dist/tests/fts5.test.js +95 -0
  46. package/dist/tests/fts5.test.js.map +1 -0
  47. package/dist/tests/helpers/db.d.ts +7 -0
  48. package/dist/tests/helpers/db.d.ts.map +1 -0
  49. package/dist/tests/helpers/db.js +46 -0
  50. package/dist/tests/helpers/db.js.map +1 -0
  51. package/dist/tests/hook-smoke.test.d.ts +2 -0
  52. package/dist/tests/hook-smoke.test.d.ts.map +1 -0
  53. package/dist/tests/hook-smoke.test.js +58 -0
  54. package/dist/tests/hook-smoke.test.js.map +1 -0
  55. package/dist/tests/init.test.d.ts +2 -0
  56. package/dist/tests/init.test.d.ts.map +1 -0
  57. package/dist/tests/init.test.js +85 -0
  58. package/dist/tests/init.test.js.map +1 -0
  59. package/dist/tests/log.test.d.ts +2 -0
  60. package/dist/tests/log.test.d.ts.map +1 -0
  61. package/dist/tests/log.test.js +68 -0
  62. package/dist/tests/log.test.js.map +1 -0
  63. package/dist/tests/read-guard.test.d.ts +2 -0
  64. package/dist/tests/read-guard.test.d.ts.map +1 -0
  65. package/dist/tests/read-guard.test.js +69 -0
  66. package/dist/tests/read-guard.test.js.map +1 -0
  67. package/dist/tests/search-hook.test.d.ts +2 -0
  68. package/dist/tests/search-hook.test.d.ts.map +1 -0
  69. package/dist/tests/search-hook.test.js +108 -0
  70. package/dist/tests/search-hook.test.js.map +1 -0
  71. package/dist/tests/session-focus.test.d.ts +2 -0
  72. package/dist/tests/session-focus.test.d.ts.map +1 -0
  73. package/dist/tests/session-focus.test.js +71 -0
  74. package/dist/tests/session-focus.test.js.map +1 -0
  75. package/dist/tests/uninstall.test.d.ts +2 -0
  76. package/dist/tests/uninstall.test.d.ts.map +1 -0
  77. package/dist/tests/uninstall.test.js +98 -0
  78. package/dist/tests/uninstall.test.js.map +1 -0
  79. package/hooks/lib/mindlore-common.cjs +36 -2
  80. package/hooks/mindlore-decision-detector.cjs +51 -0
  81. package/hooks/mindlore-fts5-sync.cjs +4 -18
  82. package/hooks/mindlore-index.cjs +4 -18
  83. package/hooks/mindlore-read-guard.cjs +62 -0
  84. package/hooks/mindlore-search.cjs +5 -18
  85. package/hooks/mindlore-session-end.cjs +74 -8
  86. package/package.json +19 -7
  87. package/plugin.json +26 -2
  88. package/skills/mindlore-decide/SKILL.md +71 -0
  89. package/skills/mindlore-ingest/SKILL.md +13 -2
  90. package/skills/mindlore-log/SKILL.md +79 -0
  91. package/skills/mindlore-query/SKILL.md +125 -0
  92. package/templates/SCHEMA.md +60 -4
  93. package/scripts/init.cjs +0 -450
  94. package/scripts/lib/constants.cjs +0 -49
  95. package/scripts/mindlore-fts5-index.cjs +0 -112
  96. package/scripts/mindlore-fts5-search.cjs +0 -119
  97. package/scripts/mindlore-health-check.cjs +0 -336
  98. package/scripts/uninstall.cjs +0 -186
package/scripts/init.cjs DELETED
@@ -1,450 +0,0 @@
1
- #!/usr/bin/env node
2
- 'use strict';
3
-
4
- /**
5
- * mindlore init — Initialize .mindlore/ knowledge base in current project.
6
- *
7
- * Usage: npx mindlore init [--recommended]
8
- *
9
- * Idempotent: running again does not destroy existing data.
10
- */
11
-
12
- const fs = require('fs');
13
- const path = require('path');
14
-
15
- // ── Constants ──────────────────────────────────────────────────────────
16
-
17
- const { MINDLORE_DIR, DB_NAME, DIRECTORIES, homedir } = require('./lib/constants.cjs');
18
-
19
- const TEMPLATE_FILES = ['INDEX.md', 'log.md'];
20
-
21
- // ── Helpers ────────────────────────────────────────────────────────────
22
-
23
- function log(msg) {
24
- console.log(` ${msg}`);
25
- }
26
-
27
- function resolvePackageRoot() {
28
- // When installed globally via npm, __dirname is inside the package
29
- // Look for templates/ relative to this script
30
- return path.resolve(__dirname, '..');
31
- }
32
-
33
- function ensureDir(dirPath) {
34
- if (!fs.existsSync(dirPath)) {
35
- fs.mkdirSync(dirPath, { recursive: true });
36
- return true;
37
- }
38
- return false;
39
- }
40
-
41
- // ── Step 1: Create .mindlore/ directories ──────────────────────────────
42
-
43
- function createDirectories(baseDir) {
44
- let created = 0;
45
- for (const dir of DIRECTORIES) {
46
- if (ensureDir(path.join(baseDir, dir))) {
47
- created++;
48
- }
49
- }
50
- return created;
51
- }
52
-
53
- // ── Step 2: Copy template files ────────────────────────────────────────
54
-
55
- function copyTemplates(baseDir, packageRoot) {
56
- const templatesDir = path.join(packageRoot, 'templates');
57
- let copied = 0;
58
-
59
- for (const file of TEMPLATE_FILES) {
60
- const dest = path.join(baseDir, file);
61
- if (!fs.existsSync(dest)) {
62
- const src = path.join(templatesDir, file);
63
- if (fs.existsSync(src)) {
64
- fs.copyFileSync(src, dest);
65
- copied++;
66
- } else {
67
- log(`WARNING: template not found: ${src}`);
68
- }
69
- }
70
- }
71
-
72
- // Copy SCHEMA.md
73
- const schemaSrc = path.join(packageRoot, 'SCHEMA.md');
74
- const schemaDest = path.join(baseDir, 'SCHEMA.md');
75
- if (!fs.existsSync(schemaDest)) {
76
- if (fs.existsSync(schemaSrc)) {
77
- fs.copyFileSync(schemaSrc, schemaDest);
78
- copied++;
79
- }
80
- }
81
-
82
- return copied;
83
- }
84
-
85
- // ── Step 3: Create FTS5 database ───────────────────────────────────────
86
-
87
- function migrateDatabase(dbPath, Database) {
88
- const db = new Database(dbPath);
89
- try {
90
- // Check if FTS5 table has the new schema (7 columns with slug, description, etc.)
91
- const info = db.pragma('table_info(mindlore_fts)');
92
- const columns = info.map((r) => r.name);
93
- if (!columns.includes('slug') || !columns.includes('description')) {
94
- log('Upgrading FTS5 schema (2 → 7 columns, porter stemmer)...');
95
- db.exec('DROP TABLE IF EXISTS mindlore_fts');
96
- db.exec(`
97
- CREATE VIRTUAL TABLE mindlore_fts
98
- USING fts5(path UNINDEXED, slug, description, type UNINDEXED, category, title, content, tokenize='porter unicode61');
99
- `);
100
- // Clear hashes so full re-index happens
101
- db.exec('DELETE FROM file_hashes');
102
- db.close();
103
- return true;
104
- }
105
- } catch (_err) {
106
- // table_info fails on FTS5 virtual tables in some versions — recreate
107
- db.exec('DROP TABLE IF EXISTS mindlore_fts');
108
- db.exec(`
109
- CREATE VIRTUAL TABLE mindlore_fts
110
- USING fts5(path UNINDEXED, slug, description, type UNINDEXED, category, title, content, tokenize='porter unicode61');
111
- `);
112
- db.exec('DELETE FROM file_hashes');
113
- db.close();
114
- return true;
115
- }
116
- db.close();
117
- return false;
118
- }
119
-
120
- function createDatabase(baseDir) {
121
- const dbPath = path.join(baseDir, DB_NAME);
122
- if (fs.existsSync(dbPath)) {
123
- let Database;
124
- try { Database = require('better-sqlite3'); } catch (_err) { return false; }
125
- const migrated = migrateDatabase(dbPath, Database);
126
- if (migrated) {
127
- log('FTS5 schema upgraded — run index to rebuild');
128
- } else {
129
- log('Database already exists, schema OK');
130
- }
131
- return migrated;
132
- }
133
-
134
- let Database;
135
- try {
136
- Database = require('better-sqlite3');
137
- } catch (_err) {
138
- log('WARNING: better-sqlite3 not installed. Run: npm install better-sqlite3');
139
- log('Database creation skipped — run mindlore init again after installing.');
140
- return false;
141
- }
142
-
143
- const db = new Database(dbPath);
144
- db.pragma('journal_mode = WAL');
145
-
146
- db.exec(`
147
- CREATE VIRTUAL TABLE IF NOT EXISTS mindlore_fts
148
- USING fts5(path UNINDEXED, slug, description, type UNINDEXED, category, title, content, tokenize='porter unicode61');
149
- `);
150
-
151
- db.exec(`
152
- CREATE TABLE IF NOT EXISTS file_hashes (
153
- path TEXT PRIMARY KEY,
154
- content_hash TEXT NOT NULL,
155
- last_indexed TEXT NOT NULL
156
- );
157
- `);
158
-
159
- db.close();
160
- return true;
161
- }
162
-
163
- // ── Step 4: Merge hooks into settings.json ─────────────────────────────
164
-
165
- function mergeHooks(packageRoot) {
166
- const settingsPath = path.join(
167
- homedir(),
168
- '.claude',
169
- 'settings.json'
170
- );
171
-
172
- if (!fs.existsSync(settingsPath)) {
173
- log('WARNING: ~/.claude/settings.json not found. Hooks not registered.');
174
- log('Create it manually or install Claude Code first.');
175
- return false;
176
- }
177
-
178
- let settings;
179
- try {
180
- const raw = fs.readFileSync(settingsPath, 'utf8');
181
- settings = JSON.parse(raw);
182
- } catch (_err) {
183
- log('WARNING: Could not parse settings.json. Hooks not registered.');
184
- return false;
185
- }
186
-
187
- // Read plugin.json for hook definitions
188
- const pluginPath = path.join(packageRoot, 'plugin.json');
189
- if (!fs.existsSync(pluginPath)) {
190
- log('WARNING: plugin.json not found. Hooks not registered.');
191
- return false;
192
- }
193
-
194
- const plugin = JSON.parse(fs.readFileSync(pluginPath, 'utf8'));
195
- if (!plugin.hooks || plugin.hooks.length === 0) {
196
- return false;
197
- }
198
-
199
- if (!settings.hooks) {
200
- settings.hooks = {};
201
- }
202
-
203
- let added = 0;
204
- for (const hook of plugin.hooks) {
205
- const event = hook.event;
206
- if (!settings.hooks[event]) {
207
- settings.hooks[event] = [];
208
- }
209
-
210
- // Check if this hook already exists (by script path containing 'mindlore-')
211
- const hookScript = path.join(packageRoot, hook.script);
212
- const hookName = path.basename(hook.script, '.cjs');
213
-
214
- const exists = settings.hooks[event].some((entry) => {
215
- // CC format: each entry is { hooks: [{ type, command }] }
216
- if (entry.hooks && Array.isArray(entry.hooks)) {
217
- return entry.hooks.some((h) => (h.command || '').includes(hookName));
218
- }
219
- // Legacy flat format check
220
- return (entry.command || '').includes(hookName);
221
- });
222
-
223
- if (!exists) {
224
- settings.hooks[event].push({
225
- hooks: [
226
- {
227
- type: 'command',
228
- command: `node "${hookScript}"`,
229
- },
230
- ],
231
- });
232
- added++;
233
- }
234
- }
235
-
236
- if (added > 0) {
237
- // Backup before writing
238
- const backupPath = settingsPath + '.mindlore-backup';
239
- if (!fs.existsSync(backupPath)) {
240
- fs.copyFileSync(settingsPath, backupPath);
241
- }
242
- fs.writeFileSync(settingsPath, JSON.stringify(settings, null, 2), 'utf8');
243
- }
244
-
245
- return added;
246
- }
247
-
248
- // ── Step 5: Add SCHEMA.md to projectDocFiles ───────────────────────────
249
-
250
- function addSchemaToProjectDocs() {
251
- const projectSettingsDir = path.join(process.cwd(), '.claude');
252
- const projectSettingsPath = path.join(projectSettingsDir, 'settings.json');
253
-
254
- let settings = {};
255
- if (fs.existsSync(projectSettingsPath)) {
256
- try {
257
- settings = JSON.parse(fs.readFileSync(projectSettingsPath, 'utf8'));
258
- } catch (_err) {
259
- settings = {};
260
- }
261
- } else {
262
- ensureDir(projectSettingsDir);
263
- }
264
-
265
- if (!settings.projectDocFiles) {
266
- settings.projectDocFiles = [];
267
- }
268
-
269
- const schemaPath = path.join(MINDLORE_DIR, 'SCHEMA.md');
270
- if (!settings.projectDocFiles.includes(schemaPath)) {
271
- settings.projectDocFiles.push(schemaPath);
272
- fs.writeFileSync(
273
- projectSettingsPath,
274
- JSON.stringify(settings, null, 2),
275
- 'utf8'
276
- );
277
- return true;
278
- }
279
- return false;
280
- }
281
-
282
- // ── Step 6: Register skills ────────────────────────────────────────────
283
-
284
- function registerSkills(packageRoot, plugin) {
285
- const skillsDir = path.join(homedir(), '.claude', 'skills');
286
- ensureDir(skillsDir);
287
-
288
- if (!plugin.skills || plugin.skills.length === 0) return 0;
289
-
290
- let added = 0;
291
- for (const skill of plugin.skills) {
292
- const skillSrcDir = path.join(packageRoot, path.dirname(skill.path));
293
- const skillDestDir = path.join(skillsDir, skill.name);
294
-
295
- ensureDir(skillDestDir);
296
- const entries = fs.readdirSync(skillSrcDir, { withFileTypes: true });
297
- for (const entry of entries) {
298
- if (!entry.isFile()) continue;
299
- fs.copyFileSync(
300
- path.join(skillSrcDir, entry.name),
301
- path.join(skillDestDir, entry.name)
302
- );
303
- }
304
- added++;
305
- }
306
-
307
- return added;
308
- }
309
-
310
- // ── Step 7: Install better-sqlite3 if needed ──────────────────────────
311
-
312
- function ensureBetterSqlite3() {
313
- try {
314
- require('better-sqlite3');
315
- return true;
316
- } catch (_err) {
317
- try {
318
- const { execSync } = require('child_process');
319
- log('Installing better-sqlite3 (native dependency)...');
320
- execSync('npm install better-sqlite3 --no-save', {
321
- cwd: process.cwd(),
322
- stdio: 'pipe',
323
- timeout: 120000,
324
- });
325
- return true;
326
- } catch (_installErr) {
327
- log('WARNING: Could not install better-sqlite3. FTS5 search disabled.');
328
- log(' Run manually: npm install better-sqlite3');
329
- return false;
330
- }
331
- }
332
- }
333
-
334
- // ── Step 8: Add .mindlore/ to .gitignore ───────────────────────────────
335
-
336
- function addToGitignore() {
337
- const gitignorePath = path.join(process.cwd(), '.gitignore');
338
- const entry = '.mindlore/';
339
-
340
- if (fs.existsSync(gitignorePath)) {
341
- const content = fs.readFileSync(gitignorePath, 'utf8');
342
- if (content.includes(entry)) {
343
- return false;
344
- }
345
- fs.appendFileSync(gitignorePath, `\n${entry}\n`, 'utf8');
346
- } else {
347
- fs.writeFileSync(gitignorePath, `${entry}\n`, 'utf8');
348
- }
349
- return true;
350
- }
351
-
352
- // ── Main ───────────────────────────────────────────────────────────────
353
-
354
- function main() {
355
- const args = process.argv.slice(2);
356
- const command = args[0];
357
-
358
- if (command === 'uninstall') {
359
- require('./uninstall.cjs');
360
- return;
361
- }
362
-
363
- if (command && command !== 'init') {
364
- console.log(`Unknown command: ${command}`);
365
- console.log('Usage: npx mindlore init [--recommended]');
366
- console.log(' npx mindlore uninstall [--all]');
367
- process.exit(1);
368
- }
369
-
370
- const isRecommended = args.includes('--recommended');
371
- const packageRoot = resolvePackageRoot();
372
- const baseDir = path.join(process.cwd(), MINDLORE_DIR);
373
-
374
- console.log('\n Mindlore — AI-native knowledge system\n');
375
-
376
- // Step 1: Directories
377
- const dirsCreated = createDirectories(baseDir);
378
- log(
379
- dirsCreated > 0
380
- ? `Created ${dirsCreated} directories in ${MINDLORE_DIR}/`
381
- : 'All directories already exist'
382
- );
383
-
384
- // Step 2: Templates
385
- const filesCopied = copyTemplates(baseDir, packageRoot);
386
- log(
387
- filesCopied > 0
388
- ? `Copied ${filesCopied} template files`
389
- : 'All templates already in place'
390
- );
391
-
392
- // Step 3: better-sqlite3 (before DB creation so it's available)
393
- ensureBetterSqlite3();
394
-
395
- // Step 4: Database
396
- const dbCreated = createDatabase(baseDir);
397
- log(dbCreated ? 'Created FTS5 database' : 'Database already exists');
398
-
399
- // Read plugin.json once for hooks + skills
400
- const pluginPath = path.join(packageRoot, 'plugin.json');
401
- const plugin = fs.existsSync(pluginPath)
402
- ? JSON.parse(fs.readFileSync(pluginPath, 'utf8'))
403
- : {};
404
-
405
- // Step 5: Hooks
406
- const hooksAdded = mergeHooks(packageRoot);
407
- if (typeof hooksAdded === 'number' && hooksAdded > 0) {
408
- log(`Registered ${hooksAdded} hooks in ~/.claude/settings.json`);
409
- } else {
410
- log('Hooks already registered (or settings.json not found)');
411
- }
412
-
413
- // Step 6: SCHEMA.md in projectDocFiles
414
- const schemaAdded = addSchemaToProjectDocs();
415
- log(
416
- schemaAdded
417
- ? 'Added SCHEMA.md to project settings'
418
- : 'SCHEMA.md already in project settings'
419
- );
420
-
421
- // Step 7: Skills
422
- const skillsAdded = registerSkills(packageRoot, plugin);
423
- log(
424
- skillsAdded > 0
425
- ? `Registered ${skillsAdded} skills in ~/.claude/skills/`
426
- : 'Skills already registered'
427
- );
428
-
429
- // Step 8: .gitignore
430
- const gitignoreAdded = addToGitignore();
431
- log(
432
- gitignoreAdded
433
- ? 'Added .mindlore/ to .gitignore'
434
- : '.mindlore/ already in .gitignore'
435
- );
436
-
437
- // Recommended profile tips
438
- if (isRecommended) {
439
- console.log('\n Recommended setup:');
440
- log('Install markitdown for better web/doc extraction:');
441
- log(' pip install markitdown');
442
- log('');
443
- log('Install context-mode for token savings:');
444
- log(' See: https://github.com/context-mode/context-mode');
445
- }
446
-
447
- console.log('\n Done! Start with: /mindlore-ingest\n');
448
- }
449
-
450
- main();
@@ -1,49 +0,0 @@
1
- 'use strict';
2
-
3
- /**
4
- * Shared constants and utilities for mindlore scripts.
5
- */
6
-
7
- const os = require('os');
8
-
9
- const MINDLORE_DIR = '.mindlore';
10
- const DB_NAME = 'mindlore.db';
11
-
12
- const DIRECTORIES = [
13
- 'raw',
14
- 'sources',
15
- 'domains',
16
- 'analyses',
17
- 'insights',
18
- 'connections',
19
- 'learnings',
20
- 'diary',
21
- 'decisions',
22
- ];
23
-
24
- const SKIP_FILES = new Set(['INDEX.md', 'SCHEMA.md', 'log.md']);
25
-
26
- const TYPE_TO_DIR = {
27
- raw: 'raw',
28
- source: 'sources',
29
- domain: 'domains',
30
- analysis: 'analyses',
31
- insight: 'insights',
32
- connection: 'connections',
33
- learning: 'learnings',
34
- decision: 'decisions',
35
- diary: 'diary',
36
- };
37
-
38
- function homedir() {
39
- return os.homedir();
40
- }
41
-
42
- module.exports = {
43
- MINDLORE_DIR,
44
- DB_NAME,
45
- DIRECTORIES,
46
- SKIP_FILES,
47
- TYPE_TO_DIR,
48
- homedir,
49
- };
@@ -1,112 +0,0 @@
1
- #!/usr/bin/env node
2
- 'use strict';
3
-
4
- /**
5
- * mindlore-fts5-index — Full re-index of .mindlore/ into FTS5 database.
6
- *
7
- * Scans all .md files, computes SHA256 content-hash, skips unchanged files.
8
- * Usage: node scripts/mindlore-fts5-index.cjs [path-to-mindlore-dir]
9
- */
10
-
11
- const fs = require('fs');
12
- const path = require('path');
13
- // ── Constants ──────────────────────────────────────────────────────────
14
-
15
- const { DB_NAME } = require('./lib/constants.cjs');
16
- const { sha256, getAllMdFiles, openDatabase, parseFrontmatter, extractFtsMetadata, SQL_FTS_INSERT } = require('../hooks/lib/mindlore-common.cjs');
17
-
18
- // ── Main ───────────────────────────────────────────────────────────────
19
-
20
- function main() {
21
- const baseDir = process.argv[2] || path.join(process.cwd(), '.mindlore');
22
- const dbPath = path.join(baseDir, DB_NAME);
23
-
24
- if (!fs.existsSync(dbPath)) {
25
- console.error(' Database not found. Run: npx mindlore init');
26
- process.exit(1);
27
- }
28
-
29
- const db = openDatabase(dbPath);
30
- if (!db) {
31
- console.error(' better-sqlite3 not installed. Run: npm install better-sqlite3');
32
- process.exit(1);
33
- }
34
-
35
- // Prepare statements
36
- const getHash = db.prepare('SELECT content_hash FROM file_hashes WHERE path = ?');
37
- const upsertHash = db.prepare(`
38
- INSERT INTO file_hashes (path, content_hash, last_indexed)
39
- VALUES (?, ?, ?)
40
- ON CONFLICT(path) DO UPDATE SET
41
- content_hash = excluded.content_hash,
42
- last_indexed = excluded.last_indexed
43
- `);
44
- const deleteFts = db.prepare('DELETE FROM mindlore_fts WHERE path = ?');
45
- const insertFts = db.prepare(SQL_FTS_INSERT);
46
-
47
- // Get all .md files
48
- const mdFiles = getAllMdFiles(baseDir);
49
- let indexed = 0;
50
- let skipped = 0;
51
- let errors = 0;
52
-
53
- const now = new Date().toISOString();
54
-
55
- const transaction = db.transaction(() => {
56
- for (const filePath of mdFiles) {
57
- try {
58
- const content = fs.readFileSync(filePath, 'utf8').replace(/\r\n/g, '\n');
59
- const hash = sha256(content);
60
-
61
- // Check if content changed
62
- const existing = getHash.get(filePath);
63
- if (existing && existing.content_hash === hash) {
64
- skipped++;
65
- continue;
66
- }
67
-
68
- // Update FTS5
69
- const { meta, body } = parseFrontmatter(content);
70
- const { slug, description, type, category, title } = extractFtsMetadata(meta, body, filePath, baseDir);
71
- deleteFts.run(filePath);
72
- insertFts.run(filePath, slug, description, type, category, title, body);
73
-
74
- // Update hash
75
- upsertHash.run(filePath, hash, now);
76
- indexed++;
77
- } catch (err) {
78
- console.error(` Error indexing ${path.basename(filePath)}: ${err.message}`);
79
- errors++;
80
- }
81
- }
82
- });
83
-
84
- transaction();
85
-
86
- // Clean up entries for deleted files
87
- const allIndexed = db.prepare('SELECT path FROM file_hashes').all();
88
- const existingPaths = new Set(mdFiles);
89
- let removed = 0;
90
-
91
- const deleteHash = db.prepare('DELETE FROM file_hashes WHERE path = ?');
92
- const cleanupTransaction = db.transaction(() => {
93
- for (const row of allIndexed) {
94
- if (!existingPaths.has(row.path)) {
95
- deleteFts.run(row.path);
96
- deleteHash.run(row.path);
97
- removed++;
98
- }
99
- }
100
- });
101
-
102
- cleanupTransaction();
103
- db.close();
104
-
105
- console.log(
106
- `\n FTS5 Index: ${indexed} indexed, ${skipped} unchanged, ${removed} removed, ${errors} errors\n`
107
- );
108
-
109
- process.exit(errors > 0 ? 1 : 0);
110
- }
111
-
112
- main();