@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,495 @@
1
+ import { Command } from 'commander';
2
+ import { withIpc } from '../ipc-helper.js';
3
+ import { writeFileSync } from 'fs';
4
+ import { resolve } from 'path';
5
+
6
+ export function dashboardCommand(): Command {
7
+ return new Command('dashboard')
8
+ .description('Generate and open the Brain dashboard with live data')
9
+ .option('-o, --output <path>', 'Output HTML file path')
10
+ .option('--no-open', 'Generate without opening in browser')
11
+ .action(async (opts) => {
12
+ await withIpc(async (client) => {
13
+ console.log('Fetching data from Brain...');
14
+
15
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
16
+ const summary: any = await client.request('analytics.summary', {});
17
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
18
+ const network: any = await client.request('synapse.stats', {});
19
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
20
+ const insights: any = await client.request('research.insights', {
21
+ activeOnly: true,
22
+ limit: 500,
23
+ });
24
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
25
+ const modules: any = await client.request('code.modules', {});
26
+
27
+ // Collect language stats
28
+ const langStats: Record<string, number> = {};
29
+ const projectSet = new Set<string>();
30
+ if (Array.isArray(modules)) {
31
+ for (const m of modules) {
32
+ langStats[m.language] = (langStats[m.language] || 0) + 1;
33
+ if (m.projectId) projectSet.add(String(m.projectId));
34
+ }
35
+ }
36
+
37
+ // Categorize insights
38
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
39
+ const insightList = Array.isArray(insights) ? insights : [];
40
+ const templates = insightList.filter((i: InsightItem) => i.type === 'template_candidate' || i.title?.includes('Template'));
41
+ const suggestions = insightList.filter((i: InsightItem) => i.type === 'suggestion' || i.type === 'project_suggestion');
42
+ const trends = insightList.filter((i: InsightItem) => i.type === 'trend' || i.type === 'pattern');
43
+ const gaps = insightList.filter((i: InsightItem) => i.type === 'gap');
44
+ const warnings = insightList.filter((i: InsightItem) => i.type === 'warning');
45
+ const synergies = insightList.filter((i: InsightItem) => i.type === 'synergy' || i.type === 'optimization');
46
+
47
+ const data = {
48
+ stats: {
49
+ modules: summary.modules?.total ?? 0,
50
+ synapses: network.totalSynapses ?? 0,
51
+ errors: summary.errors?.total ?? 0,
52
+ solutions: summary.solutions?.total ?? 0,
53
+ rules: summary.rules?.active ?? 0,
54
+ insights: insightList.length,
55
+ },
56
+ langStats,
57
+ insights: { templates, suggestions, trends, gaps, warnings, synergies },
58
+ };
59
+
60
+ const html = generateHtml(data);
61
+ const outPath = opts.output
62
+ ? resolve(opts.output)
63
+ : resolve(import.meta.dirname, '../../../dashboard.html');
64
+
65
+ writeFileSync(outPath, html, 'utf-8');
66
+ console.log(`Dashboard written to ${outPath}`);
67
+ console.log(` Modules: ${data.stats.modules}`);
68
+ console.log(` Synapses: ${data.stats.synapses}`);
69
+ console.log(` Insights: ${data.stats.insights}`);
70
+
71
+ if (opts.open !== false) {
72
+ const { exec } = await import('child_process');
73
+ exec(`start "" "${outPath}"`);
74
+ }
75
+ });
76
+ });
77
+ }
78
+
79
+ interface InsightItem {
80
+ type: string;
81
+ title: string;
82
+ description?: string;
83
+ priority?: string;
84
+ }
85
+
86
+ interface DashboardData {
87
+ stats: {
88
+ modules: number;
89
+ synapses: number;
90
+ errors: number;
91
+ solutions: number;
92
+ rules: number;
93
+ insights: number;
94
+ };
95
+ langStats: Record<string, number>;
96
+ insights: {
97
+ templates: InsightItem[];
98
+ suggestions: InsightItem[];
99
+ trends: InsightItem[];
100
+ gaps: InsightItem[];
101
+ warnings: InsightItem[];
102
+ synergies: InsightItem[];
103
+ };
104
+ }
105
+
106
+ function esc(s: string): string {
107
+ return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
108
+ }
109
+
110
+ function generateHtml(data: DashboardData): string {
111
+ const { stats, langStats, insights } = data;
112
+
113
+ // Build language chart bars
114
+ const sortedLangs = Object.entries(langStats).sort((a, b) => b[1] - a[1]);
115
+ const maxLang = sortedLangs[0]?.[1] || 1;
116
+ const langBars = sortedLangs.slice(0, 12).map(([lang, count]) => {
117
+ const pct = Math.round((count / maxLang) * 100);
118
+ return `<div class="lang-row"><span class="lang-name">${esc(lang)}</span><div class="lang-bar-bg"><div class="lang-bar" data-width="${pct}"></div></div><span class="lang-count">${count}</span></div>`;
119
+ }).join('\n');
120
+
121
+ // Build insight cards
122
+ function insightCards(items: InsightItem[], color: string): string {
123
+ if (!items.length) return '<p class="empty">Keine Insights in dieser Kategorie.</p>';
124
+ return items.slice(0, 30).map(i => {
125
+ const prio = i.priority ? `<span class="prio prio-${String(i.priority).toLowerCase()}">${esc(String(i.priority))}</span>` : '';
126
+ return `<div class="insight-card ${color}"><div class="insight-header">${prio}<strong>${esc(i.title)}</strong></div><p>${esc((i.description || '').slice(0, 200))}</p></div>`;
127
+ }).join('\n');
128
+ }
129
+
130
+ const totalKnowledge = stats.modules + stats.synapses + stats.errors + stats.solutions;
131
+ const activityLevel = Math.min(100, Math.round((stats.insights / Math.max(1, totalKnowledge)) * 1000));
132
+
133
+ return `<!DOCTYPE html>
134
+ <html lang="de">
135
+ <head>
136
+ <meta charset="UTF-8">
137
+ <meta name="viewport" content="width=device-width, initial-scale=1.0">
138
+ <title>Brain — Dashboard</title>
139
+ <style>
140
+ @import url('https://fonts.googleapis.com/css2?family=Inter:wght@300;400;500;600;700;800;900&display=swap');
141
+ *,*::before,*::after{box-sizing:border-box;margin:0;padding:0}
142
+ :root{
143
+ --bg:#04060e;--bg2:rgba(10,12,24,.7);--bg3:rgba(20,24,50,.6);--bg4:rgba(30,35,70,.5);
144
+ --glass:rgba(15,18,40,.55);--glass-border:rgba(100,120,255,.12);--glass-hover:rgba(100,120,255,.2);
145
+ --text:#e8eaf6;--text2:#8b8fb0;--text3:#4a4d6e;
146
+ --blue:#5b9cff;--red:#ff5577;--green:#3dffa0;
147
+ --purple:#b47aff;--orange:#ffb347;--cyan:#47e5ff;
148
+ --accent:linear-gradient(135deg,#b47aff,#5b9cff,#47e5ff);
149
+ --radius:16px;--radius-sm:10px;
150
+ }
151
+ html{scroll-behavior:smooth}
152
+ body{font-family:'Inter',system-ui,sans-serif;background:var(--bg);color:var(--text);line-height:1.6;min-height:100vh;overflow-x:hidden}
153
+
154
+ /* Neural canvas background */
155
+ #neural-bg{position:fixed;top:0;left:0;width:100%;height:100%;z-index:0;pointer-events:none}
156
+
157
+ /* Ambient glow orbs */
158
+ .orb{position:fixed;border-radius:50%;filter:blur(120px);opacity:.12;pointer-events:none;z-index:0}
159
+ .orb-1{width:600px;height:600px;background:var(--purple);top:-200px;left:-100px;animation:orb-float 20s ease-in-out infinite}
160
+ .orb-2{width:500px;height:500px;background:var(--blue);bottom:-150px;right:-100px;animation:orb-float 25s ease-in-out infinite reverse}
161
+ .orb-3{width:400px;height:400px;background:var(--cyan);top:40%;left:50%;animation:orb-float 18s ease-in-out infinite 5s}
162
+ @keyframes orb-float{0%,100%{transform:translate(0,0) scale(1)}33%{transform:translate(60px,-40px) scale(1.1)}66%{transform:translate(-40px,60px) scale(.9)}}
163
+
164
+ .container{max-width:1400px;margin:0 auto;padding:0 28px;position:relative;z-index:1}
165
+
166
+ /* Reveal animations */
167
+ .reveal{opacity:0;transform:translateY(30px);transition:opacity .6s ease,transform .6s ease}
168
+ .reveal.visible{opacity:1;transform:translateY(0)}
169
+ .reveal-delay-1{transition-delay:.1s}.reveal-delay-2{transition-delay:.2s}
170
+ .reveal-delay-3{transition-delay:.3s}.reveal-delay-4{transition-delay:.4s}
171
+ .reveal-delay-5{transition-delay:.5s}
172
+
173
+ section{margin-bottom:56px}
174
+
175
+ /* Header */
176
+ header{padding:60px 0 24px;text-align:center;position:relative}
177
+ .logo{display:flex;align-items:center;justify-content:center;gap:20px;margin-bottom:12px}
178
+ .logo-icon{
179
+ width:68px;height:68px;border-radius:18px;
180
+ background:linear-gradient(135deg,var(--purple),var(--blue),var(--cyan));
181
+ display:flex;align-items:center;justify-content:center;font-size:32px;
182
+ box-shadow:0 0 60px rgba(170,102,255,.35),0 0 120px rgba(90,150,255,.15);
183
+ animation:icon-breathe 4s ease-in-out infinite;
184
+ position:relative;
185
+ }
186
+ .logo-icon::after{
187
+ content:'';position:absolute;inset:-3px;border-radius:20px;
188
+ background:linear-gradient(135deg,var(--purple),var(--cyan));
189
+ opacity:.4;filter:blur(8px);z-index:-1;animation:icon-breathe 4s ease-in-out infinite reverse;
190
+ }
191
+ @keyframes icon-breathe{0%,100%{box-shadow:0 0 60px rgba(170,102,255,.35),0 0 120px rgba(90,150,255,.15)}50%{box-shadow:0 0 80px rgba(170,102,255,.5),0 0 160px rgba(90,150,255,.25)}}
192
+ .logo h1{font-size:2.8rem;font-weight:900;letter-spacing:-1px;background:linear-gradient(135deg,#fff 0%,var(--blue) 50%,var(--purple) 100%);-webkit-background-clip:text;-webkit-text-fill-color:transparent;background-clip:text}
193
+ .tagline{color:var(--text2);font-size:1.05rem;font-weight:300;letter-spacing:.5px}
194
+
195
+ /* Activity indicator */
196
+ .activity{display:inline-flex;align-items:center;gap:10px;margin-top:16px;padding:8px 20px;border-radius:30px;background:var(--glass);border:1px solid var(--glass-border);backdrop-filter:blur(20px)}
197
+ .activity-dot{width:8px;height:8px;border-radius:50%;background:var(--green);box-shadow:0 0 12px var(--green);animation:pulse-dot 2s ease-in-out infinite}
198
+ @keyframes pulse-dot{0%,100%{opacity:1;box-shadow:0 0 12px var(--green)}50%{opacity:.5;box-shadow:0 0 20px var(--green)}}
199
+ .activity-text{font-size:.8rem;color:var(--text2);font-weight:500}
200
+ .activity-bar{width:80px;height:4px;border-radius:2px;background:var(--bg4);overflow:hidden}
201
+ .activity-fill{height:100%;border-radius:2px;background:linear-gradient(90deg,var(--green),var(--cyan));transition:width 1.5s ease}
202
+
203
+ /* Nav */
204
+ nav{display:flex;justify-content:center;gap:8px;flex-wrap:wrap;padding:20px 0;margin-bottom:40px}
205
+ nav a{
206
+ color:var(--text2);text-decoration:none;padding:8px 18px;border-radius:24px;font-size:.85rem;font-weight:500;
207
+ transition:all .3s ease;border:1px solid transparent;backdrop-filter:blur(10px);
208
+ }
209
+ nav a:hover{color:var(--text);background:var(--glass);border-color:var(--glass-border);transform:translateY(-1px)}
210
+ nav a.research{
211
+ background:var(--glass);color:var(--cyan);border-color:rgba(71,229,255,.25);font-weight:600;
212
+ box-shadow:0 0 20px rgba(71,229,255,.1);animation:nav-glow 3s ease-in-out infinite alternate;
213
+ }
214
+ @keyframes nav-glow{0%{box-shadow:0 0 20px rgba(71,229,255,.1)}100%{box-shadow:0 0 35px rgba(71,229,255,.2)}}
215
+
216
+ /* Stats */
217
+ .stats-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(190px,1fr));gap:18px}
218
+ .stat-card{
219
+ background:var(--glass);border:1px solid var(--glass-border);border-radius:var(--radius);
220
+ padding:28px 22px;text-align:center;position:relative;overflow:hidden;
221
+ transition:all .35s ease;backdrop-filter:blur(20px);
222
+ }
223
+ .stat-card:hover{transform:translateY(-4px);border-color:var(--glass-hover);box-shadow:0 20px 60px rgba(0,0,0,.3)}
224
+ .stat-card::before{content:'';position:absolute;top:0;left:0;right:0;height:2px}
225
+ .stat-card::after{content:'';position:absolute;top:0;left:0;right:0;bottom:0;background:radial-gradient(ellipse at 50% 0%,rgba(255,255,255,.03),transparent 70%);pointer-events:none}
226
+ .stat-card.blue::before{background:linear-gradient(90deg,transparent,var(--blue),transparent)}
227
+ .stat-card.purple::before{background:linear-gradient(90deg,transparent,var(--purple),transparent)}
228
+ .stat-card.red::before{background:linear-gradient(90deg,transparent,var(--red),transparent)}
229
+ .stat-card.green::before{background:linear-gradient(90deg,transparent,var(--green),transparent)}
230
+ .stat-card.orange::before{background:linear-gradient(90deg,transparent,var(--orange),transparent)}
231
+ .stat-card.cyan::before{background:linear-gradient(90deg,transparent,var(--cyan),transparent)}
232
+ .stat-number{font-size:2.6rem;font-weight:900;letter-spacing:-2px}
233
+ .stat-card.blue .stat-number{color:var(--blue)}.stat-card.purple .stat-number{color:var(--purple)}
234
+ .stat-card.red .stat-number{color:var(--red)}.stat-card.green .stat-number{color:var(--green)}
235
+ .stat-card.orange .stat-number{color:var(--orange)}.stat-card.cyan .stat-number{color:var(--cyan)}
236
+ .stat-label{color:var(--text2);font-size:.82rem;margin-top:6px;font-weight:500;letter-spacing:.3px;text-transform:uppercase}
237
+
238
+ /* Section titles */
239
+ .section-title{font-size:1.5rem;font-weight:700;margin-bottom:24px;display:flex;align-items:center;gap:12px}
240
+ .section-title .icon{font-size:1.2rem;width:38px;height:38px;border-radius:var(--radius-sm);display:flex;align-items:center;justify-content:center;backdrop-filter:blur(10px)}
241
+
242
+ /* Language chart */
243
+ .lang-chart{max-width:650px}
244
+ .lang-row{display:flex;align-items:center;gap:14px;margin-bottom:10px}
245
+ .lang-name{width:100px;text-align:right;font-size:.85rem;color:var(--text2);font-weight:500}
246
+ .lang-bar-bg{flex:1;height:28px;background:var(--bg3);border-radius:6px;overflow:hidden;border:1px solid var(--glass-border)}
247
+ .lang-bar{height:100%;background:var(--accent);border-radius:6px;width:0;transition:width 1.2s cubic-bezier(.22,1,.36,1)}
248
+ .lang-count{width:50px;font-size:.85rem;color:var(--text2);font-weight:600}
249
+
250
+ /* Insight tabs */
251
+ .tab-bar{display:flex;gap:8px;flex-wrap:wrap;margin-bottom:24px}
252
+ .tab-btn{
253
+ padding:10px 20px;border-radius:24px;border:1px solid var(--glass-border);
254
+ background:var(--glass);color:var(--text2);cursor:pointer;font-size:.85rem;font-weight:500;
255
+ transition:all .3s ease;backdrop-filter:blur(10px);font-family:inherit;
256
+ }
257
+ .tab-btn:hover{border-color:var(--glass-hover);color:var(--text);transform:translateY(-1px)}
258
+ .tab-btn.active{border-color:rgba(71,229,255,.35);color:var(--cyan);background:rgba(71,229,255,.08);box-shadow:0 0 20px rgba(71,229,255,.1)}
259
+ .tab-btn .count{background:var(--bg4);padding:2px 8px;border-radius:12px;font-size:.72rem;margin-left:6px;font-weight:600}
260
+ .tab-panel{display:none}.tab-panel.active{display:block}
261
+ .insight-grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(380px,1fr));gap:14px}
262
+ .insight-card{
263
+ background:var(--glass);border:1px solid var(--glass-border);border-radius:var(--radius-sm);
264
+ padding:18px;border-left:3px solid var(--text3);transition:all .25s ease;backdrop-filter:blur(20px);
265
+ }
266
+ .insight-card:hover{transform:translateX(6px);border-color:var(--glass-hover);box-shadow:0 8px 30px rgba(0,0,0,.2)}
267
+ .insight-card.cyan{border-left-color:var(--cyan)}.insight-card.orange{border-left-color:var(--orange)}
268
+ .insight-card.green{border-left-color:var(--green)}.insight-card.red{border-left-color:var(--red)}
269
+ .insight-card.purple{border-left-color:var(--purple)}.insight-card.blue{border-left-color:var(--blue)}
270
+ .insight-header{display:flex;align-items:center;gap:8px;margin-bottom:8px;flex-wrap:wrap}
271
+ .insight-card p{color:var(--text2);font-size:.85rem;line-height:1.5}
272
+ .prio{font-size:.68rem;padding:3px 10px;border-radius:12px;text-transform:uppercase;font-weight:700;letter-spacing:.5px}
273
+ .prio-critical{background:rgba(255,85,119,.15);color:var(--red);border:1px solid rgba(255,85,119,.25)}
274
+ .prio-high{background:rgba(255,179,71,.15);color:var(--orange);border:1px solid rgba(255,179,71,.25)}
275
+ .prio-medium{background:rgba(91,156,255,.15);color:var(--blue);border:1px solid rgba(91,156,255,.25)}
276
+ .prio-low{background:rgba(139,143,176,.1);color:var(--text2);border:1px solid rgba(139,143,176,.2)}
277
+ .empty{color:var(--text3);font-style:italic;padding:24px}
278
+
279
+ /* Footer */
280
+ footer{text-align:center;padding:40px 0;border-top:1px solid var(--glass-border)}
281
+ footer p{color:var(--text3);font-size:.8rem}
282
+ footer code{background:var(--glass);padding:3px 10px;border-radius:6px;font-size:.78rem;border:1px solid var(--glass-border)}
283
+
284
+ /* Responsive */
285
+ @media(max-width:600px){.stats-grid{grid-template-columns:1fr 1fr}.insight-grid{grid-template-columns:1fr}.logo h1{font-size:2rem}}
286
+ </style>
287
+ </head>
288
+ <body>
289
+
290
+ <canvas id="neural-bg"></canvas>
291
+ <div class="orb orb-1"></div>
292
+ <div class="orb orb-2"></div>
293
+ <div class="orb orb-3"></div>
294
+
295
+ <div class="container">
296
+ <header class="reveal">
297
+ <div class="logo">
298
+ <div class="logo-icon">&#129504;</div>
299
+ <h1>Brain</h1>
300
+ </div>
301
+ <p class="tagline">Adaptive Code Intelligence</p>
302
+ <div class="activity">
303
+ <span class="activity-dot"></span>
304
+ <span class="activity-text">Neural Activity</span>
305
+ <div class="activity-bar"><div class="activity-fill" style="width:0%" data-target="${activityLevel}"></div></div>
306
+ <span class="activity-text" style="color:var(--cyan);font-weight:700">${activityLevel}%</span>
307
+ </div>
308
+ </header>
309
+
310
+ <nav class="reveal reveal-delay-1">
311
+ <a href="#stats">Stats</a>
312
+ <a href="#languages">Languages</a>
313
+ <a href="#research" class="research">&#128300; Research</a>
314
+ </nav>
315
+
316
+ <section id="stats" class="reveal reveal-delay-2">
317
+ <div class="section-title"><div class="icon" style="background:rgba(91,156,255,.1)">&#128202;</div> Neural Status</div>
318
+ <div class="stats-grid">
319
+ <div class="stat-card blue"><div class="stat-number">${stats.modules.toLocaleString()}</div><div class="stat-label">Modules</div></div>
320
+ <div class="stat-card purple"><div class="stat-number">${stats.synapses.toLocaleString()}</div><div class="stat-label">Synapses</div></div>
321
+ <div class="stat-card cyan"><div class="stat-number">${stats.insights}</div><div class="stat-label">Insights</div></div>
322
+ <div class="stat-card red"><div class="stat-number">${stats.errors}</div><div class="stat-label">Errors</div></div>
323
+ <div class="stat-card green"><div class="stat-number">${stats.solutions}</div><div class="stat-label">Solutions</div></div>
324
+ <div class="stat-card orange"><div class="stat-number">${stats.rules}</div><div class="stat-label">Rules</div></div>
325
+ </div>
326
+ </section>
327
+
328
+ <section id="languages" class="reveal reveal-delay-3">
329
+ <div class="section-title"><div class="icon" style="background:rgba(180,122,255,.1)">&#128187;</div> Languages</div>
330
+ <div class="lang-chart">${langBars}</div>
331
+ </section>
332
+
333
+ <section id="research" class="reveal reveal-delay-4">
334
+ <div class="section-title"><div class="icon" style="background:rgba(71,229,255,.1)">&#128300;</div> Research Insights</div>
335
+ <div class="tab-bar">
336
+ <button class="tab-btn active" data-tab="templates">&#127912; Templates <span class="count">${insights.templates.length}</span></button>
337
+ <button class="tab-btn" data-tab="suggestions">&#128161; Suggestions <span class="count">${insights.suggestions.length}</span></button>
338
+ <button class="tab-btn" data-tab="trends">&#128200; Trends <span class="count">${insights.trends.length}</span></button>
339
+ <button class="tab-btn" data-tab="gaps">&#9888;&#65039; Gaps <span class="count">${insights.gaps.length}</span></button>
340
+ <button class="tab-btn" data-tab="synergies">&#9889; Synergies <span class="count">${insights.synergies.length}</span></button>
341
+ <button class="tab-btn" data-tab="warnings">&#128680; Warnings <span class="count">${insights.warnings.length}</span></button>
342
+ </div>
343
+ <div class="tab-panel active" id="tab-templates"><div class="insight-grid">${insightCards(insights.templates, 'cyan')}</div></div>
344
+ <div class="tab-panel" id="tab-suggestions"><div class="insight-grid">${insightCards(insights.suggestions, 'orange')}</div></div>
345
+ <div class="tab-panel" id="tab-trends"><div class="insight-grid">${insightCards(insights.trends, 'green')}</div></div>
346
+ <div class="tab-panel" id="tab-gaps"><div class="insight-grid">${insightCards(insights.gaps, 'red')}</div></div>
347
+ <div class="tab-panel" id="tab-synergies"><div class="insight-grid">${insightCards(insights.synergies, 'purple')}</div></div>
348
+ <div class="tab-panel" id="tab-warnings"><div class="insight-grid">${insightCards(insights.warnings, 'red')}</div></div>
349
+ </section>
350
+
351
+ <footer class="reveal reveal-delay-5">
352
+ <p>Brain v1.0 &mdash; <code>brain dashboard</code></p>
353
+ </footer>
354
+ </div>
355
+
356
+ <script>
357
+ // --- Neural Network Canvas ---
358
+ (function(){
359
+ const canvas = document.getElementById('neural-bg');
360
+ const ctx = canvas.getContext('2d');
361
+ let W, H, nodes = [], mouse = {x:-1000,y:-1000};
362
+
363
+ function resize(){
364
+ W = canvas.width = window.innerWidth;
365
+ H = canvas.height = window.innerHeight;
366
+ }
367
+ resize();
368
+ window.addEventListener('resize', resize);
369
+ document.addEventListener('mousemove', e => { mouse.x = e.clientX; mouse.y = e.clientY; });
370
+
371
+ const NODE_COUNT = Math.min(80, Math.floor(window.innerWidth / 18));
372
+ const CONNECT_DIST = 180;
373
+ const MOUSE_DIST = 200;
374
+
375
+ for(let i = 0; i < NODE_COUNT; i++){
376
+ nodes.push({
377
+ x: Math.random() * W,
378
+ y: Math.random() * H,
379
+ vx: (Math.random() - 0.5) * 0.4,
380
+ vy: (Math.random() - 0.5) * 0.4,
381
+ r: Math.random() * 2 + 1,
382
+ pulse: Math.random() * Math.PI * 2,
383
+ });
384
+ }
385
+
386
+ function draw(){
387
+ ctx.clearRect(0, 0, W, H);
388
+
389
+ // Draw connections
390
+ for(let i = 0; i < nodes.length; i++){
391
+ for(let j = i + 1; j < nodes.length; j++){
392
+ const dx = nodes[i].x - nodes[j].x;
393
+ const dy = nodes[i].y - nodes[j].y;
394
+ const dist = Math.sqrt(dx*dx + dy*dy);
395
+ if(dist < CONNECT_DIST){
396
+ const alpha = (1 - dist / CONNECT_DIST) * 0.15;
397
+ ctx.strokeStyle = 'rgba(91,156,255,' + alpha + ')';
398
+ ctx.lineWidth = 0.5;
399
+ ctx.beginPath();
400
+ ctx.moveTo(nodes[i].x, nodes[i].y);
401
+ ctx.lineTo(nodes[j].x, nodes[j].y);
402
+ ctx.stroke();
403
+ }
404
+ }
405
+
406
+ // Mouse interaction
407
+ const mdx = nodes[i].x - mouse.x;
408
+ const mdy = nodes[i].y - mouse.y;
409
+ const mDist = Math.sqrt(mdx*mdx + mdy*mdy);
410
+ if(mDist < MOUSE_DIST){
411
+ const alpha = (1 - mDist / MOUSE_DIST) * 0.4;
412
+ ctx.strokeStyle = 'rgba(180,122,255,' + alpha + ')';
413
+ ctx.lineWidth = 1;
414
+ ctx.beginPath();
415
+ ctx.moveTo(nodes[i].x, nodes[i].y);
416
+ ctx.lineTo(mouse.x, mouse.y);
417
+ ctx.stroke();
418
+ }
419
+ }
420
+
421
+ // Draw nodes
422
+ const time = Date.now() * 0.001;
423
+ for(const n of nodes){
424
+ const glow = 0.4 + Math.sin(time * 1.5 + n.pulse) * 0.3;
425
+ ctx.fillStyle = 'rgba(91,156,255,' + glow + ')';
426
+ ctx.beginPath();
427
+ ctx.arc(n.x, n.y, n.r, 0, Math.PI * 2);
428
+ ctx.fill();
429
+
430
+ n.x += n.vx;
431
+ n.y += n.vy;
432
+ if(n.x < 0 || n.x > W) n.vx *= -1;
433
+ if(n.y < 0 || n.y > H) n.vy *= -1;
434
+ }
435
+
436
+ requestAnimationFrame(draw);
437
+ }
438
+ draw();
439
+ })();
440
+
441
+ // --- Reveal on scroll ---
442
+ const observer = new IntersectionObserver(entries => {
443
+ entries.forEach(e => { if(e.isIntersecting) e.target.classList.add('visible'); });
444
+ }, {threshold: 0.1});
445
+ document.querySelectorAll('.reveal').forEach(el => observer.observe(el));
446
+
447
+ // --- Tab switching ---
448
+ document.querySelectorAll('.tab-btn').forEach(btn => {
449
+ btn.addEventListener('click', () => {
450
+ document.querySelectorAll('.tab-btn').forEach(b => b.classList.remove('active'));
451
+ document.querySelectorAll('.tab-panel').forEach(p => p.classList.remove('active'));
452
+ btn.classList.add('active');
453
+ document.getElementById('tab-' + btn.dataset.tab).classList.add('active');
454
+ });
455
+ });
456
+
457
+ // --- Animate stat numbers ---
458
+ const numObserver = new IntersectionObserver(entries => {
459
+ entries.forEach(e => {
460
+ if(!e.isIntersecting) return;
461
+ const el = e.target;
462
+ if(el.dataset.animated) return;
463
+ el.dataset.animated = '1';
464
+ const target = parseInt(el.textContent.replace(/\\D/g,''), 10);
465
+ if(isNaN(target) || target === 0) return;
466
+ const duration = 1200;
467
+ const start = performance.now();
468
+ function tick(now){
469
+ const t = Math.min((now - start) / duration, 1);
470
+ const ease = 1 - Math.pow(1 - t, 3);
471
+ el.textContent = Math.round(target * ease).toLocaleString();
472
+ if(t < 1) requestAnimationFrame(tick);
473
+ }
474
+ requestAnimationFrame(tick);
475
+ });
476
+ }, {threshold: 0.5});
477
+ document.querySelectorAll('.stat-number').forEach(el => numObserver.observe(el));
478
+
479
+ // --- Animate language bars ---
480
+ setTimeout(() => {
481
+ document.querySelectorAll('.lang-bar').forEach(bar => {
482
+ bar.style.width = bar.dataset.width + '%';
483
+ });
484
+ }, 300);
485
+
486
+ // --- Activity bar ---
487
+ setTimeout(() => {
488
+ document.querySelectorAll('.activity-fill').forEach(el => {
489
+ el.style.width = el.dataset.target + '%';
490
+ });
491
+ }, 500);
492
+ </script>
493
+ </body>
494
+ </html>`;
495
+ }
@@ -0,0 +1,27 @@
1
+ import { Command } from 'commander';
2
+ import { withIpc } from '../ipc-helper.js';
3
+
4
+ export function exportCommand(): Command {
5
+ return new Command('export')
6
+ .description('Export Brain data')
7
+ .option('--format <fmt>', 'Output format: json (default)', 'json')
8
+ .action(async () => {
9
+ await withIpc(async (client) => {
10
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
11
+ const summary: any = await client.request('analytics.summary', {});
12
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
13
+ const network: any = await client.request('analytics.network', { limit: 100 });
14
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
15
+ const insights: any = await client.request('research.insights', { activeOnly: true, limit: 100 });
16
+
17
+ const data = {
18
+ exportedAt: new Date().toISOString(),
19
+ summary,
20
+ network,
21
+ insights,
22
+ };
23
+
24
+ console.log(JSON.stringify(data, null, 2));
25
+ });
26
+ });
27
+ }