@timmeck/brain 1.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.
- package/BRAIN_PLAN.md +3324 -0
- package/LICENSE +21 -0
- package/README.md +188 -0
- package/dist/brain.d.ts +11 -0
- package/dist/brain.js +166 -0
- package/dist/brain.js.map +1 -0
- package/dist/cli/commands/dashboard.d.ts +2 -0
- package/dist/cli/commands/dashboard.js +457 -0
- package/dist/cli/commands/dashboard.js.map +1 -0
- package/dist/cli/commands/export.d.ts +2 -0
- package/dist/cli/commands/export.js +25 -0
- package/dist/cli/commands/export.js.map +1 -0
- package/dist/cli/commands/import.d.ts +2 -0
- package/dist/cli/commands/import.js +173 -0
- package/dist/cli/commands/import.js.map +1 -0
- package/dist/cli/commands/insights.d.ts +2 -0
- package/dist/cli/commands/insights.js +32 -0
- package/dist/cli/commands/insights.js.map +1 -0
- package/dist/cli/commands/modules.d.ts +2 -0
- package/dist/cli/commands/modules.js +29 -0
- package/dist/cli/commands/modules.js.map +1 -0
- package/dist/cli/commands/network.d.ts +2 -0
- package/dist/cli/commands/network.js +56 -0
- package/dist/cli/commands/network.js.map +1 -0
- package/dist/cli/commands/query.d.ts +2 -0
- package/dist/cli/commands/query.js +40 -0
- package/dist/cli/commands/query.js.map +1 -0
- package/dist/cli/commands/start.d.ts +2 -0
- package/dist/cli/commands/start.js +56 -0
- package/dist/cli/commands/start.js.map +1 -0
- package/dist/cli/commands/status.d.ts +2 -0
- package/dist/cli/commands/status.js +60 -0
- package/dist/cli/commands/status.js.map +1 -0
- package/dist/cli/commands/stop.d.ts +2 -0
- package/dist/cli/commands/stop.js +34 -0
- package/dist/cli/commands/stop.js.map +1 -0
- package/dist/cli/ipc-helper.d.ts +2 -0
- package/dist/cli/ipc-helper.js +25 -0
- package/dist/cli/ipc-helper.js.map +1 -0
- package/dist/code/analyzer.d.ts +12 -0
- package/dist/code/analyzer.js +58 -0
- package/dist/code/analyzer.js.map +1 -0
- package/dist/code/fingerprint.d.ts +2 -0
- package/dist/code/fingerprint.js +84 -0
- package/dist/code/fingerprint.js.map +1 -0
- package/dist/code/matcher.d.ts +9 -0
- package/dist/code/matcher.js +37 -0
- package/dist/code/matcher.js.map +1 -0
- package/dist/code/parsers/generic.d.ts +7 -0
- package/dist/code/parsers/generic.js +22 -0
- package/dist/code/parsers/generic.js.map +1 -0
- package/dist/code/parsers/python.d.ts +7 -0
- package/dist/code/parsers/python.js +45 -0
- package/dist/code/parsers/python.js.map +1 -0
- package/dist/code/parsers/typescript.d.ts +7 -0
- package/dist/code/parsers/typescript.js +58 -0
- package/dist/code/parsers/typescript.js.map +1 -0
- package/dist/code/registry.d.ts +22 -0
- package/dist/code/registry.js +31 -0
- package/dist/code/registry.js.map +1 -0
- package/dist/code/scorer.d.ts +15 -0
- package/dist/code/scorer.js +103 -0
- package/dist/code/scorer.js.map +1 -0
- package/dist/config.d.ts +2 -0
- package/dist/config.js +110 -0
- package/dist/config.js.map +1 -0
- package/dist/db/connection.d.ts +2 -0
- package/dist/db/connection.js +19 -0
- package/dist/db/connection.js.map +1 -0
- package/dist/db/migrations/001_core_schema.d.ts +2 -0
- package/dist/db/migrations/001_core_schema.js +119 -0
- package/dist/db/migrations/001_core_schema.js.map +1 -0
- package/dist/db/migrations/002_learning_schema.d.ts +2 -0
- package/dist/db/migrations/002_learning_schema.js +37 -0
- package/dist/db/migrations/002_learning_schema.js.map +1 -0
- package/dist/db/migrations/003_code_schema.d.ts +2 -0
- package/dist/db/migrations/003_code_schema.js +52 -0
- package/dist/db/migrations/003_code_schema.js.map +1 -0
- package/dist/db/migrations/004_synapses_schema.d.ts +2 -0
- package/dist/db/migrations/004_synapses_schema.js +56 -0
- package/dist/db/migrations/004_synapses_schema.js.map +1 -0
- package/dist/db/migrations/005_fts_indexes.d.ts +2 -0
- package/dist/db/migrations/005_fts_indexes.js +77 -0
- package/dist/db/migrations/005_fts_indexes.js.map +1 -0
- package/dist/db/migrations/006_synapses_phase3.d.ts +2 -0
- package/dist/db/migrations/006_synapses_phase3.js +14 -0
- package/dist/db/migrations/006_synapses_phase3.js.map +1 -0
- package/dist/db/migrations/index.d.ts +2 -0
- package/dist/db/migrations/index.js +49 -0
- package/dist/db/migrations/index.js.map +1 -0
- package/dist/db/repositories/antipattern.repository.d.ts +26 -0
- package/dist/db/repositories/antipattern.repository.js +44 -0
- package/dist/db/repositories/antipattern.repository.js.map +1 -0
- package/dist/db/repositories/code-module.repository.d.ts +19 -0
- package/dist/db/repositories/code-module.repository.js +64 -0
- package/dist/db/repositories/code-module.repository.js.map +1 -0
- package/dist/db/repositories/error.repository.d.ts +20 -0
- package/dist/db/repositories/error.repository.js +134 -0
- package/dist/db/repositories/error.repository.js.map +1 -0
- package/dist/db/repositories/insight.repository.d.ts +18 -0
- package/dist/db/repositories/insight.repository.js +57 -0
- package/dist/db/repositories/insight.repository.js.map +1 -0
- package/dist/db/repositories/notification.repository.d.ts +24 -0
- package/dist/db/repositories/notification.repository.js +40 -0
- package/dist/db/repositories/notification.repository.js.map +1 -0
- package/dist/db/repositories/project.repository.d.ts +25 -0
- package/dist/db/repositories/project.repository.js +72 -0
- package/dist/db/repositories/project.repository.js.map +1 -0
- package/dist/db/repositories/rule.repository.d.ts +31 -0
- package/dist/db/repositories/rule.repository.js +81 -0
- package/dist/db/repositories/rule.repository.js.map +1 -0
- package/dist/db/repositories/solution.repository.d.ts +27 -0
- package/dist/db/repositories/solution.repository.js +132 -0
- package/dist/db/repositories/solution.repository.js.map +1 -0
- package/dist/db/repositories/synapse.repository.d.ts +25 -0
- package/dist/db/repositories/synapse.repository.js +115 -0
- package/dist/db/repositories/synapse.repository.js.map +1 -0
- package/dist/db/repositories/terminal.repository.d.ts +27 -0
- package/dist/db/repositories/terminal.repository.js +78 -0
- package/dist/db/repositories/terminal.repository.js.map +1 -0
- package/dist/hooks/post-tool-use.d.ts +2 -0
- package/dist/hooks/post-tool-use.js +77 -0
- package/dist/hooks/post-tool-use.js.map +1 -0
- package/dist/hooks/post-write.d.ts +2 -0
- package/dist/hooks/post-write.js +102 -0
- package/dist/hooks/post-write.js.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +47 -0
- package/dist/index.js.map +1 -0
- package/dist/ipc/client.d.ts +16 -0
- package/dist/ipc/client.js +101 -0
- package/dist/ipc/client.js.map +1 -0
- package/dist/ipc/protocol.d.ts +8 -0
- package/dist/ipc/protocol.js +29 -0
- package/dist/ipc/protocol.js.map +1 -0
- package/dist/ipc/router.d.ts +28 -0
- package/dist/ipc/router.js +70 -0
- package/dist/ipc/router.js.map +1 -0
- package/dist/ipc/server.d.ts +14 -0
- package/dist/ipc/server.js +98 -0
- package/dist/ipc/server.js.map +1 -0
- package/dist/learning/confidence-scorer.d.ts +13 -0
- package/dist/learning/confidence-scorer.js +35 -0
- package/dist/learning/confidence-scorer.js.map +1 -0
- package/dist/learning/decay.d.ts +13 -0
- package/dist/learning/decay.js +37 -0
- package/dist/learning/decay.js.map +1 -0
- package/dist/learning/learning-engine.d.ts +30 -0
- package/dist/learning/learning-engine.js +121 -0
- package/dist/learning/learning-engine.js.map +1 -0
- package/dist/learning/pattern-extractor.d.ts +16 -0
- package/dist/learning/pattern-extractor.js +61 -0
- package/dist/learning/pattern-extractor.js.map +1 -0
- package/dist/learning/rule-generator.d.ts +18 -0
- package/dist/learning/rule-generator.js +50 -0
- package/dist/learning/rule-generator.js.map +1 -0
- package/dist/matching/error-matcher.d.ts +13 -0
- package/dist/matching/error-matcher.js +84 -0
- package/dist/matching/error-matcher.js.map +1 -0
- package/dist/matching/fingerprint.d.ts +3 -0
- package/dist/matching/fingerprint.js +23 -0
- package/dist/matching/fingerprint.js.map +1 -0
- package/dist/matching/similarity.d.ts +3 -0
- package/dist/matching/similarity.js +53 -0
- package/dist/matching/similarity.js.map +1 -0
- package/dist/matching/tfidf.d.ts +15 -0
- package/dist/matching/tfidf.js +68 -0
- package/dist/matching/tfidf.js.map +1 -0
- package/dist/matching/tokenizer.d.ts +4 -0
- package/dist/matching/tokenizer.js +36 -0
- package/dist/matching/tokenizer.js.map +1 -0
- package/dist/mcp/auto-detect.d.ts +1 -0
- package/dist/mcp/auto-detect.js +81 -0
- package/dist/mcp/auto-detect.js.map +1 -0
- package/dist/mcp/server.d.ts +1 -0
- package/dist/mcp/server.js +68 -0
- package/dist/mcp/server.js.map +1 -0
- package/dist/mcp/tools.d.ts +3 -0
- package/dist/mcp/tools.js +201 -0
- package/dist/mcp/tools.js.map +1 -0
- package/dist/parsing/error-parser.d.ts +3 -0
- package/dist/parsing/error-parser.js +26 -0
- package/dist/parsing/error-parser.js.map +1 -0
- package/dist/parsing/parsers/compiler.d.ts +2 -0
- package/dist/parsing/parsers/compiler.js +83 -0
- package/dist/parsing/parsers/compiler.js.map +1 -0
- package/dist/parsing/parsers/generic.d.ts +2 -0
- package/dist/parsing/parsers/generic.js +23 -0
- package/dist/parsing/parsers/generic.js.map +1 -0
- package/dist/parsing/parsers/go.d.ts +2 -0
- package/dist/parsing/parsers/go.js +85 -0
- package/dist/parsing/parsers/go.js.map +1 -0
- package/dist/parsing/parsers/node.d.ts +2 -0
- package/dist/parsing/parsers/node.js +61 -0
- package/dist/parsing/parsers/node.js.map +1 -0
- package/dist/parsing/parsers/python.d.ts +2 -0
- package/dist/parsing/parsers/python.js +50 -0
- package/dist/parsing/parsers/python.js.map +1 -0
- package/dist/parsing/parsers/rust.d.ts +2 -0
- package/dist/parsing/parsers/rust.js +43 -0
- package/dist/parsing/parsers/rust.js.map +1 -0
- package/dist/parsing/parsers/shell.d.ts +2 -0
- package/dist/parsing/parsers/shell.js +36 -0
- package/dist/parsing/parsers/shell.js.map +1 -0
- package/dist/parsing/types.d.ts +28 -0
- package/dist/parsing/types.js +21 -0
- package/dist/parsing/types.js.map +1 -0
- package/dist/research/gap-analyzer.d.ts +23 -0
- package/dist/research/gap-analyzer.js +119 -0
- package/dist/research/gap-analyzer.js.map +1 -0
- package/dist/research/insight-generator.d.ts +23 -0
- package/dist/research/insight-generator.js +107 -0
- package/dist/research/insight-generator.js.map +1 -0
- package/dist/research/research-engine.d.ts +31 -0
- package/dist/research/research-engine.js +97 -0
- package/dist/research/research-engine.js.map +1 -0
- package/dist/research/synergy-detector.d.ts +24 -0
- package/dist/research/synergy-detector.js +109 -0
- package/dist/research/synergy-detector.js.map +1 -0
- package/dist/research/template-extractor.d.ts +18 -0
- package/dist/research/template-extractor.js +116 -0
- package/dist/research/template-extractor.js.map +1 -0
- package/dist/research/trend-analyzer.d.ts +20 -0
- package/dist/research/trend-analyzer.js +111 -0
- package/dist/research/trend-analyzer.js.map +1 -0
- package/dist/services/analytics.service.d.ts +52 -0
- package/dist/services/analytics.service.js +59 -0
- package/dist/services/analytics.service.js.map +1 -0
- package/dist/services/code.service.d.ts +39 -0
- package/dist/services/code.service.js +98 -0
- package/dist/services/code.service.js.map +1 -0
- package/dist/services/error.service.d.ts +35 -0
- package/dist/services/error.service.js +118 -0
- package/dist/services/error.service.js.map +1 -0
- package/dist/services/notification.service.d.ts +17 -0
- package/dist/services/notification.service.js +29 -0
- package/dist/services/notification.service.js.map +1 -0
- package/dist/services/prevention.service.d.ts +35 -0
- package/dist/services/prevention.service.js +82 -0
- package/dist/services/prevention.service.js.map +1 -0
- package/dist/services/research.service.d.ts +35 -0
- package/dist/services/research.service.js +60 -0
- package/dist/services/research.service.js.map +1 -0
- package/dist/services/solution.service.d.ts +30 -0
- package/dist/services/solution.service.js +73 -0
- package/dist/services/solution.service.js.map +1 -0
- package/dist/services/synapse.service.d.ts +30 -0
- package/dist/services/synapse.service.js +25 -0
- package/dist/services/synapse.service.js.map +1 -0
- package/dist/services/terminal.service.d.ts +20 -0
- package/dist/services/terminal.service.js +66 -0
- package/dist/services/terminal.service.js.map +1 -0
- package/dist/synapses/activation.d.ts +13 -0
- package/dist/synapses/activation.js +50 -0
- package/dist/synapses/activation.js.map +1 -0
- package/dist/synapses/decay.d.ts +11 -0
- package/dist/synapses/decay.js +27 -0
- package/dist/synapses/decay.js.map +1 -0
- package/dist/synapses/hebbian.d.ts +13 -0
- package/dist/synapses/hebbian.js +36 -0
- package/dist/synapses/hebbian.js.map +1 -0
- package/dist/synapses/pathfinder.d.ts +14 -0
- package/dist/synapses/pathfinder.js +50 -0
- package/dist/synapses/pathfinder.js.map +1 -0
- package/dist/synapses/synapse-manager.d.ts +30 -0
- package/dist/synapses/synapse-manager.js +72 -0
- package/dist/synapses/synapse-manager.js.map +1 -0
- package/dist/types/code.types.d.ts +47 -0
- package/dist/types/code.types.js +2 -0
- package/dist/types/code.types.js.map +1 -0
- package/dist/types/config.types.d.ts +70 -0
- package/dist/types/config.types.js +2 -0
- package/dist/types/config.types.js.map +1 -0
- package/dist/types/error.types.d.ts +60 -0
- package/dist/types/error.types.js +2 -0
- package/dist/types/error.types.js.map +1 -0
- package/dist/types/ipc.types.d.ts +11 -0
- package/dist/types/ipc.types.js +2 -0
- package/dist/types/ipc.types.js.map +1 -0
- package/dist/types/mcp.types.d.ts +46 -0
- package/dist/types/mcp.types.js +2 -0
- package/dist/types/mcp.types.js.map +1 -0
- package/dist/types/research.types.d.ts +25 -0
- package/dist/types/research.types.js +2 -0
- package/dist/types/research.types.js.map +1 -0
- package/dist/types/solution.types.d.ts +28 -0
- package/dist/types/solution.types.js +2 -0
- package/dist/types/solution.types.js.map +1 -0
- package/dist/types/synapse.types.d.ts +37 -0
- package/dist/types/synapse.types.js +2 -0
- package/dist/types/synapse.types.js.map +1 -0
- package/dist/utils/events.d.ts +59 -0
- package/dist/utils/events.js +23 -0
- package/dist/utils/events.js.map +1 -0
- package/dist/utils/hash.d.ts +1 -0
- package/dist/utils/hash.js +5 -0
- package/dist/utils/hash.js.map +1 -0
- package/dist/utils/logger.d.ts +8 -0
- package/dist/utils/logger.js +39 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/paths.d.ts +3 -0
- package/dist/utils/paths.js +18 -0
- package/dist/utils/paths.js.map +1 -0
- package/package.json +43 -0
- package/src/brain.ts +220 -0
- package/src/cli/commands/dashboard.ts +495 -0
- package/src/cli/commands/export.ts +27 -0
- package/src/cli/commands/import.ts +190 -0
- package/src/cli/commands/insights.ts +33 -0
- package/src/cli/commands/modules.ts +30 -0
- package/src/cli/commands/network.ts +61 -0
- package/src/cli/commands/query.ts +43 -0
- package/src/cli/commands/start.ts +59 -0
- package/src/cli/commands/status.ts +69 -0
- package/src/cli/commands/stop.ts +33 -0
- package/src/cli/ipc-helper.ts +21 -0
- package/src/code/analyzer.ts +77 -0
- package/src/code/fingerprint.ts +87 -0
- package/src/code/matcher.ts +64 -0
- package/src/code/parsers/generic.ts +29 -0
- package/src/code/parsers/python.ts +54 -0
- package/src/code/parsers/typescript.ts +65 -0
- package/src/code/registry.ts +60 -0
- package/src/code/scorer.ts +108 -0
- package/src/config.ts +111 -0
- package/src/db/connection.ts +22 -0
- package/src/db/migrations/001_core_schema.ts +120 -0
- package/src/db/migrations/002_learning_schema.ts +38 -0
- package/src/db/migrations/003_code_schema.ts +53 -0
- package/src/db/migrations/004_synapses_schema.ts +57 -0
- package/src/db/migrations/005_fts_indexes.ts +78 -0
- package/src/db/migrations/006_synapses_phase3.ts +17 -0
- package/src/db/migrations/index.ts +64 -0
- package/src/db/repositories/antipattern.repository.ts +66 -0
- package/src/db/repositories/code-module.repository.ts +80 -0
- package/src/db/repositories/error.repository.ts +149 -0
- package/src/db/repositories/insight.repository.ts +78 -0
- package/src/db/repositories/notification.repository.ts +66 -0
- package/src/db/repositories/project.repository.ts +93 -0
- package/src/db/repositories/rule.repository.ts +108 -0
- package/src/db/repositories/solution.repository.ts +154 -0
- package/src/db/repositories/synapse.repository.ts +153 -0
- package/src/db/repositories/terminal.repository.ts +101 -0
- package/src/hooks/post-tool-use.ts +90 -0
- package/src/hooks/post-write.ts +117 -0
- package/src/index.ts +53 -0
- package/src/ipc/client.ts +118 -0
- package/src/ipc/protocol.ts +35 -0
- package/src/ipc/router.ts +106 -0
- package/src/ipc/server.ts +110 -0
- package/src/learning/confidence-scorer.ts +47 -0
- package/src/learning/decay.ts +46 -0
- package/src/learning/learning-engine.ts +162 -0
- package/src/learning/pattern-extractor.ts +90 -0
- package/src/learning/rule-generator.ts +74 -0
- package/src/main.rs:10:5 +0 -0
- package/src/matching/error-matcher.ts +115 -0
- package/src/matching/fingerprint.ts +29 -0
- package/src/matching/similarity.ts +61 -0
- package/src/matching/tfidf.ts +74 -0
- package/src/matching/tokenizer.ts +41 -0
- package/src/mcp/auto-detect.ts +93 -0
- package/src/mcp/server.ts +73 -0
- package/src/mcp/tools.ts +290 -0
- package/src/parsing/error-parser.ts +28 -0
- package/src/parsing/parsers/compiler.ts +93 -0
- package/src/parsing/parsers/generic.ts +28 -0
- package/src/parsing/parsers/go.ts +97 -0
- package/src/parsing/parsers/node.ts +69 -0
- package/src/parsing/parsers/python.ts +62 -0
- package/src/parsing/parsers/rust.ts +50 -0
- package/src/parsing/parsers/shell.ts +42 -0
- package/src/parsing/types.ts +47 -0
- package/src/research/gap-analyzer.ts +135 -0
- package/src/research/insight-generator.ts +123 -0
- package/src/research/research-engine.ts +116 -0
- package/src/research/synergy-detector.ts +126 -0
- package/src/research/template-extractor.ts +130 -0
- package/src/research/trend-analyzer.ts +127 -0
- package/src/services/analytics.service.ts +87 -0
- package/src/services/code.service.ts +140 -0
- package/src/services/error.service.ts +164 -0
- package/src/services/notification.service.ts +41 -0
- package/src/services/prevention.service.ts +119 -0
- package/src/services/research.service.ts +93 -0
- package/src/services/solution.service.ts +116 -0
- package/src/services/synapse.service.ts +59 -0
- package/src/services/terminal.service.ts +81 -0
- package/src/synapses/activation.ts +80 -0
- package/src/synapses/decay.ts +38 -0
- package/src/synapses/hebbian.ts +69 -0
- package/src/synapses/pathfinder.ts +81 -0
- package/src/synapses/synapse-manager.ts +109 -0
- package/src/types/code.types.ts +52 -0
- package/src/types/config.types.ts +79 -0
- package/src/types/error.types.ts +67 -0
- package/src/types/ipc.types.ts +8 -0
- package/src/types/mcp.types.ts +53 -0
- package/src/types/research.types.ts +28 -0
- package/src/types/solution.types.ts +30 -0
- package/src/types/synapse.types.ts +49 -0
- package/src/utils/events.ts +45 -0
- package/src/utils/hash.ts +5 -0
- package/src/utils/logger.ts +48 -0
- package/src/utils/paths.ts +19 -0
- package/tests/fixtures/code-modules/modules.ts +83 -0
- package/tests/fixtures/errors/go.ts +9 -0
- package/tests/fixtures/errors/node.ts +24 -0
- package/tests/fixtures/errors/python.ts +21 -0
- package/tests/fixtures/errors/rust.ts +25 -0
- package/tests/fixtures/errors/shell.ts +15 -0
- package/tests/fixtures/solutions/solutions.ts +27 -0
- package/tests/helpers/setup-db.ts +52 -0
- package/tests/integration/code-flow.test.ts +86 -0
- package/tests/integration/error-flow.test.ts +83 -0
- package/tests/integration/ipc-flow.test.ts +166 -0
- package/tests/integration/learning-cycle.test.ts +82 -0
- package/tests/integration/synapse-flow.test.ts +117 -0
- package/tests/unit/code/analyzer.test.ts +58 -0
- package/tests/unit/code/fingerprint.test.ts +51 -0
- package/tests/unit/code/scorer.test.ts +55 -0
- package/tests/unit/learning/confidence-scorer.test.ts +60 -0
- package/tests/unit/learning/decay.test.ts +45 -0
- package/tests/unit/learning/pattern-extractor.test.ts +50 -0
- package/tests/unit/matching/error-matcher.test.ts +69 -0
- package/tests/unit/matching/fingerprint.test.ts +47 -0
- package/tests/unit/matching/similarity.test.ts +65 -0
- package/tests/unit/matching/tfidf.test.ts +71 -0
- package/tests/unit/matching/tokenizer.test.ts +83 -0
- package/tests/unit/parsing/parsers.test.ts +113 -0
- package/tests/unit/research/gap-analyzer.test.ts +45 -0
- package/tests/unit/research/trend-analyzer.test.ts +45 -0
- package/tests/unit/synapses/activation.test.ts +80 -0
- package/tests/unit/synapses/decay.test.ts +27 -0
- package/tests/unit/synapses/hebbian.test.ts +96 -0
- package/tests/unit/synapses/pathfinder.test.ts +72 -0
- package/tsconfig.json +18 -0
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import { getLogger } from '../utils/logger.js';
|
|
2
|
+
|
|
3
|
+
const logger = getLogger();
|
|
4
|
+
import type { ErrorService } from '../services/error.service.js';
|
|
5
|
+
import type { SolutionService } from '../services/solution.service.js';
|
|
6
|
+
import type { TerminalService } from '../services/terminal.service.js';
|
|
7
|
+
import type { PreventionService } from '../services/prevention.service.js';
|
|
8
|
+
import type { CodeService } from '../services/code.service.js';
|
|
9
|
+
import type { SynapseService } from '../services/synapse.service.js';
|
|
10
|
+
import type { ResearchService } from '../services/research.service.js';
|
|
11
|
+
import type { NotificationService } from '../services/notification.service.js';
|
|
12
|
+
import type { AnalyticsService } from '../services/analytics.service.js';
|
|
13
|
+
|
|
14
|
+
export interface Services {
|
|
15
|
+
error: ErrorService;
|
|
16
|
+
solution: SolutionService;
|
|
17
|
+
terminal: TerminalService;
|
|
18
|
+
prevention: PreventionService;
|
|
19
|
+
code: CodeService;
|
|
20
|
+
synapse: SynapseService;
|
|
21
|
+
research: ResearchService;
|
|
22
|
+
notification: NotificationService;
|
|
23
|
+
analytics: AnalyticsService;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
type MethodHandler = (params: unknown) => unknown;
|
|
27
|
+
|
|
28
|
+
export class IpcRouter {
|
|
29
|
+
private methods: Map<string, MethodHandler>;
|
|
30
|
+
|
|
31
|
+
constructor(private services: Services) {
|
|
32
|
+
this.methods = this.buildMethodMap();
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
handle(method: string, params: unknown): unknown {
|
|
36
|
+
const handler = this.methods.get(method);
|
|
37
|
+
if (!handler) {
|
|
38
|
+
throw new Error(`Unknown method: ${method}`);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
logger.debug(`IPC: ${method}`, { params });
|
|
42
|
+
const result = handler(params);
|
|
43
|
+
logger.debug(`IPC: ${method} → done`);
|
|
44
|
+
return result;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
listMethods(): string[] {
|
|
48
|
+
return [...this.methods.keys()];
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
private buildMethodMap(): Map<string, MethodHandler> {
|
|
52
|
+
const s = this.services;
|
|
53
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
54
|
+
const p = (params: unknown) => params as any;
|
|
55
|
+
|
|
56
|
+
return new Map<string, MethodHandler>([
|
|
57
|
+
// Terminal Lifecycle
|
|
58
|
+
['terminal.register', (params) => s.terminal.register(p(params))],
|
|
59
|
+
['terminal.heartbeat', (params) => s.terminal.heartbeat(p(params).uuid)],
|
|
60
|
+
['terminal.disconnect', (params) => s.terminal.disconnect(p(params).uuid)],
|
|
61
|
+
|
|
62
|
+
// Error Brain
|
|
63
|
+
['error.report', (params) => s.error.report(p(params))],
|
|
64
|
+
['error.query', (params) => s.error.query(p(params))],
|
|
65
|
+
['error.match', (params) => s.error.matchSimilar(p(params).error_id ?? p(params).errorId)],
|
|
66
|
+
['error.resolve', (params) => s.error.resolve(p(params).error_id ?? p(params).errorId, p(params).solution_id ?? p(params).solutionId)],
|
|
67
|
+
['error.get', (params) => s.error.getById(p(params).id)],
|
|
68
|
+
|
|
69
|
+
// Solutions
|
|
70
|
+
['solution.report', (params) => s.solution.report(p(params))],
|
|
71
|
+
['solution.query', (params) => s.solution.findForError(p(params).error_id ?? p(params).errorId)],
|
|
72
|
+
['solution.rate', (params) => s.solution.rateOutcome(p(params))],
|
|
73
|
+
['solution.attempt', (params) => s.solution.rateOutcome({ ...p(params), success: false })],
|
|
74
|
+
|
|
75
|
+
// Code Brain
|
|
76
|
+
['code.analyze', (params) => s.code.analyzeAndRegister(p(params))],
|
|
77
|
+
['code.find', (params) => s.code.findReusable(p(params))],
|
|
78
|
+
['code.similarity', (params) => s.code.checkSimilarity(p(params).source, p(params).language)],
|
|
79
|
+
['code.modules', (params) => s.code.listModules(p(params)?.projectId)],
|
|
80
|
+
['code.get', (params) => s.code.getById(p(params).id)],
|
|
81
|
+
|
|
82
|
+
// Prevention
|
|
83
|
+
['prevention.check', (params) => s.prevention.checkRules(p(params).errorType, p(params).message, p(params).projectId)],
|
|
84
|
+
['prevention.antipatterns', (params) => s.prevention.checkAntipatterns(p(params).errorType ?? '', p(params).message ?? p(params).error_output ?? '', p(params).projectId)],
|
|
85
|
+
|
|
86
|
+
// Synapses
|
|
87
|
+
['synapse.context', (params) => s.synapse.getErrorContext(p(params).errorId ?? p(params).error_id ?? p(params).node_id)],
|
|
88
|
+
['synapse.path', (params) => s.synapse.findPath(p(params).from_type ?? p(params).fromType, p(params).from_id ?? p(params).fromId, p(params).to_type ?? p(params).toType, p(params).to_id ?? p(params).toId)],
|
|
89
|
+
['synapse.related', (params) => s.synapse.getRelated(p(params))],
|
|
90
|
+
['synapse.stats', () => s.synapse.getNetworkStats()],
|
|
91
|
+
|
|
92
|
+
// Research / Insights
|
|
93
|
+
['research.insights', (params) => s.research.getInsights(p(params))],
|
|
94
|
+
['research.suggest', (params) => s.research.getInsights({ limit: 10, activeOnly: true, ...p(params) })],
|
|
95
|
+
['research.trends', (params) => s.research.getTrends(p(params)?.projectId, p(params)?.windowDays)],
|
|
96
|
+
|
|
97
|
+
// Notifications
|
|
98
|
+
['notification.list', (params) => s.notification.list(p(params)?.projectId)],
|
|
99
|
+
['notification.ack', (params) => s.notification.acknowledge(p(params).id)],
|
|
100
|
+
|
|
101
|
+
// Analytics
|
|
102
|
+
['analytics.summary', (params) => s.analytics.getSummary(p(params)?.projectId)],
|
|
103
|
+
['analytics.network', (params) => s.analytics.getNetworkOverview(p(params)?.limit)],
|
|
104
|
+
]);
|
|
105
|
+
}
|
|
106
|
+
}
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import net from 'node:net';
|
|
2
|
+
import { randomUUID } from 'node:crypto';
|
|
3
|
+
import { getLogger } from '../utils/logger.js';
|
|
4
|
+
|
|
5
|
+
const logger = getLogger();
|
|
6
|
+
import type { IpcMessage } from '../types/ipc.types.js';
|
|
7
|
+
import { encodeMessage, MessageDecoder } from './protocol.js';
|
|
8
|
+
import type { IpcRouter } from './router.js';
|
|
9
|
+
|
|
10
|
+
export class IpcServer {
|
|
11
|
+
private server: net.Server | null = null;
|
|
12
|
+
private clients = new Map<string, net.Socket>();
|
|
13
|
+
|
|
14
|
+
constructor(
|
|
15
|
+
private router: IpcRouter,
|
|
16
|
+
private pipeName: string,
|
|
17
|
+
) {}
|
|
18
|
+
|
|
19
|
+
start(): void {
|
|
20
|
+
this.server = net.createServer((socket) => {
|
|
21
|
+
const clientId = randomUUID();
|
|
22
|
+
this.clients.set(clientId, socket);
|
|
23
|
+
const decoder = new MessageDecoder();
|
|
24
|
+
|
|
25
|
+
logger.info(`IPC client connected: ${clientId}`);
|
|
26
|
+
|
|
27
|
+
socket.on('data', (chunk) => {
|
|
28
|
+
const messages = decoder.feed(chunk);
|
|
29
|
+
for (const msg of messages) {
|
|
30
|
+
this.handleMessage(clientId, msg, socket);
|
|
31
|
+
}
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
socket.on('close', () => {
|
|
35
|
+
logger.info(`IPC client disconnected: ${clientId}`);
|
|
36
|
+
this.clients.delete(clientId);
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
socket.on('error', (err) => {
|
|
40
|
+
logger.error(`IPC client ${clientId} error:`, err);
|
|
41
|
+
this.clients.delete(clientId);
|
|
42
|
+
});
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
this.server.on('error', (err) => {
|
|
46
|
+
logger.error('IPC server error:', err);
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
this.server.listen(this.pipeName, () => {
|
|
50
|
+
logger.info(`IPC server listening on ${this.pipeName}`);
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
private handleMessage(clientId: string, msg: IpcMessage, socket: net.Socket): void {
|
|
55
|
+
if (msg.type !== 'request' || !msg.method) return;
|
|
56
|
+
|
|
57
|
+
try {
|
|
58
|
+
const result = this.router.handle(msg.method, msg.params);
|
|
59
|
+
const response: IpcMessage = {
|
|
60
|
+
id: msg.id,
|
|
61
|
+
type: 'response',
|
|
62
|
+
result,
|
|
63
|
+
};
|
|
64
|
+
socket.write(encodeMessage(response));
|
|
65
|
+
} catch (err) {
|
|
66
|
+
const response: IpcMessage = {
|
|
67
|
+
id: msg.id,
|
|
68
|
+
type: 'response',
|
|
69
|
+
error: { code: -1, message: err instanceof Error ? err.message : String(err) },
|
|
70
|
+
};
|
|
71
|
+
socket.write(encodeMessage(response));
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
notify(clientId: string | null, notification: Omit<IpcMessage, 'id' | 'type'>): void {
|
|
76
|
+
const msg: IpcMessage = {
|
|
77
|
+
id: randomUUID(),
|
|
78
|
+
type: 'notification',
|
|
79
|
+
...notification,
|
|
80
|
+
};
|
|
81
|
+
const encoded = encodeMessage(msg);
|
|
82
|
+
|
|
83
|
+
if (clientId) {
|
|
84
|
+
const socket = this.clients.get(clientId);
|
|
85
|
+
if (socket && !socket.destroyed) {
|
|
86
|
+
socket.write(encoded);
|
|
87
|
+
}
|
|
88
|
+
} else {
|
|
89
|
+
for (const socket of this.clients.values()) {
|
|
90
|
+
if (!socket.destroyed) {
|
|
91
|
+
socket.write(encoded);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
getClientCount(): number {
|
|
98
|
+
return this.clients.size;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
stop(): void {
|
|
102
|
+
for (const socket of this.clients.values()) {
|
|
103
|
+
socket.destroy();
|
|
104
|
+
}
|
|
105
|
+
this.clients.clear();
|
|
106
|
+
this.server?.close();
|
|
107
|
+
this.server = null;
|
|
108
|
+
logger.info('IPC server stopped');
|
|
109
|
+
}
|
|
110
|
+
}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Wilson Score Interval for low-sample-size confidence.
|
|
3
|
+
* Prevents unrealistic 100% from single success/failure.
|
|
4
|
+
*/
|
|
5
|
+
export function wilsonScore(successes: number, total: number, z: number = 1.96): number {
|
|
6
|
+
if (total === 0) return 0;
|
|
7
|
+
|
|
8
|
+
const p = successes / total;
|
|
9
|
+
const z2 = z * z;
|
|
10
|
+
const n = total;
|
|
11
|
+
|
|
12
|
+
const numerator = p + z2 / (2 * n);
|
|
13
|
+
const denominator = 1 + z2 / n;
|
|
14
|
+
const margin = z * Math.sqrt((p * (1 - p) + z2 / (4 * n)) / n) / denominator;
|
|
15
|
+
|
|
16
|
+
// Lower bound of Wilson interval = conservative estimate
|
|
17
|
+
return Math.max(0, numerator / denominator - margin);
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Time-decayed confidence: recent successes count more.
|
|
22
|
+
*/
|
|
23
|
+
export function timeDecayedConfidence(
|
|
24
|
+
successes: number,
|
|
25
|
+
total: number,
|
|
26
|
+
lastUsedAt: string,
|
|
27
|
+
halfLifeDays: number,
|
|
28
|
+
): number {
|
|
29
|
+
const base = wilsonScore(successes, total);
|
|
30
|
+
const ageDays = (Date.now() - new Date(lastUsedAt).getTime()) / (1000 * 60 * 60 * 24);
|
|
31
|
+
const decay = Math.pow(0.5, ageDays / halfLifeDays);
|
|
32
|
+
return base * decay;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Combined confidence from success rate + usage frequency + recency.
|
|
37
|
+
*/
|
|
38
|
+
export function computeConfidence(
|
|
39
|
+
successCount: number,
|
|
40
|
+
failCount: number,
|
|
41
|
+
lastUsedAt: string,
|
|
42
|
+
halfLifeDays: number = 30,
|
|
43
|
+
): number {
|
|
44
|
+
const total = successCount + failCount;
|
|
45
|
+
if (total === 0) return 0;
|
|
46
|
+
return timeDecayedConfidence(successCount, total, lastUsedAt, halfLifeDays);
|
|
47
|
+
}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import type { SolutionRepository } from '../db/repositories/solution.repository.js';
|
|
2
|
+
import { wilsonScore } from './confidence-scorer.js';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Update confidence scores for all solutions based on their attempt history.
|
|
6
|
+
*/
|
|
7
|
+
export function updateSolutionConfidences(solutionRepo: SolutionRepository): number {
|
|
8
|
+
const solutions = solutionRepo.getAll();
|
|
9
|
+
let updated = 0;
|
|
10
|
+
|
|
11
|
+
for (const sol of solutions) {
|
|
12
|
+
const total = sol.success_count + sol.fail_count;
|
|
13
|
+
if (total === 0) continue;
|
|
14
|
+
|
|
15
|
+
const newConfidence = wilsonScore(sol.success_count, total);
|
|
16
|
+
if (Math.abs(newConfidence - sol.confidence) > 0.001) {
|
|
17
|
+
solutionRepo.update(sol.id, { confidence: newConfidence });
|
|
18
|
+
updated++;
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
return updated;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* Compute relevance decay factor for a timestamp.
|
|
27
|
+
*/
|
|
28
|
+
export function relevanceDecay(timestamp: string, halfLifeDays: number): number {
|
|
29
|
+
const ageDays = (Date.now() - new Date(timestamp).getTime()) / (1000 * 60 * 60 * 24);
|
|
30
|
+
return Math.pow(0.5, ageDays / halfLifeDays);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* Determine if a rule should be pruned based on its performance.
|
|
35
|
+
*/
|
|
36
|
+
export function shouldPruneRule(
|
|
37
|
+
confidence: number,
|
|
38
|
+
rejectionCount: number,
|
|
39
|
+
totalUsage: number,
|
|
40
|
+
pruneThreshold: number,
|
|
41
|
+
maxRejectionRate: number,
|
|
42
|
+
): boolean {
|
|
43
|
+
if (confidence < pruneThreshold) return true;
|
|
44
|
+
if (totalUsage > 0 && rejectionCount / totalUsage > maxRejectionRate) return true;
|
|
45
|
+
return false;
|
|
46
|
+
}
|
|
@@ -0,0 +1,162 @@
|
|
|
1
|
+
import type { LearningConfig } from '../types/config.types.js';
|
|
2
|
+
import type { ErrorRepository } from '../db/repositories/error.repository.js';
|
|
3
|
+
import type { SolutionRepository } from '../db/repositories/solution.repository.js';
|
|
4
|
+
import type { RuleRepository } from '../db/repositories/rule.repository.js';
|
|
5
|
+
import type { AntipatternRepository } from '../db/repositories/antipattern.repository.js';
|
|
6
|
+
import type { SynapseManager } from '../synapses/synapse-manager.js';
|
|
7
|
+
import { extractPatterns } from './pattern-extractor.js';
|
|
8
|
+
import { generateRules, persistRules } from './rule-generator.js';
|
|
9
|
+
import { shouldPruneRule } from './decay.js';
|
|
10
|
+
import { getLogger } from '../utils/logger.js';
|
|
11
|
+
|
|
12
|
+
export interface LearningCycleResult {
|
|
13
|
+
newPatterns: number;
|
|
14
|
+
updatedRules: number;
|
|
15
|
+
prunedRules: number;
|
|
16
|
+
newAntipatterns: number;
|
|
17
|
+
duration: number;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export class LearningEngine {
|
|
21
|
+
private timer: ReturnType<typeof setInterval> | null = null;
|
|
22
|
+
private logger = getLogger();
|
|
23
|
+
private lastCycleAt: string | null = null;
|
|
24
|
+
|
|
25
|
+
constructor(
|
|
26
|
+
private config: LearningConfig,
|
|
27
|
+
private errorRepo: ErrorRepository,
|
|
28
|
+
private solutionRepo: SolutionRepository,
|
|
29
|
+
private ruleRepo: RuleRepository,
|
|
30
|
+
private antipatternRepo: AntipatternRepository,
|
|
31
|
+
private synapseManager: SynapseManager,
|
|
32
|
+
) {}
|
|
33
|
+
|
|
34
|
+
start(): void {
|
|
35
|
+
this.logger.info(`Learning engine starting (interval: ${this.config.intervalMs}ms)`);
|
|
36
|
+
this.timer = setInterval(() => this.runCycle(), this.config.intervalMs);
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
stop(): void {
|
|
40
|
+
if (this.timer) {
|
|
41
|
+
clearInterval(this.timer);
|
|
42
|
+
this.timer = null;
|
|
43
|
+
}
|
|
44
|
+
this.logger.info('Learning engine stopped');
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
runCycle(): LearningCycleResult {
|
|
48
|
+
const start = Date.now();
|
|
49
|
+
this.logger.info('Learning cycle starting');
|
|
50
|
+
|
|
51
|
+
const result: LearningCycleResult = {
|
|
52
|
+
newPatterns: 0,
|
|
53
|
+
updatedRules: 0,
|
|
54
|
+
prunedRules: 0,
|
|
55
|
+
newAntipatterns: 0,
|
|
56
|
+
duration: 0,
|
|
57
|
+
};
|
|
58
|
+
|
|
59
|
+
// Phase 1: Collect recent errors
|
|
60
|
+
const since = this.lastCycleAt ?? new Date(Date.now() - this.config.intervalMs).toISOString();
|
|
61
|
+
const recentErrors = this.errorRepo.findUnresolved();
|
|
62
|
+
|
|
63
|
+
// Phase 2: Extract patterns
|
|
64
|
+
const patterns = extractPatterns(recentErrors, 0.7);
|
|
65
|
+
result.newPatterns = patterns.length;
|
|
66
|
+
|
|
67
|
+
// Phase 3: Enrich patterns with solution data
|
|
68
|
+
for (const pattern of patterns) {
|
|
69
|
+
let totalSuccess = 0;
|
|
70
|
+
let totalAttempts = 0;
|
|
71
|
+
|
|
72
|
+
for (const errorId of pattern.errorIds) {
|
|
73
|
+
const solutions = this.solutionRepo.findForError(errorId);
|
|
74
|
+
pattern.solutionIds.push(...solutions.map(s => s.id));
|
|
75
|
+
|
|
76
|
+
for (const sol of solutions) {
|
|
77
|
+
const rate = this.solutionRepo.successRate(sol.id);
|
|
78
|
+
totalSuccess += rate;
|
|
79
|
+
totalAttempts++;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
pattern.successRate = totalAttempts > 0 ? totalSuccess / totalAttempts : 0;
|
|
84
|
+
pattern.confidence = Math.min(
|
|
85
|
+
0.95,
|
|
86
|
+
pattern.successRate * 0.6 + Math.min(1, pattern.occurrences / 10) * 0.4,
|
|
87
|
+
);
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
// Phase 4: Generate rules from patterns
|
|
91
|
+
const rules = generateRules(patterns, this.config);
|
|
92
|
+
result.updatedRules = persistRules(rules, this.ruleRepo);
|
|
93
|
+
|
|
94
|
+
// Phase 5: Prune weak rules
|
|
95
|
+
const activeRules = this.ruleRepo.findActive();
|
|
96
|
+
for (const rule of activeRules) {
|
|
97
|
+
if (shouldPruneRule(
|
|
98
|
+
rule.confidence,
|
|
99
|
+
0, // rejection count not tracked yet
|
|
100
|
+
rule.occurrences,
|
|
101
|
+
this.config.pruneThreshold,
|
|
102
|
+
this.config.maxRejectionRate,
|
|
103
|
+
)) {
|
|
104
|
+
this.ruleRepo.update(rule.id, { active: 0 });
|
|
105
|
+
result.prunedRules++;
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// Phase 6: Detect antipatterns
|
|
110
|
+
result.newAntipatterns = this.detectAntipatterns();
|
|
111
|
+
|
|
112
|
+
// Phase 7: Run synapse decay
|
|
113
|
+
this.synapseManager.runDecay();
|
|
114
|
+
|
|
115
|
+
this.lastCycleAt = new Date().toISOString();
|
|
116
|
+
result.duration = Date.now() - start;
|
|
117
|
+
this.logger.info(`Learning cycle complete: ${result.newPatterns} patterns, ${result.updatedRules} rules, ${result.prunedRules} pruned, ${result.newAntipatterns} antipatterns (${result.duration}ms)`);
|
|
118
|
+
|
|
119
|
+
return result;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
private detectAntipatterns(): number {
|
|
123
|
+
const unresolvedErrors = this.errorRepo.findUnresolved();
|
|
124
|
+
const recurring = unresolvedErrors.filter(e => e.occurrence_count >= this.config.minOccurrences);
|
|
125
|
+
let count = 0;
|
|
126
|
+
|
|
127
|
+
for (const error of recurring) {
|
|
128
|
+
const solutions = this.solutionRepo.findForError(error.id);
|
|
129
|
+
const allFailed = solutions.length > 0 && solutions.every(s => {
|
|
130
|
+
const rate = this.solutionRepo.successRate(s.id);
|
|
131
|
+
return rate < 0.3;
|
|
132
|
+
});
|
|
133
|
+
|
|
134
|
+
if (solutions.length >= 2 && allFailed) {
|
|
135
|
+
// Check if antipattern already exists
|
|
136
|
+
const existing = this.antipatternRepo.findByProject(error.project_id);
|
|
137
|
+
const alreadyDetected = existing.some(ap =>
|
|
138
|
+
ap.pattern.includes(error.type) && ap.description.includes(error.message.substring(0, 30))
|
|
139
|
+
);
|
|
140
|
+
|
|
141
|
+
if (!alreadyDetected) {
|
|
142
|
+
this.antipatternRepo.create({
|
|
143
|
+
pattern: `${error.type}.*${error.message.substring(0, 40).replace(/[.*+?^${}()|[\]\\]/g, '\\$&')}`,
|
|
144
|
+
description: `Recurring error without solution: ${error.type}: ${error.message}`,
|
|
145
|
+
severity: solutions.length >= 4 ? 'critical' : 'warning',
|
|
146
|
+
suggestion: null,
|
|
147
|
+
occurrences: error.occurrence_count,
|
|
148
|
+
project_id: error.project_id,
|
|
149
|
+
global: 0,
|
|
150
|
+
});
|
|
151
|
+
count++;
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
return count;
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
getLastCycleAt(): string | null {
|
|
160
|
+
return this.lastCycleAt;
|
|
161
|
+
}
|
|
162
|
+
}
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
import type { ErrorRecord } from '../types/error.types.js';
|
|
2
|
+
import { tokenize } from '../matching/tokenizer.js';
|
|
3
|
+
import { cosineSimilarity } from '../matching/similarity.js';
|
|
4
|
+
|
|
5
|
+
export interface ErrorPattern {
|
|
6
|
+
errorType: string;
|
|
7
|
+
messageTemplate: string;
|
|
8
|
+
messageRegex: string;
|
|
9
|
+
filePattern: string | null;
|
|
10
|
+
occurrences: number;
|
|
11
|
+
errorIds: number[];
|
|
12
|
+
solutionIds: number[];
|
|
13
|
+
confidence: number;
|
|
14
|
+
successRate: number;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
interface Centroid {
|
|
18
|
+
errorType: string;
|
|
19
|
+
tokens: string[];
|
|
20
|
+
errorIds: number[];
|
|
21
|
+
filePattern: string | null;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Extract patterns from error records using centroid-based clustering.
|
|
26
|
+
*/
|
|
27
|
+
export function extractPatterns(
|
|
28
|
+
errors: ErrorRecord[],
|
|
29
|
+
similarityThreshold: number = 0.7,
|
|
30
|
+
): ErrorPattern[] {
|
|
31
|
+
const centroids: Centroid[] = [];
|
|
32
|
+
|
|
33
|
+
for (const error of errors) {
|
|
34
|
+
const tokens = tokenize(`${error.type} ${error.message}`);
|
|
35
|
+
let merged = false;
|
|
36
|
+
|
|
37
|
+
for (const centroid of centroids) {
|
|
38
|
+
if (centroid.errorType !== error.type) continue;
|
|
39
|
+
|
|
40
|
+
const sim = cosineSimilarity(centroid.tokens, tokens);
|
|
41
|
+
if (sim >= similarityThreshold) {
|
|
42
|
+
// Merge into existing centroid (running average)
|
|
43
|
+
const allTokens = [...centroid.tokens, ...tokens];
|
|
44
|
+
centroid.tokens = [...new Set(allTokens)];
|
|
45
|
+
centroid.errorIds.push(error.id);
|
|
46
|
+
if (!centroid.filePattern && error.file_path) {
|
|
47
|
+
centroid.filePattern = extractFilePattern(error.file_path);
|
|
48
|
+
}
|
|
49
|
+
merged = true;
|
|
50
|
+
break;
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
if (!merged) {
|
|
55
|
+
centroids.push({
|
|
56
|
+
errorType: error.type,
|
|
57
|
+
tokens,
|
|
58
|
+
errorIds: [error.id],
|
|
59
|
+
filePattern: error.file_path ? extractFilePattern(error.file_path) : null,
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
return centroids
|
|
65
|
+
.filter(c => c.errorIds.length >= 2)
|
|
66
|
+
.map(c => ({
|
|
67
|
+
errorType: c.errorType,
|
|
68
|
+
messageTemplate: c.tokens.join(' '),
|
|
69
|
+
messageRegex: buildRegex(c.tokens),
|
|
70
|
+
filePattern: c.filePattern,
|
|
71
|
+
occurrences: c.errorIds.length,
|
|
72
|
+
errorIds: c.errorIds,
|
|
73
|
+
solutionIds: [],
|
|
74
|
+
confidence: 0,
|
|
75
|
+
successRate: 0,
|
|
76
|
+
}));
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
function extractFilePattern(filePath: string): string {
|
|
80
|
+
// Extract the meaningful part: last directory + extension
|
|
81
|
+
const parts = filePath.replace(/\\/g, '/').split('/');
|
|
82
|
+
const fileName = parts[parts.length - 1] ?? '';
|
|
83
|
+
const ext = fileName.split('.').pop() ?? '';
|
|
84
|
+
return ext ? `*.${ext}` : '*';
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
function buildRegex(tokens: string[]): string {
|
|
88
|
+
const escaped = tokens.map(t => t.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'));
|
|
89
|
+
return escaped.join('.*');
|
|
90
|
+
}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import type { LearningConfig } from '../types/config.types.js';
|
|
2
|
+
import type { RuleRepository } from '../db/repositories/rule.repository.js';
|
|
3
|
+
import type { ErrorPattern } from './pattern-extractor.js';
|
|
4
|
+
import { getLogger } from '../utils/logger.js';
|
|
5
|
+
|
|
6
|
+
export interface GeneratedRule {
|
|
7
|
+
pattern: string;
|
|
8
|
+
action: string;
|
|
9
|
+
description: string;
|
|
10
|
+
confidence: number;
|
|
11
|
+
sourceErrorIds: number[];
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Generate prevention rules from extracted patterns.
|
|
16
|
+
*/
|
|
17
|
+
export function generateRules(
|
|
18
|
+
patterns: ErrorPattern[],
|
|
19
|
+
config: LearningConfig,
|
|
20
|
+
): GeneratedRule[] {
|
|
21
|
+
return patterns
|
|
22
|
+
.filter(p =>
|
|
23
|
+
p.occurrences >= config.minOccurrences &&
|
|
24
|
+
p.confidence >= config.minConfidence,
|
|
25
|
+
)
|
|
26
|
+
.map(pattern => ({
|
|
27
|
+
pattern: pattern.messageRegex,
|
|
28
|
+
action: pattern.confidence >= 0.90
|
|
29
|
+
? `Auto-fix available for ${pattern.errorType}`
|
|
30
|
+
: `Suggestion: check ${pattern.errorType} pattern (${pattern.occurrences} occurrences)`,
|
|
31
|
+
description: `Auto-generated from ${pattern.occurrences} occurrences of ${pattern.errorType}`,
|
|
32
|
+
confidence: pattern.confidence,
|
|
33
|
+
sourceErrorIds: pattern.errorIds,
|
|
34
|
+
}));
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Persist generated rules to the database.
|
|
39
|
+
*/
|
|
40
|
+
export function persistRules(
|
|
41
|
+
rules: GeneratedRule[],
|
|
42
|
+
ruleRepo: RuleRepository,
|
|
43
|
+
projectId?: number,
|
|
44
|
+
): number {
|
|
45
|
+
const logger = getLogger();
|
|
46
|
+
let created = 0;
|
|
47
|
+
|
|
48
|
+
for (const rule of rules) {
|
|
49
|
+
// Check if similar rule already exists
|
|
50
|
+
const existing = ruleRepo.findByPattern(rule.pattern);
|
|
51
|
+
if (existing.length > 0) {
|
|
52
|
+
// Update confidence of existing rule
|
|
53
|
+
const best = existing[0]!;
|
|
54
|
+
if (rule.confidence > best.confidence) {
|
|
55
|
+
ruleRepo.update(best.id, { confidence: rule.confidence });
|
|
56
|
+
}
|
|
57
|
+
continue;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
ruleRepo.create({
|
|
61
|
+
pattern: rule.pattern,
|
|
62
|
+
action: rule.action,
|
|
63
|
+
description: rule.description,
|
|
64
|
+
confidence: rule.confidence,
|
|
65
|
+
occurrences: 0,
|
|
66
|
+
active: 1,
|
|
67
|
+
project_id: projectId ?? null,
|
|
68
|
+
});
|
|
69
|
+
created++;
|
|
70
|
+
logger.info(`New rule generated: ${rule.pattern.substring(0, 50)}...`);
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
return created;
|
|
74
|
+
}
|
package/src/main.rs:10:5
ADDED
|
File without changes
|