@timmeck/brain 1.8.1 → 1.8.3

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 (164) hide show
  1. package/BRAIN_PLAN.md +3324 -3324
  2. package/LICENSE +21 -21
  3. package/dist/cli/commands/dashboard.js +595 -595
  4. package/dist/cli/commands/doctor.js +6 -1
  5. package/dist/cli/commands/doctor.js.map +1 -1
  6. package/dist/dashboard/server.js +25 -25
  7. package/dist/db/migrations/001_core_schema.js +115 -115
  8. package/dist/db/migrations/002_learning_schema.js +33 -33
  9. package/dist/db/migrations/003_code_schema.js +48 -48
  10. package/dist/db/migrations/004_synapses_schema.js +52 -52
  11. package/dist/db/migrations/005_fts_indexes.js +73 -73
  12. package/dist/db/migrations/007_feedback.js +8 -8
  13. package/dist/db/migrations/008_git_integration.js +33 -33
  14. package/dist/db/migrations/009_embeddings.js +3 -3
  15. package/dist/db/repositories/antipattern.repository.js +3 -3
  16. package/dist/db/repositories/code-module.repository.js +32 -32
  17. package/dist/db/repositories/notification.repository.js +3 -3
  18. package/dist/db/repositories/project.repository.js +21 -21
  19. package/dist/db/repositories/rule.repository.js +24 -24
  20. package/dist/db/repositories/solution.repository.js +50 -50
  21. package/dist/db/repositories/synapse.repository.js +18 -18
  22. package/dist/db/repositories/terminal.repository.js +24 -24
  23. package/dist/ipc/server.d.ts +8 -0
  24. package/dist/ipc/server.js +67 -1
  25. package/dist/ipc/server.js.map +1 -1
  26. package/dist/matching/error-matcher.js +5 -5
  27. package/dist/matching/fingerprint.js +6 -1
  28. package/dist/matching/fingerprint.js.map +1 -1
  29. package/dist/services/error.service.js +4 -3
  30. package/dist/services/error.service.js.map +1 -1
  31. package/dist/services/git.service.js +14 -14
  32. package/package.json +49 -49
  33. package/src/api/server.ts +395 -395
  34. package/src/brain.ts +266 -266
  35. package/src/cli/colors.ts +116 -116
  36. package/src/cli/commands/config.ts +169 -169
  37. package/src/cli/commands/dashboard.ts +755 -755
  38. package/src/cli/commands/doctor.ts +124 -118
  39. package/src/cli/commands/explain.ts +83 -83
  40. package/src/cli/commands/export.ts +31 -31
  41. package/src/cli/commands/import.ts +199 -199
  42. package/src/cli/commands/insights.ts +65 -65
  43. package/src/cli/commands/learn.ts +24 -24
  44. package/src/cli/commands/modules.ts +53 -53
  45. package/src/cli/commands/network.ts +67 -67
  46. package/src/cli/commands/projects.ts +42 -42
  47. package/src/cli/commands/query.ts +120 -120
  48. package/src/cli/commands/start.ts +62 -62
  49. package/src/cli/commands/status.ts +75 -75
  50. package/src/cli/commands/stop.ts +34 -34
  51. package/src/cli/ipc-helper.ts +22 -22
  52. package/src/cli/update-check.ts +63 -63
  53. package/src/code/fingerprint.ts +87 -87
  54. package/src/code/parsers/generic.ts +29 -29
  55. package/src/code/parsers/python.ts +54 -54
  56. package/src/code/parsers/typescript.ts +65 -65
  57. package/src/code/registry.ts +60 -60
  58. package/src/dashboard/server.ts +142 -142
  59. package/src/db/connection.ts +22 -22
  60. package/src/db/migrations/001_core_schema.ts +120 -120
  61. package/src/db/migrations/002_learning_schema.ts +38 -38
  62. package/src/db/migrations/003_code_schema.ts +53 -53
  63. package/src/db/migrations/004_synapses_schema.ts +57 -57
  64. package/src/db/migrations/005_fts_indexes.ts +78 -78
  65. package/src/db/migrations/006_synapses_phase3.ts +17 -17
  66. package/src/db/migrations/007_feedback.ts +13 -13
  67. package/src/db/migrations/008_git_integration.ts +38 -38
  68. package/src/db/migrations/009_embeddings.ts +8 -8
  69. package/src/db/repositories/antipattern.repository.ts +66 -66
  70. package/src/db/repositories/code-module.repository.ts +142 -142
  71. package/src/db/repositories/notification.repository.ts +66 -66
  72. package/src/db/repositories/project.repository.ts +93 -93
  73. package/src/db/repositories/rule.repository.ts +108 -108
  74. package/src/db/repositories/solution.repository.ts +154 -154
  75. package/src/db/repositories/synapse.repository.ts +153 -153
  76. package/src/db/repositories/terminal.repository.ts +101 -101
  77. package/src/embeddings/engine.ts +238 -238
  78. package/src/index.ts +63 -63
  79. package/src/ipc/client.ts +118 -118
  80. package/src/ipc/protocol.ts +35 -35
  81. package/src/ipc/router.ts +133 -133
  82. package/src/ipc/server.ts +176 -110
  83. package/src/learning/decay.ts +46 -46
  84. package/src/learning/pattern-extractor.ts +90 -90
  85. package/src/learning/rule-generator.ts +74 -74
  86. package/src/matching/error-matcher.ts +5 -5
  87. package/src/matching/fingerprint.ts +34 -29
  88. package/src/matching/similarity.ts +61 -61
  89. package/src/matching/tfidf.ts +74 -74
  90. package/src/matching/tokenizer.ts +41 -41
  91. package/src/mcp/auto-detect.ts +93 -93
  92. package/src/mcp/http-server.ts +140 -140
  93. package/src/mcp/server.ts +73 -73
  94. package/src/parsing/error-parser.ts +28 -28
  95. package/src/parsing/parsers/compiler.ts +93 -93
  96. package/src/parsing/parsers/generic.ts +28 -28
  97. package/src/parsing/parsers/go.ts +97 -97
  98. package/src/parsing/parsers/node.ts +69 -69
  99. package/src/parsing/parsers/python.ts +62 -62
  100. package/src/parsing/parsers/rust.ts +50 -50
  101. package/src/parsing/parsers/shell.ts +42 -42
  102. package/src/parsing/types.ts +47 -47
  103. package/src/research/gap-analyzer.ts +135 -135
  104. package/src/research/insight-generator.ts +123 -123
  105. package/src/research/research-engine.ts +116 -116
  106. package/src/research/synergy-detector.ts +126 -126
  107. package/src/research/template-extractor.ts +130 -130
  108. package/src/research/trend-analyzer.ts +127 -127
  109. package/src/services/code.service.ts +271 -271
  110. package/src/services/error.service.ts +4 -3
  111. package/src/services/git.service.ts +132 -132
  112. package/src/services/notification.service.ts +41 -41
  113. package/src/services/synapse.service.ts +59 -59
  114. package/src/services/terminal.service.ts +81 -81
  115. package/src/synapses/activation.ts +80 -80
  116. package/src/synapses/decay.ts +38 -38
  117. package/src/synapses/hebbian.ts +69 -69
  118. package/src/synapses/pathfinder.ts +81 -81
  119. package/src/synapses/synapse-manager.ts +109 -109
  120. package/src/types/code.types.ts +52 -52
  121. package/src/types/error.types.ts +67 -67
  122. package/src/types/ipc.types.ts +8 -8
  123. package/src/types/mcp.types.ts +53 -53
  124. package/src/types/research.types.ts +28 -28
  125. package/src/types/solution.types.ts +30 -30
  126. package/src/utils/events.ts +45 -45
  127. package/src/utils/hash.ts +5 -5
  128. package/src/utils/logger.ts +48 -48
  129. package/src/utils/paths.ts +19 -19
  130. package/tests/e2e/test_code_intelligence.py +1015 -0
  131. package/tests/e2e/test_error_memory.py +451 -0
  132. package/tests/e2e/test_full_integration.py +534 -0
  133. package/tests/fixtures/code-modules/modules.ts +83 -83
  134. package/tests/fixtures/errors/go.ts +9 -9
  135. package/tests/fixtures/errors/node.ts +24 -24
  136. package/tests/fixtures/errors/python.ts +21 -21
  137. package/tests/fixtures/errors/rust.ts +25 -25
  138. package/tests/fixtures/errors/shell.ts +15 -15
  139. package/tests/fixtures/solutions/solutions.ts +27 -27
  140. package/tests/helpers/setup-db.ts +52 -52
  141. package/tests/integration/code-flow.test.ts +86 -86
  142. package/tests/integration/error-flow.test.ts +83 -83
  143. package/tests/integration/ipc-flow.test.ts +166 -166
  144. package/tests/integration/learning-cycle.test.ts +82 -82
  145. package/tests/integration/synapse-flow.test.ts +117 -117
  146. package/tests/unit/code/analyzer.test.ts +58 -58
  147. package/tests/unit/code/fingerprint.test.ts +51 -51
  148. package/tests/unit/code/scorer.test.ts +55 -55
  149. package/tests/unit/learning/confidence-scorer.test.ts +60 -60
  150. package/tests/unit/learning/decay.test.ts +45 -45
  151. package/tests/unit/learning/pattern-extractor.test.ts +50 -50
  152. package/tests/unit/matching/error-matcher.test.ts +69 -69
  153. package/tests/unit/matching/fingerprint.test.ts +47 -47
  154. package/tests/unit/matching/similarity.test.ts +65 -65
  155. package/tests/unit/matching/tfidf.test.ts +71 -71
  156. package/tests/unit/matching/tokenizer.test.ts +83 -83
  157. package/tests/unit/parsing/parsers.test.ts +113 -113
  158. package/tests/unit/research/gap-analyzer.test.ts +45 -45
  159. package/tests/unit/research/trend-analyzer.test.ts +45 -45
  160. package/tests/unit/synapses/activation.test.ts +80 -80
  161. package/tests/unit/synapses/decay.test.ts +27 -27
  162. package/tests/unit/synapses/hebbian.test.ts +96 -96
  163. package/tests/unit/synapses/pathfinder.test.ts +72 -72
  164. package/tsconfig.json +18 -18
@@ -1,199 +1,199 @@
1
- import { Command } from 'commander';
2
- import { withIpc } from '../ipc-helper.js';
3
- import { readdirSync, readFileSync, statSync } from 'fs';
4
- import { resolve, basename, relative, extname } from 'path';
5
- import { c, icons, header, divider, progressBar } from '../colors.js';
6
-
7
- const DEFAULT_EXTENSIONS = new Set([
8
- '.ts', '.tsx', '.js', '.jsx', '.py', '.rs', '.go',
9
- '.java', '.c', '.cpp', '.h', '.hpp', '.rb', '.sh',
10
- '.html', '.css', '.scss', '.json', '.yaml', '.yml', '.toml',
11
- '.md', '.sql', '.php', '.svelte', '.vue', '.astro',
12
- ]);
13
-
14
- const EXCLUDE_DIRS = new Set([
15
- 'node_modules', 'dist', 'build', '.git', '.next',
16
- '__pycache__', 'vendor', 'coverage', '.cache', '.turbo',
17
- '.nuxt', '.output', 'target', 'out', 'venv', '.venv',
18
- 'env', '.env', 'site-packages',
19
- ]);
20
-
21
- const EXCLUDE_PATTERNS = [/\.min\./, /\.bundle\./, /\.d\.ts$/];
22
-
23
- const LANG_MAP: Record<string, string> = {
24
- ts: 'typescript', tsx: 'typescript', js: 'javascript', jsx: 'javascript',
25
- py: 'python', rs: 'rust', go: 'go', java: 'java',
26
- c: 'c', cpp: 'cpp', h: 'c', hpp: 'cpp',
27
- rb: 'ruby', sh: 'shell', bash: 'shell',
28
- html: 'html', css: 'css', scss: 'scss',
29
- json: 'json', yaml: 'yaml', yml: 'yaml', toml: 'toml',
30
- md: 'markdown', sql: 'sql', php: 'php',
31
- svelte: 'svelte', vue: 'vue', astro: 'astro',
32
- };
33
-
34
- function detectLanguage(filePath: string): string {
35
- const ext = extname(filePath).slice(1).toLowerCase();
36
- return LANG_MAP[ext] ?? ext;
37
- }
38
-
39
- function findSourceFiles(dir: string, extensions: Set<string>, maxSizeBytes: number): string[] {
40
- const files: string[] = [];
41
-
42
- function walk(current: string): void {
43
- let entries;
44
- try {
45
- entries = readdirSync(current, { withFileTypes: true });
46
- } catch {
47
- return; // skip unreadable directories
48
- }
49
-
50
- for (const entry of entries) {
51
- const fullPath = resolve(current, entry.name);
52
-
53
- if (entry.isDirectory()) {
54
- if (!EXCLUDE_DIRS.has(entry.name)) {
55
- walk(fullPath);
56
- }
57
- continue;
58
- }
59
-
60
- if (!entry.isFile()) continue;
61
-
62
- const ext = extname(entry.name).toLowerCase();
63
- if (!extensions.has(ext)) continue;
64
- if (EXCLUDE_PATTERNS.some(p => p.test(entry.name))) continue;
65
-
66
- try {
67
- const stat = statSync(fullPath);
68
- if (stat.size > maxSizeBytes) continue;
69
- files.push(fullPath);
70
- } catch {
71
- // skip unreadable files
72
- }
73
- }
74
- }
75
-
76
- walk(dir);
77
- return files.sort();
78
- }
79
-
80
- export function importCommand(): Command {
81
- return new Command('import')
82
- .description('Import source files from a project directory into Brain')
83
- .argument('<directory>', 'Project directory to scan')
84
- .option('-p, --project <name>', 'Project name (default: directory basename)')
85
- .option('-e, --extensions <list>', 'Comma-separated extensions (default: ts,tsx,js,jsx,py,rs,go,java,c,cpp,h,hpp,rb,sh,html,css,scss,json,yaml,yml,toml,md,sql,php,svelte,vue,astro)')
86
- .option('--dry-run', 'List files that would be imported without importing')
87
- .option('--max-size <kb>', 'Skip files larger than N KB', '100')
88
- .action(async (directory: string, opts) => {
89
- const dir = resolve(directory);
90
- const projectName = opts.project ?? basename(dir);
91
- const maxSizeKb = parseInt(opts.maxSize, 10) || 100;
92
- const maxSizeBytes = maxSizeKb * 1024;
93
-
94
- // Parse extensions
95
- let extensions = DEFAULT_EXTENSIONS;
96
- if (opts.extensions) {
97
- extensions = new Set(
98
- opts.extensions.split(',').map((e: string) => {
99
- const trimmed = e.trim();
100
- return trimmed.startsWith('.') ? trimmed : `.${trimmed}`;
101
- })
102
- );
103
- }
104
-
105
- // Verify directory exists
106
- try {
107
- const stat = statSync(dir);
108
- if (!stat.isDirectory()) {
109
- console.error(`Not a directory: ${dir}`);
110
- process.exit(1);
111
- }
112
- } catch {
113
- console.error(`Directory not found: ${dir}`);
114
- process.exit(1);
115
- }
116
-
117
- console.log(`${icons.search} ${c.info('Scanning')} ${c.value(dir)} ...`);
118
- const files = findSourceFiles(dir, extensions, maxSizeBytes);
119
-
120
- if (files.length === 0) {
121
- console.log(`${c.dim('No source files found.')}`);
122
- return;
123
- }
124
-
125
- console.log(`${icons.ok} Found ${c.value(files.length)} source files.\n`);
126
-
127
- if (opts.dryRun) {
128
- for (const f of files) {
129
- const rel = relative(dir, f);
130
- const lang = detectLanguage(f);
131
- console.log(` ${c.cyan(`[${lang}]`)} ${c.dim(rel)}`);
132
- }
133
- console.log(`\n${c.value(files.length)} files would be imported as project ${c.cyan(`"${projectName}"`)}.`);
134
- return;
135
- }
136
-
137
- // Import via IPC
138
- await withIpc(async (client) => {
139
- let imported = 0;
140
- let newCount = 0;
141
- let existingCount = 0;
142
- let failedCount = 0;
143
- let totalScore = 0;
144
-
145
- for (let i = 0; i < files.length; i++) {
146
- const filePath = files[i];
147
- const rel = relative(dir, filePath);
148
- const fileName = basename(filePath);
149
- const language = detectLanguage(filePath);
150
-
151
- let source: string;
152
- try {
153
- source = readFileSync(filePath, 'utf-8');
154
- } catch {
155
- failedCount++;
156
- process.stdout.write(` ${c.dim(`[${i + 1}/${files.length}]`)} ${c.dim(rel)} ${c.red('— read error')}\n`);
157
- continue;
158
- }
159
-
160
- // Skip empty files
161
- if (!source.trim()) continue;
162
-
163
- try {
164
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
165
- const result: any = await client.request('code.analyze', {
166
- project: projectName,
167
- name: fileName,
168
- filePath: rel,
169
- language,
170
- source,
171
- });
172
-
173
- const score = result.reusabilityScore ?? 0;
174
- const scoreColor = score >= 0.7 ? c.green : score >= 0.4 ? c.orange : c.red;
175
- const statusTag = result.isNew ? c.green('new') : c.dim('existing');
176
- totalScore += score;
177
- imported++;
178
-
179
- if (result.isNew) newCount++;
180
- else existingCount++;
181
-
182
- process.stdout.write(` ${c.dim(`[${i + 1}/${files.length}]`)} ${c.dim(rel)} ${c.dim(icons.arrow)} ${scoreColor(score.toFixed(2))} (${statusTag})\n`);
183
- } catch (err) {
184
- failedCount++;
185
- const msg = err instanceof Error ? err.message : String(err);
186
- process.stdout.write(` ${c.dim(`[${i + 1}/${files.length}]`)} ${c.dim(rel)} ${c.red(`— ${msg.slice(0, 80)}`)}\n`);
187
- }
188
- }
189
-
190
- const avgScore = imported > 0 ? (totalScore / imported).toFixed(2) : '0';
191
- console.log(header('Import Summary', icons.module));
192
- console.log(` ${c.label('Project:')} ${c.cyan(projectName)}`);
193
- console.log(` ${c.label('Imported:')} ${c.value(imported)} (${c.green(`${newCount} new`)}, ${c.dim(`${existingCount} existing`)})`);
194
- if (failedCount > 0) console.log(` ${c.label('Failed:')} ${c.red(failedCount)}`);
195
- console.log(` ${c.label('Avg score:')} ${c.value(avgScore)} ${progressBar(parseFloat(avgScore), 1)}`);
196
- console.log(divider());
197
- });
198
- });
199
- }
1
+ import { Command } from 'commander';
2
+ import { withIpc } from '../ipc-helper.js';
3
+ import { readdirSync, readFileSync, statSync } from 'fs';
4
+ import { resolve, basename, relative, extname } from 'path';
5
+ import { c, icons, header, divider, progressBar } from '../colors.js';
6
+
7
+ const DEFAULT_EXTENSIONS = new Set([
8
+ '.ts', '.tsx', '.js', '.jsx', '.py', '.rs', '.go',
9
+ '.java', '.c', '.cpp', '.h', '.hpp', '.rb', '.sh',
10
+ '.html', '.css', '.scss', '.json', '.yaml', '.yml', '.toml',
11
+ '.md', '.sql', '.php', '.svelte', '.vue', '.astro',
12
+ ]);
13
+
14
+ const EXCLUDE_DIRS = new Set([
15
+ 'node_modules', 'dist', 'build', '.git', '.next',
16
+ '__pycache__', 'vendor', 'coverage', '.cache', '.turbo',
17
+ '.nuxt', '.output', 'target', 'out', 'venv', '.venv',
18
+ 'env', '.env', 'site-packages',
19
+ ]);
20
+
21
+ const EXCLUDE_PATTERNS = [/\.min\./, /\.bundle\./, /\.d\.ts$/];
22
+
23
+ const LANG_MAP: Record<string, string> = {
24
+ ts: 'typescript', tsx: 'typescript', js: 'javascript', jsx: 'javascript',
25
+ py: 'python', rs: 'rust', go: 'go', java: 'java',
26
+ c: 'c', cpp: 'cpp', h: 'c', hpp: 'cpp',
27
+ rb: 'ruby', sh: 'shell', bash: 'shell',
28
+ html: 'html', css: 'css', scss: 'scss',
29
+ json: 'json', yaml: 'yaml', yml: 'yaml', toml: 'toml',
30
+ md: 'markdown', sql: 'sql', php: 'php',
31
+ svelte: 'svelte', vue: 'vue', astro: 'astro',
32
+ };
33
+
34
+ function detectLanguage(filePath: string): string {
35
+ const ext = extname(filePath).slice(1).toLowerCase();
36
+ return LANG_MAP[ext] ?? ext;
37
+ }
38
+
39
+ function findSourceFiles(dir: string, extensions: Set<string>, maxSizeBytes: number): string[] {
40
+ const files: string[] = [];
41
+
42
+ function walk(current: string): void {
43
+ let entries;
44
+ try {
45
+ entries = readdirSync(current, { withFileTypes: true });
46
+ } catch {
47
+ return; // skip unreadable directories
48
+ }
49
+
50
+ for (const entry of entries) {
51
+ const fullPath = resolve(current, entry.name);
52
+
53
+ if (entry.isDirectory()) {
54
+ if (!EXCLUDE_DIRS.has(entry.name)) {
55
+ walk(fullPath);
56
+ }
57
+ continue;
58
+ }
59
+
60
+ if (!entry.isFile()) continue;
61
+
62
+ const ext = extname(entry.name).toLowerCase();
63
+ if (!extensions.has(ext)) continue;
64
+ if (EXCLUDE_PATTERNS.some(p => p.test(entry.name))) continue;
65
+
66
+ try {
67
+ const stat = statSync(fullPath);
68
+ if (stat.size > maxSizeBytes) continue;
69
+ files.push(fullPath);
70
+ } catch {
71
+ // skip unreadable files
72
+ }
73
+ }
74
+ }
75
+
76
+ walk(dir);
77
+ return files.sort();
78
+ }
79
+
80
+ export function importCommand(): Command {
81
+ return new Command('import')
82
+ .description('Import source files from a project directory into Brain')
83
+ .argument('<directory>', 'Project directory to scan')
84
+ .option('-p, --project <name>', 'Project name (default: directory basename)')
85
+ .option('-e, --extensions <list>', 'Comma-separated extensions (default: ts,tsx,js,jsx,py,rs,go,java,c,cpp,h,hpp,rb,sh,html,css,scss,json,yaml,yml,toml,md,sql,php,svelte,vue,astro)')
86
+ .option('--dry-run', 'List files that would be imported without importing')
87
+ .option('--max-size <kb>', 'Skip files larger than N KB', '100')
88
+ .action(async (directory: string, opts) => {
89
+ const dir = resolve(directory);
90
+ const projectName = opts.project ?? basename(dir);
91
+ const maxSizeKb = parseInt(opts.maxSize, 10) || 100;
92
+ const maxSizeBytes = maxSizeKb * 1024;
93
+
94
+ // Parse extensions
95
+ let extensions = DEFAULT_EXTENSIONS;
96
+ if (opts.extensions) {
97
+ extensions = new Set(
98
+ opts.extensions.split(',').map((e: string) => {
99
+ const trimmed = e.trim();
100
+ return trimmed.startsWith('.') ? trimmed : `.${trimmed}`;
101
+ })
102
+ );
103
+ }
104
+
105
+ // Verify directory exists
106
+ try {
107
+ const stat = statSync(dir);
108
+ if (!stat.isDirectory()) {
109
+ console.error(`Not a directory: ${dir}`);
110
+ process.exit(1);
111
+ }
112
+ } catch {
113
+ console.error(`Directory not found: ${dir}`);
114
+ process.exit(1);
115
+ }
116
+
117
+ console.log(`${icons.search} ${c.info('Scanning')} ${c.value(dir)} ...`);
118
+ const files = findSourceFiles(dir, extensions, maxSizeBytes);
119
+
120
+ if (files.length === 0) {
121
+ console.log(`${c.dim('No source files found.')}`);
122
+ return;
123
+ }
124
+
125
+ console.log(`${icons.ok} Found ${c.value(files.length)} source files.\n`);
126
+
127
+ if (opts.dryRun) {
128
+ for (const f of files) {
129
+ const rel = relative(dir, f);
130
+ const lang = detectLanguage(f);
131
+ console.log(` ${c.cyan(`[${lang}]`)} ${c.dim(rel)}`);
132
+ }
133
+ console.log(`\n${c.value(files.length)} files would be imported as project ${c.cyan(`"${projectName}"`)}.`);
134
+ return;
135
+ }
136
+
137
+ // Import via IPC
138
+ await withIpc(async (client) => {
139
+ let imported = 0;
140
+ let newCount = 0;
141
+ let existingCount = 0;
142
+ let failedCount = 0;
143
+ let totalScore = 0;
144
+
145
+ for (let i = 0; i < files.length; i++) {
146
+ const filePath = files[i];
147
+ const rel = relative(dir, filePath);
148
+ const fileName = basename(filePath);
149
+ const language = detectLanguage(filePath);
150
+
151
+ let source: string;
152
+ try {
153
+ source = readFileSync(filePath, 'utf-8');
154
+ } catch {
155
+ failedCount++;
156
+ process.stdout.write(` ${c.dim(`[${i + 1}/${files.length}]`)} ${c.dim(rel)} ${c.red('— read error')}\n`);
157
+ continue;
158
+ }
159
+
160
+ // Skip empty files
161
+ if (!source.trim()) continue;
162
+
163
+ try {
164
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
165
+ const result: any = await client.request('code.analyze', {
166
+ project: projectName,
167
+ name: fileName,
168
+ filePath: rel,
169
+ language,
170
+ source,
171
+ });
172
+
173
+ const score = result.reusabilityScore ?? 0;
174
+ const scoreColor = score >= 0.7 ? c.green : score >= 0.4 ? c.orange : c.red;
175
+ const statusTag = result.isNew ? c.green('new') : c.dim('existing');
176
+ totalScore += score;
177
+ imported++;
178
+
179
+ if (result.isNew) newCount++;
180
+ else existingCount++;
181
+
182
+ process.stdout.write(` ${c.dim(`[${i + 1}/${files.length}]`)} ${c.dim(rel)} ${c.dim(icons.arrow)} ${scoreColor(score.toFixed(2))} (${statusTag})\n`);
183
+ } catch (err) {
184
+ failedCount++;
185
+ const msg = err instanceof Error ? err.message : String(err);
186
+ process.stdout.write(` ${c.dim(`[${i + 1}/${files.length}]`)} ${c.dim(rel)} ${c.red(`— ${msg.slice(0, 80)}`)}\n`);
187
+ }
188
+ }
189
+
190
+ const avgScore = imported > 0 ? (totalScore / imported).toFixed(2) : '0';
191
+ console.log(header('Import Summary', icons.module));
192
+ console.log(` ${c.label('Project:')} ${c.cyan(projectName)}`);
193
+ console.log(` ${c.label('Imported:')} ${c.value(imported)} (${c.green(`${newCount} new`)}, ${c.dim(`${existingCount} existing`)})`);
194
+ if (failedCount > 0) console.log(` ${c.label('Failed:')} ${c.red(failedCount)}`);
195
+ console.log(` ${c.label('Avg score:')} ${c.value(avgScore)} ${progressBar(parseFloat(avgScore), 1)}`);
196
+ console.log(divider());
197
+ });
198
+ });
199
+ }
@@ -1,65 +1,65 @@
1
- import { Command } from 'commander';
2
- import { withIpc } from '../ipc-helper.js';
3
- import { c, icons, header, priorityBadge, divider } from '../colors.js';
4
-
5
- const TYPE_ICONS: Record<string, string> = {
6
- trend: '📈',
7
- pattern: '🔄',
8
- gap: '⚠',
9
- synergy: '⚡',
10
- optimization: '🎯',
11
- template_candidate: '🎨',
12
- project_suggestion: '💡',
13
- warning: '🚨',
14
- suggestion: '💡',
15
- };
16
-
17
- export function insightsCommand(): Command {
18
- return new Command('insights')
19
- .description('Show research insights')
20
- .option('--type <type>', 'Filter by type: trend, pattern, gap, synergy, optimization, template_candidate, project_suggestion, warning')
21
- .option('-l, --limit <n>', 'Maximum results', '20')
22
- .action(async (opts) => {
23
- await withIpc(async (client) => {
24
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
25
- const insights: any = await client.request('research.insights', {
26
- type: opts.type,
27
- activeOnly: true,
28
- limit: parseInt(opts.limit, 10),
29
- });
30
-
31
- if (!insights?.length) {
32
- console.log(`${icons.insight} ${c.dim('No active insights.')}`);
33
- return;
34
- }
35
-
36
- console.log(header(`${insights.length} Insights`, icons.insight));
37
-
38
- // Group by type
39
- const byType: Record<string, number> = {};
40
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
41
- for (const ins of insights as any[]) {
42
- byType[ins.type] = (byType[ins.type] || 0) + 1;
43
- }
44
- const typeSummary = Object.entries(byType)
45
- .sort((a, b) => b[1] - a[1])
46
- .map(([t, count]) => `${TYPE_ICONS[t] ?? '•'} ${c.cyan(t)} ${c.dim(`(${count})`)}`)
47
- .join(' ');
48
- console.log(` ${typeSummary}\n`);
49
-
50
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
51
- for (const ins of insights as any[]) {
52
- const typeIcon = TYPE_ICONS[ins.type] ?? '•';
53
- const pBadge = priorityBadge(ins.priority ?? 0);
54
- const typeTag = c.cyan(`[${ins.type}]`);
55
-
56
- console.log(` ${typeIcon} ${typeTag} ${pBadge} ${c.value(ins.title)}`);
57
- if (ins.description) {
58
- console.log(` ${c.dim(ins.description.slice(0, 150))}`);
59
- }
60
- console.log();
61
- }
62
- console.log(divider());
63
- });
64
- });
65
- }
1
+ import { Command } from 'commander';
2
+ import { withIpc } from '../ipc-helper.js';
3
+ import { c, icons, header, priorityBadge, divider } from '../colors.js';
4
+
5
+ const TYPE_ICONS: Record<string, string> = {
6
+ trend: '📈',
7
+ pattern: '🔄',
8
+ gap: '⚠',
9
+ synergy: '⚡',
10
+ optimization: '🎯',
11
+ template_candidate: '🎨',
12
+ project_suggestion: '💡',
13
+ warning: '🚨',
14
+ suggestion: '💡',
15
+ };
16
+
17
+ export function insightsCommand(): Command {
18
+ return new Command('insights')
19
+ .description('Show research insights')
20
+ .option('--type <type>', 'Filter by type: trend, pattern, gap, synergy, optimization, template_candidate, project_suggestion, warning')
21
+ .option('-l, --limit <n>', 'Maximum results', '20')
22
+ .action(async (opts) => {
23
+ await withIpc(async (client) => {
24
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
25
+ const insights: any = await client.request('research.insights', {
26
+ type: opts.type,
27
+ activeOnly: true,
28
+ limit: parseInt(opts.limit, 10),
29
+ });
30
+
31
+ if (!insights?.length) {
32
+ console.log(`${icons.insight} ${c.dim('No active insights.')}`);
33
+ return;
34
+ }
35
+
36
+ console.log(header(`${insights.length} Insights`, icons.insight));
37
+
38
+ // Group by type
39
+ const byType: Record<string, number> = {};
40
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
41
+ for (const ins of insights as any[]) {
42
+ byType[ins.type] = (byType[ins.type] || 0) + 1;
43
+ }
44
+ const typeSummary = Object.entries(byType)
45
+ .sort((a, b) => b[1] - a[1])
46
+ .map(([t, count]) => `${TYPE_ICONS[t] ?? '•'} ${c.cyan(t)} ${c.dim(`(${count})`)}`)
47
+ .join(' ');
48
+ console.log(` ${typeSummary}\n`);
49
+
50
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
51
+ for (const ins of insights as any[]) {
52
+ const typeIcon = TYPE_ICONS[ins.type] ?? '•';
53
+ const pBadge = priorityBadge(ins.priority ?? 0);
54
+ const typeTag = c.cyan(`[${ins.type}]`);
55
+
56
+ console.log(` ${typeIcon} ${typeTag} ${pBadge} ${c.value(ins.title)}`);
57
+ if (ins.description) {
58
+ console.log(` ${c.dim(ins.description.slice(0, 150))}`);
59
+ }
60
+ console.log();
61
+ }
62
+ console.log(divider());
63
+ });
64
+ });
65
+ }
@@ -1,24 +1,24 @@
1
- import { Command } from 'commander';
2
- import { withIpc } from '../ipc-helper.js';
3
- import { c, icons, header, keyValue, divider } from '../colors.js';
4
-
5
- export function learnCommand(): Command {
6
- return new Command('learn')
7
- .description('Trigger a learning cycle manually (pattern extraction + rule generation)')
8
- .action(async () => {
9
- await withIpc(async (client) => {
10
- console.log(`${icons.brain} ${c.info('Running learning cycle...')}`);
11
-
12
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
13
- const result: any = await client.request('learning.run', {});
14
-
15
- console.log(header('Learning Cycle Complete', icons.bolt));
16
- console.log(keyValue('New patterns', result.newPatterns ?? 0));
17
- console.log(keyValue('Updated rules', result.updatedRules ?? 0));
18
- console.log(keyValue('Pruned rules', result.prunedRules ?? 0));
19
- console.log(keyValue('New anti-patterns', result.newAntipatterns ?? 0));
20
- console.log(keyValue('Duration', `${result.duration ?? 0}ms`));
21
- console.log(`\n${divider()}`);
22
- });
23
- });
24
- }
1
+ import { Command } from 'commander';
2
+ import { withIpc } from '../ipc-helper.js';
3
+ import { c, icons, header, keyValue, divider } from '../colors.js';
4
+
5
+ export function learnCommand(): Command {
6
+ return new Command('learn')
7
+ .description('Trigger a learning cycle manually (pattern extraction + rule generation)')
8
+ .action(async () => {
9
+ await withIpc(async (client) => {
10
+ console.log(`${icons.brain} ${c.info('Running learning cycle...')}`);
11
+
12
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
13
+ const result: any = await client.request('learning.run', {});
14
+
15
+ console.log(header('Learning Cycle Complete', icons.bolt));
16
+ console.log(keyValue('New patterns', result.newPatterns ?? 0));
17
+ console.log(keyValue('Updated rules', result.updatedRules ?? 0));
18
+ console.log(keyValue('Pruned rules', result.prunedRules ?? 0));
19
+ console.log(keyValue('New anti-patterns', result.newAntipatterns ?? 0));
20
+ console.log(keyValue('Duration', `${result.duration ?? 0}ms`));
21
+ console.log(`\n${divider()}`);
22
+ });
23
+ });
24
+ }