@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,190 @@
1
+ import { Command } from 'commander';
2
+ import { withIpc } from '../ipc-helper.js';
3
+ import { readdirSync, readFileSync, statSync } from 'fs';
4
+ import { resolve, basename, relative, extname } from 'path';
5
+
6
+ const DEFAULT_EXTENSIONS = new Set([
7
+ '.ts', '.tsx', '.js', '.jsx', '.py', '.rs', '.go',
8
+ '.java', '.c', '.cpp', '.h', '.hpp', '.rb', '.sh',
9
+ ]);
10
+
11
+ const EXCLUDE_DIRS = new Set([
12
+ 'node_modules', 'dist', 'build', '.git', '.next',
13
+ '__pycache__', 'vendor', 'coverage', '.cache', '.turbo',
14
+ '.nuxt', '.output', 'target', 'out', 'venv', '.venv',
15
+ 'env', '.env', 'site-packages',
16
+ ]);
17
+
18
+ const EXCLUDE_PATTERNS = [/\.min\./, /\.bundle\./, /\.d\.ts$/];
19
+
20
+ const LANG_MAP: Record<string, string> = {
21
+ ts: 'typescript', tsx: 'typescript', js: 'javascript', jsx: 'javascript',
22
+ py: 'python', rs: 'rust', go: 'go', java: 'java',
23
+ c: 'c', cpp: 'cpp', h: 'c', hpp: 'cpp',
24
+ rb: 'ruby', sh: 'shell', bash: 'shell',
25
+ };
26
+
27
+ function detectLanguage(filePath: string): string {
28
+ const ext = extname(filePath).slice(1).toLowerCase();
29
+ return LANG_MAP[ext] ?? ext;
30
+ }
31
+
32
+ function findSourceFiles(dir: string, extensions: Set<string>, maxSizeBytes: number): string[] {
33
+ const files: string[] = [];
34
+
35
+ function walk(current: string): void {
36
+ let entries;
37
+ try {
38
+ entries = readdirSync(current, { withFileTypes: true });
39
+ } catch {
40
+ return; // skip unreadable directories
41
+ }
42
+
43
+ for (const entry of entries) {
44
+ const fullPath = resolve(current, entry.name);
45
+
46
+ if (entry.isDirectory()) {
47
+ if (!EXCLUDE_DIRS.has(entry.name)) {
48
+ walk(fullPath);
49
+ }
50
+ continue;
51
+ }
52
+
53
+ if (!entry.isFile()) continue;
54
+
55
+ const ext = extname(entry.name).toLowerCase();
56
+ if (!extensions.has(ext)) continue;
57
+ if (EXCLUDE_PATTERNS.some(p => p.test(entry.name))) continue;
58
+
59
+ try {
60
+ const stat = statSync(fullPath);
61
+ if (stat.size > maxSizeBytes) continue;
62
+ files.push(fullPath);
63
+ } catch {
64
+ // skip unreadable files
65
+ }
66
+ }
67
+ }
68
+
69
+ walk(dir);
70
+ return files.sort();
71
+ }
72
+
73
+ export function importCommand(): Command {
74
+ return new Command('import')
75
+ .description('Import source files from a project directory into Brain')
76
+ .argument('<directory>', 'Project directory to scan')
77
+ .option('-p, --project <name>', 'Project name (default: directory basename)')
78
+ .option('-e, --extensions <list>', 'Comma-separated extensions (default: ts,tsx,js,jsx,py,rs,go,java,c,cpp,h,hpp,rb,sh)')
79
+ .option('--dry-run', 'List files that would be imported without importing')
80
+ .option('--max-size <kb>', 'Skip files larger than N KB', '100')
81
+ .action(async (directory: string, opts) => {
82
+ const dir = resolve(directory);
83
+ const projectName = opts.project ?? basename(dir);
84
+ const maxSizeKb = parseInt(opts.maxSize, 10) || 100;
85
+ const maxSizeBytes = maxSizeKb * 1024;
86
+
87
+ // Parse extensions
88
+ let extensions = DEFAULT_EXTENSIONS;
89
+ if (opts.extensions) {
90
+ extensions = new Set(
91
+ opts.extensions.split(',').map((e: string) => {
92
+ const trimmed = e.trim();
93
+ return trimmed.startsWith('.') ? trimmed : `.${trimmed}`;
94
+ })
95
+ );
96
+ }
97
+
98
+ // Verify directory exists
99
+ try {
100
+ const stat = statSync(dir);
101
+ if (!stat.isDirectory()) {
102
+ console.error(`Not a directory: ${dir}`);
103
+ process.exit(1);
104
+ }
105
+ } catch {
106
+ console.error(`Directory not found: ${dir}`);
107
+ process.exit(1);
108
+ }
109
+
110
+ console.log(`Scanning ${dir} ...`);
111
+ const files = findSourceFiles(dir, extensions, maxSizeBytes);
112
+
113
+ if (files.length === 0) {
114
+ console.log('No source files found.');
115
+ return;
116
+ }
117
+
118
+ console.log(`Found ${files.length} source files.\n`);
119
+
120
+ if (opts.dryRun) {
121
+ for (const f of files) {
122
+ const rel = relative(dir, f);
123
+ const lang = detectLanguage(f);
124
+ console.log(` [${lang}] ${rel}`);
125
+ }
126
+ console.log(`\n${files.length} files would be imported as project "${projectName}".`);
127
+ return;
128
+ }
129
+
130
+ // Import via IPC
131
+ await withIpc(async (client) => {
132
+ let imported = 0;
133
+ let newCount = 0;
134
+ let existingCount = 0;
135
+ let failedCount = 0;
136
+ let totalScore = 0;
137
+
138
+ for (let i = 0; i < files.length; i++) {
139
+ const filePath = files[i];
140
+ const rel = relative(dir, filePath);
141
+ const fileName = basename(filePath);
142
+ const language = detectLanguage(filePath);
143
+
144
+ let source: string;
145
+ try {
146
+ source = readFileSync(filePath, 'utf-8');
147
+ } catch {
148
+ failedCount++;
149
+ process.stdout.write(` [${i + 1}/${files.length}] ${rel} — read error\n`);
150
+ continue;
151
+ }
152
+
153
+ // Skip empty files
154
+ if (!source.trim()) continue;
155
+
156
+ try {
157
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
158
+ const result: any = await client.request('code.analyze', {
159
+ project: projectName,
160
+ name: fileName,
161
+ filePath: rel,
162
+ language,
163
+ source,
164
+ });
165
+
166
+ const score = result.reusabilityScore ?? 0;
167
+ const status = result.isNew ? 'new' : 'existing';
168
+ totalScore += score;
169
+ imported++;
170
+
171
+ if (result.isNew) newCount++;
172
+ else existingCount++;
173
+
174
+ process.stdout.write(` [${i + 1}/${files.length}] ${rel} → score: ${score.toFixed(2)} (${status})\n`);
175
+ } catch (err) {
176
+ failedCount++;
177
+ const msg = err instanceof Error ? err.message : String(err);
178
+ process.stdout.write(` [${i + 1}/${files.length}] ${rel} — failed: ${msg.slice(0, 80)}\n`);
179
+ }
180
+ }
181
+
182
+ const avgScore = imported > 0 ? (totalScore / imported).toFixed(2) : '0';
183
+ console.log(`\n--- Import Summary ---`);
184
+ console.log(` Project: ${projectName}`);
185
+ console.log(` Imported: ${imported} (${newCount} new, ${existingCount} existing)`);
186
+ if (failedCount > 0) console.log(` Failed: ${failedCount}`);
187
+ console.log(` Avg reusability score: ${avgScore}`);
188
+ });
189
+ });
190
+ }
@@ -0,0 +1,33 @@
1
+ import { Command } from 'commander';
2
+ import { withIpc } from '../ipc-helper.js';
3
+
4
+ export function insightsCommand(): Command {
5
+ return new Command('insights')
6
+ .description('Show research insights')
7
+ .option('--type <type>', 'Filter by type: trend, pattern, gap, synergy, optimization, template_candidate, project_suggestion, warning')
8
+ .option('-l, --limit <n>', 'Maximum results', '20')
9
+ .action(async (opts) => {
10
+ await withIpc(async (client) => {
11
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
12
+ const insights: any = await client.request('research.insights', {
13
+ type: opts.type,
14
+ activeOnly: true,
15
+ limit: parseInt(opts.limit, 10),
16
+ });
17
+
18
+ if (!insights?.length) {
19
+ console.log('No active insights.');
20
+ return;
21
+ }
22
+
23
+ console.log(`${insights.length} insights:\n`);
24
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
25
+ for (const ins of insights as any[]) {
26
+ const priority = ins.priority >= 8 ? 'HIGH' : ins.priority >= 5 ? 'MEDIUM' : 'LOW';
27
+ console.log(` [${ins.type}] [${priority}] ${ins.title}`);
28
+ if (ins.description) console.log(` ${ins.description.slice(0, 150)}`);
29
+ console.log();
30
+ }
31
+ });
32
+ });
33
+ }
@@ -0,0 +1,30 @@
1
+ import { Command } from 'commander';
2
+ import { withIpc } from '../ipc-helper.js';
3
+
4
+ export function modulesCommand(): Command {
5
+ return new Command('modules')
6
+ .description('List registered code modules')
7
+ .option('--language <lang>', 'Filter by language')
8
+ .action(async (opts) => {
9
+ await withIpc(async (client) => {
10
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
11
+ const modules: any = await client.request('code.modules', {
12
+ language: opts.language,
13
+ });
14
+
15
+ if (!modules?.length) {
16
+ console.log('No code modules registered.');
17
+ return;
18
+ }
19
+
20
+ console.log(`${modules.length} code modules:\n`);
21
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
22
+ for (const mod of modules as any[]) {
23
+ console.log(` #${mod.id} [${mod.language}] ${mod.name}`);
24
+ if (mod.description) console.log(` ${mod.description.slice(0, 120)}`);
25
+ console.log(` File: ${mod.filePath} Reusability: ${mod.reusabilityScore ?? '?'}`);
26
+ console.log();
27
+ }
28
+ });
29
+ });
30
+ }
@@ -0,0 +1,61 @@
1
+ import { Command } from 'commander';
2
+ import { withIpc } from '../ipc-helper.js';
3
+
4
+ export function networkCommand(): Command {
5
+ return new Command('network')
6
+ .description('Explore the synapse network')
7
+ .option('--node <type:id>', 'Node to explore (e.g., error:42)')
8
+ .option('-l, --limit <n>', 'Max synapses to show', '20')
9
+ .action(async (opts) => {
10
+ await withIpc(async (client) => {
11
+ if (opts.node) {
12
+ const [nodeType, nodeIdStr] = opts.node.split(':');
13
+ const nodeId = parseInt(nodeIdStr, 10);
14
+
15
+ if (!nodeType || isNaN(nodeId)) {
16
+ console.error('Invalid node format. Use: --node error:42');
17
+ return;
18
+ }
19
+
20
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
21
+ const related: any = await client.request('synapse.related', {
22
+ nodeType,
23
+ nodeId,
24
+ maxDepth: 2,
25
+ });
26
+
27
+ if (!related?.length) {
28
+ console.log(`No connections found for ${nodeType}:${nodeId}`);
29
+ return;
30
+ }
31
+
32
+ console.log(`Connections from ${nodeType}:${nodeId}:\n`);
33
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
34
+ for (const r of related as any[]) {
35
+ console.log(` → ${r.nodeType}:${r.nodeId} (weight: ${(r.activation ?? r.weight ?? 0).toFixed(3)})`);
36
+ }
37
+ } else {
38
+ // Show general network stats
39
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
40
+ const stats: any = await client.request('synapse.stats', {});
41
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
42
+ const overview: any = await client.request('analytics.network', {
43
+ limit: parseInt(opts.limit, 10),
44
+ });
45
+
46
+ console.log('Synapse Network Overview:\n');
47
+ console.log(` Total synapses: ${stats.totalSynapses ?? 0}`);
48
+ console.log(` Average weight: ${(stats.avgWeight ?? 0).toFixed(3)}`);
49
+ console.log();
50
+
51
+ if (overview?.strongestSynapses?.length) {
52
+ console.log('Strongest connections:');
53
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
54
+ for (const s of overview.strongestSynapses as any[]) {
55
+ console.log(` ${s.source} → ${s.target} [${s.type}] weight: ${(s.weight ?? 0).toFixed(3)}`);
56
+ }
57
+ }
58
+ }
59
+ });
60
+ });
61
+ }
@@ -0,0 +1,43 @@
1
+ import { Command } from 'commander';
2
+ import { withIpc } from '../ipc-helper.js';
3
+
4
+ export function queryCommand(): Command {
5
+ return new Command('query')
6
+ .description('Search for errors and solutions')
7
+ .argument('<search>', 'Error message or description to search for')
8
+ .option('-l, --limit <n>', 'Maximum results', '10')
9
+ .action(async (search: string, opts) => {
10
+ await withIpc(async (client) => {
11
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
12
+ const results: any = await client.request('error.query', {
13
+ search,
14
+ limit: parseInt(opts.limit, 10),
15
+ });
16
+
17
+ if (!results?.length) {
18
+ console.log('No matching errors found.');
19
+ return;
20
+ }
21
+
22
+ console.log(`Found ${results.length} errors:\n`);
23
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
24
+ for (const err of results as any[]) {
25
+ const status = err.resolved ? 'RESOLVED' : 'OPEN';
26
+ console.log(` #${err.id} [${status}] ${err.errorType ?? 'unknown'}`);
27
+ console.log(` ${(err.message ?? '').slice(0, 120)}`);
28
+
29
+ // Get solutions for this error
30
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
31
+ const solutions: any = await client.request('solution.query', { error_id: err.id });
32
+ if (solutions?.length > 0) {
33
+ console.log(` Solutions: ${solutions.length}`);
34
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
35
+ for (const sol of solutions.slice(0, 3) as any[]) {
36
+ console.log(` - #${sol.id}: ${(sol.description ?? '').slice(0, 100)}`);
37
+ }
38
+ }
39
+ console.log();
40
+ }
41
+ });
42
+ });
43
+ }
@@ -0,0 +1,59 @@
1
+ import { Command } from 'commander';
2
+ import { spawn } from 'node:child_process';
3
+ import fs from 'node:fs';
4
+ import path from 'node:path';
5
+ import { getDataDir } from '../../utils/paths.js';
6
+
7
+ export function startCommand(): Command {
8
+ return new Command('start')
9
+ .description('Start the Brain daemon')
10
+ .option('-f, --foreground', 'Run in foreground (no detach)')
11
+ .option('-c, --config <path>', 'Config file path')
12
+ .action((opts) => {
13
+ const pidPath = path.join(getDataDir(), 'brain.pid');
14
+
15
+ // Check if already running
16
+ if (fs.existsSync(pidPath)) {
17
+ const pid = parseInt(fs.readFileSync(pidPath, 'utf8').trim(), 10);
18
+ try {
19
+ process.kill(pid, 0); // Check if process exists
20
+ console.log(`Brain daemon is already running (PID: ${pid})`);
21
+ return;
22
+ } catch {
23
+ // PID file stale, remove it
24
+ fs.unlinkSync(pidPath);
25
+ }
26
+ }
27
+
28
+ if (opts.foreground) {
29
+ // Run in foreground — import dynamically to avoid loading everything at CLI parse time
30
+ import('../../brain.js').then(({ BrainCore }) => {
31
+ const core = new BrainCore();
32
+ core.start(opts.config);
33
+ });
34
+ return;
35
+ }
36
+
37
+ // Spawn detached daemon
38
+ const args = ['daemon'];
39
+ if (opts.config) args.push('-c', opts.config);
40
+
41
+ const entryPoint = path.resolve(import.meta.dirname, '../../index.js');
42
+ const child = spawn(process.execPath, [entryPoint, ...args], {
43
+ detached: true,
44
+ stdio: 'ignore',
45
+ });
46
+ child.unref();
47
+
48
+ console.log(`Brain daemon starting (PID: ${child.pid})`);
49
+
50
+ // Wait briefly for PID file to appear
51
+ setTimeout(() => {
52
+ if (fs.existsSync(pidPath)) {
53
+ console.log('Brain daemon started successfully.');
54
+ } else {
55
+ console.log('Brain daemon may still be starting. Check: brain status');
56
+ }
57
+ }, 1000);
58
+ });
59
+ }
@@ -0,0 +1,69 @@
1
+ import { Command } from 'commander';
2
+ import fs from 'node:fs';
3
+ import path from 'node:path';
4
+ import { getDataDir } from '../../utils/paths.js';
5
+ import { withIpc } from '../ipc-helper.js';
6
+
7
+ export function statusCommand(): Command {
8
+ return new Command('status')
9
+ .description('Show Brain daemon status')
10
+ .action(async () => {
11
+ const pidPath = path.join(getDataDir(), 'brain.pid');
12
+
13
+ if (!fs.existsSync(pidPath)) {
14
+ console.log('Brain Daemon: NOT RUNNING');
15
+ return;
16
+ }
17
+
18
+ const pid = parseInt(fs.readFileSync(pidPath, 'utf8').trim(), 10);
19
+ let running = false;
20
+ try {
21
+ process.kill(pid, 0);
22
+ running = true;
23
+ } catch { /* not running */ }
24
+
25
+ if (!running) {
26
+ console.log('Brain Daemon: NOT RUNNING (stale PID file)');
27
+ return;
28
+ }
29
+
30
+ console.log(`Brain Daemon: RUNNING (PID ${pid})`);
31
+
32
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
33
+ await withIpc(async (client) => {
34
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
35
+ const summary: any = await client.request('analytics.summary', {});
36
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
37
+ const network: any = await client.request('synapse.stats', {});
38
+
39
+ const dbPath = path.join(getDataDir(), 'brain.db');
40
+ let dbSize = '?';
41
+ try {
42
+ const stat = fs.statSync(dbPath);
43
+ dbSize = `${(stat.size / 1024 / 1024).toFixed(1)} MB`;
44
+ } catch { /* ignore */ }
45
+
46
+ console.log(`Database: ${dbPath} (${dbSize})`);
47
+ console.log();
48
+
49
+ console.log('Error Brain:');
50
+ console.log(` Errors: ${summary.errors?.total ?? 0} total, ${summary.errors?.unresolved ?? 0} unresolved, ${summary.errors?.last7d ?? 0} last 7d`);
51
+ console.log(` Solutions: ${summary.solutions?.total ?? 0}`);
52
+ console.log(` Rules: ${summary.rules?.active ?? 0} active`);
53
+ console.log(` Anti-Patterns: ${summary.antipatterns?.total ?? 0}`);
54
+ console.log();
55
+
56
+ console.log('Code Brain:');
57
+ console.log(` Modules: ${summary.modules?.total ?? 0} registered`);
58
+ console.log();
59
+
60
+ console.log('Synapse Network:');
61
+ console.log(` Synapses: ${network.totalSynapses ?? 0}`);
62
+ console.log(` Avg weight: ${(network.avgWeight ?? 0).toFixed(2)}`);
63
+ console.log();
64
+
65
+ console.log('Research Brain:');
66
+ console.log(` Insights: ${summary.insights?.active ?? 0} active`);
67
+ });
68
+ });
69
+ }
@@ -0,0 +1,33 @@
1
+ import { Command } from 'commander';
2
+ import fs from 'node:fs';
3
+ import path from 'node:path';
4
+ import { getDataDir } from '../../utils/paths.js';
5
+
6
+ export function stopCommand(): Command {
7
+ return new Command('stop')
8
+ .description('Stop the Brain daemon')
9
+ .action(() => {
10
+ const pidPath = path.join(getDataDir(), 'brain.pid');
11
+
12
+ if (!fs.existsSync(pidPath)) {
13
+ console.log('Brain daemon is not running (no PID file found).');
14
+ return;
15
+ }
16
+
17
+ const pid = parseInt(fs.readFileSync(pidPath, 'utf8').trim(), 10);
18
+
19
+ try {
20
+ process.kill(pid, 'SIGTERM');
21
+ console.log(`Brain daemon stopped (PID: ${pid})`);
22
+ } catch (err) {
23
+ if ((err as NodeJS.ErrnoException).code === 'ESRCH') {
24
+ console.log('Brain daemon was not running (stale PID file removed).');
25
+ } else {
26
+ console.error(`Failed to stop daemon: ${err}`);
27
+ }
28
+ }
29
+
30
+ // Clean up PID file
31
+ try { fs.unlinkSync(pidPath); } catch { /* ignore */ }
32
+ });
33
+ }
@@ -0,0 +1,21 @@
1
+ import { IpcClient } from '../ipc/client.js';
2
+ import { getPipeName } from '../utils/paths.js';
3
+
4
+ export async function withIpc<T>(fn: (client: IpcClient) => Promise<T>): Promise<T> {
5
+ const client = new IpcClient(getPipeName(), 5000);
6
+ try {
7
+ await client.connect();
8
+ return await fn(client);
9
+ } catch (err) {
10
+ if (err instanceof Error && err.message.includes('ENOENT')) {
11
+ console.error('Brain daemon is not running. Start it with: brain start');
12
+ } else if (err instanceof Error && err.message.includes('ECONNREFUSED')) {
13
+ console.error('Brain daemon is not responding. Try: brain stop && brain start');
14
+ } else {
15
+ console.error(`Error: ${err instanceof Error ? err.message : err}`);
16
+ }
17
+ process.exit(1);
18
+ } finally {
19
+ client.disconnect();
20
+ }
21
+ }
@@ -0,0 +1,77 @@
1
+ import type { ExportInfo } from '../types/code.types.js';
2
+ import * as tsParser from './parsers/typescript.js';
3
+ import * as pyParser from './parsers/python.js';
4
+ import * as genericParser from './parsers/generic.js';
5
+
6
+ export interface AnalysisResult {
7
+ exports: ExportInfo[];
8
+ externalDeps: string[];
9
+ internalDeps: string[];
10
+ isPure: boolean;
11
+ hasTypeAnnotations: boolean;
12
+ linesOfCode: number;
13
+ }
14
+
15
+ const SIDE_EFFECT_PATTERNS = [
16
+ 'fs.', 'process.exit', 'process.env', 'console.', 'fetch(',
17
+ 'XMLHttpRequest', 'document.', 'window.',
18
+ 'global.', 'require(',
19
+ ];
20
+
21
+ function getParser(language: string) {
22
+ switch (language) {
23
+ case 'typescript':
24
+ case 'javascript':
25
+ return tsParser;
26
+ case 'python':
27
+ return pyParser;
28
+ default:
29
+ return genericParser;
30
+ }
31
+ }
32
+
33
+ export function analyzeCode(source: string, language: string): AnalysisResult {
34
+ const parser = getParser(language);
35
+ const exports = parser.extractExports(source);
36
+ const { external, internal } = parser.extractImports(source);
37
+ const isPure = checkPurity(source);
38
+ const typed = parser.hasTypeAnnotations(source);
39
+ const linesOfCode = source.split('\n').filter(l => l.trim().length > 0).length;
40
+
41
+ return {
42
+ exports,
43
+ externalDeps: external,
44
+ internalDeps: internal,
45
+ isPure,
46
+ hasTypeAnnotations: typed,
47
+ linesOfCode,
48
+ };
49
+ }
50
+
51
+ export function checkPurity(source: string): boolean {
52
+ return !SIDE_EFFECT_PATTERNS.some(p => source.includes(p));
53
+ }
54
+
55
+ export function measureCohesion(exports: ExportInfo[]): number {
56
+ if (exports.length <= 1) return 1.0;
57
+
58
+ const names = exports.map(e =>
59
+ e.name
60
+ .replace(/([A-Z]+)([A-Z][a-z])/g, '$1 $2')
61
+ .replace(/([a-z\d])([A-Z])/g, '$1 $2')
62
+ .toLowerCase()
63
+ .split(/\s+/)
64
+ );
65
+
66
+ const vocab = new Set<string>();
67
+ names.forEach(tokens => tokens.forEach(t => vocab.add(t)));
68
+
69
+ let sharedTokens = 0;
70
+ for (const token of vocab) {
71
+ const count = names.filter(n => n.includes(token)).length;
72
+ if (count > 1) sharedTokens += count;
73
+ }
74
+
75
+ const maxPossible = names.length * vocab.size;
76
+ return maxPossible === 0 ? 0 : sharedTokens / maxPossible;
77
+ }