@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,30 +1,30 @@
1
- export interface SolutionRecord {
2
- id: number;
3
- description: string;
4
- commands: string | null;
5
- code_change: string | null;
6
- source: string;
7
- confidence: number;
8
- success_count: number;
9
- fail_count: number;
10
- created_at: string;
11
- updated_at: string;
12
- }
13
-
14
- export interface ErrorSolution {
15
- id: number;
16
- error_id: number;
17
- solution_id: number;
18
- applied_at: string | null;
19
- success: number | null;
20
- }
21
-
22
- export interface SolutionAttempt {
23
- id: number;
24
- error_solution_id: number;
25
- terminal_id: number | null;
26
- attempted_at: string;
27
- success: number;
28
- output: string | null;
29
- duration_ms: number | null;
30
- }
1
+ export interface SolutionRecord {
2
+ id: number;
3
+ description: string;
4
+ commands: string | null;
5
+ code_change: string | null;
6
+ source: string;
7
+ confidence: number;
8
+ success_count: number;
9
+ fail_count: number;
10
+ created_at: string;
11
+ updated_at: string;
12
+ }
13
+
14
+ export interface ErrorSolution {
15
+ id: number;
16
+ error_id: number;
17
+ solution_id: number;
18
+ applied_at: string | null;
19
+ success: number | null;
20
+ }
21
+
22
+ export interface SolutionAttempt {
23
+ id: number;
24
+ error_solution_id: number;
25
+ terminal_id: number | null;
26
+ attempted_at: string;
27
+ success: number;
28
+ output: string | null;
29
+ duration_ms: number | null;
30
+ }
@@ -1,45 +1,45 @@
1
- import { EventEmitter } from 'node:events';
2
-
3
- export interface BrainEvents {
4
- 'error:reported': { errorId: number; projectId: number; fingerprint: string };
5
- 'error:resolved': { errorId: number; solutionId: number };
6
- 'solution:applied': { errorId: number; solutionId: number; success: boolean };
7
- 'solution:created': { solutionId: number };
8
- 'module:registered': { moduleId: number; projectId: number };
9
- 'module:updated': { moduleId: number };
10
- 'synapse:created': { synapseId: number; sourceType: string; targetType: string };
11
- 'synapse:strengthened': { synapseId: number; newWeight: number };
12
- 'insight:created': { insightId: number; type: string };
13
- 'rule:learned': { ruleId: number; pattern: string };
14
- 'terminal:connected': { terminalId: number; uuid: string };
15
- 'terminal:disconnected': { terminalId: number };
16
- }
17
-
18
- export type BrainEventName = keyof BrainEvents;
19
-
20
- export class TypedEventBus extends EventEmitter {
21
- emit<K extends BrainEventName>(event: K, data: BrainEvents[K]): boolean {
22
- return super.emit(event, data);
23
- }
24
-
25
- on<K extends BrainEventName>(event: K, listener: (data: BrainEvents[K]) => void): this {
26
- return super.on(event, listener);
27
- }
28
-
29
- once<K extends BrainEventName>(event: K, listener: (data: BrainEvents[K]) => void): this {
30
- return super.once(event, listener);
31
- }
32
-
33
- off<K extends BrainEventName>(event: K, listener: (data: BrainEvents[K]) => void): this {
34
- return super.off(event, listener);
35
- }
36
- }
37
-
38
- let busInstance: TypedEventBus | null = null;
39
-
40
- export function getEventBus(): TypedEventBus {
41
- if (!busInstance) {
42
- busInstance = new TypedEventBus();
43
- }
44
- return busInstance;
45
- }
1
+ import { EventEmitter } from 'node:events';
2
+
3
+ export interface BrainEvents {
4
+ 'error:reported': { errorId: number; projectId: number; fingerprint: string };
5
+ 'error:resolved': { errorId: number; solutionId: number };
6
+ 'solution:applied': { errorId: number; solutionId: number; success: boolean };
7
+ 'solution:created': { solutionId: number };
8
+ 'module:registered': { moduleId: number; projectId: number };
9
+ 'module:updated': { moduleId: number };
10
+ 'synapse:created': { synapseId: number; sourceType: string; targetType: string };
11
+ 'synapse:strengthened': { synapseId: number; newWeight: number };
12
+ 'insight:created': { insightId: number; type: string };
13
+ 'rule:learned': { ruleId: number; pattern: string };
14
+ 'terminal:connected': { terminalId: number; uuid: string };
15
+ 'terminal:disconnected': { terminalId: number };
16
+ }
17
+
18
+ export type BrainEventName = keyof BrainEvents;
19
+
20
+ export class TypedEventBus extends EventEmitter {
21
+ emit<K extends BrainEventName>(event: K, data: BrainEvents[K]): boolean {
22
+ return super.emit(event, data);
23
+ }
24
+
25
+ on<K extends BrainEventName>(event: K, listener: (data: BrainEvents[K]) => void): this {
26
+ return super.on(event, listener);
27
+ }
28
+
29
+ once<K extends BrainEventName>(event: K, listener: (data: BrainEvents[K]) => void): this {
30
+ return super.once(event, listener);
31
+ }
32
+
33
+ off<K extends BrainEventName>(event: K, listener: (data: BrainEvents[K]) => void): this {
34
+ return super.off(event, listener);
35
+ }
36
+ }
37
+
38
+ let busInstance: TypedEventBus | null = null;
39
+
40
+ export function getEventBus(): TypedEventBus {
41
+ if (!busInstance) {
42
+ busInstance = new TypedEventBus();
43
+ }
44
+ return busInstance;
45
+ }
package/src/utils/hash.ts CHANGED
@@ -1,5 +1,5 @@
1
- import { createHash } from 'node:crypto';
2
-
3
- export function sha256(input: string): string {
4
- return createHash('sha256').update(input).digest('hex');
5
- }
1
+ import { createHash } from 'node:crypto';
2
+
3
+ export function sha256(input: string): string {
4
+ return createHash('sha256').update(input).digest('hex');
5
+ }
@@ -1,48 +1,48 @@
1
- import winston from 'winston';
2
- import path from 'node:path';
3
- import { getDataDir } from './paths.js';
4
-
5
- const { combine, timestamp, printf, colorize } = winston.format;
6
-
7
- const logFormat = printf(({ level, message, timestamp, ...meta }) => {
8
- const metaStr = Object.keys(meta).length ? ` ${JSON.stringify(meta)}` : '';
9
- return `${timestamp} [${level}]${metaStr} ${message}`;
10
- });
11
-
12
- let loggerInstance: winston.Logger | null = null;
13
-
14
- export function createLogger(opts?: { level?: string; file?: string; maxSize?: number; maxFiles?: number }): winston.Logger {
15
- if (loggerInstance) return loggerInstance;
16
-
17
- const level = opts?.level ?? process.env['BRAIN_LOG_LEVEL'] ?? 'info';
18
- const logFile = opts?.file ?? path.join(getDataDir(), 'brain.log');
19
- const maxSize = opts?.maxSize ?? 10 * 1024 * 1024; // 10MB
20
- const maxFiles = opts?.maxFiles ?? 3;
21
-
22
- const transports: winston.transport[] = [
23
- new winston.transports.File({
24
- filename: logFile,
25
- maxsize: maxSize,
26
- maxFiles,
27
- format: combine(timestamp(), logFormat),
28
- }),
29
- ];
30
-
31
- if (process.env['NODE_ENV'] !== 'production') {
32
- transports.push(
33
- new winston.transports.Console({
34
- format: combine(colorize(), timestamp(), logFormat),
35
- })
36
- );
37
- }
38
-
39
- loggerInstance = winston.createLogger({ level, transports });
40
- return loggerInstance;
41
- }
42
-
43
- export function getLogger(): winston.Logger {
44
- if (!loggerInstance) {
45
- return createLogger();
46
- }
47
- return loggerInstance;
48
- }
1
+ import winston from 'winston';
2
+ import path from 'node:path';
3
+ import { getDataDir } from './paths.js';
4
+
5
+ const { combine, timestamp, printf, colorize } = winston.format;
6
+
7
+ const logFormat = printf(({ level, message, timestamp, ...meta }) => {
8
+ const metaStr = Object.keys(meta).length ? ` ${JSON.stringify(meta)}` : '';
9
+ return `${timestamp} [${level}]${metaStr} ${message}`;
10
+ });
11
+
12
+ let loggerInstance: winston.Logger | null = null;
13
+
14
+ export function createLogger(opts?: { level?: string; file?: string; maxSize?: number; maxFiles?: number }): winston.Logger {
15
+ if (loggerInstance) return loggerInstance;
16
+
17
+ const level = opts?.level ?? process.env['BRAIN_LOG_LEVEL'] ?? 'info';
18
+ const logFile = opts?.file ?? path.join(getDataDir(), 'brain.log');
19
+ const maxSize = opts?.maxSize ?? 10 * 1024 * 1024; // 10MB
20
+ const maxFiles = opts?.maxFiles ?? 3;
21
+
22
+ const transports: winston.transport[] = [
23
+ new winston.transports.File({
24
+ filename: logFile,
25
+ maxsize: maxSize,
26
+ maxFiles,
27
+ format: combine(timestamp(), logFormat),
28
+ }),
29
+ ];
30
+
31
+ if (process.env['NODE_ENV'] !== 'production') {
32
+ transports.push(
33
+ new winston.transports.Console({
34
+ format: combine(colorize(), timestamp(), logFormat),
35
+ })
36
+ );
37
+ }
38
+
39
+ loggerInstance = winston.createLogger({ level, transports });
40
+ return loggerInstance;
41
+ }
42
+
43
+ export function getLogger(): winston.Logger {
44
+ if (!loggerInstance) {
45
+ return createLogger();
46
+ }
47
+ return loggerInstance;
48
+ }
@@ -1,19 +1,19 @@
1
- import path from 'node:path';
2
- import os from 'node:os';
3
-
4
- export function normalizePath(filePath: string): string {
5
- return filePath.replace(/\\/g, '/');
6
- }
7
-
8
- export function getDataDir(): string {
9
- const envDir = process.env['BRAIN_DATA_DIR'];
10
- if (envDir) return path.resolve(envDir);
11
- return path.join(os.homedir(), '.brain');
12
- }
13
-
14
- export function getPipeName(name: string = 'brain'): string {
15
- if (process.platform === 'win32') {
16
- return `\\\\.\\pipe\\${name}`;
17
- }
18
- return path.join(os.tmpdir(), `${name}.sock`);
19
- }
1
+ import path from 'node:path';
2
+ import os from 'node:os';
3
+
4
+ export function normalizePath(filePath: string): string {
5
+ return filePath.replace(/\\/g, '/');
6
+ }
7
+
8
+ export function getDataDir(): string {
9
+ const envDir = process.env['BRAIN_DATA_DIR'];
10
+ if (envDir) return path.resolve(envDir);
11
+ return path.join(os.homedir(), '.brain');
12
+ }
13
+
14
+ export function getPipeName(name: string = 'brain'): string {
15
+ if (process.platform === 'win32') {
16
+ return `\\\\.\\pipe\\${name}`;
17
+ }
18
+ return path.join(os.tmpdir(), `${name}.sock`);
19
+ }