@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.
Files changed (437) hide show
  1. package/BRAIN_PLAN.md +3324 -0
  2. package/LICENSE +21 -0
  3. package/README.md +188 -0
  4. package/dist/brain.d.ts +11 -0
  5. package/dist/brain.js +166 -0
  6. package/dist/brain.js.map +1 -0
  7. package/dist/cli/commands/dashboard.d.ts +2 -0
  8. package/dist/cli/commands/dashboard.js +457 -0
  9. package/dist/cli/commands/dashboard.js.map +1 -0
  10. package/dist/cli/commands/export.d.ts +2 -0
  11. package/dist/cli/commands/export.js +25 -0
  12. package/dist/cli/commands/export.js.map +1 -0
  13. package/dist/cli/commands/import.d.ts +2 -0
  14. package/dist/cli/commands/import.js +173 -0
  15. package/dist/cli/commands/import.js.map +1 -0
  16. package/dist/cli/commands/insights.d.ts +2 -0
  17. package/dist/cli/commands/insights.js +32 -0
  18. package/dist/cli/commands/insights.js.map +1 -0
  19. package/dist/cli/commands/modules.d.ts +2 -0
  20. package/dist/cli/commands/modules.js +29 -0
  21. package/dist/cli/commands/modules.js.map +1 -0
  22. package/dist/cli/commands/network.d.ts +2 -0
  23. package/dist/cli/commands/network.js +56 -0
  24. package/dist/cli/commands/network.js.map +1 -0
  25. package/dist/cli/commands/query.d.ts +2 -0
  26. package/dist/cli/commands/query.js +40 -0
  27. package/dist/cli/commands/query.js.map +1 -0
  28. package/dist/cli/commands/start.d.ts +2 -0
  29. package/dist/cli/commands/start.js +56 -0
  30. package/dist/cli/commands/start.js.map +1 -0
  31. package/dist/cli/commands/status.d.ts +2 -0
  32. package/dist/cli/commands/status.js +60 -0
  33. package/dist/cli/commands/status.js.map +1 -0
  34. package/dist/cli/commands/stop.d.ts +2 -0
  35. package/dist/cli/commands/stop.js +34 -0
  36. package/dist/cli/commands/stop.js.map +1 -0
  37. package/dist/cli/ipc-helper.d.ts +2 -0
  38. package/dist/cli/ipc-helper.js +25 -0
  39. package/dist/cli/ipc-helper.js.map +1 -0
  40. package/dist/code/analyzer.d.ts +12 -0
  41. package/dist/code/analyzer.js +58 -0
  42. package/dist/code/analyzer.js.map +1 -0
  43. package/dist/code/fingerprint.d.ts +2 -0
  44. package/dist/code/fingerprint.js +84 -0
  45. package/dist/code/fingerprint.js.map +1 -0
  46. package/dist/code/matcher.d.ts +9 -0
  47. package/dist/code/matcher.js +37 -0
  48. package/dist/code/matcher.js.map +1 -0
  49. package/dist/code/parsers/generic.d.ts +7 -0
  50. package/dist/code/parsers/generic.js +22 -0
  51. package/dist/code/parsers/generic.js.map +1 -0
  52. package/dist/code/parsers/python.d.ts +7 -0
  53. package/dist/code/parsers/python.js +45 -0
  54. package/dist/code/parsers/python.js.map +1 -0
  55. package/dist/code/parsers/typescript.d.ts +7 -0
  56. package/dist/code/parsers/typescript.js +58 -0
  57. package/dist/code/parsers/typescript.js.map +1 -0
  58. package/dist/code/registry.d.ts +22 -0
  59. package/dist/code/registry.js +31 -0
  60. package/dist/code/registry.js.map +1 -0
  61. package/dist/code/scorer.d.ts +15 -0
  62. package/dist/code/scorer.js +103 -0
  63. package/dist/code/scorer.js.map +1 -0
  64. package/dist/config.d.ts +2 -0
  65. package/dist/config.js +110 -0
  66. package/dist/config.js.map +1 -0
  67. package/dist/db/connection.d.ts +2 -0
  68. package/dist/db/connection.js +19 -0
  69. package/dist/db/connection.js.map +1 -0
  70. package/dist/db/migrations/001_core_schema.d.ts +2 -0
  71. package/dist/db/migrations/001_core_schema.js +119 -0
  72. package/dist/db/migrations/001_core_schema.js.map +1 -0
  73. package/dist/db/migrations/002_learning_schema.d.ts +2 -0
  74. package/dist/db/migrations/002_learning_schema.js +37 -0
  75. package/dist/db/migrations/002_learning_schema.js.map +1 -0
  76. package/dist/db/migrations/003_code_schema.d.ts +2 -0
  77. package/dist/db/migrations/003_code_schema.js +52 -0
  78. package/dist/db/migrations/003_code_schema.js.map +1 -0
  79. package/dist/db/migrations/004_synapses_schema.d.ts +2 -0
  80. package/dist/db/migrations/004_synapses_schema.js +56 -0
  81. package/dist/db/migrations/004_synapses_schema.js.map +1 -0
  82. package/dist/db/migrations/005_fts_indexes.d.ts +2 -0
  83. package/dist/db/migrations/005_fts_indexes.js +77 -0
  84. package/dist/db/migrations/005_fts_indexes.js.map +1 -0
  85. package/dist/db/migrations/006_synapses_phase3.d.ts +2 -0
  86. package/dist/db/migrations/006_synapses_phase3.js +14 -0
  87. package/dist/db/migrations/006_synapses_phase3.js.map +1 -0
  88. package/dist/db/migrations/index.d.ts +2 -0
  89. package/dist/db/migrations/index.js +49 -0
  90. package/dist/db/migrations/index.js.map +1 -0
  91. package/dist/db/repositories/antipattern.repository.d.ts +26 -0
  92. package/dist/db/repositories/antipattern.repository.js +44 -0
  93. package/dist/db/repositories/antipattern.repository.js.map +1 -0
  94. package/dist/db/repositories/code-module.repository.d.ts +19 -0
  95. package/dist/db/repositories/code-module.repository.js +64 -0
  96. package/dist/db/repositories/code-module.repository.js.map +1 -0
  97. package/dist/db/repositories/error.repository.d.ts +20 -0
  98. package/dist/db/repositories/error.repository.js +134 -0
  99. package/dist/db/repositories/error.repository.js.map +1 -0
  100. package/dist/db/repositories/insight.repository.d.ts +18 -0
  101. package/dist/db/repositories/insight.repository.js +57 -0
  102. package/dist/db/repositories/insight.repository.js.map +1 -0
  103. package/dist/db/repositories/notification.repository.d.ts +24 -0
  104. package/dist/db/repositories/notification.repository.js +40 -0
  105. package/dist/db/repositories/notification.repository.js.map +1 -0
  106. package/dist/db/repositories/project.repository.d.ts +25 -0
  107. package/dist/db/repositories/project.repository.js +72 -0
  108. package/dist/db/repositories/project.repository.js.map +1 -0
  109. package/dist/db/repositories/rule.repository.d.ts +31 -0
  110. package/dist/db/repositories/rule.repository.js +81 -0
  111. package/dist/db/repositories/rule.repository.js.map +1 -0
  112. package/dist/db/repositories/solution.repository.d.ts +27 -0
  113. package/dist/db/repositories/solution.repository.js +132 -0
  114. package/dist/db/repositories/solution.repository.js.map +1 -0
  115. package/dist/db/repositories/synapse.repository.d.ts +25 -0
  116. package/dist/db/repositories/synapse.repository.js +115 -0
  117. package/dist/db/repositories/synapse.repository.js.map +1 -0
  118. package/dist/db/repositories/terminal.repository.d.ts +27 -0
  119. package/dist/db/repositories/terminal.repository.js +78 -0
  120. package/dist/db/repositories/terminal.repository.js.map +1 -0
  121. package/dist/hooks/post-tool-use.d.ts +2 -0
  122. package/dist/hooks/post-tool-use.js +77 -0
  123. package/dist/hooks/post-tool-use.js.map +1 -0
  124. package/dist/hooks/post-write.d.ts +2 -0
  125. package/dist/hooks/post-write.js +102 -0
  126. package/dist/hooks/post-write.js.map +1 -0
  127. package/dist/index.d.ts +2 -0
  128. package/dist/index.js +47 -0
  129. package/dist/index.js.map +1 -0
  130. package/dist/ipc/client.d.ts +16 -0
  131. package/dist/ipc/client.js +101 -0
  132. package/dist/ipc/client.js.map +1 -0
  133. package/dist/ipc/protocol.d.ts +8 -0
  134. package/dist/ipc/protocol.js +29 -0
  135. package/dist/ipc/protocol.js.map +1 -0
  136. package/dist/ipc/router.d.ts +28 -0
  137. package/dist/ipc/router.js +70 -0
  138. package/dist/ipc/router.js.map +1 -0
  139. package/dist/ipc/server.d.ts +14 -0
  140. package/dist/ipc/server.js +98 -0
  141. package/dist/ipc/server.js.map +1 -0
  142. package/dist/learning/confidence-scorer.d.ts +13 -0
  143. package/dist/learning/confidence-scorer.js +35 -0
  144. package/dist/learning/confidence-scorer.js.map +1 -0
  145. package/dist/learning/decay.d.ts +13 -0
  146. package/dist/learning/decay.js +37 -0
  147. package/dist/learning/decay.js.map +1 -0
  148. package/dist/learning/learning-engine.d.ts +30 -0
  149. package/dist/learning/learning-engine.js +121 -0
  150. package/dist/learning/learning-engine.js.map +1 -0
  151. package/dist/learning/pattern-extractor.d.ts +16 -0
  152. package/dist/learning/pattern-extractor.js +61 -0
  153. package/dist/learning/pattern-extractor.js.map +1 -0
  154. package/dist/learning/rule-generator.d.ts +18 -0
  155. package/dist/learning/rule-generator.js +50 -0
  156. package/dist/learning/rule-generator.js.map +1 -0
  157. package/dist/matching/error-matcher.d.ts +13 -0
  158. package/dist/matching/error-matcher.js +84 -0
  159. package/dist/matching/error-matcher.js.map +1 -0
  160. package/dist/matching/fingerprint.d.ts +3 -0
  161. package/dist/matching/fingerprint.js +23 -0
  162. package/dist/matching/fingerprint.js.map +1 -0
  163. package/dist/matching/similarity.d.ts +3 -0
  164. package/dist/matching/similarity.js +53 -0
  165. package/dist/matching/similarity.js.map +1 -0
  166. package/dist/matching/tfidf.d.ts +15 -0
  167. package/dist/matching/tfidf.js +68 -0
  168. package/dist/matching/tfidf.js.map +1 -0
  169. package/dist/matching/tokenizer.d.ts +4 -0
  170. package/dist/matching/tokenizer.js +36 -0
  171. package/dist/matching/tokenizer.js.map +1 -0
  172. package/dist/mcp/auto-detect.d.ts +1 -0
  173. package/dist/mcp/auto-detect.js +81 -0
  174. package/dist/mcp/auto-detect.js.map +1 -0
  175. package/dist/mcp/server.d.ts +1 -0
  176. package/dist/mcp/server.js +68 -0
  177. package/dist/mcp/server.js.map +1 -0
  178. package/dist/mcp/tools.d.ts +3 -0
  179. package/dist/mcp/tools.js +201 -0
  180. package/dist/mcp/tools.js.map +1 -0
  181. package/dist/parsing/error-parser.d.ts +3 -0
  182. package/dist/parsing/error-parser.js +26 -0
  183. package/dist/parsing/error-parser.js.map +1 -0
  184. package/dist/parsing/parsers/compiler.d.ts +2 -0
  185. package/dist/parsing/parsers/compiler.js +83 -0
  186. package/dist/parsing/parsers/compiler.js.map +1 -0
  187. package/dist/parsing/parsers/generic.d.ts +2 -0
  188. package/dist/parsing/parsers/generic.js +23 -0
  189. package/dist/parsing/parsers/generic.js.map +1 -0
  190. package/dist/parsing/parsers/go.d.ts +2 -0
  191. package/dist/parsing/parsers/go.js +85 -0
  192. package/dist/parsing/parsers/go.js.map +1 -0
  193. package/dist/parsing/parsers/node.d.ts +2 -0
  194. package/dist/parsing/parsers/node.js +61 -0
  195. package/dist/parsing/parsers/node.js.map +1 -0
  196. package/dist/parsing/parsers/python.d.ts +2 -0
  197. package/dist/parsing/parsers/python.js +50 -0
  198. package/dist/parsing/parsers/python.js.map +1 -0
  199. package/dist/parsing/parsers/rust.d.ts +2 -0
  200. package/dist/parsing/parsers/rust.js +43 -0
  201. package/dist/parsing/parsers/rust.js.map +1 -0
  202. package/dist/parsing/parsers/shell.d.ts +2 -0
  203. package/dist/parsing/parsers/shell.js +36 -0
  204. package/dist/parsing/parsers/shell.js.map +1 -0
  205. package/dist/parsing/types.d.ts +28 -0
  206. package/dist/parsing/types.js +21 -0
  207. package/dist/parsing/types.js.map +1 -0
  208. package/dist/research/gap-analyzer.d.ts +23 -0
  209. package/dist/research/gap-analyzer.js +119 -0
  210. package/dist/research/gap-analyzer.js.map +1 -0
  211. package/dist/research/insight-generator.d.ts +23 -0
  212. package/dist/research/insight-generator.js +107 -0
  213. package/dist/research/insight-generator.js.map +1 -0
  214. package/dist/research/research-engine.d.ts +31 -0
  215. package/dist/research/research-engine.js +97 -0
  216. package/dist/research/research-engine.js.map +1 -0
  217. package/dist/research/synergy-detector.d.ts +24 -0
  218. package/dist/research/synergy-detector.js +109 -0
  219. package/dist/research/synergy-detector.js.map +1 -0
  220. package/dist/research/template-extractor.d.ts +18 -0
  221. package/dist/research/template-extractor.js +116 -0
  222. package/dist/research/template-extractor.js.map +1 -0
  223. package/dist/research/trend-analyzer.d.ts +20 -0
  224. package/dist/research/trend-analyzer.js +111 -0
  225. package/dist/research/trend-analyzer.js.map +1 -0
  226. package/dist/services/analytics.service.d.ts +52 -0
  227. package/dist/services/analytics.service.js +59 -0
  228. package/dist/services/analytics.service.js.map +1 -0
  229. package/dist/services/code.service.d.ts +39 -0
  230. package/dist/services/code.service.js +98 -0
  231. package/dist/services/code.service.js.map +1 -0
  232. package/dist/services/error.service.d.ts +35 -0
  233. package/dist/services/error.service.js +118 -0
  234. package/dist/services/error.service.js.map +1 -0
  235. package/dist/services/notification.service.d.ts +17 -0
  236. package/dist/services/notification.service.js +29 -0
  237. package/dist/services/notification.service.js.map +1 -0
  238. package/dist/services/prevention.service.d.ts +35 -0
  239. package/dist/services/prevention.service.js +82 -0
  240. package/dist/services/prevention.service.js.map +1 -0
  241. package/dist/services/research.service.d.ts +35 -0
  242. package/dist/services/research.service.js +60 -0
  243. package/dist/services/research.service.js.map +1 -0
  244. package/dist/services/solution.service.d.ts +30 -0
  245. package/dist/services/solution.service.js +73 -0
  246. package/dist/services/solution.service.js.map +1 -0
  247. package/dist/services/synapse.service.d.ts +30 -0
  248. package/dist/services/synapse.service.js +25 -0
  249. package/dist/services/synapse.service.js.map +1 -0
  250. package/dist/services/terminal.service.d.ts +20 -0
  251. package/dist/services/terminal.service.js +66 -0
  252. package/dist/services/terminal.service.js.map +1 -0
  253. package/dist/synapses/activation.d.ts +13 -0
  254. package/dist/synapses/activation.js +50 -0
  255. package/dist/synapses/activation.js.map +1 -0
  256. package/dist/synapses/decay.d.ts +11 -0
  257. package/dist/synapses/decay.js +27 -0
  258. package/dist/synapses/decay.js.map +1 -0
  259. package/dist/synapses/hebbian.d.ts +13 -0
  260. package/dist/synapses/hebbian.js +36 -0
  261. package/dist/synapses/hebbian.js.map +1 -0
  262. package/dist/synapses/pathfinder.d.ts +14 -0
  263. package/dist/synapses/pathfinder.js +50 -0
  264. package/dist/synapses/pathfinder.js.map +1 -0
  265. package/dist/synapses/synapse-manager.d.ts +30 -0
  266. package/dist/synapses/synapse-manager.js +72 -0
  267. package/dist/synapses/synapse-manager.js.map +1 -0
  268. package/dist/types/code.types.d.ts +47 -0
  269. package/dist/types/code.types.js +2 -0
  270. package/dist/types/code.types.js.map +1 -0
  271. package/dist/types/config.types.d.ts +70 -0
  272. package/dist/types/config.types.js +2 -0
  273. package/dist/types/config.types.js.map +1 -0
  274. package/dist/types/error.types.d.ts +60 -0
  275. package/dist/types/error.types.js +2 -0
  276. package/dist/types/error.types.js.map +1 -0
  277. package/dist/types/ipc.types.d.ts +11 -0
  278. package/dist/types/ipc.types.js +2 -0
  279. package/dist/types/ipc.types.js.map +1 -0
  280. package/dist/types/mcp.types.d.ts +46 -0
  281. package/dist/types/mcp.types.js +2 -0
  282. package/dist/types/mcp.types.js.map +1 -0
  283. package/dist/types/research.types.d.ts +25 -0
  284. package/dist/types/research.types.js +2 -0
  285. package/dist/types/research.types.js.map +1 -0
  286. package/dist/types/solution.types.d.ts +28 -0
  287. package/dist/types/solution.types.js +2 -0
  288. package/dist/types/solution.types.js.map +1 -0
  289. package/dist/types/synapse.types.d.ts +37 -0
  290. package/dist/types/synapse.types.js +2 -0
  291. package/dist/types/synapse.types.js.map +1 -0
  292. package/dist/utils/events.d.ts +59 -0
  293. package/dist/utils/events.js +23 -0
  294. package/dist/utils/events.js.map +1 -0
  295. package/dist/utils/hash.d.ts +1 -0
  296. package/dist/utils/hash.js +5 -0
  297. package/dist/utils/hash.js.map +1 -0
  298. package/dist/utils/logger.d.ts +8 -0
  299. package/dist/utils/logger.js +39 -0
  300. package/dist/utils/logger.js.map +1 -0
  301. package/dist/utils/paths.d.ts +3 -0
  302. package/dist/utils/paths.js +18 -0
  303. package/dist/utils/paths.js.map +1 -0
  304. package/package.json +43 -0
  305. package/src/brain.ts +220 -0
  306. package/src/cli/commands/dashboard.ts +495 -0
  307. package/src/cli/commands/export.ts +27 -0
  308. package/src/cli/commands/import.ts +190 -0
  309. package/src/cli/commands/insights.ts +33 -0
  310. package/src/cli/commands/modules.ts +30 -0
  311. package/src/cli/commands/network.ts +61 -0
  312. package/src/cli/commands/query.ts +43 -0
  313. package/src/cli/commands/start.ts +59 -0
  314. package/src/cli/commands/status.ts +69 -0
  315. package/src/cli/commands/stop.ts +33 -0
  316. package/src/cli/ipc-helper.ts +21 -0
  317. package/src/code/analyzer.ts +77 -0
  318. package/src/code/fingerprint.ts +87 -0
  319. package/src/code/matcher.ts +64 -0
  320. package/src/code/parsers/generic.ts +29 -0
  321. package/src/code/parsers/python.ts +54 -0
  322. package/src/code/parsers/typescript.ts +65 -0
  323. package/src/code/registry.ts +60 -0
  324. package/src/code/scorer.ts +108 -0
  325. package/src/config.ts +111 -0
  326. package/src/db/connection.ts +22 -0
  327. package/src/db/migrations/001_core_schema.ts +120 -0
  328. package/src/db/migrations/002_learning_schema.ts +38 -0
  329. package/src/db/migrations/003_code_schema.ts +53 -0
  330. package/src/db/migrations/004_synapses_schema.ts +57 -0
  331. package/src/db/migrations/005_fts_indexes.ts +78 -0
  332. package/src/db/migrations/006_synapses_phase3.ts +17 -0
  333. package/src/db/migrations/index.ts +64 -0
  334. package/src/db/repositories/antipattern.repository.ts +66 -0
  335. package/src/db/repositories/code-module.repository.ts +80 -0
  336. package/src/db/repositories/error.repository.ts +149 -0
  337. package/src/db/repositories/insight.repository.ts +78 -0
  338. package/src/db/repositories/notification.repository.ts +66 -0
  339. package/src/db/repositories/project.repository.ts +93 -0
  340. package/src/db/repositories/rule.repository.ts +108 -0
  341. package/src/db/repositories/solution.repository.ts +154 -0
  342. package/src/db/repositories/synapse.repository.ts +153 -0
  343. package/src/db/repositories/terminal.repository.ts +101 -0
  344. package/src/hooks/post-tool-use.ts +90 -0
  345. package/src/hooks/post-write.ts +117 -0
  346. package/src/index.ts +53 -0
  347. package/src/ipc/client.ts +118 -0
  348. package/src/ipc/protocol.ts +35 -0
  349. package/src/ipc/router.ts +106 -0
  350. package/src/ipc/server.ts +110 -0
  351. package/src/learning/confidence-scorer.ts +47 -0
  352. package/src/learning/decay.ts +46 -0
  353. package/src/learning/learning-engine.ts +162 -0
  354. package/src/learning/pattern-extractor.ts +90 -0
  355. package/src/learning/rule-generator.ts +74 -0
  356. package/src/main.rs:10:5 +0 -0
  357. package/src/matching/error-matcher.ts +115 -0
  358. package/src/matching/fingerprint.ts +29 -0
  359. package/src/matching/similarity.ts +61 -0
  360. package/src/matching/tfidf.ts +74 -0
  361. package/src/matching/tokenizer.ts +41 -0
  362. package/src/mcp/auto-detect.ts +93 -0
  363. package/src/mcp/server.ts +73 -0
  364. package/src/mcp/tools.ts +290 -0
  365. package/src/parsing/error-parser.ts +28 -0
  366. package/src/parsing/parsers/compiler.ts +93 -0
  367. package/src/parsing/parsers/generic.ts +28 -0
  368. package/src/parsing/parsers/go.ts +97 -0
  369. package/src/parsing/parsers/node.ts +69 -0
  370. package/src/parsing/parsers/python.ts +62 -0
  371. package/src/parsing/parsers/rust.ts +50 -0
  372. package/src/parsing/parsers/shell.ts +42 -0
  373. package/src/parsing/types.ts +47 -0
  374. package/src/research/gap-analyzer.ts +135 -0
  375. package/src/research/insight-generator.ts +123 -0
  376. package/src/research/research-engine.ts +116 -0
  377. package/src/research/synergy-detector.ts +126 -0
  378. package/src/research/template-extractor.ts +130 -0
  379. package/src/research/trend-analyzer.ts +127 -0
  380. package/src/services/analytics.service.ts +87 -0
  381. package/src/services/code.service.ts +140 -0
  382. package/src/services/error.service.ts +164 -0
  383. package/src/services/notification.service.ts +41 -0
  384. package/src/services/prevention.service.ts +119 -0
  385. package/src/services/research.service.ts +93 -0
  386. package/src/services/solution.service.ts +116 -0
  387. package/src/services/synapse.service.ts +59 -0
  388. package/src/services/terminal.service.ts +81 -0
  389. package/src/synapses/activation.ts +80 -0
  390. package/src/synapses/decay.ts +38 -0
  391. package/src/synapses/hebbian.ts +69 -0
  392. package/src/synapses/pathfinder.ts +81 -0
  393. package/src/synapses/synapse-manager.ts +109 -0
  394. package/src/types/code.types.ts +52 -0
  395. package/src/types/config.types.ts +79 -0
  396. package/src/types/error.types.ts +67 -0
  397. package/src/types/ipc.types.ts +8 -0
  398. package/src/types/mcp.types.ts +53 -0
  399. package/src/types/research.types.ts +28 -0
  400. package/src/types/solution.types.ts +30 -0
  401. package/src/types/synapse.types.ts +49 -0
  402. package/src/utils/events.ts +45 -0
  403. package/src/utils/hash.ts +5 -0
  404. package/src/utils/logger.ts +48 -0
  405. package/src/utils/paths.ts +19 -0
  406. package/tests/fixtures/code-modules/modules.ts +83 -0
  407. package/tests/fixtures/errors/go.ts +9 -0
  408. package/tests/fixtures/errors/node.ts +24 -0
  409. package/tests/fixtures/errors/python.ts +21 -0
  410. package/tests/fixtures/errors/rust.ts +25 -0
  411. package/tests/fixtures/errors/shell.ts +15 -0
  412. package/tests/fixtures/solutions/solutions.ts +27 -0
  413. package/tests/helpers/setup-db.ts +52 -0
  414. package/tests/integration/code-flow.test.ts +86 -0
  415. package/tests/integration/error-flow.test.ts +83 -0
  416. package/tests/integration/ipc-flow.test.ts +166 -0
  417. package/tests/integration/learning-cycle.test.ts +82 -0
  418. package/tests/integration/synapse-flow.test.ts +117 -0
  419. package/tests/unit/code/analyzer.test.ts +58 -0
  420. package/tests/unit/code/fingerprint.test.ts +51 -0
  421. package/tests/unit/code/scorer.test.ts +55 -0
  422. package/tests/unit/learning/confidence-scorer.test.ts +60 -0
  423. package/tests/unit/learning/decay.test.ts +45 -0
  424. package/tests/unit/learning/pattern-extractor.test.ts +50 -0
  425. package/tests/unit/matching/error-matcher.test.ts +69 -0
  426. package/tests/unit/matching/fingerprint.test.ts +47 -0
  427. package/tests/unit/matching/similarity.test.ts +65 -0
  428. package/tests/unit/matching/tfidf.test.ts +71 -0
  429. package/tests/unit/matching/tokenizer.test.ts +83 -0
  430. package/tests/unit/parsing/parsers.test.ts +113 -0
  431. package/tests/unit/research/gap-analyzer.test.ts +45 -0
  432. package/tests/unit/research/trend-analyzer.test.ts +45 -0
  433. package/tests/unit/synapses/activation.test.ts +80 -0
  434. package/tests/unit/synapses/decay.test.ts +27 -0
  435. package/tests/unit/synapses/hebbian.test.ts +96 -0
  436. package/tests/unit/synapses/pathfinder.test.ts +72 -0
  437. package/tsconfig.json +18 -0
@@ -0,0 +1,290 @@
1
+ import { z } from 'zod';
2
+ import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
3
+ import type { IpcClient } from '../ipc/client.js';
4
+
5
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
6
+ type AnyResult = any;
7
+
8
+ function textResult(data: unknown): { content: Array<{ type: 'text'; text: string }> } {
9
+ const text = typeof data === 'string' ? data : JSON.stringify(data, null, 2);
10
+ return { content: [{ type: 'text' as const, text }] };
11
+ }
12
+
13
+ export function registerTools(server: McpServer, ipc: IpcClient): void {
14
+
15
+ // === Error Brain Tools ===
16
+
17
+ server.tool(
18
+ 'brain_report_error',
19
+ 'Report an error that occurred. Brain stores it, matches against known errors, returns solutions if available.',
20
+ {
21
+ error_output: z.string().describe('The raw error output from the terminal'),
22
+ command: z.string().optional().describe('The command that caused the error'),
23
+ task_context: z.string().optional().describe('What was the user trying to accomplish'),
24
+ working_directory: z.string().optional().describe('Working directory when error occurred'),
25
+ project: z.string().optional().describe('Project name'),
26
+ },
27
+ async (params) => {
28
+ const result: AnyResult = await ipc.request('error.report', {
29
+ project: params.project ?? 'default',
30
+ errorOutput: params.error_output,
31
+ filePath: params.working_directory,
32
+ });
33
+ let response = `Error #${result.errorId} recorded (${result.isNew ? 'new' : 'seen before'}).`;
34
+ if (result.matches?.length > 0) {
35
+ const best = result.matches[0];
36
+ response += `\nSimilar error found (#${best.errorId}, ${Math.round(best.score * 100)}% match).`;
37
+ }
38
+ return textResult(response);
39
+ },
40
+ );
41
+
42
+ server.tool(
43
+ 'brain_query_error',
44
+ 'Search for similar errors and their solutions in the Brain database.',
45
+ {
46
+ query: z.string().describe('Error message or description to search for'),
47
+ project_only: z.boolean().optional().describe('Only search in current project'),
48
+ },
49
+ async (params) => {
50
+ const results: AnyResult = await ipc.request('error.query', {
51
+ search: params.query,
52
+ });
53
+ if (!results?.length) return textResult('No matching errors found.');
54
+ const lines = results.map((e: AnyResult) =>
55
+ `#${e.id} [${e.errorType}] ${e.message?.slice(0, 120)}${e.resolved ? ' (resolved)' : ''}`
56
+ );
57
+ return textResult(`Found ${results.length} errors:\n${lines.join('\n')}`);
58
+ },
59
+ );
60
+
61
+ server.tool(
62
+ 'brain_report_solution',
63
+ 'Report a successful solution for an error. Brain will learn from this.',
64
+ {
65
+ error_id: z.number().describe('The error ID this solution fixes'),
66
+ description: z.string().describe('What was done to fix the error'),
67
+ commands: z.string().optional().describe('Commands used to fix'),
68
+ code_change: z.string().optional().describe('Code changes or diff'),
69
+ },
70
+ async (params) => {
71
+ const solutionId: AnyResult = await ipc.request('solution.report', {
72
+ errorId: params.error_id,
73
+ description: params.description,
74
+ commands: params.commands,
75
+ codeChange: params.code_change,
76
+ });
77
+ return textResult(`Solution #${solutionId} recorded for error #${params.error_id}. Brain will use this to help with similar errors in the future.`);
78
+ },
79
+ );
80
+
81
+ server.tool(
82
+ 'brain_report_attempt',
83
+ 'Report a failed solution attempt. Brain learns what does NOT work.',
84
+ {
85
+ error_id: z.number().describe('The error ID'),
86
+ solution_id: z.number().describe('The solution ID that was attempted'),
87
+ description: z.string().optional().describe('What was tried'),
88
+ output: z.string().optional().describe('Output of the failed attempt'),
89
+ },
90
+ async (params) => {
91
+ await ipc.request('solution.rate', {
92
+ errorId: params.error_id,
93
+ solutionId: params.solution_id,
94
+ success: false,
95
+ output: params.output,
96
+ });
97
+ return textResult(`Failed attempt recorded for error #${params.error_id}. Brain will avoid suggesting this approach for similar errors.`);
98
+ },
99
+ );
100
+
101
+ // === Code Brain Tools ===
102
+
103
+ server.tool(
104
+ 'brain_find_reusable_code',
105
+ 'Search for reusable code modules from other projects. Use when starting new functionality.',
106
+ {
107
+ purpose: z.string().describe('What the code should do (e.g., "retry with backoff", "JWT authentication")'),
108
+ language: z.string().optional().describe('Programming language'),
109
+ },
110
+ async (params) => {
111
+ const results: AnyResult = await ipc.request('code.find', {
112
+ query: params.purpose,
113
+ language: params.language,
114
+ });
115
+ if (!results?.length) return textResult('No reusable code modules found.');
116
+ const lines = results.map((m: AnyResult) =>
117
+ `#${m.id} [${m.language}] ${m.name} — ${m.description ?? 'no description'} (reusability: ${m.reusabilityScore ?? '?'})`
118
+ );
119
+ return textResult(`Found ${results.length} modules:\n${lines.join('\n')}`);
120
+ },
121
+ );
122
+
123
+ server.tool(
124
+ 'brain_register_code',
125
+ 'Register a code module as reusable. Brain will analyze it and make it available to other projects.',
126
+ {
127
+ source_code: z.string().describe('The source code'),
128
+ file_path: z.string().describe('File path relative to project root'),
129
+ project: z.string().optional().describe('Project name'),
130
+ name: z.string().optional().describe('Module name (optional - Brain auto-detects)'),
131
+ language: z.string().optional().describe('Programming language'),
132
+ description: z.string().optional().describe('What this code does'),
133
+ },
134
+ async (params) => {
135
+ const result: AnyResult = await ipc.request('code.analyze', {
136
+ project: params.project ?? 'default',
137
+ name: params.name ?? params.file_path.split('/').pop() ?? 'unknown',
138
+ filePath: params.file_path,
139
+ language: params.language ?? detectLanguage(params.file_path),
140
+ source: params.source_code,
141
+ description: params.description,
142
+ });
143
+ return textResult(`Module #${result.moduleId} registered (${result.isNew ? 'new' : 'updated'}). Reusability score: ${result.reusabilityScore}.`);
144
+ },
145
+ );
146
+
147
+ server.tool(
148
+ 'brain_check_code_similarity',
149
+ 'Check if similar code already exists in other projects before writing new code.',
150
+ {
151
+ source_code: z.string().describe('The code to check'),
152
+ language: z.string().optional().describe('Programming language'),
153
+ file_path: z.string().optional().describe('File path for context'),
154
+ },
155
+ async (params) => {
156
+ const results: AnyResult = await ipc.request('code.similarity', {
157
+ source: params.source_code,
158
+ language: params.language ?? detectLanguage(params.file_path ?? ''),
159
+ });
160
+ if (!results?.length) return textResult('No similar code found. This appears to be unique.');
161
+ const lines = results.map((m: AnyResult) =>
162
+ `Module #${m.moduleId}: ${Math.round(m.score * 100)}% match (${m.matchType})`
163
+ );
164
+ return textResult(`Found ${results.length} similar modules:\n${lines.join('\n')}`);
165
+ },
166
+ );
167
+
168
+ // === Synapse Network Tools ===
169
+
170
+ server.tool(
171
+ 'brain_explore',
172
+ 'Explore what Brain knows about a topic. Uses spreading activation through the synapse network.',
173
+ {
174
+ node_type: z.string().describe('Type: error, solution, code_module, project'),
175
+ node_id: z.number().describe('ID of the node to explore from'),
176
+ max_depth: z.number().optional().describe('How many hops to follow (default: 3)'),
177
+ },
178
+ async (params) => {
179
+ const context: AnyResult = await ipc.request('synapse.context', {
180
+ errorId: params.node_id,
181
+ });
182
+ const sections: string[] = [];
183
+ if (context.solutions?.length) sections.push(`Solutions: ${context.solutions.length} found`);
184
+ if (context.relatedErrors?.length) sections.push(`Related errors: ${context.relatedErrors.length}`);
185
+ if (context.relevantModules?.length) sections.push(`Relevant modules: ${context.relevantModules.length}`);
186
+ if (context.preventionRules?.length) sections.push(`Prevention rules: ${context.preventionRules.length}`);
187
+ if (context.insights?.length) sections.push(`Insights: ${context.insights.length}`);
188
+ return textResult(sections.length ? sections.join('\n') : 'No connections found for this node.');
189
+ },
190
+ );
191
+
192
+ server.tool(
193
+ 'brain_connections',
194
+ 'Find how two things are connected in Brain (e.g., how an error relates to a code module).',
195
+ {
196
+ from_type: z.string().describe('Source type: error, solution, code_module, project'),
197
+ from_id: z.number().describe('Source ID'),
198
+ to_type: z.string().describe('Target type'),
199
+ to_id: z.number().describe('Target ID'),
200
+ },
201
+ async (params) => {
202
+ const path: AnyResult = await ipc.request('synapse.path', params);
203
+ if (!path) return textResult('No connection found between these nodes.');
204
+ return textResult(path);
205
+ },
206
+ );
207
+
208
+ // === Research Brain Tools ===
209
+
210
+ server.tool(
211
+ 'brain_insights',
212
+ 'Get research insights: trends, gaps, synergies, template candidates, and project suggestions.',
213
+ {
214
+ type: z.string().optional().describe('Filter by type: trend, pattern, gap, synergy, optimization, template_candidate, project_suggestion, warning'),
215
+ priority: z.string().optional().describe('Minimum priority: low, medium, high, critical'),
216
+ },
217
+ async (params) => {
218
+ const insights: AnyResult = await ipc.request('research.insights', {
219
+ type: params.type,
220
+ activeOnly: true,
221
+ limit: 20,
222
+ });
223
+ if (!insights?.length) return textResult('No active insights.');
224
+ const lines = insights.map((i: AnyResult) =>
225
+ `[${i.type}] ${i.title}: ${i.description?.slice(0, 150)}`
226
+ );
227
+ return textResult(`${insights.length} insights:\n${lines.join('\n')}`);
228
+ },
229
+ );
230
+
231
+ server.tool(
232
+ 'brain_suggest',
233
+ 'Ask Brain for suggestions: what to build next, what to improve, what patterns to extract.',
234
+ {
235
+ context: z.string().describe('Current context or question'),
236
+ },
237
+ async (params) => {
238
+ const suggestions: AnyResult = await ipc.request('research.suggest', {
239
+ context: params.context,
240
+ });
241
+ return textResult(suggestions);
242
+ },
243
+ );
244
+
245
+ // === Status & Notifications ===
246
+
247
+ server.tool(
248
+ 'brain_status',
249
+ 'Get current Brain status: errors, solutions, code modules, synapse network, insights.',
250
+ {},
251
+ async () => {
252
+ const summary: AnyResult = await ipc.request('analytics.summary', {});
253
+ const network: AnyResult = await ipc.request('synapse.stats', {});
254
+ const lines = [
255
+ `Errors: ${summary.errors?.total ?? 0} total, ${summary.errors?.unresolved ?? 0} unresolved`,
256
+ `Solutions: ${summary.solutions?.total ?? 0}`,
257
+ `Rules: ${summary.rules?.active ?? 0} active`,
258
+ `Code modules: ${summary.modules?.total ?? 0}`,
259
+ `Insights: ${summary.insights?.active ?? 0} active`,
260
+ `Synapses: ${network.totalSynapses ?? 0} connections`,
261
+ ];
262
+ return textResult(lines.join('\n'));
263
+ },
264
+ );
265
+
266
+ server.tool(
267
+ 'brain_notifications',
268
+ 'Get pending notifications (new solutions, recurring errors, research insights).',
269
+ {},
270
+ async () => {
271
+ const notifications: AnyResult = await ipc.request('notification.list', {});
272
+ if (!notifications?.length) return textResult('No pending notifications.');
273
+ const lines = notifications.map((n: AnyResult) =>
274
+ `[${n.type}] ${n.title}: ${n.message?.slice(0, 120)}`
275
+ );
276
+ return textResult(`${notifications.length} notifications:\n${lines.join('\n')}`);
277
+ },
278
+ );
279
+ }
280
+
281
+ function detectLanguage(filePath: string): string {
282
+ const ext = filePath.split('.').pop()?.toLowerCase() ?? '';
283
+ const map: Record<string, string> = {
284
+ ts: 'typescript', tsx: 'typescript', js: 'javascript', jsx: 'javascript',
285
+ py: 'python', rs: 'rust', go: 'go', java: 'java',
286
+ c: 'c', cpp: 'cpp', h: 'c', hpp: 'cpp',
287
+ rb: 'ruby', sh: 'shell', bash: 'shell',
288
+ };
289
+ return map[ext] ?? ext;
290
+ }
@@ -0,0 +1,28 @@
1
+ import { ErrorParserRegistry } from './types.js';
2
+ import { nodeParser } from './parsers/node.js';
3
+ import { pythonParser } from './parsers/python.js';
4
+ import { rustParser } from './parsers/rust.js';
5
+ import { goParser } from './parsers/go.js';
6
+ import { shellParser } from './parsers/shell.js';
7
+ import { compilerParser } from './parsers/compiler.js';
8
+ import { genericParser } from './parsers/generic.js';
9
+
10
+ let registryInstance: ErrorParserRegistry | null = null;
11
+
12
+ export function getParserRegistry(): ErrorParserRegistry {
13
+ if (!registryInstance) {
14
+ registryInstance = new ErrorParserRegistry();
15
+ registryInstance.register(nodeParser);
16
+ registryInstance.register(pythonParser);
17
+ registryInstance.register(rustParser);
18
+ registryInstance.register(goParser);
19
+ registryInstance.register(shellParser);
20
+ registryInstance.register(compilerParser);
21
+ registryInstance.register(genericParser);
22
+ }
23
+ return registryInstance;
24
+ }
25
+
26
+ export function parseError(input: string) {
27
+ return getParserRegistry().parse(input);
28
+ }
@@ -0,0 +1,93 @@
1
+ import type { ErrorParser, ParsedError, StackFrame } from '../types.js';
2
+ import path from 'node:path';
3
+
4
+ const GCC_RE = /^(.+?):(\d+):(\d+): (error|warning|fatal error): (.+)/m;
5
+ const JAVAC_RE = /^(.+\.java):(\d+): error: (.+)/m;
6
+ const GENERIC_COMPILER_RE = /^(.+?):(\d+)(?::(\d+))?: (?:error|fatal): (.+)/m;
7
+
8
+ export const compilerParser: ErrorParser = {
9
+ name: 'compiler',
10
+ priority: 7,
11
+
12
+ canParse(input: string): boolean {
13
+ return (
14
+ GCC_RE.test(input) ||
15
+ JAVAC_RE.test(input) ||
16
+ /compilation failed|fatal error:/.test(input)
17
+ );
18
+ },
19
+
20
+ parse(input: string): ParsedError | null {
21
+ let gccMatch = GCC_RE.exec(input);
22
+ if (gccMatch) {
23
+ const filePath = gccMatch[1]!;
24
+ const frame: StackFrame = {
25
+ function_name: null,
26
+ file_path: filePath,
27
+ line_number: parseInt(gccMatch[2]!, 10),
28
+ column_number: parseInt(gccMatch[3]!, 10),
29
+ normalized: `<compiler>@${path.basename(filePath)}`,
30
+ };
31
+ return {
32
+ errorType: gccMatch[4] === 'warning' ? 'CompilerWarning' : 'CompilerError',
33
+ message: gccMatch[5]!,
34
+ stackTrace: input,
35
+ frames: [frame],
36
+ sourceFile: filePath,
37
+ sourceLine: frame.line_number,
38
+ language: detectLanguage(filePath),
39
+ };
40
+ }
41
+
42
+ const javacMatch = JAVAC_RE.exec(input);
43
+ if (javacMatch) {
44
+ const filePath = javacMatch[1]!;
45
+ return {
46
+ errorType: 'CompilerError',
47
+ message: javacMatch[3]!,
48
+ stackTrace: input,
49
+ frames: [{
50
+ function_name: null,
51
+ file_path: filePath,
52
+ line_number: parseInt(javacMatch[2]!, 10),
53
+ column_number: null,
54
+ normalized: `<compiler>@${path.basename(filePath)}`,
55
+ }],
56
+ sourceFile: filePath,
57
+ sourceLine: parseInt(javacMatch[2]!, 10),
58
+ language: 'java',
59
+ };
60
+ }
61
+
62
+ const genericMatch = GENERIC_COMPILER_RE.exec(input);
63
+ if (genericMatch) {
64
+ const filePath = genericMatch[1]!;
65
+ return {
66
+ errorType: 'CompilerError',
67
+ message: genericMatch[4]!,
68
+ stackTrace: input,
69
+ frames: [{
70
+ function_name: null,
71
+ file_path: filePath,
72
+ line_number: parseInt(genericMatch[2]!, 10),
73
+ column_number: genericMatch[3] ? parseInt(genericMatch[3], 10) : null,
74
+ normalized: `<compiler>@${path.basename(filePath)}`,
75
+ }],
76
+ sourceFile: filePath,
77
+ sourceLine: parseInt(genericMatch[2]!, 10),
78
+ language: detectLanguage(filePath),
79
+ };
80
+ }
81
+
82
+ return null;
83
+ },
84
+ };
85
+
86
+ function detectLanguage(filePath: string): string {
87
+ const ext = path.extname(filePath).toLowerCase();
88
+ const map: Record<string, string> = {
89
+ '.c': 'c', '.h': 'c', '.cpp': 'cpp', '.cc': 'cpp', '.cxx': 'cpp',
90
+ '.java': 'java', '.rs': 'rust', '.go': 'go', '.swift': 'swift',
91
+ };
92
+ return map[ext] ?? 'unknown';
93
+ }
@@ -0,0 +1,28 @@
1
+ import type { ErrorParser, ParsedError } from '../types.js';
2
+
3
+ const ERROR_LINE_RE = /(?:error|Error|ERROR)[\s:]+(.+)/;
4
+
5
+ export const genericParser: ErrorParser = {
6
+ name: 'generic',
7
+ priority: 0,
8
+
9
+ canParse(_input: string): boolean {
10
+ return true;
11
+ },
12
+
13
+ parse(input: string): ParsedError | null {
14
+ const match = ERROR_LINE_RE.exec(input);
15
+ const firstLine = input.trim().split('\n')[0] ?? input;
16
+ const message = match ? match[1]! : firstLine;
17
+
18
+ return {
19
+ errorType: 'UnknownError',
20
+ message: message.trim(),
21
+ stackTrace: null,
22
+ frames: [],
23
+ sourceFile: null,
24
+ sourceLine: null,
25
+ language: null,
26
+ };
27
+ },
28
+ };
@@ -0,0 +1,97 @@
1
+ import type { ErrorParser, ParsedError, StackFrame } from '../types.js';
2
+ import path from 'node:path';
3
+
4
+ const GO_FILE_ERROR_RE = /^\.?\/?(.+\.go):(\d+):(\d+): (.+)/m;
5
+ const GO_PANIC_RE = /^panic: (.+)/m;
6
+ const GO_GOROUTINE_RE = /^goroutine \d+ \[.+\]:/m;
7
+ const GO_STACK_RE = /^\t(.+\.go):(\d+)/gm;
8
+ const GO_FUNC_RE = /^(.+)\(.*\)$/gm;
9
+
10
+ export const goParser: ErrorParser = {
11
+ name: 'go',
12
+ priority: 10,
13
+
14
+ canParse(input: string): boolean {
15
+ return (
16
+ GO_FILE_ERROR_RE.test(input) ||
17
+ GO_PANIC_RE.test(input) ||
18
+ /^fatal error:/.test(input)
19
+ );
20
+ },
21
+
22
+ parse(input: string): ParsedError | null {
23
+ const panicMatch = GO_PANIC_RE.exec(input);
24
+ if (panicMatch) {
25
+ return parsePanic(input, panicMatch[1]!);
26
+ }
27
+
28
+ const fileMatch = GO_FILE_ERROR_RE.exec(input);
29
+ if (fileMatch) {
30
+ return {
31
+ errorType: 'CompilerError',
32
+ message: fileMatch[4]!,
33
+ stackTrace: input,
34
+ frames: [{
35
+ function_name: null,
36
+ file_path: fileMatch[1]!,
37
+ line_number: parseInt(fileMatch[2]!, 10),
38
+ column_number: parseInt(fileMatch[3]!, 10),
39
+ normalized: `<compiler>@${path.basename(fileMatch[1]!)}`,
40
+ }],
41
+ sourceFile: fileMatch[1]!,
42
+ sourceLine: parseInt(fileMatch[2]!, 10),
43
+ language: 'go',
44
+ };
45
+ }
46
+
47
+ const fatalMatch = /^fatal error: (.+)/m.exec(input);
48
+ if (fatalMatch) {
49
+ return {
50
+ errorType: 'FatalError',
51
+ message: fatalMatch[1]!,
52
+ stackTrace: input,
53
+ frames: [],
54
+ sourceFile: null,
55
+ sourceLine: null,
56
+ language: 'go',
57
+ };
58
+ }
59
+
60
+ return null;
61
+ },
62
+ };
63
+
64
+ function parsePanic(input: string, message: string): ParsedError {
65
+ const frames: StackFrame[] = [];
66
+ const lines = input.split('\n');
67
+ let i = 0;
68
+
69
+ while (i < lines.length) {
70
+ if (lines[i]!.startsWith('\t')) {
71
+ const stackMatch = /^\t(.+\.go):(\d+)/.exec(lines[i]!);
72
+ if (stackMatch) {
73
+ const funcLine = i > 0 ? lines[i - 1]!.trim() : null;
74
+ const funcName = funcLine?.replace(/\(.*\)$/, '') ?? null;
75
+ frames.push({
76
+ function_name: funcName,
77
+ file_path: stackMatch[1]!,
78
+ line_number: parseInt(stackMatch[2]!, 10),
79
+ column_number: null,
80
+ normalized: `${funcName || '<anon>'}@${path.basename(stackMatch[1]!)}`,
81
+ });
82
+ }
83
+ }
84
+ i++;
85
+ }
86
+
87
+ const topFrame = frames[0];
88
+ return {
89
+ errorType: 'PanicError',
90
+ message,
91
+ stackTrace: input,
92
+ frames,
93
+ sourceFile: topFrame?.file_path ?? null,
94
+ sourceLine: topFrame?.line_number ?? null,
95
+ language: 'go',
96
+ };
97
+ }
@@ -0,0 +1,69 @@
1
+ import type { ErrorParser, ParsedError, StackFrame } from '../types.js';
2
+ import path from 'node:path';
3
+
4
+ const V8_STACK_RE = /at (?:(.+?) )?\((.+?):(\d+):(\d+)\)/;
5
+ const V8_STACK_BARE_RE = /at (.+?):(\d+):(\d+)/;
6
+ const ERROR_TYPE_RE = /^(\w+(?:Error|Exception|Warning)?): (.+)$/m;
7
+
8
+ function parseFrames(input: string): StackFrame[] {
9
+ const frames: StackFrame[] = [];
10
+ for (const line of input.split('\n')) {
11
+ const trimmed = line.trim();
12
+ let match = V8_STACK_RE.exec(trimmed);
13
+ if (match) {
14
+ frames.push({
15
+ function_name: match[1] || null,
16
+ file_path: match[2]!,
17
+ line_number: parseInt(match[3]!, 10),
18
+ column_number: parseInt(match[4]!, 10),
19
+ normalized: `${match[1] || '<anon>'}@${path.basename(match[2]!)}`,
20
+ });
21
+ continue;
22
+ }
23
+ match = V8_STACK_BARE_RE.exec(trimmed);
24
+ if (match) {
25
+ frames.push({
26
+ function_name: null,
27
+ file_path: match[1]!,
28
+ line_number: parseInt(match[2]!, 10),
29
+ column_number: parseInt(match[3]!, 10),
30
+ normalized: `<anon>@${path.basename(match[1]!)}`,
31
+ });
32
+ }
33
+ }
34
+ return frames;
35
+ }
36
+
37
+ export const nodeParser: ErrorParser = {
38
+ name: 'node',
39
+ priority: 10,
40
+
41
+ canParse(input: string): boolean {
42
+ return (
43
+ /at .+ \(.+:\d+:\d+\)/.test(input) ||
44
+ /at .+:\d+:\d+/.test(input) ||
45
+ /^\w*Error:/.test(input) ||
46
+ /^\w*TypeError:/.test(input)
47
+ );
48
+ },
49
+
50
+ parse(input: string): ParsedError | null {
51
+ const typeMatch = ERROR_TYPE_RE.exec(input);
52
+ if (!typeMatch) return null;
53
+
54
+ const errorType = typeMatch[1]!;
55
+ const message = typeMatch[2]!;
56
+ const frames = parseFrames(input);
57
+ const topFrame = frames[0];
58
+
59
+ return {
60
+ errorType,
61
+ message,
62
+ stackTrace: input,
63
+ frames,
64
+ sourceFile: topFrame?.file_path ?? null,
65
+ sourceLine: topFrame?.line_number ?? null,
66
+ language: 'javascript',
67
+ };
68
+ },
69
+ };
@@ -0,0 +1,62 @@
1
+ import type { ErrorParser, ParsedError, StackFrame } from '../types.js';
2
+ import path from 'node:path';
3
+
4
+ const FRAME_RE = /File "(.+?)", line (\d+)(?:, in (.+))?/g;
5
+ const ERROR_LINE_RE = /^(\w+(?:Error|Exception|Warning)?): (.+)$/m;
6
+
7
+ export const pythonParser: ErrorParser = {
8
+ name: 'python',
9
+ priority: 10,
10
+
11
+ canParse(input: string): boolean {
12
+ return (
13
+ /Traceback \(most recent call last\)/.test(input) ||
14
+ /File ".+", line \d+/.test(input)
15
+ );
16
+ },
17
+
18
+ parse(input: string): ParsedError | null {
19
+ const frames: StackFrame[] = [];
20
+ let match: RegExpExecArray | null;
21
+ const frameRe = new RegExp(FRAME_RE.source, 'g');
22
+
23
+ while ((match = frameRe.exec(input)) !== null) {
24
+ frames.push({
25
+ function_name: match[3] || null,
26
+ file_path: match[1]!,
27
+ line_number: parseInt(match[2]!, 10),
28
+ column_number: null,
29
+ normalized: `${match[3] || '<module>'}@${path.basename(match[1]!)}`,
30
+ });
31
+ }
32
+
33
+ const lines = input.trim().split('\n');
34
+ let errorType = 'PythonError';
35
+ let message = '';
36
+
37
+ for (let i = lines.length - 1; i >= 0; i--) {
38
+ const errMatch = ERROR_LINE_RE.exec(lines[i]!);
39
+ if (errMatch) {
40
+ errorType = errMatch[1]!;
41
+ message = errMatch[2]!;
42
+ break;
43
+ }
44
+ }
45
+
46
+ if (!message && lines.length > 0) {
47
+ message = lines[lines.length - 1]!;
48
+ }
49
+
50
+ const topFrame = frames[frames.length - 1];
51
+
52
+ return {
53
+ errorType,
54
+ message,
55
+ stackTrace: input,
56
+ frames,
57
+ sourceFile: topFrame?.file_path ?? null,
58
+ sourceLine: topFrame?.line_number ?? null,
59
+ language: 'python',
60
+ };
61
+ },
62
+ };