@mindrian_os/cli 1.13.0-beta.10 → 1.13.0-beta.44

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 (590) hide show
  1. package/CHANGELOG.md +348 -13
  2. package/README.md +76 -579
  3. package/bin/cli.js +166 -40
  4. package/lib/core/active-plugin-root.cjs +207 -0
  5. package/package.json +7 -13
  6. package/.claude-plugin/plugin.json +0 -21
  7. package/.mcp.json +0 -9
  8. package/agents/brain-query.md +0 -80
  9. package/agents/framework-runner.md +0 -237
  10. package/agents/grading.md +0 -188
  11. package/agents/investor.md +0 -128
  12. package/agents/larry-extended.md +0 -135
  13. package/agents/opportunity-scanner.md +0 -91
  14. package/agents/persona-analyst.md +0 -132
  15. package/agents/research.md +0 -89
  16. package/agents/reverse-salient-agent.md +0 -27
  17. package/bin/mindrian-mcp-server.cjs +0 -182
  18. package/bin/mindrian-tools.cjs +0 -765
  19. package/commands/act.md +0 -433
  20. package/commands/admin.md +0 -404
  21. package/commands/analyze-needs.md +0 -36
  22. package/commands/analyze-systems.md +0 -33
  23. package/commands/analyze-timing.md +0 -36
  24. package/commands/auto-explore.md +0 -64
  25. package/commands/beautiful-question.md +0 -34
  26. package/commands/brain-derive.md +0 -78
  27. package/commands/build-knowledge.md +0 -36
  28. package/commands/build-thesis.md +0 -40
  29. package/commands/causal.md +0 -228
  30. package/commands/challenge-assumptions.md +0 -33
  31. package/commands/compare-ventures.md +0 -77
  32. package/commands/dashboard.md +0 -110
  33. package/commands/deep-grade.md +0 -76
  34. package/commands/diagnose.md +0 -52
  35. package/commands/diagnostics.md +0 -145
  36. package/commands/doctor.md +0 -151
  37. package/commands/dominant-designs.md +0 -34
  38. package/commands/explain-decision.md +0 -87
  39. package/commands/explore-domains.md +0 -36
  40. package/commands/explore-futures.md +0 -34
  41. package/commands/explore-trends.md +0 -36
  42. package/commands/export.md +0 -103
  43. package/commands/file-meeting.md +0 -724
  44. package/commands/find-analogies.md +0 -182
  45. package/commands/find-bottlenecks.md +0 -56
  46. package/commands/find-connections.md +0 -70
  47. package/commands/funding.md +0 -81
  48. package/commands/grade.md +0 -197
  49. package/commands/graph.md +0 -128
  50. package/commands/hat-briefing.md +0 -119
  51. package/commands/heal.md +0 -196
  52. package/commands/help.md +0 -399
  53. package/commands/hmi-status.md +0 -172
  54. package/commands/jtbd.md +0 -241
  55. package/commands/leadership.md +0 -73
  56. package/commands/lean-canvas.md +0 -34
  57. package/commands/macro-trends.md +0 -34
  58. package/commands/map-unknowns.md +0 -34
  59. package/commands/memory.md +0 -173
  60. package/commands/models.md +0 -175
  61. package/commands/mos-reason.md +0 -279
  62. package/commands/mullins.md +0 -114
  63. package/commands/new-project.md +0 -481
  64. package/commands/onboard.md +0 -434
  65. package/commands/operator.md +0 -149
  66. package/commands/opportunities.md +0 -144
  67. package/commands/organize.md +0 -497
  68. package/commands/persona.md +0 -192
  69. package/commands/pipeline.md +0 -106
  70. package/commands/present.md +0 -91
  71. package/commands/publish.md +0 -201
  72. package/commands/query.md +0 -124
  73. package/commands/radar.md +0 -72
  74. package/commands/reanalyze.md +0 -91
  75. package/commands/research.md +0 -190
  76. package/commands/room.md +0 -352
  77. package/commands/rooms.md +0 -598
  78. package/commands/root-cause.md +0 -34
  79. package/commands/rs-experts.md +0 -79
  80. package/commands/rs-explain.md +0 -94
  81. package/commands/rs-fetch.md +0 -88
  82. package/commands/rs-thesis.md +0 -79
  83. package/commands/scenario-plan.md +0 -34
  84. package/commands/scheduled-tasks.md +0 -285
  85. package/commands/score-innovation.md +0 -37
  86. package/commands/scout.md +0 -239
  87. package/commands/setup.md +0 -618
  88. package/commands/snapshot.md +0 -147
  89. package/commands/speakers.md +0 -84
  90. package/commands/splash.md +0 -28
  91. package/commands/status.md +0 -75
  92. package/commands/structure-argument.md +0 -36
  93. package/commands/suggest-next.md +0 -74
  94. package/commands/systems-thinking.md +0 -34
  95. package/commands/think-hats.md +0 -36
  96. package/commands/update.md +0 -181
  97. package/commands/user-needs.md +0 -34
  98. package/commands/validate.md +0 -34
  99. package/commands/value-proposition.md +0 -55
  100. package/commands/vault.md +0 -180
  101. package/commands/visualize.md +0 -52
  102. package/commands/whitespace.md +0 -501
  103. package/commands/wiki.md +0 -69
  104. package/hooks/hooks.json +0 -381
  105. package/hooks/run-hook.cmd +0 -64
  106. package/lib/__init__.py +0 -0
  107. package/lib/__pycache__/__init__.cpython-312.pyc +0 -0
  108. package/lib/agents/auto-explore-agent.cjs +0 -1043
  109. package/lib/agents/reverse-salient-agent.cjs +0 -679
  110. package/lib/agents/tension-hook-agent.cjs +0 -544
  111. package/lib/chat/chat-context.js +0 -185
  112. package/lib/chat/chat-panel.js +0 -721
  113. package/lib/chat/fabric-chat.cjs +0 -288
  114. package/lib/chat/generative-tools.js +0 -219
  115. package/lib/conversation/ROOM.md +0 -39
  116. package/lib/conversation/classifier-rules.json +0 -38
  117. package/lib/conversation/classifier.cjs +0 -264
  118. package/lib/conversation/operator.cjs +0 -287
  119. package/lib/copy/115-spec-strings.cjs +0 -55
  120. package/lib/core/__init__.py +0 -0
  121. package/lib/core/__nav-stub.cjs +0 -14
  122. package/lib/core/__pycache__/__init__.cpython-312.pyc +0 -0
  123. package/lib/core/__pycache__/rs-math.cpython-312.pyc +0 -0
  124. package/lib/core/__pycache__/rs_cache.cpython-312.pyc +0 -0
  125. package/lib/core/__pycache__/rs_corpus.cpython-312.pyc +0 -0
  126. package/lib/core/__pycache__/rs_hybrid.cpython-312.pyc +0 -0
  127. package/lib/core/__pycache__/rs_math.cpython-312.pyc +0 -0
  128. package/lib/core/__pycache__/rs_rooms.cpython-312.pyc +0 -0
  129. package/lib/core/artifact-id.cjs +0 -148
  130. package/lib/core/asset-ops.cjs +0 -151
  131. package/lib/core/auto-commit-throttle.cjs +0 -129
  132. package/lib/core/bearer-token.cjs +0 -199
  133. package/lib/core/brain-client.cjs +0 -865
  134. package/lib/core/brain-derivation-prompts.cjs +0 -326
  135. package/lib/core/brain-derivation-queue.cjs +0 -431
  136. package/lib/core/brain-derivation.cjs +0 -580
  137. package/lib/core/brain-md-schema.cjs +0 -528
  138. package/lib/core/brain-md-staleness.cjs +0 -357
  139. package/lib/core/brain-response-sanitize.cjs +0 -188
  140. package/lib/core/bridge-writer.cjs +0 -477
  141. package/lib/core/chat-context-builder.cjs +0 -253
  142. package/lib/core/cross-room-aggregator.cjs +0 -762
  143. package/lib/core/daily-briefing.cjs +0 -438
  144. package/lib/core/decision-capture.cjs +0 -618
  145. package/lib/core/deep-links.cjs +0 -82
  146. package/lib/core/dispatch-optimizer.cjs +0 -354
  147. package/lib/core/dual-path-detector.cjs +0 -84
  148. package/lib/core/dual-path-detector.test.cjs +0 -334
  149. package/lib/core/exports-log.cjs +0 -79
  150. package/lib/core/feynman-minto-invariants.cjs +0 -605
  151. package/lib/core/folder-memory-async.cjs +0 -338
  152. package/lib/core/folder-memory-shared.cjs +0 -890
  153. package/lib/core/folder-memory.cjs +0 -416
  154. package/lib/core/framework-chain-composer.cjs +0 -411
  155. package/lib/core/frontmatter-schemas.cjs +0 -330
  156. package/lib/core/git-ops.cjs +0 -141
  157. package/lib/core/graph-ops.cjs +0 -258
  158. package/lib/core/hat-persistence.cjs +0 -362
  159. package/lib/core/index.cjs +0 -60
  160. package/lib/core/integration-registry.cjs +0 -232
  161. package/lib/core/intelligence-cascade.cjs +0 -661
  162. package/lib/core/lazygraph-ops.cjs +0 -1057
  163. package/lib/core/lru-cache.cjs +0 -139
  164. package/lib/core/mcp-profiles.cjs +0 -182
  165. package/lib/core/meeting-ops.cjs +0 -54
  166. package/lib/core/memory-ops.cjs +0 -600
  167. package/lib/core/migrations/ROOM.md +0 -33
  168. package/lib/core/migrations/phase-109-nodes-provenance.cjs +0 -339
  169. package/lib/core/migrations/phase-109-session-focus.cjs +0 -99
  170. package/lib/core/model-profiles.cjs +0 -246
  171. package/lib/core/mullins-scaffold.cjs +0 -160
  172. package/lib/core/nav-dial.cjs +0 -316
  173. package/lib/core/navigation/ROOM.md +0 -15
  174. package/lib/core/navigation/explanation.cjs +0 -43
  175. package/lib/core/navigation/focus.cjs +0 -135
  176. package/lib/core/navigation/ingestion.cjs +0 -82
  177. package/lib/core/navigation/insights.cjs +0 -350
  178. package/lib/core/navigation/memory-events.cjs +0 -118
  179. package/lib/core/navigation/neighborhood.cjs +0 -78
  180. package/lib/core/navigation/packet.cjs +0 -182
  181. package/lib/core/navigation/room-home.cjs +0 -127
  182. package/lib/core/navigation/transitions.cjs +0 -82
  183. package/lib/core/navigation-engine-shared.cjs +0 -242
  184. package/lib/core/navigation-engine.cjs +0 -664
  185. package/lib/core/navigation.cjs +0 -60
  186. package/lib/core/nl-graph-queries.cjs +0 -164
  187. package/lib/core/offer-presenter.cjs +0 -406
  188. package/lib/core/opportunity-extractor.cjs +0 -183
  189. package/lib/core/opportunity-ops.cjs +0 -1371
  190. package/lib/core/persona-ops.cjs +0 -537
  191. package/lib/core/persona-taxonomy.cjs +0 -190
  192. package/lib/core/platform-gates.cjs +0 -120
  193. package/lib/core/platform.cjs +0 -257
  194. package/lib/core/proactive-intelligence.cjs +0 -528
  195. package/lib/core/problem-type-router.cjs +0 -315
  196. package/lib/core/reasoning-ops.cjs +0 -639
  197. package/lib/core/reverse-salient-persona-suffix.cjs +0 -115
  198. package/lib/core/room-classifier-strict-mode.cjs +0 -229
  199. package/lib/core/room-db.cjs +0 -127
  200. package/lib/core/room-ops-async.cjs +0 -92
  201. package/lib/core/room-ops-shared.cjs +0 -64
  202. package/lib/core/room-ops-sync.cjs +0 -70
  203. package/lib/core/room-ops.cjs +0 -32
  204. package/lib/core/room-type-detector.cjs +0 -386
  205. package/lib/core/rs-brain-substrate-prompts.cjs +0 -129
  206. package/lib/core/rs-brain-substrate.cjs +0 -570
  207. package/lib/core/rs-breakthrough-scorer.cjs +0 -255
  208. package/lib/core/rs-canon-violations.cjs +0 -82
  209. package/lib/core/rs-chain-feeder.cjs +0 -343
  210. package/lib/core/rs-commercial-assessor.cjs +0 -280
  211. package/lib/core/rs-differential-scorer.cjs +0 -376
  212. package/lib/core/rs-domain-analyzer.cjs +0 -385
  213. package/lib/core/rs-egress-prompts.cjs +0 -113
  214. package/lib/core/rs-egress-telemetry.cjs +0 -225
  215. package/lib/core/rs-egress-violations.cjs +0 -53
  216. package/lib/core/rs-expert-mapper.cjs +0 -467
  217. package/lib/core/rs-fetcher-academic.cjs +0 -697
  218. package/lib/core/rs-fetcher-experts.cjs +0 -314
  219. package/lib/core/rs-fetcher-industry.cjs +0 -731
  220. package/lib/core/rs-fetcher-patents.cjs +0 -564
  221. package/lib/core/rs-innovation-classifier.cjs +0 -194
  222. package/lib/core/rs-mind-map.cjs +0 -656
  223. package/lib/core/rs-neo4j-writer.cjs +0 -388
  224. package/lib/core/rs-nl-to-query.cjs +0 -425
  225. package/lib/core/rs-pinecone-bridge.cjs +0 -303
  226. package/lib/core/rs-preprocessor.cjs +0 -350
  227. package/lib/core/rs-query-matrix.cjs +0 -316
  228. package/lib/core/rs-query-to-text.cjs +0 -438
  229. package/lib/core/rs-sqlite-mirror.cjs +0 -443
  230. package/lib/core/rs-thesis-generator.cjs +0 -188
  231. package/lib/core/rs_cache.py +0 -479
  232. package/lib/core/rs_corpus.py +0 -468
  233. package/lib/core/rs_hybrid.py +0 -586
  234. package/lib/core/rs_math.py +0 -287
  235. package/lib/core/rs_rooms.py +0 -193
  236. package/lib/core/scheduled-scanner.cjs +0 -463
  237. package/lib/core/scratchpad-ops.cjs +0 -201
  238. package/lib/core/section-8-trace-schema.cjs +0 -138
  239. package/lib/core/section-registry.cjs +0 -111
  240. package/lib/core/session-state.cjs +0 -144
  241. package/lib/core/shallow-doc-parser.cjs +0 -174
  242. package/lib/core/shallow-doc-parser.test.cjs +0 -226
  243. package/lib/core/skill-activation-router.cjs +0 -284
  244. package/lib/core/state-ops.cjs +0 -46
  245. package/lib/core/statusline-cache.cjs +0 -266
  246. package/lib/core/token-estimator.cjs +0 -348
  247. package/lib/core/user-archetype.cjs +0 -239
  248. package/lib/core/user-md-ops.cjs +0 -524
  249. package/lib/core/visual-ops.cjs +0 -624
  250. package/lib/core/write-lock.cjs +0 -149
  251. package/lib/graph/canvas-graph.js +0 -467
  252. package/lib/graph/constellation-config.cjs +0 -299
  253. package/lib/graph/graph-detail-panel.js +0 -165
  254. package/lib/hmi/ROOM.md +0 -47
  255. package/lib/hmi/across-session-memory.cjs +0 -604
  256. package/lib/hmi/cross-room-memory.cjs +0 -575
  257. package/lib/hmi/decoy-tier.cjs +0 -395
  258. package/lib/hmi/jtbd-classifier.cjs +0 -219
  259. package/lib/hmi/jtbd-state.cjs +0 -199
  260. package/lib/hmi/jtbd-taxonomy.json +0 -392
  261. package/lib/hmi/selector-dispatcher.cjs +0 -546
  262. package/lib/hmi/selector-telemetry.cjs +0 -263
  263. package/lib/hmi/shape-f0-renderer.cjs +0 -139
  264. package/lib/hmi/shape-f1-fallback.cjs +0 -80
  265. package/lib/hmi/shape-f1-renderer.cjs +0 -138
  266. package/lib/hmi/shape-f2-renderer.cjs +0 -132
  267. package/lib/hmi/shape-f3-renderer.cjs +0 -66
  268. package/lib/hmi/shape-f4-renderer.cjs +0 -72
  269. package/lib/hmi/shape-f5-renderer.cjs +0 -155
  270. package/lib/hmi/shape-f6-plan-review-renderer.cjs +0 -312
  271. package/lib/hmi/shape-f6-renderer.cjs +0 -144
  272. package/lib/hmi/shape-g-renderer.cjs +0 -219
  273. package/lib/hmi/shape-h-renderer.cjs +0 -222
  274. package/lib/hmi/tier-check.cjs +0 -63
  275. package/lib/import/PRECONDITIONS.md +0 -41
  276. package/lib/import/branding.cjs +0 -210
  277. package/lib/import/branding.test.cjs +0 -235
  278. package/lib/import/classifications-sync.cjs +0 -104
  279. package/lib/import/classifications-sync.test.cjs +0 -129
  280. package/lib/import/enricher.cjs +0 -296
  281. package/lib/import/enricher.test.cjs +0 -273
  282. package/lib/import/integration.test.cjs +0 -376
  283. package/lib/import/manifest.cjs +0 -129
  284. package/lib/import/manifest.schema.json +0 -185
  285. package/lib/import/manifest.test.cjs +0 -123
  286. package/lib/import/meeting-detector.cjs +0 -92
  287. package/lib/import/meeting-detector.test.cjs +0 -100
  288. package/lib/import/person-detector.cjs +0 -229
  289. package/lib/import/person-detector.test.cjs +0 -149
  290. package/lib/import/report.cjs +0 -186
  291. package/lib/import/report.test.cjs +0 -186
  292. package/lib/import/room-md-scaffolder.cjs +0 -49
  293. package/lib/import/router.cjs +0 -224
  294. package/lib/import/router.test.cjs +0 -356
  295. package/lib/import/run-all-tests.cjs +0 -36
  296. package/lib/import/smoke-test.cjs +0 -213
  297. package/lib/import/smoke-test.test.cjs +0 -148
  298. package/lib/import/test-fixtures/collision-vault/preexisting-room/STATE.md +0 -8
  299. package/lib/import/test-fixtures/collision-vault/preexisting-room/problem-definition/onboarding/onboarding.md +0 -7
  300. package/lib/import/test-fixtures/collision-vault/source/onboarding.md +0 -5
  301. package/lib/import/test-fixtures/obsidian-vault/.obsidian/workspace.json +0 -1
  302. package/lib/import/test-fixtures/obsidian-vault/notes/with-wikilinks.md +0 -4
  303. package/lib/import/test-fixtures/tiny-vault/notes/2026-01-15-team-sync.md +0 -9
  304. package/lib/import/test-fixtures/tiny-vault/notes/empty.md +0 -3
  305. package/lib/import/test-fixtures/tiny-vault/notes/onboarding.md +0 -5
  306. package/lib/import/test-fixtures/tiny-vault/notes/pricing.md +0 -5
  307. package/lib/import/test-fixtures/tiny-vault/notes/random.md +0 -4
  308. package/lib/import/undo.test.cjs +0 -199
  309. package/lib/import/vault-scanner.cjs +0 -105
  310. package/lib/import/vault-scanner.test.cjs +0 -67
  311. package/lib/mcp/app-html/dashboard.html +0 -316
  312. package/lib/mcp/app-html/graph.html +0 -428
  313. package/lib/mcp/app-html/mindrian-platform.html +0 -1841
  314. package/lib/mcp/app-html/wiki.html +0 -383
  315. package/lib/mcp/app-views.cjs +0 -322
  316. package/lib/mcp/brain-router.cjs +0 -418
  317. package/lib/mcp/capability-registry.cjs +0 -62
  318. package/lib/mcp/larry-context.cjs +0 -46
  319. package/lib/mcp/larry-server-instructions.md +0 -114
  320. package/lib/mcp/pipeline-state.cjs +0 -275
  321. package/lib/mcp/prompts.cjs +0 -302
  322. package/lib/mcp/resources.cjs +0 -227
  323. package/lib/mcp/session-catchup.cjs +0 -327
  324. package/lib/mcp/surface-detect.cjs +0 -75
  325. package/lib/mcp/tool-router.cjs +0 -1034
  326. package/lib/memory/aaak-compress.cjs +0 -403
  327. package/lib/memory/aaak-compress.test.cjs +0 -288
  328. package/lib/memory/async-artifact-auto-commit.test.cjs +0 -223
  329. package/lib/memory/bearer-token.test.cjs +0 -315
  330. package/lib/memory/brain-cache-lru.test.cjs +0 -259
  331. package/lib/memory/brain-client-query-shape.test.cjs +0 -160
  332. package/lib/memory/brain-derivation-graceful-degradation.test.cjs +0 -1019
  333. package/lib/memory/brain-derivation-queue.test.cjs +0 -539
  334. package/lib/memory/brain-derivation.test.cjs +0 -634
  335. package/lib/memory/brain-derive-command.test.cjs +0 -534
  336. package/lib/memory/brain-md-invariants-validator.test.cjs +0 -704
  337. package/lib/memory/brain-md-schema.test.cjs +0 -467
  338. package/lib/memory/brain-md-staleness.test.cjs +0 -525
  339. package/lib/memory/brain-server-resolution.test.cjs +0 -314
  340. package/lib/memory/chat-context.test.cjs +0 -128
  341. package/lib/memory/cross-room-aggregator.test.cjs +0 -909
  342. package/lib/memory/dashboard-server.test.cjs +0 -256
  343. package/lib/memory/debouncer-drain-at-prompt.test.cjs +0 -389
  344. package/lib/memory/decision-capture.test.cjs +0 -632
  345. package/lib/memory/decision-capture.worker.cjs +0 -70
  346. package/lib/memory/explain-decision-command.test.cjs +0 -521
  347. package/lib/memory/explain-decision-footer.test.cjs +0 -316
  348. package/lib/memory/explored-materials-store.cjs +0 -392
  349. package/lib/memory/feynman-minto-guardian.test.cjs +0 -736
  350. package/lib/memory/feynman-minto-invariants.test.cjs +0 -511
  351. package/lib/memory/feynman-prompts-drift.test.cjs +0 -144
  352. package/lib/memory/feynman-prompts.cjs +0 -151
  353. package/lib/memory/feynman-prompts.test.cjs +0 -96
  354. package/lib/memory/folder-memory-quadruple.test.cjs +0 -548
  355. package/lib/memory/folder-memory.test.cjs +0 -503
  356. package/lib/memory/framework-chain-composer.test.cjs +0 -515
  357. package/lib/memory/frontmatter-schema-validator.test.cjs +0 -290
  358. package/lib/memory/heal-command.test.cjs +0 -604
  359. package/lib/memory/index-artifact-transaction.test.cjs +0 -333
  360. package/lib/memory/lazygraph-rs-discoveries-view.test.cjs +0 -122
  361. package/lib/memory/mcp-input-validation.test.cjs +0 -240
  362. package/lib/memory/mcp-server-brain-deps.test.cjs +0 -270
  363. package/lib/memory/mcp-stack-fallback.test.cjs +0 -433
  364. package/lib/memory/minto-debouncer.test.cjs +0 -407
  365. package/lib/memory/minto-debouncer.worker.cjs +0 -46
  366. package/lib/memory/minto-migration-v88.test.cjs +0 -265
  367. package/lib/memory/minto-schema-v88.test.cjs +0 -390
  368. package/lib/memory/mos-status-renderer.test.cjs +0 -631
  369. package/lib/memory/narrative-schema.cjs +0 -376
  370. package/lib/memory/narrative-schema.test.cjs +0 -209
  371. package/lib/memory/nav-dial.test.cjs +0 -414
  372. package/lib/memory/navigation-engine-core.test.cjs +0 -722
  373. package/lib/memory/navigation-invariants.test.cjs +0 -483
  374. package/lib/memory/offer-presenter.test.cjs +0 -554
  375. package/lib/memory/on-stop-snapshot.test.cjs +0 -404
  376. package/lib/memory/pending-tension-store.cjs +0 -373
  377. package/lib/memory/post-compact-reinjection.test.cjs +0 -854
  378. package/lib/memory/post-write-triple.test.cjs +0 -317
  379. package/lib/memory/pre-compact-snapshot.test.cjs +0 -495
  380. package/lib/memory/problem-type-router.test.cjs +0 -656
  381. package/lib/memory/query-efficiency-telemetry.test.cjs +0 -370
  382. package/lib/memory/recompile-room-references.test.cjs +0 -392
  383. package/lib/memory/recompile-room-references.worker.cjs +0 -42
  384. package/lib/memory/record-decision-dual-write.test.cjs +0 -454
  385. package/lib/memory/room-classifier-strict-mode.test.cjs +0 -417
  386. package/lib/memory/room-minto-hook.test.cjs +0 -398
  387. package/lib/memory/rs-discovery-engine.test.cjs +0 -323
  388. package/lib/memory/run-feynman-tests.cjs +0 -1239
  389. package/lib/memory/security-trifecta.test.cjs +0 -312
  390. package/lib/memory/session-start-brain-staleness.test.cjs +0 -363
  391. package/lib/memory/session-start-triple-injection.test.cjs +0 -514
  392. package/lib/memory/sessionstart-banner-formatter.cjs +0 -318
  393. package/lib/memory/sessionstart-minto-banner.test.cjs +0 -373
  394. package/lib/memory/skill-activation-router.test.cjs +0 -419
  395. package/lib/memory/stamp-artifact-write.test.cjs +0 -304
  396. package/lib/memory/statusline-active-room.test.cjs +0 -315
  397. package/lib/memory/statusline-minto-segment.test.cjs +0 -292
  398. package/lib/memory/sync-async-entry-points.test.cjs +0 -204
  399. package/lib/memory/test-bridge-writer-enhanced.cjs +0 -452
  400. package/lib/memory/test-rs-brain-substrate-shape.cjs +0 -529
  401. package/lib/memory/test-rs-brain-substrate.cjs +0 -636
  402. package/lib/memory/test-rs-breakthrough-scorer.cjs +0 -375
  403. package/lib/memory/test-rs-canon-violations.cjs +0 -218
  404. package/lib/memory/test-rs-chain-feeder-core.cjs +0 -344
  405. package/lib/memory/test-rs-chain-feeder-skill-spawn.cjs +0 -297
  406. package/lib/memory/test-rs-commercial-assessor.cjs +0 -385
  407. package/lib/memory/test-rs-differential-scorer.cjs +0 -480
  408. package/lib/memory/test-rs-discovery-engine.cjs +0 -603
  409. package/lib/memory/test-rs-domain-analyzer.cjs +0 -492
  410. package/lib/memory/test-rs-egress-primitives.cjs +0 -420
  411. package/lib/memory/test-rs-expert-mapper.cjs +0 -547
  412. package/lib/memory/test-rs-explain-command.cjs +0 -443
  413. package/lib/memory/test-rs-fetcher-academic.cjs +0 -848
  414. package/lib/memory/test-rs-fetcher-experts.cjs +0 -496
  415. package/lib/memory/test-rs-fetcher-industry.cjs +0 -702
  416. package/lib/memory/test-rs-fetcher-patents.cjs +0 -674
  417. package/lib/memory/test-rs-innovation-classifier.cjs +0 -301
  418. package/lib/memory/test-rs-mind-map.cjs +0 -646
  419. package/lib/memory/test-rs-neo4j-writer.cjs +0 -518
  420. package/lib/memory/test-rs-nl-to-query.cjs +0 -449
  421. package/lib/memory/test-rs-pinecone-bridge.cjs +0 -277
  422. package/lib/memory/test-rs-preprocessor.cjs +0 -433
  423. package/lib/memory/test-rs-query-matrix.cjs +0 -391
  424. package/lib/memory/test-rs-query-to-text.cjs +0 -551
  425. package/lib/memory/test-rs-sqlite-mirror.cjs +0 -649
  426. package/lib/memory/test-rs-thesis-generator.cjs +0 -360
  427. package/lib/memory/triple-context-formatter.cjs +0 -473
  428. package/lib/memory/triple-context-formatter.test.cjs +0 -442
  429. package/lib/memory/user-md-persona.test.cjs +0 -565
  430. package/lib/memory/userpromptsubmit-integration.test.cjs +0 -690
  431. package/lib/memory/validators/README.md +0 -157
  432. package/lib/memory/validators/brain-md-invariants.cjs +0 -475
  433. package/lib/memory/validators/brain-substrate-invariants.cjs +0 -285
  434. package/lib/memory/validators/external-academic-invariants.cjs +0 -249
  435. package/lib/memory/validators/external-industry-invariants.cjs +0 -271
  436. package/lib/memory/validators/external-patents-invariants.cjs +0 -266
  437. package/lib/memory/validators/minto-invariants.cjs +0 -62
  438. package/lib/memory/validators/navigation-invariants.cjs +0 -340
  439. package/lib/memory/validators/queue-health.cjs +0 -95
  440. package/lib/memory/validators/snapshot-integrity.cjs +0 -129
  441. package/lib/memory/validators/stale-lifecycle.cjs +0 -116
  442. package/lib/memory/vault-section-minto-generator-atomic.test.cjs +0 -556
  443. package/lib/memory/vault-section-minto-generator-atomic.worker.cjs +0 -73
  444. package/lib/memory/write-lock-atomic.test.cjs +0 -137
  445. package/lib/memory/write-lock-atomic.worker.cjs +0 -55
  446. package/lib/parity/check-parity.cjs +0 -83
  447. package/lib/presentation/presentation-server.cjs +0 -101
  448. package/lib/presentation/presentation-watcher.cjs +0 -123
  449. package/lib/quickview/hub-server.cjs +0 -719
  450. package/lib/quickview/server.cjs +0 -533
  451. package/lib/render/JTBD-PALETTES.md +0 -145
  452. package/lib/render/ROOM.md +0 -59
  453. package/lib/render/render-v2.cjs +0 -486
  454. package/lib/render/render-v2.test.cjs +0 -267
  455. package/lib/render/render.cjs +0 -65
  456. package/lib/state/ROOM.md +0 -46
  457. package/lib/state/state-md-parser.cjs +0 -215
  458. package/lib/statusline/ROOM.md +0 -38
  459. package/lib/statusline/banner-suppression.cjs +0 -50
  460. package/lib/statusline/surface-detect.cjs +0 -85
  461. package/lib/update-bootstrap.sh.template +0 -145
  462. package/lib/vault/frontmatter-schema.cjs +0 -297
  463. package/lib/vault/room-scanner.cjs +0 -352
  464. package/lib/vault/wikilink-builder.cjs +0 -231
  465. package/lib/vault/wikilink-builder.test.cjs +0 -182
  466. package/lib/wiki/graph-links.cjs +0 -281
  467. package/lib/wiki/page-renderer.cjs +0 -229
  468. package/lib/wiki/wiki-chat.cjs +0 -81
  469. package/lib/wiki/wiki-layout.cjs +0 -1459
  470. package/lib/wiki/wiki-search.cjs +0 -142
  471. package/lib/wiki/wiki-server.cjs +0 -678
  472. package/lib/wiki/wiki-watcher.cjs +0 -105
  473. package/pipelines/analogy/01-decompose.md +0 -80
  474. package/pipelines/analogy/02-abstract.md +0 -87
  475. package/pipelines/analogy/03-search.md +0 -135
  476. package/pipelines/analogy/04-transfer.md +0 -101
  477. package/pipelines/analogy/05-validate.md +0 -106
  478. package/pipelines/analogy/CHAIN.md +0 -56
  479. package/pipelines/discovery/01-explore-domains.md +0 -44
  480. package/pipelines/discovery/02-think-hats.md +0 -50
  481. package/pipelines/discovery/03-analyze-needs.md +0 -54
  482. package/pipelines/discovery/CHAIN.md +0 -37
  483. package/pipelines/thesis/01-structure-argument.md +0 -45
  484. package/pipelines/thesis/02-challenge-assumptions.md +0 -48
  485. package/pipelines/thesis/03-build-thesis.md +0 -54
  486. package/pipelines/thesis/CHAIN.md +0 -37
  487. package/references/brain/causal-directives.md +0 -91
  488. package/references/brain/causal-enrichment.cypher +0 -165
  489. package/references/brain/command-triggers-schema.md +0 -226
  490. package/references/brain/graph-architecture.md +0 -317
  491. package/references/brain/query-patterns.md +0 -460
  492. package/references/brain/room-hierarchy-schema.md +0 -218
  493. package/references/brain/schema.md +0 -76
  494. package/references/capability-radar/capabilities-index.md +0 -241
  495. package/references/capability-radar/changelog-cache.md +0 -81
  496. package/references/causal/causal-schema.md +0 -103
  497. package/references/design/email-template-standard.md +0 -155
  498. package/references/design/graph-visualization-standard.md +0 -178
  499. package/references/document-generation.md +0 -179
  500. package/references/hsi/HSI-TOOLS-REFERENCE.md +0 -222
  501. package/references/import-config.md +0 -141
  502. package/references/integrations/detection-patterns.md +0 -101
  503. package/references/meeting/artifact-template.md +0 -377
  504. package/references/meeting/cross-meeting-intelligence.md +0 -216
  505. package/references/meeting/cross-relationship-patterns.md +0 -202
  506. package/references/meeting/live-join-interface.md +0 -244
  507. package/references/meeting/section-mapping.md +0 -192
  508. package/references/meeting/segment-classification.md +0 -258
  509. package/references/meeting/speaker-profile-template.md +0 -219
  510. package/references/meeting/summary-template.md +0 -348
  511. package/references/meeting/transcript-patterns.md +0 -226
  512. package/references/methodology/analyze-needs.md +0 -135
  513. package/references/methodology/analyze-systems.md +0 -121
  514. package/references/methodology/analyze-timing.md +0 -149
  515. package/references/methodology/beautiful-question.md +0 -109
  516. package/references/methodology/build-knowledge.md +0 -161
  517. package/references/methodology/build-thesis.md +0 -237
  518. package/references/methodology/challenge-assumptions.md +0 -127
  519. package/references/methodology/diagnose.md +0 -169
  520. package/references/methodology/dominant-designs.md +0 -212
  521. package/references/methodology/explore-domains.md +0 -147
  522. package/references/methodology/explore-futures.md +0 -163
  523. package/references/methodology/explore-trends.md +0 -129
  524. package/references/methodology/find-bottlenecks.md +0 -131
  525. package/references/methodology/grade.md +0 -211
  526. package/references/methodology/index.md +0 -97
  527. package/references/methodology/leadership.md +0 -200
  528. package/references/methodology/lean-canvas.md +0 -116
  529. package/references/methodology/macro-trends.md +0 -192
  530. package/references/methodology/map-unknowns.md +0 -137
  531. package/references/methodology/mullins-7-domains.md +0 -104
  532. package/references/methodology/problem-types.md +0 -65
  533. package/references/methodology/root-cause.md +0 -178
  534. package/references/methodology/sapphire-encoding.md +0 -355
  535. package/references/methodology/scenario-plan.md +0 -178
  536. package/references/methodology/score-innovation.md +0 -154
  537. package/references/methodology/structure-argument.md +0 -158
  538. package/references/methodology/systems-thinking.md +0 -159
  539. package/references/methodology/think-hats.md +0 -147
  540. package/references/methodology/triz-matrix.json +0 -751
  541. package/references/methodology/triz-principles.md +0 -501
  542. package/references/methodology/user-needs.md +0 -199
  543. package/references/methodology/validate.md +0 -163
  544. package/references/methodology/value-proposition.md +0 -244
  545. package/references/opportunities/funding-lifecycle.md +0 -103
  546. package/references/opportunities/grant-api-patterns.md +0 -99
  547. package/references/opportunities/opportunity-template.md +0 -84
  548. package/references/personality/assessment-philosophy.md +0 -72
  549. package/references/personality/lexicon.md +0 -100
  550. package/references/personality/persona-chains.md +0 -56
  551. package/references/personality/pws-lexicon-full.md +0 -499
  552. package/references/personality/voice-dna.md +0 -156
  553. package/references/personas/hat-perspectives.md +0 -76
  554. package/references/personas/persona-template.md +0 -63
  555. package/references/pipeline/act-output-contract.md +0 -88
  556. package/references/pipeline/chains-index.md +0 -39
  557. package/references/pws-profile-generation.md +0 -79
  558. package/references/reasoning/reasoning-schema.md +0 -143
  559. package/references/reasoning/reasoning-template.md +0 -68
  560. package/references/reasoning/run-template.md +0 -38
  561. package/references/research/RESEARCH_14_CLAUDE_CODE_SOURCE_ARCHITECTURE.md +0 -209
  562. package/references/research/RESEARCH_15_V1.8_OPTIMIZATION_JTBD.md +0 -375
  563. package/references/research/RESEARCH_16_NATIVE_FIRST_PLUGIN_ARCHITECTURE.md +0 -575
  564. package/references/research/RESEARCH_17_MCP_UI_FRAMEWORKS.md +0 -272
  565. package/references/taxonomy/TAXONOMY.md +0 -192
  566. package/references/templates/MINTO.md +0 -36
  567. package/references/user-research/2026-04-05-leah-lawrence-session.md +0 -202
  568. package/references/vault-kit/README.md +0 -35
  569. package/references/vault-kit/app.json +0 -12
  570. package/references/vault-kit/appearance.json +0 -12
  571. package/references/vault-kit/graph.json +0 -35
  572. package/references/vault-kit/snippets/mindrian-destijl.css +0 -297
  573. package/references/vault-kit/templates/new-artifact.md +0 -37
  574. package/references/vault-kit/templates/new-meeting-note.md +0 -35
  575. package/references/vault-kit/templates/new-team-profile.md +0 -29
  576. package/references/vault-kit/templates/new-xref.md +0 -35
  577. package/references/visual/symbol-system.md +0 -151
  578. package/skills/MOSDeckEngine/SKILL.md +0 -325
  579. package/skills/brain-connector/SKILL.md +0 -114
  580. package/skills/context-engine/SKILL.md +0 -147
  581. package/skills/conversation-mode/SKILL.md +0 -102
  582. package/skills/larry-personality/SKILL.md +0 -219
  583. package/skills/larry-personality/framework-chains.md +0 -92
  584. package/skills/larry-personality/mode-engine.md +0 -185
  585. package/skills/mullins-scaffold/SKILL.md +0 -61
  586. package/skills/mullins-scaffold/scaffold.json +0 -146
  587. package/skills/pws-methodology/SKILL.md +0 -49
  588. package/skills/room-passive/SKILL.md +0 -165
  589. package/skills/room-proactive/SKILL.md +0 -250
  590. package/skills/ui-system/SKILL.md +0 -277
@@ -1,865 +0,0 @@
1
- 'use strict';
2
-
3
- /**
4
- * Brain HTTP Client — calls mindrian-brain.onrender.com
5
- *
6
- * Replaces direct MCP tool calls (mcp__neo4j-brain__*, mcp__pinecone-brain__*)
7
- * with a single HTTP API that handles Neo4j + Pinecone behind one key.
8
- *
9
- * Falls back gracefully:
10
- * 1. If MINDRIAN_BRAIN_KEY is set → calls Brain API
11
- * 2. If Brain API returns Pinecone quota error → retries with Neo4j-only
12
- * 3. If no key → returns null (Tier 0, no Brain)
13
- *
14
- * Usage in commands/skills:
15
- * const brain = require('./brain-client.cjs');
16
- * const result = await brain.query('MATCH (f:Framework) RETURN f.name LIMIT 5');
17
- * const result = await brain.search('innovation framework');
18
- * const schema = await brain.schema();
19
- */
20
-
21
- const BRAIN_URL = process.env.MINDRIAN_BRAIN_URL || 'https://mindrian-brain.onrender.com';
22
-
23
- // Phase 87-07 (CASCADE-06): Brain session cache with 5-minute TTL.
24
- // Every callTool() previously re-ran the `initialize` handshake (~1 network
25
- // round-trip). With a long-lived MCP server this is wasted work -- sessions
26
- // live longer than the ~60s transport timeout. Cache the initialized
27
- // sessionId (keyed by api-key-hash) for 5 minutes.
28
- //
29
- // R-87-07-RACE (audit): two concurrent callTool() invocations with the same
30
- // api_key previously both saw a cache miss, both initialized, and the second
31
- // overwrote the first -- one of the two initialize handshakes was wasted.
32
- // Fix: cache the init *Promise*, not the resolved session. The first caller
33
- // stores { promise: initSession(apiKey), expiresAt }; concurrent callers
34
- // within the TTL `await entry.promise`. On rejection we remove the entry so
35
- // the next caller re-initializes fresh.
36
- //
37
- // Hash: sha256 truncated to 16 hex chars (64 bits of key space, zero realistic
38
- // collision). A cheaper non-crypto hash was considered but its narrower int
39
- // space has non-zero collision probability once the design extends across
40
- // users; sha256 is effectively free at these volumes and eliminates the
41
- // concern entirely (R-87-07-RACE).
42
- const crypto = require('node:crypto');
43
- const SESSION_TTL_MS = 5 * 60 * 1000; // 5 minutes
44
- /** @type {Map<string, {promise: Promise<string>, expiresAt: number}>} */
45
- const sessionCache = new Map();
46
-
47
- function _hashKey(key) {
48
- return crypto.createHash('sha256').update(String(key)).digest('hex').slice(0, 16);
49
- }
50
-
51
- /**
52
- * SEC-01: Sanitize any user-origin string before interpolation into a
53
- * Cypher query. Whitelist from 87-CONTEXT.md lines 121-127:
54
- * [a-zA-Z0-9 ._-]
55
- * Every other char (including `"`, `'`, backtick, newline, `{`, `}`, `$`,
56
- * `\`, `;`, `/`, `*`) is stripped. Null/undefined return ''. Non-strings
57
- * are coerced via String() defensively so the caller never crashes.
58
- *
59
- * This replaces the legacy single-quote-escape pattern that only
60
- * escaped one metacharacter (double-quote) and was trivially bypassable
61
- * via backticks, newlines, `${...}` expansions, or Cypher comments.
62
- *
63
- * @param {*} value
64
- * @returns {string}
65
- */
66
- function sanitizeCypherInput(value) {
67
- if (value === null || value === undefined) return '';
68
- if (typeof value !== 'string') {
69
- try { value = String(value); } catch (_e) { return ''; }
70
- }
71
- return value.replace(/[^a-zA-Z0-9 ._-]/g, '');
72
- }
73
-
74
- /**
75
- * SEC-02: Refuse to load a Brain API key from a .env file whose permissions
76
- * expose it to group or world readers. Unix semantics only -- on Windows
77
- * POSIX mode bits are not meaningful for NTFS ACLs, so we return true and
78
- * warn once per process.
79
- *
80
- * mode & 0o077 !== 0 => any group/world bit is set => reject
81
- * mode 0o600 (-rw-------) and 0o400 (-r--------) pass; 0o644, 0o664 fail.
82
- *
83
- * On stat failure we return false (no key beats a key we cannot verify).
84
- *
85
- * @param {string} envPath
86
- * @returns {boolean}
87
- */
88
- function checkFilePermissions(envPath) {
89
- try {
90
- const fs = require('fs');
91
- if (process.platform === 'win32') {
92
- if (!checkFilePermissions._warned) {
93
- process.stderr.write(
94
- '[mindrian-os] Note: API key file permission check is Linux/macOS only; '
95
- + 'on Windows rely on NTFS ACLs.\n'
96
- );
97
- checkFilePermissions._warned = true;
98
- }
99
- return true;
100
- }
101
- const stat = fs.statSync(envPath);
102
- if ((stat.mode & 0o077) !== 0) {
103
- process.stderr.write(
104
- `[mindrian-os] Refusing to load API key from ${envPath}: `
105
- + `permissions too open (must be 0600). chmod 600 ${envPath}\n`
106
- );
107
- return false;
108
- }
109
- return true;
110
- } catch (_e) {
111
- return false;
112
- }
113
- }
114
- checkFilePermissions._warned = false;
115
-
116
- /**
117
- * Get the Brain API key from environment.
118
- * Checks: MINDRIAN_BRAIN_KEY, then falls back to reading .env in CWD.
119
- * Every .env candidate path is gated by checkFilePermissions (SEC-02).
120
- */
121
- function getApiKey() {
122
- if (process.env.MINDRIAN_BRAIN_KEY) {
123
- return process.env.MINDRIAN_BRAIN_KEY;
124
- }
125
- // Try reading .env from CWD
126
- try {
127
- const fs = require('fs');
128
- const path = require('path');
129
- const envPath = path.join(process.cwd(), '.env');
130
- if (fs.existsSync(envPath) && checkFilePermissions(envPath)) {
131
- const content = fs.readFileSync(envPath, 'utf8');
132
- const match = content.match(/MINDRIAN_BRAIN_KEY=(.+)/);
133
- if (match) return match[1].trim();
134
- }
135
- } catch (e) {}
136
- // Fallback: try reading ~/.mindrian.env (global backup)
137
- try {
138
- const fs = require('fs');
139
- const path = require('path');
140
- const globalEnvPath = path.join(require('os').homedir(), '.mindrian.env');
141
- if (fs.existsSync(globalEnvPath) && checkFilePermissions(globalEnvPath)) {
142
- const content = fs.readFileSync(globalEnvPath, 'utf8');
143
- const match = content.match(/MINDRIAN_BRAIN_KEY=(.+)/);
144
- if (match) return match[1].trim();
145
- }
146
- } catch (e) {}
147
- return null;
148
- }
149
-
150
- /**
151
- * Check if Brain is available (key exists).
152
- */
153
- function isAvailable() {
154
- return !!getApiKey();
155
- }
156
-
157
- /**
158
- * Phase 87-07: ensure we have a valid initialized Brain session for the given
159
- * api key, reusing the cached one if non-expired. Uses the pending-promise
160
- * pattern so concurrent callers share a single in-flight init (R-87-07-RACE).
161
- *
162
- * Returns the resolved session marker (an opaque string -- the Brain Streamable
163
- * HTTP transport does not require us to echo a sessionId on subsequent requests
164
- * inside the same cache window, but awaiting this promise proves the key is
165
- * valid against the Brain endpoint exactly once per TTL window).
166
- *
167
- * On any init rejection (network error, 401, etc.) the cache entry is removed
168
- * in the .catch() tail so the next caller retries fresh rather than inheriting
169
- * a poisoned promise.
170
- *
171
- * Sentinel `{ error: 'invalid_key' }` is returned *through* the promise (not
172
- * thrown) so callers treat 401 identically to the pre-cache flow.
173
- *
174
- * @param {string} apiKey
175
- * @returns {Promise<string|{error:string,message:string}|null>}
176
- */
177
- async function _ensureSession(apiKey) {
178
- const keyHash = _hashKey(apiKey);
179
- const cached = sessionCache.get(keyHash);
180
- if (cached && cached.expiresAt > Date.now()) {
181
- // Cache hit. Works whether the promise is still pending (concurrent init
182
- // in flight) or already resolved (TTL reuse). Awaiting a resolved promise
183
- // is a microtask no-op, so the fast path stays fast.
184
- return cached.promise;
185
- }
186
- // Cache miss. Build the promise FIRST, install it in the cache BEFORE the
187
- // first real await, so concurrent callers within the same event-loop tick
188
- // see the same in-flight promise (R-87-07-RACE pending-promise pattern).
189
- const promise = (async () => {
190
- const initRes = await fetch(`${BRAIN_URL}/mcp`, {
191
- method: 'POST',
192
- headers: {
193
- 'Content-Type': 'application/json',
194
- 'Accept': 'application/json, text/event-stream',
195
- 'Authorization': `Bearer ${apiKey}`,
196
- },
197
- body: JSON.stringify({
198
- jsonrpc: '2.0',
199
- id: 1,
200
- method: 'initialize',
201
- params: {
202
- protocolVersion: '2024-11-05',
203
- capabilities: {},
204
- clientInfo: { name: 'mindrian-cli', version: '1.0.0' },
205
- },
206
- }),
207
- });
208
- if (!initRes.ok) {
209
- if (initRes.status === 401) {
210
- return { error: 'invalid_key', message: 'Brain API key is invalid.' };
211
- }
212
- // Any other non-OK status becomes a throw so the cache entry is purged
213
- // by the .catch() below and the next caller retries.
214
- throw new Error(`Brain init HTTP ${initRes.status}`);
215
- }
216
- // Opaque session marker. Subsequent tools/call requests don't need to
217
- // echo this back -- the transport is stateless at the HTTP level. What
218
- // matters is that we validated the key is live within this TTL window.
219
- return 'validated-' + Date.now();
220
- })();
221
- sessionCache.set(keyHash, { promise, expiresAt: Date.now() + SESSION_TTL_MS });
222
- // On reject, purge the entry so the next caller initializes fresh. Swallow
223
- // here (we re-throw in the awaiter below) so Node doesn't see an
224
- // unhandledRejection on the cache handle itself.
225
- promise.catch(() => { sessionCache.delete(keyHash); });
226
- return promise;
227
- }
228
-
229
- /**
230
- * Call a Brain MCP tool via HTTP.
231
- * @param {string} toolName - e.g., 'brain_query', 'brain_search', 'brain_schema'
232
- * @param {object} args - tool arguments
233
- * @returns {object|null} - result or null if unavailable
234
- */
235
- async function callTool(toolName, args) {
236
- const key = getApiKey();
237
- if (!key) return null;
238
-
239
- try {
240
- // Phase 87-07: reuse cached Brain session (5-min TTL) instead of
241
- // re-running initialize on every callTool. Concurrent callers share
242
- // the in-flight promise via the pending-promise pattern.
243
- const session = await _ensureSession(key);
244
- if (session && typeof session === 'object' && session.error === 'invalid_key') {
245
- return session;
246
- }
247
- if (!session) return null;
248
-
249
- // Call the tool
250
- const toolRes = await fetch(`${BRAIN_URL}/mcp`, {
251
- method: 'POST',
252
- headers: {
253
- 'Content-Type': 'application/json',
254
- 'Accept': 'application/json, text/event-stream',
255
- 'Authorization': `Bearer ${key}`,
256
- },
257
- body: JSON.stringify({
258
- jsonrpc: '2.0',
259
- id: 2,
260
- method: 'tools/call',
261
- params: { name: toolName, arguments: args },
262
- }),
263
- });
264
-
265
- if (!toolRes.ok) return null;
266
-
267
- const text = await toolRes.text();
268
- // Parse SSE response
269
- const dataLine = text.split('\n').find(l => l.startsWith('data: '));
270
- if (!dataLine) return null;
271
-
272
- const parsed = JSON.parse(dataLine.slice(6));
273
- if (parsed.result && parsed.result.content) {
274
- const textContent = parsed.result.content.find(c => c.type === 'text');
275
- if (textContent) {
276
- try {
277
- return JSON.parse(textContent.text);
278
- } catch (e) {
279
- return { text: textContent.text };
280
- }
281
- }
282
- }
283
- return parsed.result || null;
284
- } catch (err) {
285
- // Network error, timeout, etc.
286
- return null;
287
- }
288
- }
289
-
290
- /**
291
- * Query Neo4j via Brain (Cypher query).
292
- * This does NOT use Pinecone, no embedding quota consumed.
293
- *
294
- * NOTE (Finding I, v1.10.9 hotfix 2026-04-15): the Brain MCP brain_query
295
- * tool expects the parameter name `cypher`, not `query`. Previously this
296
- * function sent { query: cypher } which tripped an MCP input validation
297
- * error (code -32602, path ["cypher"], "Required"). Downstream scripts
298
- * like fetch-brain-baseline.cjs and compute-whitespace-gaps.py then
299
- * silently fell through to empty-baseline mode even though Brain was
300
- * fully reachable and the key was valid. Witnessed against the live
301
- * iia-deeptech-centers room on 2026-04-15. brain_search uses `query`
302
- * which is why Pinecone semantic search kept working and masked this.
303
- *
304
- * NOTE (2026-05-11, graph-on-graph P0): `query` now accepts an optional
305
- * second argument `params` and forwards it to the `brain_query` MCP tool
306
- * as { cypher, params }. The Brain tool declares `params:
307
- * z.record(z.any()).optional()`, so a parameterized Cypher gets its
308
- * bindings through cleanly. `params` MUST be a generic-handles-only object
309
- * — framework names, phase identifiers, problem types per Canon Part 8 —
310
- * NEVER user content (artifact bodies, meeting text, personal identifiers,
311
- * proprietary numbers). Previously the second arg was silently dropped, so
312
- * callers (rs-explain-command.cjs, rs-thesis-command.cjs, rs-nl-to-query)
313
- * that generated parameterized Cypher had their bindings disappear or were
314
- * pushed toward unsafe string interpolation. A param-less call still sends
315
- * only { cypher } and behaves exactly as before.
316
- *
317
- * NOTE (2026-05-11, graph-on-graph P0 cont.): RESULT-SHAPE NORMALIZATION.
318
- * The Brain MCP `brain_query` tool serializes its result as
319
- * `JSON.stringify(records)` where `records` is a BARE ARRAY of row objects.
320
- * `callTool` returns that array directly (or `{ text: 'Error: ...' }` on a
321
- * Cypher error, or `null` when the Brain is unreachable / no API key).
322
- * Consumers across the codebase — brain-router.cjs, brain-derivation.cjs's
323
- * `renderRecords`, rs-chain-feeder.cjs, rs-experts-command.cjs,
324
- * rs-explain-command.cjs, rs-thesis-command.cjs — all read `result.records`,
325
- * so the bare-array shape silently dropped every row. `query` therefore now
326
- * ALWAYS returns `{ records: [...] }` on a successful brain_query; an
327
- * unreachable Brain / missing key still returns `null`; a Cypher-error
328
- * response (`{ text: 'Error: ...' }` or `{ error: ... }`) passes through
329
- * unchanged so callers that inspect the failure can still see it; any other
330
- * unexpected shape collapses to `{ records: [] }` so callers never crash.
331
- * `search`, `smartSearch`, `schema`, `stats`, `write`, `callTool` are
332
- * deliberately untouched — only `query` is normalized.
333
- */
334
- async function query(cypher, params) {
335
- const args = { cypher: cypher };
336
- if (params && typeof params === 'object' && Object.keys(params).length > 0) {
337
- args.params = params;
338
- }
339
- const result = await callTool('brain_query', args);
340
- if (result == null) return null; // unreachable / no API key
341
- if (Array.isArray(result)) return { records: result }; // the normal brain_query shape
342
- if (result && Array.isArray(result.records)) return result; // already normalized (defensive)
343
- if (result && (result.error || result.text)) return result; // error / message passthrough
344
- return { records: [] }; // unexpected shape -> empty, never crash
345
- }
346
-
347
- /**
348
- * Search Pinecone via Brain (semantic search).
349
- * If quota exhausted, returns error with fallback suggestion.
350
- */
351
- async function search(queryText, options = {}) {
352
- const result = await callTool('brain_search', {
353
- query: queryText,
354
- namespace: options.namespace || undefined,
355
- topK: options.topK || 5,
356
- });
357
-
358
- // Check for Pinecone quota exhaustion
359
- if (result && result.text && result.text.includes('RESOURCE_EXHAUSTED')) {
360
- return {
361
- error: 'pinecone_quota_exhausted',
362
- message: 'Pinecone embedding quota exhausted for this month. Using Neo4j Cypher fallback.',
363
- fallback: 'neo4j',
364
- };
365
- }
366
-
367
- return result;
368
- }
369
-
370
- /**
371
- * Search with automatic fallback: Pinecone first, Neo4j Cypher if quota exhausted.
372
- */
373
- async function smartSearch(queryText, options = {}) {
374
- // Try Pinecone first
375
- const pineconeResult = await search(queryText, options);
376
-
377
- if (pineconeResult && pineconeResult.error === 'pinecone_quota_exhausted') {
378
- // Fallback to Neo4j full-text search
379
- const cypher = `
380
- CALL db.index.fulltext.queryNodes("framework_search", $query)
381
- YIELD node, score
382
- RETURN node.name AS name, node.description AS description, score
383
- LIMIT ${options.topK || 5}
384
- `;
385
- const neo4jResult = await query(cypher.replace('$query', `"${sanitizeCypherInput(queryText)}"`));
386
- if (neo4jResult) {
387
- neo4jResult._source = 'neo4j_fallback';
388
- neo4jResult._note = 'Pinecone quota exhausted. Results from Neo4j Cypher fulltext search.';
389
- }
390
- return neo4jResult;
391
- }
392
-
393
- return pineconeResult;
394
- }
395
-
396
- /**
397
- * Get Neo4j schema.
398
- */
399
- async function schema() {
400
- return callTool('brain_schema', {});
401
- }
402
-
403
- /**
404
- * Get Pinecone stats.
405
- */
406
- async function stats() {
407
- return callTool('brain_stats', {});
408
- }
409
-
410
- /**
411
- * Enrich local graph with causal edges from Brain's teaching graph.
412
- *
413
- * Queries the Brain Neo4j for causal framework chains relevant to the
414
- * given problem type or section keywords. Returns structured causal data
415
- * suitable for writing to local SQLite graph as CAUSES/ROOT_CAUSE_OF edges.
416
- *
417
- * @param {string} problemType - Room problem type (e.g., 'market-validation')
418
- * @param {string[]} sectionKeywords - Keywords from room sections for context
419
- * @param {object} [options] - Optional config
420
- * @param {number} [options.maxChainDepth=3] - Maximum causal chain depth
421
- * @param {number} [options.minConfidence=0.5] - Minimum confidence threshold
422
- * @returns {Promise<{ causes: Array, rootCauses: Array } | null>}
423
- * causes: [{ from, to, mechanism, confidence, framework }]
424
- * rootCauses: [{ from, to, chainLength, intermediateCauses, confidence }]
425
- */
426
- async function enrichCausalEdges(problemType, sectionKeywords, options = {}) {
427
- if (!isAvailable()) return null;
428
-
429
- // SEC-01 defence-in-depth: coerce + bound numeric interpolants so a hostile
430
- // non-number (e.g. an object with .toString() side-effects) cannot reach
431
- // the Cypher string.
432
- const maxDepth = Math.max(1, Math.min(10, Number(options.maxChainDepth) || 3));
433
- const minConf = Math.max(0, Math.min(1, Number(options.minConfidence) || 0.5));
434
- const keywordFilter = sectionKeywords && sectionKeywords.length > 0
435
- ? sectionKeywords.map(k => `"${sanitizeCypherInput(k)}"`).join(', ')
436
- : '';
437
-
438
- // Query 1: Direct causal relationships from framework chains
439
- const causesCypher = `
440
- MATCH (f1:Framework)-[r:ADDRESSES_PROBLEM_TYPE]->(pt:ProblemType)
441
- WHERE pt.name CONTAINS "${sanitizeCypherInput(problemType || '')}"
442
- WITH f1
443
- MATCH (f1)-[co:CO_OCCURS]->(f2:Framework)
444
- WHERE co.weight >= ${minConf}
445
- RETURN f1.name AS cause_framework,
446
- f2.name AS effect_framework,
447
- co.weight AS confidence,
448
- f1.description AS mechanism
449
- LIMIT 20
450
- `;
451
-
452
- // Query 2: Root cause chains (multi-hop framework dependencies)
453
- const rootCauseCypher = `
454
- MATCH path = (root:Framework)-[:CO_OCCURS*1..${maxDepth}]->(leaf:Framework)
455
- WHERE root <> leaf
456
- ${keywordFilter ? `AND ANY(k IN [${keywordFilter}] WHERE root.name CONTAINS k OR root.description CONTAINS k)` : ''}
457
- WITH root, leaf, path, length(path) AS depth
458
- WHERE depth >= 2
459
- RETURN root.name AS root_cause,
460
- leaf.name AS symptom,
461
- depth AS chain_length,
462
- [n IN nodes(path) | n.name] AS chain_nodes
463
- LIMIT 10
464
- `;
465
-
466
- try {
467
- const [causesResult, rootCausesResult] = await Promise.all([
468
- query(causesCypher),
469
- query(rootCauseCypher),
470
- ]);
471
-
472
- const causes = [];
473
- const rootCauses = [];
474
-
475
- // Parse causes
476
- if (causesResult && Array.isArray(causesResult.records)) {
477
- for (const rec of causesResult.records) {
478
- causes.push({
479
- from: rec.cause_framework || rec[0],
480
- to: rec.effect_framework || rec[1],
481
- mechanism: rec.mechanism || rec[3] || '',
482
- confidence: parseFloat(rec.confidence || rec[2] || 0),
483
- framework: rec.cause_framework || rec[0] || '',
484
- });
485
- }
486
- }
487
-
488
- // Parse root causes
489
- if (rootCausesResult && Array.isArray(rootCausesResult.records)) {
490
- for (const rec of rootCausesResult.records) {
491
- rootCauses.push({
492
- from: rec.root_cause || rec[0],
493
- to: rec.symptom || rec[1],
494
- chainLength: parseInt(rec.chain_length || rec[2] || 1, 10),
495
- intermediateCauses: rec.chain_nodes || rec[3] || [],
496
- confidence: 1.0 / (parseInt(rec.chain_length || rec[2] || 1, 10) + 1),
497
- });
498
- }
499
- }
500
-
501
- return { causes, rootCauses };
502
- } catch (err) {
503
- // Brain query failed -- return null for graceful degradation
504
- return null;
505
- }
506
- }
507
-
508
- /**
509
- * Hat-aware framework recommendation.
510
- *
511
- * Reads persistent hat states and adjusts Brain framework queries:
512
- * - Black Hat concerns boost risk-related frameworks (Risk Matrix, SWOT threats)
513
- * - Yellow Hat opportunities boost HSI scoring and opportunity frameworks
514
- * - Blue Hat methodology notes avoid repeating ineffective frameworks
515
- *
516
- * @param {string} roomDir - Absolute path to room directory
517
- * @param {string} problemType - Room problem type
518
- * @param {object} [options] - Optional config
519
- * @param {number} [options.topK=5] - Number of frameworks to return
520
- * @returns {Promise<{ frameworks: Array, hat_influence: object } | null>}
521
- */
522
- async function hatAwareRecommend(roomDir, problemType, options = {}) {
523
- if (!isAvailable()) return null;
524
-
525
- // Lazy-require to avoid circular dependency at module load time
526
- const { loadAllHatStates } = require('./hat-persistence.cjs');
527
- const hatStates = loadAllHatStates(roomDir);
528
- // SEC-01 defence-in-depth: bound topK numeric interpolation.
529
- const topK = Math.max(1, Math.min(100, Number(options.topK) || 5));
530
-
531
- const hatInfluence = {
532
- risk_boost: false,
533
- opportunity_boost: false,
534
- avoid_frameworks: [],
535
- };
536
-
537
- // Black Hat: if concerns exist, boost risk-related frameworks
538
- const blackConcerns = hatStates.black.top_concerns || [];
539
- const riskBoost = blackConcerns.length > 0;
540
- hatInfluence.risk_boost = riskBoost;
541
-
542
- // Yellow Hat: if opportunities exist, boost HSI/opportunity frameworks
543
- const yellowOpps = hatStates.yellow.top_opportunities || [];
544
- const oppBoost = yellowOpps.length > 0;
545
- hatInfluence.opportunity_boost = oppBoost;
546
-
547
- // Blue Hat: methodology notes may flag ineffective frameworks to avoid
548
- const blueNotes = hatStates.blue.methodology_notes || [];
549
- const avoidPatterns = blueNotes
550
- .filter(n => /ineffective|didn't work|not useful|skip|avoid/i.test(n))
551
- .map(n => {
552
- // Extract framework name from notes like "SWOT was ineffective for this stage"
553
- const match = n.match(/^(\w[\w\s]+?)\s+(?:was|is|were|proved)\s/i);
554
- return match ? match[1].trim() : null;
555
- })
556
- .filter(Boolean);
557
- hatInfluence.avoid_frameworks = avoidPatterns;
558
-
559
- // Build Cypher query with hat-influenced scoring
560
- const safeProblemType = sanitizeCypherInput(problemType || '');
561
- const avoidClause = avoidPatterns.length > 0
562
- ? `AND NOT ANY(avoid IN [${avoidPatterns.map(a => `"${sanitizeCypherInput(a)}"`).join(', ')}] WHERE f.name CONTAINS avoid)`
563
- : '';
564
-
565
- // Query: frameworks for problem type, with hat-influenced ordering
566
- const cypher = `
567
- MATCH (f:Framework)-[:ADDRESSES_PROBLEM_TYPE]->(pt:ProblemType)
568
- WHERE pt.name CONTAINS "${safeProblemType}"
569
- ${avoidClause}
570
- WITH f
571
- OPTIONAL MATCH (f)-[co:CO_OCCURS]->(f2:Framework)
572
- WITH f, count(co) AS connections
573
- RETURN f.name AS name,
574
- f.description AS description,
575
- connections,
576
- CASE
577
- WHEN ${riskBoost ? 'true' : 'false'} AND (f.name CONTAINS 'Risk' OR f.name CONTAINS 'SWOT' OR f.name CONTAINS 'Failure') THEN connections + 10
578
- WHEN ${oppBoost ? 'true' : 'false'} AND (f.name CONTAINS 'HSI' OR f.name CONTAINS 'Opportunity' OR f.name CONTAINS 'Innovation') THEN connections + 10
579
- ELSE connections
580
- END AS hat_score
581
- ORDER BY hat_score DESC
582
- LIMIT ${topK}
583
- `;
584
-
585
- try {
586
- const result = await query(cypher);
587
- const frameworks = [];
588
-
589
- if (result && Array.isArray(result.records)) {
590
- for (const rec of result.records) {
591
- frameworks.push({
592
- name: rec.name || rec[0],
593
- description: rec.description || rec[1],
594
- connections: parseInt(rec.connections || rec[2] || 0, 10),
595
- hat_score: parseInt(rec.hat_score || rec[3] || 0, 10),
596
- });
597
- }
598
- }
599
-
600
- return {
601
- frameworks,
602
- hat_influence: hatInfluence,
603
- black_concerns: blackConcerns.slice(0, 3),
604
- yellow_opportunities: yellowOpps.slice(0, 3),
605
- blue_avoid: avoidPatterns,
606
- };
607
- } catch (err) {
608
- return null;
609
- }
610
- }
611
-
612
- /**
613
- * Suggest validation steps for a banked opportunity using Brain framework chains.
614
- *
615
- * Queries Brain Neo4j for frameworks that ADDRESSES_PROBLEM_TYPE matching the
616
- * opportunity's domain/problem, then follows FEEDS_INTO chains to build a
617
- * suggested validation sequence.
618
- *
619
- * @param {Object} opportunity - Opportunity object with at minimum: problem, domain, knight_position
620
- * @param {Object} [options] - Optional config
621
- * @param {number} [options.maxSteps=5] - Maximum validation steps to return
622
- * @param {number} [options.chainDepth=3] - Maximum FEEDS_INTO chain depth
623
- * @returns {Promise<{ steps: Array<{framework: string, reason: string, order: number}>, chain_source: string } | null>}
624
- * Returns null if Brain unavailable (Tier 0 graceful degradation)
625
- */
626
- async function suggestValidationSteps(opportunity, options = {}) {
627
- if (!isAvailable()) return null;
628
- if (!opportunity || !opportunity.problem) return null;
629
-
630
- const maxSteps = options.maxSteps || 5;
631
- const chainDepth = options.chainDepth || 3;
632
- const safeProblem = sanitizeCypherInput(opportunity.problem || '').substring(0, 200);
633
- const safeDomain = sanitizeCypherInput(opportunity.domain || '').substring(0, 100);
634
- const safeKnight = opportunity.knight_position || 'uncertainty';
635
-
636
- // Query 1: Find frameworks that address this problem type / domain
637
- const matchCypher = `
638
- MATCH (f:Framework)-[:ADDRESSES_PROBLEM_TYPE]->(pt:ProblemType)
639
- WHERE pt.name CONTAINS "${safeDomain}"
640
- OR pt.description CONTAINS "${safeDomain}"
641
- RETURN f.name AS name, f.description AS description
642
- LIMIT 10
643
- `;
644
-
645
- // Query 2: Follow FEEDS_INTO chains from matched frameworks
646
- const chainCypher = `
647
- MATCH (f:Framework)-[:ADDRESSES_PROBLEM_TYPE]->(pt:ProblemType)
648
- WHERE pt.name CONTAINS "${safeDomain}"
649
- OR pt.description CONTAINS "${safeDomain}"
650
- WITH f LIMIT 3
651
- MATCH path = (f)-[:FEEDS_INTO*1..${chainDepth}]->(next:Framework)
652
- RETURN f.name AS start_framework,
653
- next.name AS next_framework,
654
- next.description AS next_description,
655
- length(path) AS depth
656
- ORDER BY depth ASC
657
- LIMIT ${maxSteps * 2}
658
- `;
659
-
660
- try {
661
- const [matchResult, chainResult] = await Promise.all([
662
- query(matchCypher),
663
- query(chainCypher),
664
- ]);
665
-
666
- const steps = [];
667
- const seen = new Set();
668
-
669
- // First: add the entry-point frameworks
670
- if (matchResult && Array.isArray(matchResult.records)) {
671
- for (const rec of matchResult.records) {
672
- const name = rec.name || rec[0];
673
- if (name && !seen.has(name) && steps.length < maxSteps) {
674
- seen.add(name);
675
- steps.push({
676
- framework: name,
677
- reason: safeKnight === 'uncertainty'
678
- ? `Explore this ${safeDomain} uncertainty with ${name}`
679
- : `Validate this ${safeDomain} risk using ${name}`,
680
- order: steps.length + 1,
681
- });
682
- }
683
- }
684
- }
685
-
686
- // Then: add FEEDS_INTO chain steps
687
- if (chainResult && Array.isArray(chainResult.records)) {
688
- for (const rec of chainResult.records) {
689
- const name = rec.next_framework || rec[1];
690
- const desc = rec.next_description || rec[2] || '';
691
- if (name && !seen.has(name) && steps.length < maxSteps) {
692
- seen.add(name);
693
- steps.push({
694
- framework: name,
695
- reason: desc ? `Then apply ${name}: ${desc.substring(0, 120)}` : `Then apply ${name} (follows from chain)`,
696
- order: steps.length + 1,
697
- });
698
- }
699
- }
700
- }
701
-
702
- if (steps.length === 0) return null;
703
-
704
- return {
705
- steps,
706
- chain_source: 'brain_feeds_into',
707
- };
708
- } catch (err) {
709
- // Brain query failed -- graceful degradation
710
- return null;
711
- }
712
- }
713
-
714
- /**
715
- * Write Cypher to Neo4j via Brain (write operations).
716
- * Used by sync-rooms-brain for creating Room/RoomGroup nodes and edges.
717
- * Returns null if Brain is unavailable -- never throws.
718
- *
719
- * NOTE (Finding I sibling, v1.10.9 hotfix 2026-04-15): same param-name
720
- * mismatch as brain_query had. Brain MCP brain_write expects `cypher`,
721
- * not `query`. This function had the mirror bug since inception but
722
- * never fired in production because sync-rooms-brain is rarely invoked
723
- * against the live Brain. Caught by the plan-checker audit for Phase 85.
724
- *
725
- * @param {string} cypher - Cypher write query
726
- * @returns {Promise<object|null>}
727
- */
728
- async function write(cypher) {
729
- return callTool('brain_write', { cypher: cypher });
730
- }
731
-
732
- /**
733
- * Tier 0 fallback: hardcoded persona framework chains.
734
- * Used when Brain is unavailable or query returns no results.
735
- *
736
- * @param {string} persona - 'tto', 'researcher', or 'business'
737
- * @returns {{ persona: string, chain: Array<{framework: string, description: string, order: number}>, source: 'tier0' }}
738
- */
739
- function getTier0Chain(persona) {
740
- const chains = {
741
- tto: [
742
- { framework: 'Domain Exploration', description: 'What domains could this technology touch?', order: 1 },
743
- { framework: 'Problem Definition', description: 'What specific problems does it solve?', order: 2 },
744
- { framework: 'JTBD Analysis', description: 'Who needs this solved and what progress do they want?', order: 3 },
745
- { framework: 'Value Proposition', description: 'What is the value and how to deliver it?', order: 4 },
746
- ],
747
- researcher: [
748
- { framework: 'Problem Exploration', description: 'What problem does the research address?', order: 1 },
749
- { framework: 'JTBD Analysis', description: 'Who cares about this problem?', order: 2 },
750
- { framework: 'Value Proposition', description: 'What would a solution look like?', order: 3 },
751
- { framework: 'Lean Canvas', description: 'How to deliver and sustain this?', order: 4 },
752
- ],
753
- business: [
754
- { framework: 'Opportunity Recognition', description: 'What opportunity exists in the market?', order: 1 },
755
- { framework: 'Market Analysis', description: 'How big is this market?', order: 2 },
756
- { framework: 'Problem Definition', description: 'What specific problem for whom?', order: 3 },
757
- { framework: 'Competitive Analysis', description: 'Who else is trying and what is your edge?', order: 4 },
758
- ],
759
- };
760
-
761
- return {
762
- persona: persona,
763
- chain: chains[persona] || chains.researcher,
764
- source: 'tier0',
765
- };
766
- }
767
-
768
- /**
769
- * Get ordered framework chain for a persona.
770
- *
771
- * Brain-connected: queries FEEDS_INTO edges for dynamic chains.
772
- * Brain-unavailable: returns Tier 0 hardcoded chains from getTier0Chain().
773
- *
774
- * Per CONV-02: Persona-aware chain routing.
775
- * Per CONV-03: Brain framework chain selection with Tier 0 fallback.
776
- *
777
- * @param {string} persona - 'tto', 'researcher', or 'business' (case-insensitive)
778
- * @returns {Promise<{ persona: string, chain: Array<{framework: string, description: string, order: number}>, source: 'brain'|'tier0' }>}
779
- */
780
- async function getFrameworkChain(persona) {
781
- const personaLower = (persona || '').toLowerCase();
782
-
783
- // Persona-to-entry-framework mapping
784
- const entryMap = {
785
- tto: 'Domain Exploration',
786
- researcher: 'Problem Exploration',
787
- business: 'Opportunity Recognition',
788
- };
789
-
790
- const entryFramework = entryMap[personaLower];
791
- if (!entryFramework) {
792
- // Unknown persona - fall back to Tier 0 default (researcher chain)
793
- return getTier0Chain('researcher');
794
- }
795
-
796
- // Try Brain first
797
- if (isAvailable()) {
798
- try {
799
- const safeEntry = sanitizeCypherInput(entryFramework);
800
- const cypher = `
801
- MATCH path = (start:Framework)-[:FEEDS_INTO*1..4]->(next:Framework)
802
- WHERE start.name CONTAINS "${safeEntry}"
803
- RETURN start.name AS start_name,
804
- [n IN nodes(path) | n.name] AS chain,
805
- [n IN nodes(path) | n.description] AS descriptions,
806
- length(path) AS depth
807
- ORDER BY depth ASC
808
- LIMIT 5
809
- `;
810
- const result = await query(cypher);
811
-
812
- if (result && Array.isArray(result.records) && result.records.length > 0) {
813
- // Build ordered chain from longest path
814
- const longestPath = result.records[result.records.length - 1];
815
- const chain = longestPath.chain || longestPath[1] || [];
816
- const descriptions = longestPath.descriptions || longestPath[2] || [];
817
-
818
- if (chain.length > 0) {
819
- return {
820
- persona: personaLower,
821
- chain: chain.map((name, i) => ({
822
- framework: name,
823
- description: descriptions[i] || '',
824
- order: i + 1,
825
- })),
826
- source: 'brain',
827
- };
828
- }
829
- }
830
- } catch (err) {
831
- // Brain query failed - fall through to Tier 0
832
- }
833
- }
834
-
835
- // Tier 0 fallback
836
- return getTier0Chain(personaLower);
837
- }
838
-
839
- module.exports = {
840
- isAvailable,
841
- getApiKey,
842
- callTool,
843
- query,
844
- write,
845
- search,
846
- smartSearch,
847
- schema,
848
- stats,
849
- enrichCausalEdges,
850
- hatAwareRecommend,
851
- suggestValidationSteps,
852
- getFrameworkChain,
853
- // SEC-01/SEC-02 + CASCADE-06 test surface: not part of the public API.
854
- // See lib/memory/security-trifecta.test.cjs + brain-cache-lru.test.cjs.
855
- // Helpers are small and pure. sessionCache + _ensureSession + _hashKey +
856
- // SESSION_TTL_MS are exposed for Phase 87-07 cache-behavior tests.
857
- _test: {
858
- sanitizeCypherInput,
859
- checkFilePermissions,
860
- sessionCache,
861
- SESSION_TTL_MS,
862
- _hashKey,
863
- _ensureSession,
864
- },
865
- };