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