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.
- package/README.md +7 -1
- package/dist/scripts/init.d.ts +10 -0
- package/dist/scripts/init.d.ts.map +1 -0
- package/dist/scripts/init.js +375 -0
- package/dist/scripts/init.js.map +1 -0
- package/dist/scripts/lib/constants.d.ts +29 -0
- package/dist/scripts/lib/constants.d.ts.map +1 -0
- package/dist/scripts/lib/constants.js +58 -0
- package/dist/scripts/lib/constants.js.map +1 -0
- package/dist/scripts/mindlore-fts5-index.d.ts +9 -0
- package/dist/scripts/mindlore-fts5-index.d.ts.map +1 -0
- package/dist/scripts/mindlore-fts5-index.js +89 -0
- package/dist/scripts/mindlore-fts5-index.js.map +1 -0
- package/dist/scripts/mindlore-fts5-search.d.ts +10 -0
- package/dist/scripts/mindlore-fts5-search.d.ts.map +1 -0
- package/dist/scripts/mindlore-fts5-search.js +108 -0
- package/dist/scripts/mindlore-fts5-search.js.map +1 -0
- package/dist/scripts/mindlore-health-check.d.ts +10 -0
- package/dist/scripts/mindlore-health-check.d.ts.map +1 -0
- package/dist/scripts/mindlore-health-check.js +337 -0
- package/dist/scripts/mindlore-health-check.js.map +1 -0
- package/dist/scripts/uninstall.d.ts +10 -0
- package/dist/scripts/uninstall.d.ts.map +1 -0
- package/dist/scripts/uninstall.js +143 -0
- package/dist/scripts/uninstall.js.map +1 -0
- package/dist/tests/compounding.test.d.ts +8 -0
- package/dist/tests/compounding.test.d.ts.map +1 -0
- package/dist/tests/compounding.test.js +51 -0
- package/dist/tests/compounding.test.js.map +1 -0
- package/dist/tests/decision.test.d.ts +2 -0
- package/dist/tests/decision.test.d.ts.map +1 -0
- package/dist/tests/decision.test.js +61 -0
- package/dist/tests/decision.test.js.map +1 -0
- package/dist/tests/dedup.test.d.ts +2 -0
- package/dist/tests/dedup.test.d.ts.map +1 -0
- package/dist/tests/dedup.test.js +74 -0
- package/dist/tests/dedup.test.js.map +1 -0
- package/dist/tests/frontmatter.test.d.ts +2 -0
- package/dist/tests/frontmatter.test.d.ts.map +1 -0
- package/dist/tests/frontmatter.test.js +90 -0
- package/dist/tests/frontmatter.test.js.map +1 -0
- package/dist/tests/fts5.test.d.ts +2 -0
- package/dist/tests/fts5.test.d.ts.map +1 -0
- package/dist/tests/fts5.test.js +95 -0
- package/dist/tests/fts5.test.js.map +1 -0
- package/dist/tests/helpers/db.d.ts +7 -0
- package/dist/tests/helpers/db.d.ts.map +1 -0
- package/dist/tests/helpers/db.js +46 -0
- package/dist/tests/helpers/db.js.map +1 -0
- package/dist/tests/hook-smoke.test.d.ts +2 -0
- package/dist/tests/hook-smoke.test.d.ts.map +1 -0
- package/dist/tests/hook-smoke.test.js +58 -0
- package/dist/tests/hook-smoke.test.js.map +1 -0
- package/dist/tests/init.test.d.ts +2 -0
- package/dist/tests/init.test.d.ts.map +1 -0
- package/dist/tests/init.test.js +85 -0
- package/dist/tests/init.test.js.map +1 -0
- package/dist/tests/log.test.d.ts +2 -0
- package/dist/tests/log.test.d.ts.map +1 -0
- package/dist/tests/log.test.js +68 -0
- package/dist/tests/log.test.js.map +1 -0
- package/dist/tests/read-guard.test.d.ts +2 -0
- package/dist/tests/read-guard.test.d.ts.map +1 -0
- package/dist/tests/read-guard.test.js +69 -0
- package/dist/tests/read-guard.test.js.map +1 -0
- package/dist/tests/search-hook.test.d.ts +2 -0
- package/dist/tests/search-hook.test.d.ts.map +1 -0
- package/dist/tests/search-hook.test.js +108 -0
- package/dist/tests/search-hook.test.js.map +1 -0
- package/dist/tests/session-focus.test.d.ts +2 -0
- package/dist/tests/session-focus.test.d.ts.map +1 -0
- package/dist/tests/session-focus.test.js +71 -0
- package/dist/tests/session-focus.test.js.map +1 -0
- package/dist/tests/uninstall.test.d.ts +2 -0
- package/dist/tests/uninstall.test.d.ts.map +1 -0
- package/dist/tests/uninstall.test.js +98 -0
- package/dist/tests/uninstall.test.js.map +1 -0
- package/hooks/mindlore-read-guard.cjs +1 -1
- package/package.json +19 -7
- package/scripts/init.cjs +0 -448
- package/scripts/lib/constants.cjs +0 -49
- package/scripts/mindlore-fts5-index.cjs +0 -112
- package/scripts/mindlore-fts5-search.cjs +0 -119
- package/scripts/mindlore-health-check.cjs +0 -389
- package/scripts/uninstall.cjs +0 -186
package/package.json
CHANGED
|
@@ -1,17 +1,22 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "mindlore",
|
|
3
|
-
"version": "0.2.
|
|
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.
|
|
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
|
-
"
|
|
13
|
-
"
|
|
14
|
-
"
|
|
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
|
-
"
|
|
56
|
+
"ts-jest": "^29.4.9",
|
|
57
|
+
"typescript": "^6.0.2"
|
|
46
58
|
},
|
|
47
59
|
"files": [
|
|
48
|
-
"
|
|
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();
|