@timmeck/brain 1.0.0 → 1.1.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/BRAIN_PLAN.md +3324 -3324
- package/LICENSE +21 -21
- package/README.md +194 -188
- package/dist/brain.js +2 -0
- package/dist/brain.js.map +1 -1
- package/dist/cli/colors.d.ts +50 -0
- package/dist/cli/colors.js +106 -0
- package/dist/cli/colors.js.map +1 -0
- package/dist/cli/commands/config.d.ts +2 -0
- package/dist/cli/commands/config.js +165 -0
- package/dist/cli/commands/config.js.map +1 -0
- package/dist/cli/commands/dashboard.js +222 -8
- package/dist/cli/commands/dashboard.js.map +1 -1
- package/dist/cli/commands/export.js +3 -0
- package/dist/cli/commands/export.js.map +1 -1
- package/dist/cli/commands/import.js +24 -15
- package/dist/cli/commands/import.js.map +1 -1
- package/dist/cli/commands/insights.js +33 -6
- package/dist/cli/commands/insights.js.map +1 -1
- package/dist/cli/commands/learn.d.ts +2 -0
- package/dist/cli/commands/learn.js +22 -0
- package/dist/cli/commands/learn.js.map +1 -0
- package/dist/cli/commands/modules.js +25 -6
- package/dist/cli/commands/modules.js.map +1 -1
- package/dist/cli/commands/network.js +15 -9
- package/dist/cli/commands/network.js.map +1 -1
- package/dist/cli/commands/query.js +92 -25
- package/dist/cli/commands/query.js.map +1 -1
- package/dist/cli/commands/start.js +8 -5
- package/dist/cli/commands/start.js.map +1 -1
- package/dist/cli/commands/status.js +21 -16
- package/dist/cli/commands/status.js.map +1 -1
- package/dist/cli/commands/stop.js +5 -4
- package/dist/cli/commands/stop.js.map +1 -1
- package/dist/cli/ipc-helper.js +4 -3
- package/dist/cli/ipc-helper.js.map +1 -1
- package/dist/cli/update-check.d.ts +2 -0
- package/dist/cli/update-check.js +58 -0
- package/dist/cli/update-check.js.map +1 -0
- package/dist/db/migrations/001_core_schema.js +115 -115
- package/dist/db/migrations/002_learning_schema.js +33 -33
- package/dist/db/migrations/003_code_schema.js +48 -48
- package/dist/db/migrations/004_synapses_schema.js +52 -52
- package/dist/db/migrations/005_fts_indexes.js +73 -73
- package/dist/db/migrations/index.js +6 -6
- package/dist/db/repositories/antipattern.repository.js +3 -3
- package/dist/db/repositories/code-module.repository.d.ts +1 -0
- package/dist/db/repositories/code-module.repository.js +8 -0
- package/dist/db/repositories/code-module.repository.js.map +1 -1
- package/dist/db/repositories/error.repository.js +46 -46
- package/dist/db/repositories/insight.repository.js +3 -3
- package/dist/db/repositories/notification.repository.js +3 -3
- package/dist/db/repositories/project.repository.js +21 -21
- package/dist/db/repositories/rule.repository.js +24 -24
- package/dist/db/repositories/solution.repository.js +50 -50
- package/dist/db/repositories/synapse.repository.js +18 -18
- package/dist/db/repositories/terminal.repository.js +24 -24
- package/dist/index.js +4 -0
- package/dist/index.js.map +1 -1
- package/dist/ipc/router.d.ts +2 -0
- package/dist/ipc/router.js +7 -1
- package/dist/ipc/router.js.map +1 -1
- package/dist/services/code.service.d.ts +1 -1
- package/dist/services/code.service.js +5 -2
- package/dist/services/code.service.js.map +1 -1
- package/package.json +5 -4
- package/src/brain.ts +3 -0
- package/src/cli/colors.ts +116 -0
- package/src/cli/commands/config.ts +169 -0
- package/src/cli/commands/dashboard.ts +231 -8
- package/src/cli/commands/export.ts +4 -0
- package/src/cli/commands/import.ts +24 -15
- package/src/cli/commands/insights.ts +37 -5
- package/src/cli/commands/learn.ts +24 -0
- package/src/cli/commands/modules.ts +28 -5
- package/src/cli/commands/network.ts +15 -9
- package/src/cli/commands/query.ts +103 -26
- package/src/cli/commands/start.ts +8 -5
- package/src/cli/commands/status.ts +22 -16
- package/src/cli/commands/stop.ts +5 -4
- package/src/cli/ipc-helper.ts +4 -3
- package/src/cli/update-check.ts +63 -0
- package/src/code/analyzer.ts +77 -77
- package/src/code/fingerprint.ts +87 -87
- package/src/code/matcher.ts +64 -64
- package/src/code/parsers/generic.ts +29 -29
- package/src/code/parsers/python.ts +54 -54
- package/src/code/parsers/typescript.ts +65 -65
- package/src/code/registry.ts +60 -60
- package/src/code/scorer.ts +108 -108
- package/src/config.ts +111 -111
- package/src/db/connection.ts +22 -22
- package/src/db/migrations/001_core_schema.ts +120 -120
- package/src/db/migrations/002_learning_schema.ts +38 -38
- package/src/db/migrations/003_code_schema.ts +53 -53
- package/src/db/migrations/004_synapses_schema.ts +57 -57
- package/src/db/migrations/005_fts_indexes.ts +78 -78
- package/src/db/migrations/006_synapses_phase3.ts +17 -17
- package/src/db/migrations/index.ts +64 -64
- package/src/db/repositories/antipattern.repository.ts +66 -66
- package/src/db/repositories/code-module.repository.ts +9 -0
- package/src/db/repositories/error.repository.ts +149 -149
- package/src/db/repositories/insight.repository.ts +78 -78
- package/src/db/repositories/notification.repository.ts +66 -66
- package/src/db/repositories/project.repository.ts +93 -93
- package/src/db/repositories/rule.repository.ts +108 -108
- package/src/db/repositories/solution.repository.ts +154 -154
- package/src/db/repositories/synapse.repository.ts +153 -153
- package/src/db/repositories/terminal.repository.ts +101 -101
- package/src/hooks/post-tool-use.ts +90 -90
- package/src/hooks/post-write.ts +117 -117
- package/src/index.ts +4 -0
- package/src/ipc/client.ts +118 -118
- package/src/ipc/protocol.ts +35 -35
- package/src/ipc/router.ts +9 -1
- package/src/ipc/server.ts +110 -110
- package/src/learning/confidence-scorer.ts +47 -47
- package/src/learning/decay.ts +46 -46
- package/src/learning/learning-engine.ts +162 -162
- package/src/learning/pattern-extractor.ts +90 -90
- package/src/learning/rule-generator.ts +74 -74
- package/src/matching/error-matcher.ts +115 -115
- package/src/matching/fingerprint.ts +29 -29
- package/src/matching/similarity.ts +61 -61
- package/src/matching/tfidf.ts +74 -74
- package/src/matching/tokenizer.ts +41 -41
- package/src/mcp/auto-detect.ts +93 -93
- package/src/mcp/server.ts +73 -73
- package/src/mcp/tools.ts +290 -290
- package/src/parsing/error-parser.ts +28 -28
- package/src/parsing/parsers/compiler.ts +93 -93
- package/src/parsing/parsers/generic.ts +28 -28
- package/src/parsing/parsers/go.ts +97 -97
- package/src/parsing/parsers/node.ts +69 -69
- package/src/parsing/parsers/python.ts +62 -62
- package/src/parsing/parsers/rust.ts +50 -50
- package/src/parsing/parsers/shell.ts +42 -42
- package/src/parsing/types.ts +47 -47
- package/src/research/gap-analyzer.ts +135 -135
- package/src/research/insight-generator.ts +123 -123
- package/src/research/research-engine.ts +116 -116
- package/src/research/synergy-detector.ts +126 -126
- package/src/research/template-extractor.ts +130 -130
- package/src/research/trend-analyzer.ts +127 -127
- package/src/services/analytics.service.ts +87 -87
- package/src/services/code.service.ts +5 -2
- package/src/services/error.service.ts +164 -164
- package/src/services/notification.service.ts +41 -41
- package/src/services/prevention.service.ts +119 -119
- package/src/services/research.service.ts +93 -93
- package/src/services/solution.service.ts +116 -116
- package/src/services/synapse.service.ts +59 -59
- package/src/services/terminal.service.ts +81 -81
- package/src/synapses/activation.ts +80 -80
- package/src/synapses/decay.ts +38 -38
- package/src/synapses/hebbian.ts +69 -69
- package/src/synapses/pathfinder.ts +81 -81
- package/src/synapses/synapse-manager.ts +109 -109
- package/src/types/code.types.ts +52 -52
- package/src/types/config.types.ts +79 -79
- package/src/types/error.types.ts +67 -67
- package/src/types/ipc.types.ts +8 -8
- package/src/types/mcp.types.ts +53 -53
- package/src/types/research.types.ts +28 -28
- package/src/types/solution.types.ts +30 -30
- package/src/types/synapse.types.ts +49 -49
- package/src/utils/events.ts +45 -45
- package/src/utils/hash.ts +5 -5
- package/src/utils/logger.ts +48 -48
- package/src/utils/paths.ts +19 -19
- package/tests/fixtures/code-modules/modules.ts +83 -83
- package/tests/fixtures/errors/go.ts +9 -9
- package/tests/fixtures/errors/node.ts +24 -24
- package/tests/fixtures/errors/python.ts +21 -21
- package/tests/fixtures/errors/rust.ts +25 -25
- package/tests/fixtures/errors/shell.ts +15 -15
- package/tests/fixtures/solutions/solutions.ts +27 -27
- package/tests/helpers/setup-db.ts +52 -52
- package/tests/integration/code-flow.test.ts +86 -86
- package/tests/integration/error-flow.test.ts +83 -83
- package/tests/integration/ipc-flow.test.ts +166 -166
- package/tests/integration/learning-cycle.test.ts +82 -82
- package/tests/integration/synapse-flow.test.ts +117 -117
- package/tests/unit/code/analyzer.test.ts +58 -58
- package/tests/unit/code/fingerprint.test.ts +51 -51
- package/tests/unit/code/scorer.test.ts +55 -55
- package/tests/unit/learning/confidence-scorer.test.ts +60 -60
- package/tests/unit/learning/decay.test.ts +45 -45
- package/tests/unit/learning/pattern-extractor.test.ts +50 -50
- package/tests/unit/matching/error-matcher.test.ts +69 -69
- package/tests/unit/matching/fingerprint.test.ts +47 -47
- package/tests/unit/matching/similarity.test.ts +65 -65
- package/tests/unit/matching/tfidf.test.ts +71 -71
- package/tests/unit/matching/tokenizer.test.ts +83 -83
- package/tests/unit/parsing/parsers.test.ts +113 -113
- package/tests/unit/research/gap-analyzer.test.ts +45 -45
- package/tests/unit/research/trend-analyzer.test.ts +45 -45
- package/tests/unit/synapses/activation.test.ts +80 -80
- package/tests/unit/synapses/decay.test.ts +27 -27
- package/tests/unit/synapses/hebbian.test.ts +96 -96
- package/tests/unit/synapses/pathfinder.test.ts +72 -72
- package/tsconfig.json +18 -18
|
@@ -1,81 +1,81 @@
|
|
|
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
|
+
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,109 +1,109 @@
|
|
|
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
|
-
getNetworkStats(): NetworkStats {
|
|
101
|
-
return {
|
|
102
|
-
totalNodes: this.repo.countNodes(),
|
|
103
|
-
totalSynapses: this.repo.totalCount(),
|
|
104
|
-
avgWeight: this.repo.avgWeight(),
|
|
105
|
-
nodesByType: {} as Record<NodeType, number>,
|
|
106
|
-
synapsesByType: this.repo.countByType() as Record<SynapseType, number>,
|
|
107
|
-
};
|
|
108
|
-
}
|
|
109
|
-
}
|
|
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
|
+
getNetworkStats(): NetworkStats {
|
|
101
|
+
return {
|
|
102
|
+
totalNodes: this.repo.countNodes(),
|
|
103
|
+
totalSynapses: this.repo.totalCount(),
|
|
104
|
+
avgWeight: this.repo.avgWeight(),
|
|
105
|
+
nodesByType: {} as Record<NodeType, number>,
|
|
106
|
+
synapsesByType: this.repo.countByType() as Record<SynapseType, number>,
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
}
|
package/src/types/code.types.ts
CHANGED
|
@@ -1,52 +1,52 @@
|
|
|
1
|
-
export interface CodeModuleRecord {
|
|
2
|
-
id: number;
|
|
3
|
-
project_id: number;
|
|
4
|
-
name: string;
|
|
5
|
-
file_path: string;
|
|
6
|
-
language: string;
|
|
7
|
-
fingerprint: string;
|
|
8
|
-
description: string | null;
|
|
9
|
-
source_hash: string;
|
|
10
|
-
lines_of_code: number;
|
|
11
|
-
complexity: number | null;
|
|
12
|
-
reusability_score: number;
|
|
13
|
-
created_at: string;
|
|
14
|
-
updated_at: string;
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
export interface ModuleUsage {
|
|
18
|
-
id: number;
|
|
19
|
-
module_id: number;
|
|
20
|
-
used_in_project_id: number;
|
|
21
|
-
used_in_file: string;
|
|
22
|
-
usage_type: string;
|
|
23
|
-
first_used: string;
|
|
24
|
-
last_used: string;
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
export interface ModuleSimilarity {
|
|
28
|
-
id: number;
|
|
29
|
-
module_a_id: number;
|
|
30
|
-
module_b_id: number;
|
|
31
|
-
similarity_score: number;
|
|
32
|
-
computed_at: string;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
export interface CodeUnit {
|
|
36
|
-
name: string;
|
|
37
|
-
filePath: string;
|
|
38
|
-
language: string;
|
|
39
|
-
source: string;
|
|
40
|
-
exports: ExportInfo[];
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
export interface ExportInfo {
|
|
44
|
-
name: string;
|
|
45
|
-
type: 'function' | 'class' | 'constant' | 'type' | 'interface' | 'variable';
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
export interface ReusabilitySignal {
|
|
49
|
-
name: string;
|
|
50
|
-
score: number;
|
|
51
|
-
reason: string;
|
|
52
|
-
}
|
|
1
|
+
export interface CodeModuleRecord {
|
|
2
|
+
id: number;
|
|
3
|
+
project_id: number;
|
|
4
|
+
name: string;
|
|
5
|
+
file_path: string;
|
|
6
|
+
language: string;
|
|
7
|
+
fingerprint: string;
|
|
8
|
+
description: string | null;
|
|
9
|
+
source_hash: string;
|
|
10
|
+
lines_of_code: number;
|
|
11
|
+
complexity: number | null;
|
|
12
|
+
reusability_score: number;
|
|
13
|
+
created_at: string;
|
|
14
|
+
updated_at: string;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export interface ModuleUsage {
|
|
18
|
+
id: number;
|
|
19
|
+
module_id: number;
|
|
20
|
+
used_in_project_id: number;
|
|
21
|
+
used_in_file: string;
|
|
22
|
+
usage_type: string;
|
|
23
|
+
first_used: string;
|
|
24
|
+
last_used: string;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
export interface ModuleSimilarity {
|
|
28
|
+
id: number;
|
|
29
|
+
module_a_id: number;
|
|
30
|
+
module_b_id: number;
|
|
31
|
+
similarity_score: number;
|
|
32
|
+
computed_at: string;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export interface CodeUnit {
|
|
36
|
+
name: string;
|
|
37
|
+
filePath: string;
|
|
38
|
+
language: string;
|
|
39
|
+
source: string;
|
|
40
|
+
exports: ExportInfo[];
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export interface ExportInfo {
|
|
44
|
+
name: string;
|
|
45
|
+
type: 'function' | 'class' | 'constant' | 'type' | 'interface' | 'variable';
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
export interface ReusabilitySignal {
|
|
49
|
+
name: string;
|
|
50
|
+
score: number;
|
|
51
|
+
reason: string;
|
|
52
|
+
}
|
|
@@ -1,79 +1,79 @@
|
|
|
1
|
-
export interface IpcConfig {
|
|
2
|
-
pipeName: string;
|
|
3
|
-
timeout: number;
|
|
4
|
-
}
|
|
5
|
-
|
|
6
|
-
export interface LearningConfig {
|
|
7
|
-
intervalMs: number;
|
|
8
|
-
minOccurrences: number;
|
|
9
|
-
minSuccessRate: number;
|
|
10
|
-
minConfidence: number;
|
|
11
|
-
pruneThreshold: number;
|
|
12
|
-
maxRejectionRate: number;
|
|
13
|
-
decayHalfLifeDays: number;
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
export interface TerminalConfig {
|
|
17
|
-
staleTimeout: number;
|
|
18
|
-
maxConnected: number;
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
export interface MatchingConfig {
|
|
22
|
-
fingerprintFields: string[];
|
|
23
|
-
similarityThreshold: number;
|
|
24
|
-
maxResults: number;
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
export interface CodeConfig {
|
|
28
|
-
supportedLanguages: string[];
|
|
29
|
-
maxModuleSize: number;
|
|
30
|
-
similarityThreshold: number;
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
export interface SynapsesConfig {
|
|
34
|
-
initialWeight: number;
|
|
35
|
-
learningRate: number;
|
|
36
|
-
decayHalfLifeDays: number;
|
|
37
|
-
pruneThreshold: number;
|
|
38
|
-
decayAfterDays: number;
|
|
39
|
-
maxDepth: number;
|
|
40
|
-
minActivationWeight: number;
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
export interface ResearchConfig {
|
|
44
|
-
intervalMs: number;
|
|
45
|
-
initialDelayMs: number;
|
|
46
|
-
minDataPoints: number;
|
|
47
|
-
trendWindowDays: number;
|
|
48
|
-
gapMinOccurrences: number;
|
|
49
|
-
synergyMinWeight: number;
|
|
50
|
-
templateMinAdaptations: number;
|
|
51
|
-
insightExpiryDays: number;
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
export interface LogConfig {
|
|
55
|
-
level: string;
|
|
56
|
-
file: string;
|
|
57
|
-
maxSize: number;
|
|
58
|
-
maxFiles: number;
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
export interface RetentionConfig {
|
|
62
|
-
errorDays: number;
|
|
63
|
-
solutionDays: number;
|
|
64
|
-
insightDays: number;
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
export interface BrainConfig {
|
|
68
|
-
dataDir: string;
|
|
69
|
-
dbPath: string;
|
|
70
|
-
ipc: IpcConfig;
|
|
71
|
-
learning: LearningConfig;
|
|
72
|
-
terminal: TerminalConfig;
|
|
73
|
-
matching: MatchingConfig;
|
|
74
|
-
code: CodeConfig;
|
|
75
|
-
synapses: SynapsesConfig;
|
|
76
|
-
research: ResearchConfig;
|
|
77
|
-
log: LogConfig;
|
|
78
|
-
retention: RetentionConfig;
|
|
79
|
-
}
|
|
1
|
+
export interface IpcConfig {
|
|
2
|
+
pipeName: string;
|
|
3
|
+
timeout: number;
|
|
4
|
+
}
|
|
5
|
+
|
|
6
|
+
export interface LearningConfig {
|
|
7
|
+
intervalMs: number;
|
|
8
|
+
minOccurrences: number;
|
|
9
|
+
minSuccessRate: number;
|
|
10
|
+
minConfidence: number;
|
|
11
|
+
pruneThreshold: number;
|
|
12
|
+
maxRejectionRate: number;
|
|
13
|
+
decayHalfLifeDays: number;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export interface TerminalConfig {
|
|
17
|
+
staleTimeout: number;
|
|
18
|
+
maxConnected: number;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export interface MatchingConfig {
|
|
22
|
+
fingerprintFields: string[];
|
|
23
|
+
similarityThreshold: number;
|
|
24
|
+
maxResults: number;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
export interface CodeConfig {
|
|
28
|
+
supportedLanguages: string[];
|
|
29
|
+
maxModuleSize: number;
|
|
30
|
+
similarityThreshold: number;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export interface SynapsesConfig {
|
|
34
|
+
initialWeight: number;
|
|
35
|
+
learningRate: number;
|
|
36
|
+
decayHalfLifeDays: number;
|
|
37
|
+
pruneThreshold: number;
|
|
38
|
+
decayAfterDays: number;
|
|
39
|
+
maxDepth: number;
|
|
40
|
+
minActivationWeight: number;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export interface ResearchConfig {
|
|
44
|
+
intervalMs: number;
|
|
45
|
+
initialDelayMs: number;
|
|
46
|
+
minDataPoints: number;
|
|
47
|
+
trendWindowDays: number;
|
|
48
|
+
gapMinOccurrences: number;
|
|
49
|
+
synergyMinWeight: number;
|
|
50
|
+
templateMinAdaptations: number;
|
|
51
|
+
insightExpiryDays: number;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
export interface LogConfig {
|
|
55
|
+
level: string;
|
|
56
|
+
file: string;
|
|
57
|
+
maxSize: number;
|
|
58
|
+
maxFiles: number;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
export interface RetentionConfig {
|
|
62
|
+
errorDays: number;
|
|
63
|
+
solutionDays: number;
|
|
64
|
+
insightDays: number;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
export interface BrainConfig {
|
|
68
|
+
dataDir: string;
|
|
69
|
+
dbPath: string;
|
|
70
|
+
ipc: IpcConfig;
|
|
71
|
+
learning: LearningConfig;
|
|
72
|
+
terminal: TerminalConfig;
|
|
73
|
+
matching: MatchingConfig;
|
|
74
|
+
code: CodeConfig;
|
|
75
|
+
synapses: SynapsesConfig;
|
|
76
|
+
research: ResearchConfig;
|
|
77
|
+
log: LogConfig;
|
|
78
|
+
retention: RetentionConfig;
|
|
79
|
+
}
|