@timmeck/brain 1.8.5 → 2.0.0

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 (221) hide show
  1. package/.dockerignore +11 -0
  2. package/Dockerfile +21 -0
  3. package/README.md +19 -0
  4. package/assets/brain-avatar-256.png +0 -0
  5. package/assets/brain-avatar-512.png +0 -0
  6. package/assets/brain-avatar.png +0 -0
  7. package/brain.log +1164 -0
  8. package/{src/cli/commands/dashboard.ts → dashboard.html} +688 -807
  9. package/dist/api/server.d.ts +4 -18
  10. package/dist/api/server.js +4 -173
  11. package/dist/api/server.js.map +1 -1
  12. package/dist/brain.d.ts +6 -0
  13. package/dist/brain.js +56 -4
  14. package/dist/brain.js.map +1 -1
  15. package/dist/cli/colors.d.ts +4 -25
  16. package/dist/cli/colors.js +3 -89
  17. package/dist/cli/colors.js.map +1 -1
  18. package/dist/cli/commands/peers.d.ts +2 -0
  19. package/dist/cli/commands/peers.js +38 -0
  20. package/dist/cli/commands/peers.js.map +1 -0
  21. package/dist/cli/commands/start.js +54 -17
  22. package/dist/cli/commands/start.js.map +1 -1
  23. package/dist/db/connection.d.ts +1 -2
  24. package/dist/db/connection.js +1 -18
  25. package/dist/db/connection.js.map +1 -1
  26. package/dist/index.js +3 -1
  27. package/dist/index.js.map +1 -1
  28. package/dist/ipc/__tests__/protocol.test.d.ts +1 -0
  29. package/dist/ipc/__tests__/protocol.test.js +117 -0
  30. package/dist/ipc/__tests__/protocol.test.js.map +1 -0
  31. package/dist/ipc/client.d.ts +1 -16
  32. package/dist/ipc/client.js +1 -100
  33. package/dist/ipc/client.js.map +1 -1
  34. package/dist/ipc/protocol.d.ts +1 -8
  35. package/dist/ipc/protocol.js +1 -28
  36. package/dist/ipc/protocol.js.map +1 -1
  37. package/dist/ipc/router.js +8 -0
  38. package/dist/ipc/router.js.map +1 -1
  39. package/dist/ipc/server.d.ts +1 -22
  40. package/dist/ipc/server.js +1 -163
  41. package/dist/ipc/server.js.map +1 -1
  42. package/dist/mcp/http-server.d.ts +1 -7
  43. package/dist/mcp/http-server.js +6 -117
  44. package/dist/mcp/http-server.js.map +1 -1
  45. package/dist/mcp/server.js +5 -61
  46. package/dist/mcp/server.js.map +1 -1
  47. package/dist/signals/__tests__/fingerprint.test.d.ts +1 -0
  48. package/dist/signals/__tests__/fingerprint.test.js +118 -0
  49. package/dist/signals/__tests__/fingerprint.test.js.map +1 -0
  50. package/dist/types/ipc.types.d.ts +1 -11
  51. package/dist/utils/__tests__/hash.test.d.ts +1 -0
  52. package/dist/utils/__tests__/hash.test.js +32 -0
  53. package/dist/utils/__tests__/hash.test.js.map +1 -0
  54. package/dist/utils/__tests__/paths.test.d.ts +1 -0
  55. package/dist/utils/__tests__/paths.test.js +75 -0
  56. package/dist/utils/__tests__/paths.test.js.map +1 -0
  57. package/dist/utils/events.d.ts +4 -8
  58. package/dist/utils/events.js +2 -14
  59. package/dist/utils/events.js.map +1 -1
  60. package/dist/utils/hash.d.ts +1 -1
  61. package/dist/utils/hash.js +1 -4
  62. package/dist/utils/hash.js.map +1 -1
  63. package/dist/utils/logger.d.ts +3 -2
  64. package/dist/utils/logger.js +8 -35
  65. package/dist/utils/logger.js.map +1 -1
  66. package/dist/utils/paths.d.ts +2 -1
  67. package/dist/utils/paths.js +4 -13
  68. package/dist/utils/paths.js.map +1 -1
  69. package/gen_avatar.py +142 -0
  70. package/package.json +2 -1
  71. package/BRAIN_PLAN.md +0 -3324
  72. package/src/api/server.ts +0 -395
  73. package/src/brain.ts +0 -266
  74. package/src/cli/colors.ts +0 -116
  75. package/src/cli/commands/config.ts +0 -169
  76. package/src/cli/commands/doctor.ts +0 -124
  77. package/src/cli/commands/explain.ts +0 -83
  78. package/src/cli/commands/export.ts +0 -31
  79. package/src/cli/commands/import.ts +0 -199
  80. package/src/cli/commands/insights.ts +0 -65
  81. package/src/cli/commands/learn.ts +0 -24
  82. package/src/cli/commands/modules.ts +0 -53
  83. package/src/cli/commands/network.ts +0 -67
  84. package/src/cli/commands/projects.ts +0 -42
  85. package/src/cli/commands/query.ts +0 -120
  86. package/src/cli/commands/start.ts +0 -62
  87. package/src/cli/commands/status.ts +0 -75
  88. package/src/cli/commands/stop.ts +0 -34
  89. package/src/cli/ipc-helper.ts +0 -22
  90. package/src/cli/update-check.ts +0 -63
  91. package/src/code/analyzer.ts +0 -117
  92. package/src/code/fingerprint.ts +0 -87
  93. package/src/code/matcher.ts +0 -129
  94. package/src/code/parsers/generic.ts +0 -29
  95. package/src/code/parsers/python.ts +0 -54
  96. package/src/code/parsers/typescript.ts +0 -65
  97. package/src/code/registry.ts +0 -60
  98. package/src/code/scorer.ts +0 -120
  99. package/src/config.ts +0 -135
  100. package/src/dashboard/server.ts +0 -142
  101. package/src/db/connection.ts +0 -22
  102. package/src/db/migrations/001_core_schema.ts +0 -120
  103. package/src/db/migrations/002_learning_schema.ts +0 -38
  104. package/src/db/migrations/003_code_schema.ts +0 -53
  105. package/src/db/migrations/004_synapses_schema.ts +0 -57
  106. package/src/db/migrations/005_fts_indexes.ts +0 -78
  107. package/src/db/migrations/006_synapses_phase3.ts +0 -17
  108. package/src/db/migrations/007_feedback.ts +0 -13
  109. package/src/db/migrations/008_git_integration.ts +0 -38
  110. package/src/db/migrations/009_embeddings.ts +0 -8
  111. package/src/db/migrations/index.ts +0 -70
  112. package/src/db/repositories/antipattern.repository.ts +0 -66
  113. package/src/db/repositories/code-module.repository.ts +0 -142
  114. package/src/db/repositories/error.repository.ts +0 -189
  115. package/src/db/repositories/insight.repository.ts +0 -99
  116. package/src/db/repositories/notification.repository.ts +0 -66
  117. package/src/db/repositories/project.repository.ts +0 -93
  118. package/src/db/repositories/rule.repository.ts +0 -108
  119. package/src/db/repositories/solution.repository.ts +0 -154
  120. package/src/db/repositories/synapse.repository.ts +0 -163
  121. package/src/db/repositories/terminal.repository.ts +0 -101
  122. package/src/embeddings/engine.ts +0 -238
  123. package/src/hooks/post-tool-use.ts +0 -92
  124. package/src/hooks/post-write.ts +0 -129
  125. package/src/index.ts +0 -63
  126. package/src/ipc/client.ts +0 -118
  127. package/src/ipc/protocol.ts +0 -35
  128. package/src/ipc/router.ts +0 -133
  129. package/src/ipc/server.ts +0 -176
  130. package/src/learning/confidence-scorer.ts +0 -80
  131. package/src/learning/decay.ts +0 -46
  132. package/src/learning/learning-engine.ts +0 -170
  133. package/src/learning/pattern-extractor.ts +0 -90
  134. package/src/learning/rule-generator.ts +0 -74
  135. package/src/main.rs:10:5 +0 -0
  136. package/src/matching/error-matcher.ts +0 -166
  137. package/src/matching/fingerprint.ts +0 -34
  138. package/src/matching/similarity.ts +0 -61
  139. package/src/matching/tfidf.ts +0 -74
  140. package/src/matching/tokenizer.ts +0 -41
  141. package/src/mcp/auto-detect.ts +0 -93
  142. package/src/mcp/http-server.ts +0 -140
  143. package/src/mcp/server.ts +0 -73
  144. package/src/mcp/tools.ts +0 -328
  145. package/src/parsing/error-parser.ts +0 -28
  146. package/src/parsing/parsers/compiler.ts +0 -93
  147. package/src/parsing/parsers/generic.ts +0 -28
  148. package/src/parsing/parsers/go.ts +0 -97
  149. package/src/parsing/parsers/node.ts +0 -69
  150. package/src/parsing/parsers/python.ts +0 -62
  151. package/src/parsing/parsers/rust.ts +0 -50
  152. package/src/parsing/parsers/shell.ts +0 -42
  153. package/src/parsing/types.ts +0 -47
  154. package/src/research/gap-analyzer.ts +0 -135
  155. package/src/research/insight-generator.ts +0 -123
  156. package/src/research/research-engine.ts +0 -116
  157. package/src/research/synergy-detector.ts +0 -126
  158. package/src/research/template-extractor.ts +0 -130
  159. package/src/research/trend-analyzer.ts +0 -127
  160. package/src/services/analytics.service.ts +0 -226
  161. package/src/services/code.service.ts +0 -271
  162. package/src/services/error.service.ts +0 -266
  163. package/src/services/git.service.ts +0 -132
  164. package/src/services/notification.service.ts +0 -41
  165. package/src/services/prevention.service.ts +0 -159
  166. package/src/services/research.service.ts +0 -98
  167. package/src/services/solution.service.ts +0 -174
  168. package/src/services/synapse.service.ts +0 -59
  169. package/src/services/terminal.service.ts +0 -81
  170. package/src/synapses/activation.ts +0 -80
  171. package/src/synapses/decay.ts +0 -38
  172. package/src/synapses/hebbian.ts +0 -69
  173. package/src/synapses/pathfinder.ts +0 -81
  174. package/src/synapses/synapse-manager.ts +0 -113
  175. package/src/types/code.types.ts +0 -52
  176. package/src/types/config.types.ts +0 -103
  177. package/src/types/error.types.ts +0 -67
  178. package/src/types/ipc.types.ts +0 -8
  179. package/src/types/mcp.types.ts +0 -53
  180. package/src/types/research.types.ts +0 -28
  181. package/src/types/solution.types.ts +0 -30
  182. package/src/types/synapse.types.ts +0 -50
  183. package/src/utils/events.ts +0 -45
  184. package/src/utils/hash.ts +0 -5
  185. package/src/utils/logger.ts +0 -48
  186. package/src/utils/paths.ts +0 -19
  187. package/tests/e2e/test_code_intelligence.py +0 -1015
  188. package/tests/e2e/test_error_memory.py +0 -451
  189. package/tests/e2e/test_full_integration.py +0 -534
  190. package/tests/fixtures/code-modules/modules.ts +0 -83
  191. package/tests/fixtures/errors/go.ts +0 -9
  192. package/tests/fixtures/errors/node.ts +0 -24
  193. package/tests/fixtures/errors/python.ts +0 -21
  194. package/tests/fixtures/errors/rust.ts +0 -25
  195. package/tests/fixtures/errors/shell.ts +0 -15
  196. package/tests/fixtures/solutions/solutions.ts +0 -27
  197. package/tests/helpers/setup-db.ts +0 -52
  198. package/tests/integration/code-flow.test.ts +0 -86
  199. package/tests/integration/error-flow.test.ts +0 -83
  200. package/tests/integration/ipc-flow.test.ts +0 -166
  201. package/tests/integration/learning-cycle.test.ts +0 -82
  202. package/tests/integration/synapse-flow.test.ts +0 -117
  203. package/tests/unit/code/analyzer.test.ts +0 -58
  204. package/tests/unit/code/fingerprint.test.ts +0 -51
  205. package/tests/unit/code/scorer.test.ts +0 -55
  206. package/tests/unit/learning/confidence-scorer.test.ts +0 -60
  207. package/tests/unit/learning/decay.test.ts +0 -45
  208. package/tests/unit/learning/pattern-extractor.test.ts +0 -50
  209. package/tests/unit/matching/error-matcher.test.ts +0 -69
  210. package/tests/unit/matching/fingerprint.test.ts +0 -47
  211. package/tests/unit/matching/similarity.test.ts +0 -65
  212. package/tests/unit/matching/tfidf.test.ts +0 -71
  213. package/tests/unit/matching/tokenizer.test.ts +0 -83
  214. package/tests/unit/parsing/parsers.test.ts +0 -113
  215. package/tests/unit/research/gap-analyzer.test.ts +0 -45
  216. package/tests/unit/research/trend-analyzer.test.ts +0 -45
  217. package/tests/unit/synapses/activation.test.ts +0 -80
  218. package/tests/unit/synapses/decay.test.ts +0 -27
  219. package/tests/unit/synapses/hebbian.test.ts +0 -96
  220. package/tests/unit/synapses/pathfinder.test.ts +0 -72
  221. package/tsconfig.json +0 -18
@@ -1,174 +0,0 @@
1
- import type { SolutionRecord } from '../types/solution.types.js';
2
- import type { SolutionRepository } from '../db/repositories/solution.repository.js';
3
- import type { SynapseManager } from '../synapses/synapse-manager.js';
4
- import { getEventBus } from '../utils/events.js';
5
- import { getLogger } from '../utils/logger.js';
6
-
7
- export interface ReportSolutionInput {
8
- errorId: number;
9
- description: string;
10
- commands?: string;
11
- codeChange?: string;
12
- source?: string;
13
- }
14
-
15
- export interface RateOutcomeInput {
16
- errorId: number;
17
- solutionId: number;
18
- success: boolean;
19
- terminalId?: number;
20
- output?: string;
21
- durationMs?: number;
22
- }
23
-
24
- export class SolutionService {
25
- private logger = getLogger();
26
- private eventBus = getEventBus();
27
-
28
- constructor(
29
- private solutionRepo: SolutionRepository,
30
- private synapseManager: SynapseManager,
31
- ) {}
32
-
33
- report(input: ReportSolutionInput): number {
34
- const solutionId = this.solutionRepo.create({
35
- description: input.description,
36
- commands: input.commands ?? null,
37
- code_change: input.codeChange ?? null,
38
- source: input.source ?? 'manual',
39
- confidence: 0.5,
40
- });
41
-
42
- this.solutionRepo.linkToError(input.errorId, solutionId);
43
-
44
- // Create synapse: solution solves error
45
- this.synapseManager.strengthen(
46
- { type: 'solution', id: solutionId },
47
- { type: 'error', id: input.errorId },
48
- 'solves',
49
- );
50
-
51
- this.eventBus.emit('solution:created', { solutionId });
52
- this.logger.info(`Solution reported (id=${solutionId}) for error ${input.errorId}`);
53
-
54
- return solutionId;
55
- }
56
-
57
- rateOutcome(input: RateOutcomeInput): void {
58
- // Record the attempt
59
- const errorSolutions = this.solutionRepo.findForError(input.errorId);
60
- const link = errorSolutions.find(s => s.id === input.solutionId);
61
- if (!link) {
62
- this.solutionRepo.linkToError(input.errorId, input.solutionId);
63
- }
64
-
65
- // Find the error_solution link id via DB
66
- this.solutionRepo.recordAttempt({
67
- errorSolutionId: input.errorId, // will be resolved via link
68
- terminalId: input.terminalId,
69
- success: input.success ? 1 : 0,
70
- output: input.output,
71
- durationMs: input.durationMs,
72
- });
73
-
74
- // Update synapse
75
- if (input.success) {
76
- this.synapseManager.strengthen(
77
- { type: 'solution', id: input.solutionId },
78
- { type: 'error', id: input.errorId },
79
- 'solves',
80
- { outcome: 'success' },
81
- );
82
- // Update solution confidence based on success rate
83
- const rate = this.solutionRepo.successRate(input.solutionId);
84
- this.solutionRepo.update(input.solutionId, { confidence: rate });
85
- } else {
86
- const synapse = this.synapseManager.find(
87
- { type: 'solution', id: input.solutionId },
88
- { type: 'error', id: input.errorId },
89
- 'solves',
90
- );
91
- if (synapse) {
92
- this.synapseManager.weaken(synapse.id, 0.7);
93
- }
94
- const rate = this.solutionRepo.successRate(input.solutionId);
95
- this.solutionRepo.update(input.solutionId, { confidence: rate });
96
- }
97
-
98
- this.eventBus.emit('solution:applied', {
99
- errorId: input.errorId,
100
- solutionId: input.solutionId,
101
- success: input.success,
102
- });
103
- }
104
-
105
- findForError(errorId: number): SolutionRecord[] {
106
- return this.solutionRepo.findForError(errorId);
107
- }
108
-
109
- getById(id: number): SolutionRecord | undefined {
110
- return this.solutionRepo.getById(id);
111
- }
112
-
113
- successRate(solutionId: number): number {
114
- return this.solutionRepo.successRate(solutionId);
115
- }
116
-
117
- analyzeEfficiency(): {
118
- avgDurationMs: number;
119
- slowSolutions: Array<{ solutionId: number; avgDuration: number; description: string }>;
120
- successRateOverall: number;
121
- totalAttempts: number;
122
- } {
123
- const allSolutions = this.solutionRepo.getAll();
124
- let totalDuration = 0;
125
- let totalAttempts = 0;
126
- let totalSuccessRate = 0;
127
- let solutionCount = 0;
128
- const solutionDurations: Array<{ solutionId: number; avgDuration: number; description: string }> = [];
129
-
130
- for (const solution of allSolutions) {
131
- const rate = this.solutionRepo.successRate(solution.id);
132
- if (solution.success_count + solution.fail_count > 0) {
133
- totalSuccessRate += rate;
134
- solutionCount++;
135
- }
136
-
137
- // Check attempts for duration data
138
- const attempts = this.solutionRepo.getAttempts(solution.id);
139
- if (attempts.length > 0) {
140
- let solDuration = 0;
141
- let solAttemptCount = 0;
142
- for (const attempt of attempts) {
143
- if (attempt.duration_ms && attempt.duration_ms > 0) {
144
- solDuration += attempt.duration_ms;
145
- solAttemptCount++;
146
- totalDuration += attempt.duration_ms;
147
- totalAttempts++;
148
- }
149
- }
150
- if (solAttemptCount > 0) {
151
- solutionDurations.push({
152
- solutionId: solution.id,
153
- avgDuration: solDuration / solAttemptCount,
154
- description: solution.description,
155
- });
156
- }
157
- }
158
- }
159
-
160
- // Find slow solutions (above 2x average)
161
- const avgDurationMs = totalAttempts > 0 ? totalDuration / totalAttempts : 0;
162
- const slowSolutions = solutionDurations
163
- .filter(s => s.avgDuration > avgDurationMs * 2)
164
- .sort((a, b) => b.avgDuration - a.avgDuration)
165
- .slice(0, 10);
166
-
167
- return {
168
- avgDurationMs,
169
- slowSolutions,
170
- successRateOverall: solutionCount > 0 ? totalSuccessRate / solutionCount : 0,
171
- totalAttempts,
172
- };
173
- }
174
- }
@@ -1,59 +0,0 @@
1
- import type { NodeType, SynapseRecord, NetworkStats } from '../types/synapse.types.js';
2
- import type { SynapseManager } from '../synapses/synapse-manager.js';
3
- import type { ActivationResult } from '../synapses/activation.js';
4
- import type { SynapsePath } from '../synapses/pathfinder.js';
5
-
6
- export interface ErrorContext {
7
- solutions: ActivationResult[];
8
- relatedErrors: ActivationResult[];
9
- relevantModules: ActivationResult[];
10
- preventionRules: ActivationResult[];
11
- insights: ActivationResult[];
12
- }
13
-
14
- export interface RelatedQuery {
15
- nodeType: NodeType;
16
- nodeId: number;
17
- maxDepth?: number;
18
- minWeight?: number;
19
- }
20
-
21
- export class SynapseService {
22
- constructor(private manager: SynapseManager) {}
23
-
24
- getErrorContext(errorId: number): ErrorContext {
25
- return this.manager.getErrorContext(errorId);
26
- }
27
-
28
- findPath(
29
- fromType: NodeType,
30
- fromId: number,
31
- toType: NodeType,
32
- toId: number,
33
- ): SynapsePath | null {
34
- return this.manager.findPath(
35
- { type: fromType, id: fromId },
36
- { type: toType, id: toId },
37
- );
38
- }
39
-
40
- getRelated(query: RelatedQuery): ActivationResult[] {
41
- return this.manager.activate(
42
- { type: query.nodeType, id: query.nodeId },
43
- query.maxDepth,
44
- query.minWeight,
45
- );
46
- }
47
-
48
- getNetworkStats(): NetworkStats {
49
- return this.manager.getNetworkStats();
50
- }
51
-
52
- getStrongestSynapses(limit?: number): SynapseRecord[] {
53
- return this.manager.getStrongestSynapses(limit);
54
- }
55
-
56
- runDecay(): { decayed: number; pruned: number } {
57
- return this.manager.runDecay();
58
- }
59
- }
@@ -1,81 +0,0 @@
1
- import type { TerminalRepository } from '../db/repositories/terminal.repository.js';
2
- import { getEventBus } from '../utils/events.js';
3
- import { getLogger } from '../utils/logger.js';
4
-
5
- export interface RegisterTerminalInput {
6
- uuid: string;
7
- projectId?: number;
8
- pid?: number;
9
- shell?: string;
10
- cwd?: string;
11
- }
12
-
13
- export class TerminalService {
14
- private logger = getLogger();
15
- private eventBus = getEventBus();
16
-
17
- constructor(
18
- private terminalRepo: TerminalRepository,
19
- private staleTimeout: number,
20
- ) {}
21
-
22
- register(input: RegisterTerminalInput): number {
23
- const existing = this.terminalRepo.findByUuid(input.uuid);
24
- if (existing) {
25
- this.terminalRepo.update(existing.id, {
26
- last_seen: new Date().toISOString(),
27
- disconnected_at: null,
28
- project_id: input.projectId ?? existing.project_id,
29
- cwd: input.cwd ?? existing.cwd,
30
- });
31
- this.logger.info(`Terminal reconnected (id=${existing.id}, uuid=${input.uuid})`);
32
- this.eventBus.emit('terminal:connected', { terminalId: existing.id, uuid: input.uuid });
33
- return existing.id;
34
- }
35
-
36
- const id = this.terminalRepo.create({
37
- uuid: input.uuid,
38
- project_id: input.projectId ?? null,
39
- pid: input.pid ?? null,
40
- shell: input.shell ?? null,
41
- cwd: input.cwd ?? null,
42
- });
43
-
44
- this.logger.info(`Terminal registered (id=${id}, uuid=${input.uuid})`);
45
- this.eventBus.emit('terminal:connected', { terminalId: id, uuid: input.uuid });
46
- return id;
47
- }
48
-
49
- heartbeat(uuid: string): void {
50
- const terminal = this.terminalRepo.findByUuid(uuid);
51
- if (terminal) {
52
- this.terminalRepo.update(terminal.id, {
53
- last_seen: new Date().toISOString(),
54
- });
55
- }
56
- }
57
-
58
- disconnect(uuid: string): void {
59
- const terminal = this.terminalRepo.findByUuid(uuid);
60
- if (terminal) {
61
- this.terminalRepo.update(terminal.id, {
62
- disconnected_at: new Date().toISOString(),
63
- });
64
- this.eventBus.emit('terminal:disconnected', { terminalId: terminal.id });
65
- this.logger.info(`Terminal disconnected (id=${terminal.id}, uuid=${uuid})`);
66
- }
67
- }
68
-
69
- cleanup(): number {
70
- const cutoff = new Date(Date.now() - this.staleTimeout).toISOString();
71
- const count = this.terminalRepo.cleanupStale(cutoff);
72
- if (count > 0) {
73
- this.logger.info(`Cleaned up ${count} stale terminal(s)`);
74
- }
75
- return count;
76
- }
77
-
78
- getConnected() {
79
- return this.terminalRepo.findConnected();
80
- }
81
- }
@@ -1,80 +0,0 @@
1
- import type { NodeType } from '../types/synapse.types.js';
2
- import type { SynapseRepository } from '../db/repositories/synapse.repository.js';
3
-
4
- export interface ActivationNode {
5
- type: NodeType;
6
- id: number;
7
- }
8
-
9
- export interface ActivationResult {
10
- node: ActivationNode;
11
- activation: number;
12
- depth: number;
13
- path: string[];
14
- }
15
-
16
- export function spreadingActivation(
17
- repo: SynapseRepository,
18
- startNode: ActivationNode,
19
- maxDepth: number = 3,
20
- minWeight: number = 0.2,
21
- ): ActivationResult[] {
22
- const visited = new Set<string>();
23
- const results: ActivationResult[] = [];
24
-
25
- const queue: Array<{
26
- node: ActivationNode;
27
- depth: number;
28
- pathWeight: number;
29
- path: string[];
30
- }> = [{ node: startNode, depth: 0, pathWeight: 1.0, path: [] }];
31
-
32
- while (queue.length > 0) {
33
- const current = queue.shift()!;
34
- const key = `${current.node.type}:${current.node.id}`;
35
-
36
- if (visited.has(key)) continue;
37
- if (current.depth > maxDepth) continue;
38
- if (current.pathWeight < minWeight) continue;
39
-
40
- visited.add(key);
41
-
42
- if (current.depth > 0) {
43
- results.push({
44
- node: current.node,
45
- activation: current.pathWeight,
46
- depth: current.depth,
47
- path: current.path,
48
- });
49
- }
50
-
51
- const outgoing = repo.getOutgoing(current.node.type, current.node.id);
52
- const incoming = repo.getIncoming(current.node.type, current.node.id);
53
-
54
- for (const synapse of outgoing) {
55
- const nextWeight = current.pathWeight * synapse.weight;
56
- if (nextWeight >= minWeight) {
57
- queue.push({
58
- node: { type: synapse.target_type, id: synapse.target_id },
59
- depth: current.depth + 1,
60
- pathWeight: nextWeight,
61
- path: [...current.path, `--${synapse.synapse_type}-->`],
62
- });
63
- }
64
- }
65
-
66
- for (const synapse of incoming) {
67
- const nextWeight = current.pathWeight * synapse.weight;
68
- if (nextWeight >= minWeight) {
69
- queue.push({
70
- node: { type: synapse.source_type, id: synapse.source_id },
71
- depth: current.depth + 1,
72
- pathWeight: nextWeight,
73
- path: [...current.path, `<--${synapse.synapse_type}--`],
74
- });
75
- }
76
- }
77
- }
78
-
79
- return results.sort((a, b) => b.activation - a.activation);
80
- }
@@ -1,38 +0,0 @@
1
- import type { SynapseRepository } from '../db/repositories/synapse.repository.js';
2
-
3
- export interface DecayConfig {
4
- decayHalfLifeDays: number;
5
- decayAfterDays: number;
6
- pruneThreshold: number;
7
- }
8
-
9
- export function timeDecayFactor(lastActivatedAt: string, halfLifeDays: number): number {
10
- const now = Date.now();
11
- const activated = new Date(lastActivatedAt).getTime();
12
- const ageDays = (now - activated) / (1000 * 60 * 60 * 24);
13
- return Math.pow(0.5, ageDays / halfLifeDays);
14
- }
15
-
16
- export function decayAll(repo: SynapseRepository, config: DecayConfig): { decayed: number; pruned: number } {
17
- const cutoff = new Date();
18
- cutoff.setDate(cutoff.getDate() - config.decayAfterDays);
19
-
20
- const stale = repo.findInactiveSince(cutoff.toISOString());
21
- let pruned = 0;
22
- let decayed = 0;
23
-
24
- for (const synapse of stale) {
25
- const factor = timeDecayFactor(synapse.last_activated_at, config.decayHalfLifeDays);
26
- const newWeight = synapse.weight * factor;
27
-
28
- if (newWeight < config.pruneThreshold) {
29
- repo.delete(synapse.id);
30
- pruned++;
31
- } else {
32
- repo.update(synapse.id, { weight: newWeight });
33
- decayed++;
34
- }
35
- }
36
-
37
- return { decayed, pruned };
38
- }
@@ -1,69 +0,0 @@
1
- import type { SynapseRecord, NodeType, SynapseType } from '../types/synapse.types.js';
2
- import type { SynapseRepository } from '../db/repositories/synapse.repository.js';
3
-
4
- export interface HebbianConfig {
5
- initialWeight: number;
6
- learningRate: number;
7
- pruneThreshold: number;
8
- }
9
-
10
- export interface NodeRef {
11
- type: NodeType;
12
- id: number;
13
- }
14
-
15
- export function strengthen(
16
- repo: SynapseRepository,
17
- source: NodeRef,
18
- target: NodeRef,
19
- synapseType: SynapseType,
20
- config: HebbianConfig,
21
- context?: Record<string, unknown>,
22
- ): SynapseRecord {
23
- const existing = repo.findBySourceTarget(
24
- source.type, source.id, target.type, target.id, synapseType,
25
- );
26
-
27
- if (existing) {
28
- // Hebbian: weight grows logarithmically, saturates at 1.0
29
- const newWeight = Math.min(
30
- 1.0,
31
- existing.weight + (1.0 - existing.weight) * config.learningRate,
32
- );
33
- repo.update(existing.id, {
34
- weight: newWeight,
35
- activation_count: existing.activation_count + 1,
36
- last_activated_at: new Date().toISOString(),
37
- });
38
- return { ...existing, weight: newWeight, activation_count: existing.activation_count + 1 };
39
- }
40
-
41
- const id = repo.create({
42
- source_type: source.type,
43
- source_id: source.id,
44
- target_type: target.type,
45
- target_id: target.id,
46
- synapse_type: synapseType,
47
- weight: config.initialWeight,
48
- metadata: context ? JSON.stringify(context) : null,
49
- });
50
-
51
- return repo.getById(id)!;
52
- }
53
-
54
- export function weaken(
55
- repo: SynapseRepository,
56
- synapseId: number,
57
- config: HebbianConfig,
58
- factor: number = 0.5,
59
- ): void {
60
- const synapse = repo.getById(synapseId);
61
- if (!synapse) return;
62
-
63
- const newWeight = synapse.weight * factor;
64
- if (newWeight < config.pruneThreshold) {
65
- repo.delete(synapseId);
66
- } else {
67
- repo.update(synapseId, { weight: newWeight });
68
- }
69
- }
@@ -1,81 +0,0 @@
1
- import type { NodeType, SynapseRecord } from '../types/synapse.types.js';
2
- import type { SynapseRepository } from '../db/repositories/synapse.repository.js';
3
-
4
- export interface PathNode {
5
- type: NodeType;
6
- id: number;
7
- }
8
-
9
- export interface SynapsePath {
10
- from: PathNode;
11
- to: PathNode;
12
- synapses: SynapseRecord[];
13
- totalWeight: number;
14
- hops: number;
15
- }
16
-
17
- export function findPath(
18
- repo: SynapseRepository,
19
- from: PathNode,
20
- to: PathNode,
21
- maxDepth: number = 5,
22
- ): SynapsePath | null {
23
- const visited = new Set<string>();
24
-
25
- const queue: Array<{
26
- node: PathNode;
27
- path: SynapseRecord[];
28
- totalWeight: number;
29
- }> = [{ node: from, path: [], totalWeight: 1.0 }];
30
-
31
- let bestPath: SynapsePath | null = null;
32
-
33
- while (queue.length > 0) {
34
- const current = queue.shift()!;
35
- const key = `${current.node.type}:${current.node.id}`;
36
-
37
- if (visited.has(key)) continue;
38
- visited.add(key);
39
-
40
- if (current.node.type === to.type && current.node.id === to.id) {
41
- if (!bestPath || current.totalWeight > bestPath.totalWeight) {
42
- bestPath = {
43
- from,
44
- to,
45
- synapses: current.path,
46
- totalWeight: current.totalWeight,
47
- hops: current.path.length,
48
- };
49
- }
50
- continue;
51
- }
52
-
53
- if (current.path.length >= maxDepth) continue;
54
-
55
- const outgoing = repo.getOutgoing(current.node.type, current.node.id);
56
- for (const synapse of outgoing) {
57
- const targetKey = `${synapse.target_type}:${synapse.target_id}`;
58
- if (!visited.has(targetKey)) {
59
- queue.push({
60
- node: { type: synapse.target_type, id: synapse.target_id },
61
- path: [...current.path, synapse],
62
- totalWeight: current.totalWeight * synapse.weight,
63
- });
64
- }
65
- }
66
-
67
- const incoming = repo.getIncoming(current.node.type, current.node.id);
68
- for (const synapse of incoming) {
69
- const sourceKey = `${synapse.source_type}:${synapse.source_id}`;
70
- if (!visited.has(sourceKey)) {
71
- queue.push({
72
- node: { type: synapse.source_type, id: synapse.source_id },
73
- path: [...current.path, synapse],
74
- totalWeight: current.totalWeight * synapse.weight,
75
- });
76
- }
77
- }
78
- }
79
-
80
- return bestPath;
81
- }
@@ -1,113 +0,0 @@
1
- import type { SynapsesConfig } from '../types/config.types.js';
2
- import type { NodeType, SynapseType, SynapseRecord, NetworkStats } from '../types/synapse.types.js';
3
- import type { SynapseRepository } from '../db/repositories/synapse.repository.js';
4
- import { strengthen, weaken, type NodeRef } from './hebbian.js';
5
- import { decayAll } from './decay.js';
6
- import { spreadingActivation, type ActivationResult } from './activation.js';
7
- import { findPath, type SynapsePath } from './pathfinder.js';
8
- import { getLogger } from '../utils/logger.js';
9
-
10
- export class SynapseManager {
11
- private logger = getLogger();
12
-
13
- constructor(
14
- private repo: SynapseRepository,
15
- private config: SynapsesConfig,
16
- ) {}
17
-
18
- strengthen(
19
- source: NodeRef,
20
- target: NodeRef,
21
- synapseType: SynapseType,
22
- context?: Record<string, unknown>,
23
- ): SynapseRecord {
24
- this.logger.debug(`Strengthening synapse ${source.type}:${source.id} --${synapseType}--> ${target.type}:${target.id}`);
25
- return strengthen(this.repo, source, target, synapseType, {
26
- initialWeight: this.config.initialWeight,
27
- learningRate: this.config.learningRate,
28
- pruneThreshold: this.config.pruneThreshold,
29
- }, context);
30
- }
31
-
32
- weaken(synapseId: number, factor: number = 0.5): void {
33
- this.logger.debug(`Weakening synapse ${synapseId} by factor ${factor}`);
34
- weaken(this.repo, synapseId, {
35
- initialWeight: this.config.initialWeight,
36
- learningRate: this.config.learningRate,
37
- pruneThreshold: this.config.pruneThreshold,
38
- }, factor);
39
- }
40
-
41
- find(
42
- source: NodeRef,
43
- target: NodeRef,
44
- synapseType: SynapseType,
45
- ): SynapseRecord | undefined {
46
- return this.repo.findBySourceTarget(
47
- source.type, source.id, target.type, target.id, synapseType,
48
- );
49
- }
50
-
51
- activate(
52
- startNode: NodeRef,
53
- maxDepth?: number,
54
- minWeight?: number,
55
- ): ActivationResult[] {
56
- return spreadingActivation(
57
- this.repo,
58
- startNode,
59
- maxDepth ?? this.config.maxDepth,
60
- minWeight ?? this.config.minActivationWeight,
61
- );
62
- }
63
-
64
- findPath(from: NodeRef, to: NodeRef, maxDepth?: number): SynapsePath | null {
65
- return findPath(this.repo, from, to, maxDepth ?? this.config.maxDepth + 2);
66
- }
67
-
68
- runDecay(): { decayed: number; pruned: number } {
69
- this.logger.info('Running synapse decay cycle');
70
- const result = decayAll(this.repo, {
71
- decayHalfLifeDays: this.config.decayHalfLifeDays,
72
- decayAfterDays: this.config.decayAfterDays,
73
- pruneThreshold: this.config.pruneThreshold,
74
- });
75
- this.logger.info(`Decay complete: ${result.decayed} decayed, ${result.pruned} pruned`);
76
- return result;
77
- }
78
-
79
- getErrorContext(errorId: number): {
80
- solutions: ActivationResult[];
81
- relatedErrors: ActivationResult[];
82
- relevantModules: ActivationResult[];
83
- preventionRules: ActivationResult[];
84
- insights: ActivationResult[];
85
- } {
86
- const all = this.activate({ type: 'error', id: errorId });
87
- return {
88
- solutions: all.filter(a => a.node.type === 'solution'),
89
- relatedErrors: all.filter(a => a.node.type === 'error'),
90
- relevantModules: all.filter(a => a.node.type === 'code_module'),
91
- preventionRules: all.filter(a => a.node.type === 'rule'),
92
- insights: all.filter(a => a.node.type === 'insight'),
93
- };
94
- }
95
-
96
- getStrongestSynapses(limit: number = 20): SynapseRecord[] {
97
- return this.repo.topByWeight(limit);
98
- }
99
-
100
- getDiverseSynapses(perType: number = 25): SynapseRecord[] {
101
- return this.repo.topDiverse(perType);
102
- }
103
-
104
- getNetworkStats(): NetworkStats {
105
- return {
106
- totalNodes: this.repo.countNodes(),
107
- totalSynapses: this.repo.totalCount(),
108
- avgWeight: this.repo.avgWeight(),
109
- nodesByType: {} as Record<NodeType, number>,
110
- synapsesByType: this.repo.countByType() as Record<SynapseType, number>,
111
- };
112
- }
113
- }