@mindrian_os/install 1.13.0-beta.11

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 (597) hide show
  1. package/.claude-plugin/plugin.json +21 -0
  2. package/.mcp.json +9 -0
  3. package/CHANGELOG.md +3333 -0
  4. package/LICENSE +123 -0
  5. package/README.md +673 -0
  6. package/agents/brain-query.md +80 -0
  7. package/agents/framework-runner.md +237 -0
  8. package/agents/grading.md +188 -0
  9. package/agents/investor.md +128 -0
  10. package/agents/larry-extended.md +135 -0
  11. package/agents/opportunity-scanner.md +91 -0
  12. package/agents/persona-analyst.md +132 -0
  13. package/agents/research.md +89 -0
  14. package/agents/reverse-salient-agent.md +27 -0
  15. package/bin/cli.js +142 -0
  16. package/bin/mindrian-mcp-server.cjs +182 -0
  17. package/bin/mindrian-tools.cjs +765 -0
  18. package/commands/act.md +439 -0
  19. package/commands/admin.md +404 -0
  20. package/commands/analyze-needs.md +42 -0
  21. package/commands/analyze-systems.md +39 -0
  22. package/commands/analyze-timing.md +42 -0
  23. package/commands/auto-explore.md +64 -0
  24. package/commands/beautiful-question.md +40 -0
  25. package/commands/brain-derive.md +78 -0
  26. package/commands/build-knowledge.md +42 -0
  27. package/commands/build-thesis.md +46 -0
  28. package/commands/causal.md +234 -0
  29. package/commands/challenge-assumptions.md +33 -0
  30. package/commands/compare-ventures.md +83 -0
  31. package/commands/dashboard.md +110 -0
  32. package/commands/deep-grade.md +82 -0
  33. package/commands/diagnose.md +58 -0
  34. package/commands/diagnostics.md +151 -0
  35. package/commands/doctor.md +151 -0
  36. package/commands/dominant-designs.md +40 -0
  37. package/commands/explain-decision.md +87 -0
  38. package/commands/explore-domains.md +42 -0
  39. package/commands/explore-futures.md +40 -0
  40. package/commands/explore-trends.md +42 -0
  41. package/commands/export.md +103 -0
  42. package/commands/file-meeting.md +724 -0
  43. package/commands/find-analogies.md +188 -0
  44. package/commands/find-bottlenecks.md +62 -0
  45. package/commands/find-connections.md +76 -0
  46. package/commands/funding.md +81 -0
  47. package/commands/grade.md +203 -0
  48. package/commands/graph.md +128 -0
  49. package/commands/hat-briefing.md +125 -0
  50. package/commands/heal.md +196 -0
  51. package/commands/help.md +399 -0
  52. package/commands/hmi-status.md +172 -0
  53. package/commands/jtbd.md +241 -0
  54. package/commands/leadership.md +73 -0
  55. package/commands/lean-canvas.md +40 -0
  56. package/commands/macro-trends.md +40 -0
  57. package/commands/map-unknowns.md +40 -0
  58. package/commands/memory.md +173 -0
  59. package/commands/models.md +175 -0
  60. package/commands/mos-reason.md +285 -0
  61. package/commands/mullins.md +120 -0
  62. package/commands/new-project.md +481 -0
  63. package/commands/onboard.md +434 -0
  64. package/commands/operator.md +149 -0
  65. package/commands/opportunities.md +144 -0
  66. package/commands/organize.md +497 -0
  67. package/commands/persona.md +198 -0
  68. package/commands/pipeline.md +112 -0
  69. package/commands/present.md +91 -0
  70. package/commands/publish.md +201 -0
  71. package/commands/query.md +124 -0
  72. package/commands/radar.md +72 -0
  73. package/commands/reanalyze.md +91 -0
  74. package/commands/research.md +196 -0
  75. package/commands/room.md +352 -0
  76. package/commands/rooms.md +598 -0
  77. package/commands/root-cause.md +40 -0
  78. package/commands/rs-experts.md +85 -0
  79. package/commands/rs-explain.md +100 -0
  80. package/commands/rs-fetch.md +94 -0
  81. package/commands/rs-thesis.md +85 -0
  82. package/commands/scenario-plan.md +40 -0
  83. package/commands/scheduled-tasks.md +285 -0
  84. package/commands/score-innovation.md +43 -0
  85. package/commands/scout.md +239 -0
  86. package/commands/setup.md +618 -0
  87. package/commands/snapshot.md +147 -0
  88. package/commands/speakers.md +84 -0
  89. package/commands/splash.md +28 -0
  90. package/commands/status.md +75 -0
  91. package/commands/structure-argument.md +42 -0
  92. package/commands/suggest-next.md +80 -0
  93. package/commands/systems-thinking.md +40 -0
  94. package/commands/think-hats.md +42 -0
  95. package/commands/update.md +181 -0
  96. package/commands/user-needs.md +40 -0
  97. package/commands/validate.md +40 -0
  98. package/commands/value-proposition.md +61 -0
  99. package/commands/vault.md +180 -0
  100. package/commands/visualize.md +52 -0
  101. package/commands/whitespace.md +507 -0
  102. package/commands/wiki.md +69 -0
  103. package/hooks/hooks.json +381 -0
  104. package/hooks/run-hook.cmd +64 -0
  105. package/lib/__init__.py +0 -0
  106. package/lib/__pycache__/__init__.cpython-312.pyc +0 -0
  107. package/lib/agents/auto-explore-agent.cjs +1043 -0
  108. package/lib/agents/reverse-salient-agent.cjs +679 -0
  109. package/lib/agents/tension-hook-agent.cjs +544 -0
  110. package/lib/brain/ROOM.md +44 -0
  111. package/lib/brain/chain-recommender.cjs +301 -0
  112. package/lib/chat/chat-context.js +185 -0
  113. package/lib/chat/chat-panel.js +721 -0
  114. package/lib/chat/fabric-chat.cjs +288 -0
  115. package/lib/chat/generative-tools.js +219 -0
  116. package/lib/conversation/ROOM.md +39 -0
  117. package/lib/conversation/classifier-rules.json +38 -0
  118. package/lib/conversation/classifier.cjs +264 -0
  119. package/lib/conversation/operator.cjs +287 -0
  120. package/lib/copy/115-spec-strings.cjs +55 -0
  121. package/lib/core/__init__.py +0 -0
  122. package/lib/core/__nav-stub.cjs +14 -0
  123. package/lib/core/__pycache__/__init__.cpython-312.pyc +0 -0
  124. package/lib/core/__pycache__/rs-math.cpython-312.pyc +0 -0
  125. package/lib/core/__pycache__/rs_cache.cpython-312.pyc +0 -0
  126. package/lib/core/__pycache__/rs_corpus.cpython-312.pyc +0 -0
  127. package/lib/core/__pycache__/rs_hybrid.cpython-312.pyc +0 -0
  128. package/lib/core/__pycache__/rs_math.cpython-312.pyc +0 -0
  129. package/lib/core/__pycache__/rs_rooms.cpython-312.pyc +0 -0
  130. package/lib/core/artifact-id.cjs +148 -0
  131. package/lib/core/asset-ops.cjs +151 -0
  132. package/lib/core/auto-commit-throttle.cjs +129 -0
  133. package/lib/core/bearer-token.cjs +199 -0
  134. package/lib/core/brain-client.cjs +865 -0
  135. package/lib/core/brain-derivation-prompts.cjs +326 -0
  136. package/lib/core/brain-derivation-queue.cjs +431 -0
  137. package/lib/core/brain-derivation.cjs +580 -0
  138. package/lib/core/brain-md-schema.cjs +528 -0
  139. package/lib/core/brain-md-staleness.cjs +357 -0
  140. package/lib/core/brain-response-sanitize.cjs +188 -0
  141. package/lib/core/bridge-writer.cjs +477 -0
  142. package/lib/core/chat-context-builder.cjs +253 -0
  143. package/lib/core/cross-room-aggregator.cjs +762 -0
  144. package/lib/core/daily-briefing.cjs +438 -0
  145. package/lib/core/decision-capture.cjs +618 -0
  146. package/lib/core/deep-links.cjs +82 -0
  147. package/lib/core/dispatch-optimizer.cjs +354 -0
  148. package/lib/core/dual-path-detector.cjs +84 -0
  149. package/lib/core/dual-path-detector.test.cjs +334 -0
  150. package/lib/core/exports-log.cjs +79 -0
  151. package/lib/core/feynman-minto-invariants.cjs +605 -0
  152. package/lib/core/folder-memory-async.cjs +338 -0
  153. package/lib/core/folder-memory-shared.cjs +890 -0
  154. package/lib/core/folder-memory.cjs +416 -0
  155. package/lib/core/framework-chain-composer.cjs +411 -0
  156. package/lib/core/frontmatter-schemas.cjs +330 -0
  157. package/lib/core/git-ops.cjs +141 -0
  158. package/lib/core/graph-ops.cjs +258 -0
  159. package/lib/core/hat-persistence.cjs +362 -0
  160. package/lib/core/index.cjs +60 -0
  161. package/lib/core/integration-registry.cjs +232 -0
  162. package/lib/core/intelligence-cascade.cjs +661 -0
  163. package/lib/core/lazygraph-ops.cjs +1057 -0
  164. package/lib/core/lru-cache.cjs +139 -0
  165. package/lib/core/mcp-profiles.cjs +182 -0
  166. package/lib/core/meeting-ops.cjs +54 -0
  167. package/lib/core/memory-ops.cjs +600 -0
  168. package/lib/core/migrations/ROOM.md +33 -0
  169. package/lib/core/migrations/phase-109-nodes-provenance.cjs +339 -0
  170. package/lib/core/migrations/phase-109-session-focus.cjs +99 -0
  171. package/lib/core/model-profiles.cjs +246 -0
  172. package/lib/core/mullins-scaffold.cjs +160 -0
  173. package/lib/core/nav-dial.cjs +316 -0
  174. package/lib/core/navigation/ROOM.md +15 -0
  175. package/lib/core/navigation/explanation.cjs +43 -0
  176. package/lib/core/navigation/focus.cjs +135 -0
  177. package/lib/core/navigation/ingestion.cjs +82 -0
  178. package/lib/core/navigation/insights.cjs +350 -0
  179. package/lib/core/navigation/memory-events.cjs +118 -0
  180. package/lib/core/navigation/neighborhood.cjs +78 -0
  181. package/lib/core/navigation/packet.cjs +182 -0
  182. package/lib/core/navigation/room-home.cjs +127 -0
  183. package/lib/core/navigation/transitions.cjs +82 -0
  184. package/lib/core/navigation-engine-shared.cjs +242 -0
  185. package/lib/core/navigation-engine.cjs +664 -0
  186. package/lib/core/navigation.cjs +60 -0
  187. package/lib/core/nl-graph-queries.cjs +164 -0
  188. package/lib/core/offer-presenter.cjs +406 -0
  189. package/lib/core/opportunity-extractor.cjs +183 -0
  190. package/lib/core/opportunity-ops.cjs +1371 -0
  191. package/lib/core/persona-ops.cjs +537 -0
  192. package/lib/core/persona-taxonomy.cjs +190 -0
  193. package/lib/core/platform-gates.cjs +120 -0
  194. package/lib/core/platform.cjs +257 -0
  195. package/lib/core/proactive-intelligence.cjs +528 -0
  196. package/lib/core/problem-type-router.cjs +315 -0
  197. package/lib/core/reasoning-ops.cjs +639 -0
  198. package/lib/core/reverse-salient-persona-suffix.cjs +115 -0
  199. package/lib/core/room-classifier-strict-mode.cjs +229 -0
  200. package/lib/core/room-db.cjs +127 -0
  201. package/lib/core/room-ops-async.cjs +92 -0
  202. package/lib/core/room-ops-shared.cjs +64 -0
  203. package/lib/core/room-ops-sync.cjs +70 -0
  204. package/lib/core/room-ops.cjs +32 -0
  205. package/lib/core/room-type-detector.cjs +386 -0
  206. package/lib/core/rs-brain-substrate-prompts.cjs +129 -0
  207. package/lib/core/rs-brain-substrate.cjs +570 -0
  208. package/lib/core/rs-breakthrough-scorer.cjs +255 -0
  209. package/lib/core/rs-canon-violations.cjs +82 -0
  210. package/lib/core/rs-chain-feeder.cjs +343 -0
  211. package/lib/core/rs-commercial-assessor.cjs +280 -0
  212. package/lib/core/rs-differential-scorer.cjs +376 -0
  213. package/lib/core/rs-domain-analyzer.cjs +385 -0
  214. package/lib/core/rs-egress-prompts.cjs +113 -0
  215. package/lib/core/rs-egress-telemetry.cjs +225 -0
  216. package/lib/core/rs-egress-violations.cjs +53 -0
  217. package/lib/core/rs-expert-mapper.cjs +467 -0
  218. package/lib/core/rs-fetcher-academic.cjs +697 -0
  219. package/lib/core/rs-fetcher-experts.cjs +314 -0
  220. package/lib/core/rs-fetcher-industry.cjs +731 -0
  221. package/lib/core/rs-fetcher-patents.cjs +564 -0
  222. package/lib/core/rs-innovation-classifier.cjs +194 -0
  223. package/lib/core/rs-mind-map.cjs +656 -0
  224. package/lib/core/rs-neo4j-writer.cjs +388 -0
  225. package/lib/core/rs-nl-to-query.cjs +425 -0
  226. package/lib/core/rs-pinecone-bridge.cjs +303 -0
  227. package/lib/core/rs-preprocessor.cjs +350 -0
  228. package/lib/core/rs-query-matrix.cjs +316 -0
  229. package/lib/core/rs-query-to-text.cjs +438 -0
  230. package/lib/core/rs-sqlite-mirror.cjs +443 -0
  231. package/lib/core/rs-thesis-generator.cjs +188 -0
  232. package/lib/core/rs_cache.py +479 -0
  233. package/lib/core/rs_corpus.py +468 -0
  234. package/lib/core/rs_hybrid.py +586 -0
  235. package/lib/core/rs_math.py +287 -0
  236. package/lib/core/rs_rooms.py +193 -0
  237. package/lib/core/scheduled-scanner.cjs +463 -0
  238. package/lib/core/scratchpad-ops.cjs +201 -0
  239. package/lib/core/section-8-trace-schema.cjs +138 -0
  240. package/lib/core/section-registry.cjs +111 -0
  241. package/lib/core/session-state.cjs +144 -0
  242. package/lib/core/shallow-doc-parser.cjs +174 -0
  243. package/lib/core/shallow-doc-parser.test.cjs +226 -0
  244. package/lib/core/skill-activation-router.cjs +284 -0
  245. package/lib/core/state-ops.cjs +46 -0
  246. package/lib/core/statusline-cache.cjs +266 -0
  247. package/lib/core/token-estimator.cjs +348 -0
  248. package/lib/core/user-archetype.cjs +239 -0
  249. package/lib/core/user-md-ops.cjs +524 -0
  250. package/lib/core/visual-ops.cjs +624 -0
  251. package/lib/core/write-lock.cjs +149 -0
  252. package/lib/graph/canvas-graph.js +467 -0
  253. package/lib/graph/constellation-config.cjs +299 -0
  254. package/lib/graph/graph-detail-panel.js +165 -0
  255. package/lib/hmi/ROOM.md +47 -0
  256. package/lib/hmi/across-session-memory.cjs +604 -0
  257. package/lib/hmi/cross-room-memory.cjs +575 -0
  258. package/lib/hmi/decoy-tier.cjs +395 -0
  259. package/lib/hmi/jtbd-classifier.cjs +219 -0
  260. package/lib/hmi/jtbd-state.cjs +199 -0
  261. package/lib/hmi/jtbd-taxonomy.json +392 -0
  262. package/lib/hmi/selector-dispatcher.cjs +546 -0
  263. package/lib/hmi/selector-telemetry.cjs +263 -0
  264. package/lib/hmi/shape-f0-renderer.cjs +139 -0
  265. package/lib/hmi/shape-f1-fallback.cjs +80 -0
  266. package/lib/hmi/shape-f1-renderer.cjs +138 -0
  267. package/lib/hmi/shape-f2-renderer.cjs +132 -0
  268. package/lib/hmi/shape-f3-renderer.cjs +66 -0
  269. package/lib/hmi/shape-f4-renderer.cjs +72 -0
  270. package/lib/hmi/shape-f5-renderer.cjs +155 -0
  271. package/lib/hmi/shape-f6-plan-review-renderer.cjs +312 -0
  272. package/lib/hmi/shape-f6-renderer.cjs +144 -0
  273. package/lib/hmi/shape-g-renderer.cjs +219 -0
  274. package/lib/hmi/shape-h-renderer.cjs +222 -0
  275. package/lib/hmi/tier-check.cjs +63 -0
  276. package/lib/import/PRECONDITIONS.md +41 -0
  277. package/lib/import/branding.cjs +210 -0
  278. package/lib/import/branding.test.cjs +235 -0
  279. package/lib/import/classifications-sync.cjs +104 -0
  280. package/lib/import/classifications-sync.test.cjs +129 -0
  281. package/lib/import/enricher.cjs +296 -0
  282. package/lib/import/enricher.test.cjs +273 -0
  283. package/lib/import/integration.test.cjs +376 -0
  284. package/lib/import/manifest.cjs +129 -0
  285. package/lib/import/manifest.schema.json +185 -0
  286. package/lib/import/manifest.test.cjs +123 -0
  287. package/lib/import/meeting-detector.cjs +92 -0
  288. package/lib/import/meeting-detector.test.cjs +100 -0
  289. package/lib/import/person-detector.cjs +229 -0
  290. package/lib/import/person-detector.test.cjs +149 -0
  291. package/lib/import/report.cjs +186 -0
  292. package/lib/import/report.test.cjs +186 -0
  293. package/lib/import/room-md-scaffolder.cjs +49 -0
  294. package/lib/import/router.cjs +224 -0
  295. package/lib/import/router.test.cjs +356 -0
  296. package/lib/import/run-all-tests.cjs +36 -0
  297. package/lib/import/smoke-test.cjs +213 -0
  298. package/lib/import/smoke-test.test.cjs +148 -0
  299. package/lib/import/test-fixtures/collision-vault/preexisting-room/STATE.md +8 -0
  300. package/lib/import/test-fixtures/collision-vault/preexisting-room/problem-definition/onboarding/onboarding.md +7 -0
  301. package/lib/import/test-fixtures/collision-vault/source/onboarding.md +5 -0
  302. package/lib/import/test-fixtures/obsidian-vault/.obsidian/workspace.json +1 -0
  303. package/lib/import/test-fixtures/obsidian-vault/notes/with-wikilinks.md +4 -0
  304. package/lib/import/test-fixtures/tiny-vault/notes/2026-01-15-team-sync.md +9 -0
  305. package/lib/import/test-fixtures/tiny-vault/notes/empty.md +3 -0
  306. package/lib/import/test-fixtures/tiny-vault/notes/onboarding.md +5 -0
  307. package/lib/import/test-fixtures/tiny-vault/notes/pricing.md +5 -0
  308. package/lib/import/test-fixtures/tiny-vault/notes/random.md +4 -0
  309. package/lib/import/undo.test.cjs +199 -0
  310. package/lib/import/vault-scanner.cjs +105 -0
  311. package/lib/import/vault-scanner.test.cjs +67 -0
  312. package/lib/mcp/app-html/dashboard.html +316 -0
  313. package/lib/mcp/app-html/graph.html +428 -0
  314. package/lib/mcp/app-html/mindrian-platform.html +1841 -0
  315. package/lib/mcp/app-html/wiki.html +383 -0
  316. package/lib/mcp/app-views.cjs +322 -0
  317. package/lib/mcp/brain-router.cjs +418 -0
  318. package/lib/mcp/capability-registry.cjs +62 -0
  319. package/lib/mcp/larry-context.cjs +46 -0
  320. package/lib/mcp/larry-server-instructions.md +114 -0
  321. package/lib/mcp/pipeline-state.cjs +275 -0
  322. package/lib/mcp/prompts.cjs +302 -0
  323. package/lib/mcp/resources.cjs +227 -0
  324. package/lib/mcp/session-catchup.cjs +327 -0
  325. package/lib/mcp/surface-detect.cjs +75 -0
  326. package/lib/mcp/tool-router.cjs +1034 -0
  327. package/lib/memory/aaak-compress.cjs +403 -0
  328. package/lib/memory/aaak-compress.test.cjs +288 -0
  329. package/lib/memory/async-artifact-auto-commit.test.cjs +223 -0
  330. package/lib/memory/bearer-token.test.cjs +315 -0
  331. package/lib/memory/brain-cache-lru.test.cjs +259 -0
  332. package/lib/memory/brain-client-query-shape.test.cjs +160 -0
  333. package/lib/memory/brain-derivation-graceful-degradation.test.cjs +1019 -0
  334. package/lib/memory/brain-derivation-queue.test.cjs +539 -0
  335. package/lib/memory/brain-derivation.test.cjs +634 -0
  336. package/lib/memory/brain-derive-command.test.cjs +534 -0
  337. package/lib/memory/brain-md-invariants-validator.test.cjs +704 -0
  338. package/lib/memory/brain-md-schema.test.cjs +467 -0
  339. package/lib/memory/brain-md-staleness.test.cjs +525 -0
  340. package/lib/memory/brain-server-resolution.test.cjs +314 -0
  341. package/lib/memory/chain-recommender.test.cjs +233 -0
  342. package/lib/memory/chat-context.test.cjs +128 -0
  343. package/lib/memory/command-registry.test.cjs +220 -0
  344. package/lib/memory/cross-room-aggregator.test.cjs +909 -0
  345. package/lib/memory/dashboard-server.test.cjs +256 -0
  346. package/lib/memory/debouncer-drain-at-prompt.test.cjs +389 -0
  347. package/lib/memory/decision-capture.test.cjs +632 -0
  348. package/lib/memory/decision-capture.worker.cjs +70 -0
  349. package/lib/memory/explain-decision-command.test.cjs +521 -0
  350. package/lib/memory/explain-decision-footer.test.cjs +316 -0
  351. package/lib/memory/explored-materials-store.cjs +392 -0
  352. package/lib/memory/feynman-minto-guardian.test.cjs +736 -0
  353. package/lib/memory/feynman-minto-invariants.test.cjs +511 -0
  354. package/lib/memory/feynman-prompts-drift.test.cjs +144 -0
  355. package/lib/memory/feynman-prompts.cjs +151 -0
  356. package/lib/memory/feynman-prompts.test.cjs +96 -0
  357. package/lib/memory/folder-memory-quadruple.test.cjs +548 -0
  358. package/lib/memory/folder-memory.test.cjs +503 -0
  359. package/lib/memory/framework-chain-composer.test.cjs +515 -0
  360. package/lib/memory/frontmatter-schema-validator.test.cjs +290 -0
  361. package/lib/memory/heal-command.test.cjs +604 -0
  362. package/lib/memory/index-artifact-transaction.test.cjs +333 -0
  363. package/lib/memory/lazygraph-rs-discoveries-view.test.cjs +122 -0
  364. package/lib/memory/mcp-input-validation.test.cjs +240 -0
  365. package/lib/memory/mcp-server-brain-deps.test.cjs +270 -0
  366. package/lib/memory/mcp-stack-fallback.test.cjs +433 -0
  367. package/lib/memory/minto-debouncer.test.cjs +407 -0
  368. package/lib/memory/minto-debouncer.worker.cjs +46 -0
  369. package/lib/memory/minto-migration-v88.test.cjs +265 -0
  370. package/lib/memory/minto-schema-v88.test.cjs +390 -0
  371. package/lib/memory/mos-status-renderer.test.cjs +631 -0
  372. package/lib/memory/narrative-schema.cjs +376 -0
  373. package/lib/memory/narrative-schema.test.cjs +209 -0
  374. package/lib/memory/nav-dial.test.cjs +414 -0
  375. package/lib/memory/navigation-engine-core.test.cjs +722 -0
  376. package/lib/memory/navigation-invariants.test.cjs +483 -0
  377. package/lib/memory/offer-presenter.test.cjs +554 -0
  378. package/lib/memory/on-stop-snapshot.test.cjs +404 -0
  379. package/lib/memory/pending-tension-store.cjs +373 -0
  380. package/lib/memory/post-compact-reinjection.test.cjs +854 -0
  381. package/lib/memory/post-write-triple.test.cjs +317 -0
  382. package/lib/memory/pre-compact-snapshot.test.cjs +495 -0
  383. package/lib/memory/problem-type-router.test.cjs +656 -0
  384. package/lib/memory/query-efficiency-telemetry.test.cjs +370 -0
  385. package/lib/memory/recompile-room-references.test.cjs +392 -0
  386. package/lib/memory/recompile-room-references.worker.cjs +42 -0
  387. package/lib/memory/record-decision-dual-write.test.cjs +454 -0
  388. package/lib/memory/room-classifier-strict-mode.test.cjs +417 -0
  389. package/lib/memory/room-minto-hook.test.cjs +398 -0
  390. package/lib/memory/rs-discovery-engine.test.cjs +323 -0
  391. package/lib/memory/run-feynman-tests.cjs +1247 -0
  392. package/lib/memory/security-trifecta.test.cjs +312 -0
  393. package/lib/memory/session-start-brain-staleness.test.cjs +363 -0
  394. package/lib/memory/session-start-triple-injection.test.cjs +514 -0
  395. package/lib/memory/sessionstart-banner-formatter.cjs +318 -0
  396. package/lib/memory/sessionstart-minto-banner.test.cjs +373 -0
  397. package/lib/memory/skill-activation-router.test.cjs +419 -0
  398. package/lib/memory/stamp-artifact-write.test.cjs +304 -0
  399. package/lib/memory/statusline-active-room.test.cjs +315 -0
  400. package/lib/memory/statusline-minto-segment.test.cjs +292 -0
  401. package/lib/memory/sync-async-entry-points.test.cjs +204 -0
  402. package/lib/memory/test-bridge-writer-enhanced.cjs +452 -0
  403. package/lib/memory/test-rs-brain-substrate-shape.cjs +529 -0
  404. package/lib/memory/test-rs-brain-substrate.cjs +636 -0
  405. package/lib/memory/test-rs-breakthrough-scorer.cjs +375 -0
  406. package/lib/memory/test-rs-canon-violations.cjs +218 -0
  407. package/lib/memory/test-rs-chain-feeder-core.cjs +344 -0
  408. package/lib/memory/test-rs-chain-feeder-skill-spawn.cjs +297 -0
  409. package/lib/memory/test-rs-commercial-assessor.cjs +385 -0
  410. package/lib/memory/test-rs-differential-scorer.cjs +480 -0
  411. package/lib/memory/test-rs-discovery-engine.cjs +603 -0
  412. package/lib/memory/test-rs-domain-analyzer.cjs +492 -0
  413. package/lib/memory/test-rs-egress-primitives.cjs +420 -0
  414. package/lib/memory/test-rs-expert-mapper.cjs +547 -0
  415. package/lib/memory/test-rs-explain-command.cjs +443 -0
  416. package/lib/memory/test-rs-fetcher-academic.cjs +848 -0
  417. package/lib/memory/test-rs-fetcher-experts.cjs +496 -0
  418. package/lib/memory/test-rs-fetcher-industry.cjs +702 -0
  419. package/lib/memory/test-rs-fetcher-patents.cjs +674 -0
  420. package/lib/memory/test-rs-innovation-classifier.cjs +301 -0
  421. package/lib/memory/test-rs-mind-map.cjs +646 -0
  422. package/lib/memory/test-rs-neo4j-writer.cjs +518 -0
  423. package/lib/memory/test-rs-nl-to-query.cjs +449 -0
  424. package/lib/memory/test-rs-pinecone-bridge.cjs +277 -0
  425. package/lib/memory/test-rs-preprocessor.cjs +433 -0
  426. package/lib/memory/test-rs-query-matrix.cjs +391 -0
  427. package/lib/memory/test-rs-query-to-text.cjs +551 -0
  428. package/lib/memory/test-rs-sqlite-mirror.cjs +649 -0
  429. package/lib/memory/test-rs-thesis-generator.cjs +360 -0
  430. package/lib/memory/triple-context-formatter.cjs +473 -0
  431. package/lib/memory/triple-context-formatter.test.cjs +442 -0
  432. package/lib/memory/user-md-persona.test.cjs +565 -0
  433. package/lib/memory/userpromptsubmit-integration.test.cjs +690 -0
  434. package/lib/memory/validators/README.md +157 -0
  435. package/lib/memory/validators/brain-md-invariants.cjs +475 -0
  436. package/lib/memory/validators/brain-substrate-invariants.cjs +285 -0
  437. package/lib/memory/validators/external-academic-invariants.cjs +249 -0
  438. package/lib/memory/validators/external-industry-invariants.cjs +271 -0
  439. package/lib/memory/validators/external-patents-invariants.cjs +266 -0
  440. package/lib/memory/validators/minto-invariants.cjs +62 -0
  441. package/lib/memory/validators/navigation-invariants.cjs +340 -0
  442. package/lib/memory/validators/queue-health.cjs +95 -0
  443. package/lib/memory/validators/snapshot-integrity.cjs +129 -0
  444. package/lib/memory/validators/stale-lifecycle.cjs +116 -0
  445. package/lib/memory/vault-section-minto-generator-atomic.test.cjs +556 -0
  446. package/lib/memory/vault-section-minto-generator-atomic.worker.cjs +73 -0
  447. package/lib/memory/write-lock-atomic.test.cjs +137 -0
  448. package/lib/memory/write-lock-atomic.worker.cjs +55 -0
  449. package/lib/parity/check-parity.cjs +83 -0
  450. package/lib/presentation/presentation-server.cjs +101 -0
  451. package/lib/presentation/presentation-watcher.cjs +123 -0
  452. package/lib/quickview/hub-server.cjs +719 -0
  453. package/lib/quickview/server.cjs +533 -0
  454. package/lib/render/JTBD-PALETTES.md +145 -0
  455. package/lib/render/ROOM.md +59 -0
  456. package/lib/render/render-v2.cjs +486 -0
  457. package/lib/render/render-v2.test.cjs +267 -0
  458. package/lib/render/render.cjs +65 -0
  459. package/lib/state/ROOM.md +46 -0
  460. package/lib/state/state-md-parser.cjs +215 -0
  461. package/lib/statusline/ROOM.md +38 -0
  462. package/lib/statusline/banner-suppression.cjs +50 -0
  463. package/lib/statusline/surface-detect.cjs +85 -0
  464. package/lib/update-bootstrap.sh.template +145 -0
  465. package/lib/vault/frontmatter-schema.cjs +297 -0
  466. package/lib/vault/room-scanner.cjs +352 -0
  467. package/lib/vault/wikilink-builder.cjs +231 -0
  468. package/lib/vault/wikilink-builder.test.cjs +182 -0
  469. package/lib/wiki/graph-links.cjs +281 -0
  470. package/lib/wiki/page-renderer.cjs +229 -0
  471. package/lib/wiki/wiki-chat.cjs +81 -0
  472. package/lib/wiki/wiki-layout.cjs +1459 -0
  473. package/lib/wiki/wiki-search.cjs +142 -0
  474. package/lib/wiki/wiki-server.cjs +678 -0
  475. package/lib/wiki/wiki-watcher.cjs +105 -0
  476. package/lib/workflow/ROOM.md +47 -0
  477. package/lib/workflow/command-resolver.cjs +155 -0
  478. package/lib/workflow/command-resolver.test.cjs +235 -0
  479. package/package.json +44 -0
  480. package/pipelines/analogy/01-decompose.md +80 -0
  481. package/pipelines/analogy/02-abstract.md +87 -0
  482. package/pipelines/analogy/03-search.md +135 -0
  483. package/pipelines/analogy/04-transfer.md +101 -0
  484. package/pipelines/analogy/05-validate.md +106 -0
  485. package/pipelines/analogy/CHAIN.md +56 -0
  486. package/pipelines/discovery/01-explore-domains.md +44 -0
  487. package/pipelines/discovery/02-think-hats.md +50 -0
  488. package/pipelines/discovery/03-analyze-needs.md +54 -0
  489. package/pipelines/discovery/CHAIN.md +37 -0
  490. package/pipelines/thesis/01-structure-argument.md +45 -0
  491. package/pipelines/thesis/02-challenge-assumptions.md +48 -0
  492. package/pipelines/thesis/03-build-thesis.md +54 -0
  493. package/pipelines/thesis/CHAIN.md +37 -0
  494. package/references/brain/causal-directives.md +91 -0
  495. package/references/brain/causal-enrichment.cypher +165 -0
  496. package/references/brain/command-triggers-schema.md +226 -0
  497. package/references/brain/graph-architecture.md +317 -0
  498. package/references/brain/query-patterns.md +460 -0
  499. package/references/brain/room-hierarchy-schema.md +218 -0
  500. package/references/brain/schema.md +76 -0
  501. package/references/capability-radar/capabilities-index.md +241 -0
  502. package/references/capability-radar/changelog-cache.md +81 -0
  503. package/references/causal/causal-schema.md +103 -0
  504. package/references/design/email-template-standard.md +155 -0
  505. package/references/design/graph-visualization-standard.md +178 -0
  506. package/references/document-generation.md +179 -0
  507. package/references/hsi/HSI-TOOLS-REFERENCE.md +222 -0
  508. package/references/import-config.md +141 -0
  509. package/references/integrations/detection-patterns.md +101 -0
  510. package/references/meeting/artifact-template.md +377 -0
  511. package/references/meeting/cross-meeting-intelligence.md +216 -0
  512. package/references/meeting/cross-relationship-patterns.md +202 -0
  513. package/references/meeting/live-join-interface.md +244 -0
  514. package/references/meeting/section-mapping.md +192 -0
  515. package/references/meeting/segment-classification.md +258 -0
  516. package/references/meeting/speaker-profile-template.md +219 -0
  517. package/references/meeting/summary-template.md +348 -0
  518. package/references/meeting/transcript-patterns.md +226 -0
  519. package/references/methodology/analyze-needs.md +135 -0
  520. package/references/methodology/analyze-systems.md +121 -0
  521. package/references/methodology/analyze-timing.md +149 -0
  522. package/references/methodology/beautiful-question.md +109 -0
  523. package/references/methodology/build-knowledge.md +161 -0
  524. package/references/methodology/build-thesis.md +237 -0
  525. package/references/methodology/challenge-assumptions.md +127 -0
  526. package/references/methodology/diagnose.md +169 -0
  527. package/references/methodology/dominant-designs.md +212 -0
  528. package/references/methodology/explore-domains.md +147 -0
  529. package/references/methodology/explore-futures.md +163 -0
  530. package/references/methodology/explore-trends.md +129 -0
  531. package/references/methodology/find-bottlenecks.md +131 -0
  532. package/references/methodology/grade.md +211 -0
  533. package/references/methodology/index.md +97 -0
  534. package/references/methodology/leadership.md +200 -0
  535. package/references/methodology/lean-canvas.md +116 -0
  536. package/references/methodology/macro-trends.md +192 -0
  537. package/references/methodology/map-unknowns.md +137 -0
  538. package/references/methodology/mullins-7-domains.md +104 -0
  539. package/references/methodology/problem-types.md +65 -0
  540. package/references/methodology/root-cause.md +178 -0
  541. package/references/methodology/sapphire-encoding.md +355 -0
  542. package/references/methodology/scenario-plan.md +178 -0
  543. package/references/methodology/score-innovation.md +154 -0
  544. package/references/methodology/structure-argument.md +158 -0
  545. package/references/methodology/systems-thinking.md +159 -0
  546. package/references/methodology/think-hats.md +147 -0
  547. package/references/methodology/triz-matrix.json +751 -0
  548. package/references/methodology/triz-principles.md +501 -0
  549. package/references/methodology/user-needs.md +199 -0
  550. package/references/methodology/validate.md +163 -0
  551. package/references/methodology/value-proposition.md +244 -0
  552. package/references/opportunities/funding-lifecycle.md +103 -0
  553. package/references/opportunities/grant-api-patterns.md +99 -0
  554. package/references/opportunities/opportunity-template.md +84 -0
  555. package/references/personality/assessment-philosophy.md +72 -0
  556. package/references/personality/lexicon.md +100 -0
  557. package/references/personality/persona-chains.md +56 -0
  558. package/references/personality/pws-lexicon-full.md +499 -0
  559. package/references/personality/voice-dna.md +156 -0
  560. package/references/personas/hat-perspectives.md +76 -0
  561. package/references/personas/persona-template.md +63 -0
  562. package/references/pipeline/act-output-contract.md +88 -0
  563. package/references/pipeline/chains-index.md +39 -0
  564. package/references/pws-profile-generation.md +79 -0
  565. package/references/reasoning/reasoning-schema.md +143 -0
  566. package/references/reasoning/reasoning-template.md +68 -0
  567. package/references/reasoning/run-template.md +38 -0
  568. package/references/research/RESEARCH_14_CLAUDE_CODE_SOURCE_ARCHITECTURE.md +209 -0
  569. package/references/research/RESEARCH_15_V1.8_OPTIMIZATION_JTBD.md +375 -0
  570. package/references/research/RESEARCH_16_NATIVE_FIRST_PLUGIN_ARCHITECTURE.md +575 -0
  571. package/references/research/RESEARCH_17_MCP_UI_FRAMEWORKS.md +272 -0
  572. package/references/taxonomy/TAXONOMY.md +192 -0
  573. package/references/templates/MINTO.md +36 -0
  574. package/references/user-research/2026-04-05-leah-lawrence-session.md +202 -0
  575. package/references/vault-kit/README.md +35 -0
  576. package/references/vault-kit/app.json +12 -0
  577. package/references/vault-kit/appearance.json +12 -0
  578. package/references/vault-kit/graph.json +35 -0
  579. package/references/vault-kit/snippets/mindrian-destijl.css +297 -0
  580. package/references/vault-kit/templates/new-artifact.md +37 -0
  581. package/references/vault-kit/templates/new-meeting-note.md +35 -0
  582. package/references/vault-kit/templates/new-team-profile.md +29 -0
  583. package/references/vault-kit/templates/new-xref.md +35 -0
  584. package/references/visual/symbol-system.md +151 -0
  585. package/skills/MOSDeckEngine/SKILL.md +325 -0
  586. package/skills/brain-connector/SKILL.md +114 -0
  587. package/skills/context-engine/SKILL.md +147 -0
  588. package/skills/conversation-mode/SKILL.md +102 -0
  589. package/skills/larry-personality/SKILL.md +219 -0
  590. package/skills/larry-personality/framework-chains.md +92 -0
  591. package/skills/larry-personality/mode-engine.md +185 -0
  592. package/skills/mullins-scaffold/SKILL.md +61 -0
  593. package/skills/mullins-scaffold/scaffold.json +146 -0
  594. package/skills/pws-methodology/SKILL.md +49 -0
  595. package/skills/room-passive/SKILL.md +165 -0
  596. package/skills/room-proactive/SKILL.md +250 -0
  597. package/skills/ui-system/SKILL.md +277 -0
@@ -0,0 +1,865 @@
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
+ };