@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,267 @@
1
+ #!/usr/bin/env node
2
+ 'use strict';
3
+
4
+ /*
5
+ * Copyright (c) 2026 Mindrian. BSL 1.1.
6
+ *
7
+ * Phase 102-01 -- renderer contract regression fence.
8
+ *
9
+ * History:
10
+ * - Phase 99-03 shipped a positional 4-arg pass-through stub at this
11
+ * module path with the contract `render(zones, mode, operator, tier)
12
+ * -> envelope` and tests targeted that envelope.
13
+ * - Phase 102-01 (CONTEXT D-10, RESEARCH §2) evolves the renderer to
14
+ * a destructured single-arg signature `render({ zones, mode,
15
+ * operator, tier, jtbd, tokenBudget, roomDir, provenance }) ->
16
+ * { rendered, contract }`. The legacy 4-arg signature lives on at
17
+ * lib/render/render.cjs (v1 shim) for unmigrated callers.
18
+ * - This file is the new contract regression fence for Phase 102. The
19
+ * scope mirrors the 99-03 fence (operator vocab + defaults + error
20
+ * paths + envelope stability + mode/tier passthrough) but exercises
21
+ * the v2 destructured surface.
22
+ *
23
+ * Scenarios (8 IIFE blocks):
24
+ * 1. 5-operator round-trip via v2 destructured API. JUST_TALK is the
25
+ * explicit suppress case; the other 4 round-trip into composed output.
26
+ * 2. JUST_TALK default when operator omitted (undefined) -- explicit
27
+ * operator: undefined is treated as "no operator" (renderer does
28
+ * not throw and returns a normal envelope).
29
+ * 3. JUST_TALK default when operator null -- same as above; null is a
30
+ * no-op operator value, NOT 'JUST_TALK' (which is a real value).
31
+ * 4. v2 must not throw on unknown operator strings -- Phase 102 owns
32
+ * operator semantics; only specific operators (JUST_TALK,
33
+ * METHODOLOGY) trigger gates. Unknown operators fall through to
34
+ * composition so callers can route arbitrary string values without
35
+ * a defensive try/catch wrapper.
36
+ * 5. Envelope shape stable -- exactly 2 keys: rendered + contract. No
37
+ * `_stub` provenance leak (Phase 99-03 sentinel removed).
38
+ * 6. mode passthrough -- 'A' / 'B' / 'tier-0' all pass through; v2
39
+ * does NOT validate mode (Phase 102-04 owns Mode B prefix logic).
40
+ * 7. tier passthrough -- 0 / 1 / 2 / 3 all pass through; v2 does NOT
41
+ * validate tier (Phase 90 follow-on owns tier-check semantics).
42
+ * 8. OPERATORS export is frozen and equals the 5 canonical names.
43
+ *
44
+ * Registered in lib/memory/run-feynman-tests.cjs (same registration the
45
+ * 99-03 fence used; Phase 102 inherits the slot).
46
+ */
47
+
48
+ const assert = require('node:assert/strict');
49
+ const path = require('node:path');
50
+
51
+ const RENDER_MODULE = path.resolve(__dirname, 'render-v2.cjs');
52
+ const { render, OPERATORS } = require(RENDER_MODULE);
53
+
54
+ let passed = 0;
55
+ let failed = 0;
56
+
57
+ function ok(name) {
58
+ passed += 1;
59
+ process.stdout.write(' ok ' + name + '\n');
60
+ }
61
+
62
+ function fail(name, err) {
63
+ failed += 1;
64
+ process.stdout.write(' FAIL ' + name + '\n');
65
+ if (err) process.stdout.write(' ' + (err.message || String(err)) + '\n');
66
+ }
67
+
68
+ const CANONICAL = [
69
+ 'JUST_TALK',
70
+ 'EXPLORE_CAPTURE',
71
+ 'BUILD_ROOM',
72
+ 'METHODOLOGY',
73
+ 'DECISION_GATE',
74
+ ];
75
+
76
+ // ---------------------------------------------------------------------------
77
+ // Scenario 1: 5-operator round-trip via the v2 destructured API.
78
+ // ---------------------------------------------------------------------------
79
+ (function test1_fiveOperatorRoundTrip() {
80
+ for (const op of CANONICAL) {
81
+ const label = '5-operator round-trip: ' + op;
82
+ try {
83
+ const env = render({
84
+ zones: { header: '-- h --', body: 'b' },
85
+ mode: 'A',
86
+ operator: op,
87
+ tier: 1,
88
+ });
89
+ assert.ok(env && typeof env === 'object', 'envelope is an object');
90
+ assert.equal(typeof env.rendered, 'string', 'rendered is a string');
91
+ assert.ok(env.contract && typeof env.contract === 'object', 'contract is an object');
92
+
93
+ if (op === 'JUST_TALK') {
94
+ // Explicit suppress per Phase 102-01 algorithm step 6.
95
+ assert.equal(env.rendered, '', 'JUST_TALK rendered === ""');
96
+ assert.equal(env.contract.suppressed, true, 'JUST_TALK contract.suppressed === true');
97
+ assert.equal(env.contract.reason, 'just-talk', 'JUST_TALK contract.reason === "just-talk"');
98
+ } else {
99
+ // All other operators compose normally.
100
+ assert.ok(env.rendered.indexOf('-- h --') !== -1, op + ' header round-trips');
101
+ assert.ok(env.rendered.indexOf('b') !== -1, op + ' body round-trips');
102
+ }
103
+ ok(label);
104
+ } catch (e) { fail(label, e); }
105
+ }
106
+ })();
107
+
108
+ // ---------------------------------------------------------------------------
109
+ // Scenario 2: undefined operator does not crash, renders normally.
110
+ // ---------------------------------------------------------------------------
111
+ (function test2_undefinedOperator() {
112
+ const label = 'undefined operator does not crash';
113
+ try {
114
+ const env = render({
115
+ zones: { header: 'h', body: 'b' },
116
+ mode: 'A',
117
+ operator: undefined,
118
+ tier: 1,
119
+ });
120
+ assert.equal(typeof env.rendered, 'string', 'rendered is a string');
121
+ assert.ok(env.rendered.indexOf('h') !== -1, 'header rendered without an operator');
122
+ assert.ok(env.rendered.indexOf('b') !== -1, 'body rendered without an operator');
123
+ ok(label);
124
+ } catch (e) { fail(label, e); }
125
+ })();
126
+
127
+ // ---------------------------------------------------------------------------
128
+ // Scenario 3: null operator does not crash, renders normally.
129
+ // ---------------------------------------------------------------------------
130
+ (function test3_nullOperator() {
131
+ const label = 'null operator does not crash';
132
+ try {
133
+ const env = render({
134
+ zones: { header: 'h', body: 'b' },
135
+ mode: 'A',
136
+ operator: null,
137
+ tier: 1,
138
+ });
139
+ assert.equal(typeof env.rendered, 'string', 'rendered is a string');
140
+ assert.ok(env.rendered.indexOf('h') !== -1, 'header rendered with null operator');
141
+ ok(label);
142
+ } catch (e) { fail(label, e); }
143
+ })();
144
+
145
+ // ---------------------------------------------------------------------------
146
+ // Scenario 4: unknown operator does not throw -- v2 falls through to
147
+ // composition. Phase 99-03 stub threw; Phase 102 deliberately tolerates
148
+ // unknown operators so upstream callers (hooks, /mos:operator command)
149
+ // can route arbitrary strings without defensive try/catch wrappers.
150
+ // ---------------------------------------------------------------------------
151
+ (function test4_unknownOperatorTolerated() {
152
+ const label = 'unknown operator does not throw (v2 tolerates)';
153
+ try {
154
+ let thrown = null;
155
+ let env = null;
156
+ try {
157
+ env = render({
158
+ zones: { header: 'h', body: 'b' },
159
+ mode: 'A',
160
+ operator: 'NOT_A_REAL_OPERATOR',
161
+ tier: 1,
162
+ });
163
+ } catch (e) { thrown = e; }
164
+ assert.equal(thrown, null, 'unknown operator must NOT throw');
165
+ assert.ok(env && typeof env.rendered === 'string', 'still returns a normal envelope');
166
+ assert.ok(env.rendered.indexOf('h') !== -1, 'composes normally for unknown operator');
167
+ ok(label);
168
+ } catch (e) { fail(label, e); }
169
+ })();
170
+
171
+ // ---------------------------------------------------------------------------
172
+ // Scenario 5: Envelope shape stable (exactly 2 keys, no _stub leak).
173
+ // ---------------------------------------------------------------------------
174
+ (function test5_envelopeShapeStable() {
175
+ const label = 'envelope shape stable: exactly 2 keys (rendered + contract)';
176
+ try {
177
+ const env = render({
178
+ zones: { header: 'h' },
179
+ mode: 'A',
180
+ operator: 'BUILD_ROOM',
181
+ tier: 1,
182
+ });
183
+ const keys = Object.keys(env).sort();
184
+ assert.deepEqual(keys, ['contract', 'rendered'], 'envelope keys are exactly {rendered, contract}');
185
+ assert.ok(!('_stub' in env), 'no Phase 99-03 _stub provenance leak');
186
+ assert.ok(!('zones' in env), 'no zones leak (Phase 99-03 pass-through removed)');
187
+ ok(label);
188
+ } catch (e) { fail(label, e); }
189
+ })();
190
+
191
+ // ---------------------------------------------------------------------------
192
+ // Scenario 6: mode passthrough.
193
+ // ---------------------------------------------------------------------------
194
+ (function test6_modePassthrough() {
195
+ const label = 'mode passthrough: A / B / tier-0 all render without throwing';
196
+ try {
197
+ for (const mode of ['A', 'B', 'tier-0']) {
198
+ const env = render({
199
+ zones: { header: 'h' },
200
+ mode: mode,
201
+ operator: 'BUILD_ROOM',
202
+ tier: 1,
203
+ });
204
+ assert.equal(typeof env.rendered, 'string', 'mode ' + mode + ' renders');
205
+ }
206
+ // Stub does NOT validate mode -- arbitrary string also passes through.
207
+ const env = render({
208
+ zones: { header: 'h' },
209
+ mode: 'arbitrary-mode',
210
+ operator: 'BUILD_ROOM',
211
+ tier: 1,
212
+ });
213
+ assert.equal(typeof env.rendered, 'string', 'arbitrary mode renders');
214
+ ok(label);
215
+ } catch (e) { fail(label, e); }
216
+ })();
217
+
218
+ // ---------------------------------------------------------------------------
219
+ // Scenario 7: tier passthrough.
220
+ // ---------------------------------------------------------------------------
221
+ (function test7_tierPassthrough() {
222
+ const label = 'tier passthrough: 0 / 1 / 2 / 3 all render without throwing';
223
+ try {
224
+ for (const tier of [0, 1, 2, 3]) {
225
+ const env = render({
226
+ zones: { header: 'h' },
227
+ mode: 'A',
228
+ operator: 'BUILD_ROOM',
229
+ tier: tier,
230
+ });
231
+ assert.equal(typeof env.rendered, 'string', 'tier ' + tier + ' renders');
232
+ }
233
+ // Stub does NOT validate tier -- string tier also passes through.
234
+ const env = render({
235
+ zones: { header: 'h' },
236
+ mode: 'A',
237
+ operator: 'BUILD_ROOM',
238
+ tier: 'arbitrary-tier',
239
+ });
240
+ assert.equal(typeof env.rendered, 'string', 'arbitrary tier renders');
241
+ ok(label);
242
+ } catch (e) { fail(label, e); }
243
+ })();
244
+
245
+ // ---------------------------------------------------------------------------
246
+ // Scenario 8: OPERATORS export is frozen and equals the 5 canonical names.
247
+ // ---------------------------------------------------------------------------
248
+ (function test8_operatorsExportFrozen() {
249
+ const label = 'OPERATORS export is frozen and equals 5 canonical names';
250
+ try {
251
+ assert.ok(Array.isArray(OPERATORS), 'OPERATORS is an array');
252
+ assert.deepEqual(
253
+ [...OPERATORS].sort(),
254
+ [...CANONICAL].sort(),
255
+ 'OPERATORS equals 5 canonical names'
256
+ );
257
+ assert.ok(Object.isFrozen(OPERATORS), 'OPERATORS is frozen');
258
+ ok(label);
259
+ } catch (e) { fail(label, e); }
260
+ })();
261
+
262
+ process.stdout.write('\n');
263
+ process.stdout.write(
264
+ 'Phase 102-01 renderer contract regression: ' +
265
+ passed + ' passed, ' + failed + ' failed\n'
266
+ );
267
+ process.exit(failed === 0 ? 0 : 1);
@@ -0,0 +1,65 @@
1
+ #!/usr/bin/env node
2
+ 'use strict';
3
+
4
+ /*
5
+ * Copyright (c) 2026 Mindrian. BSL 1.1.
6
+ *
7
+ * Phase 102-01 -- v1 renderer shim (legacy 4-arg positional signature).
8
+ *
9
+ * Per Phase 102 CONTEXT.md D-10, this module ships in the same wave as
10
+ * lib/render/render-v2.cjs and exposes the legacy 4-arg signature so
11
+ * any caller that has not yet migrated to the v2 destructured API keeps
12
+ * working unchanged. Phase 104 migrates each call site explicitly;
13
+ * Phase 105 polling flags any unmigrated call site as a compliance gap.
14
+ *
15
+ * render(zones, mode, operator, tier) -> string
16
+ *
17
+ * The shim:
18
+ * - Forwards to render-v2's destructured API.
19
+ * - Fills jtbd=null, tokenBudget={used:0,total:1}, roomDir from env or
20
+ * process.cwd(), provenance=null per CONTEXT D-10.
21
+ * - Returns just the `rendered` string (legacy callers expect a string,
22
+ * not the {rendered, contract} envelope).
23
+ *
24
+ * Constraints (Phase 87 invariant):
25
+ * - Zero new runtime dependencies.
26
+ * - CJS only.
27
+ * - The shim has zero rendering logic of its own; all muscle lives in
28
+ * render-v2. If render-v2's behavior changes, the shim adapts for
29
+ * free.
30
+ *
31
+ * Canon parts:
32
+ * 7 (Reuse Before Build) -- one renderer module, two entry points;
33
+ * no parallel formatter implementations.
34
+ * 8 (Graph Boundary) -- inherits Canon Part 8 compliance from
35
+ * render-v2 by construction (no Brain calls anywhere in the chain).
36
+ */
37
+
38
+ const v2 = require('./render-v2.cjs');
39
+
40
+ /**
41
+ * Legacy 4-arg renderer entry point.
42
+ *
43
+ * @param {object|null|undefined} zones - 4-zone payload
44
+ * ({ header, body, signals?, footer? }).
45
+ * @param {string|null|undefined} mode - 'A' | 'B' | 'tier-0'.
46
+ * @param {string|null|undefined} operator - one of OPERATORS or null.
47
+ * @param {number|string|null|undefined} tier - 0|1|2|3.
48
+ * @returns {string} Composed 4-zone string ('' when operator suppresses).
49
+ */
50
+ function render(zones, mode, operator, tier) {
51
+ const result = v2.render({
52
+ zones: zones,
53
+ mode: mode,
54
+ operator: operator,
55
+ tier: tier,
56
+ jtbd: null,
57
+ tokenBudget: { used: 0, total: 1 },
58
+ roomDir: process.env.MINDRIAN_ROOM_DIR || process.cwd(),
59
+ provenance: null,
60
+ });
61
+ // v1 callers expect a string, not the {rendered, contract} envelope.
62
+ return result.rendered;
63
+ }
64
+
65
+ module.exports = { render, OPERATORS: v2.OPERATORS };
@@ -0,0 +1,46 @@
1
+ ---
2
+ folder: lib/state
3
+ founding_phase: 100
4
+ purpose: STATE.md parsing utilities -- shared across phases that read project / room STATE.md
5
+ icm_layer: 3
6
+ canon_parts: [3, 7]
7
+ license: BSL-1.1
8
+ ---
9
+
10
+ # lib/state/ -- STATE.md Parsing Utilities
11
+
12
+ This directory is the home for reusable parsers that read `STATE.md` (project-level
13
+ at `.planning/STATE.md` and per-room at `<roomDir>/STATE.md`).
14
+
15
+ ## Files
16
+
17
+ | File | Purpose | Founded |
18
+ | --------------------- | ---------------------------------------------------------- | ------- |
19
+ | `state-md-parser.cjs` | Decisions section parser (bullet, numbered, table formats) | 100-02 |
20
+
21
+ ## Why this directory exists
22
+
23
+ Phase 100's heuristic JTBD classifier needs to read the last 3 decisions from
24
+ STATE.md as Stratum 3 input (per Phase 100 CONTEXT D-04). Phase 100 CONTEXT
25
+ `code_context` calls out the parser as a side-effect deliverable to keep STATE.md
26
+ parsing centralized rather than duplicated across phases (101-106 will all read
27
+ the same source).
28
+
29
+ Anything that reads `## Decisions` or `### Decisions` from a STATE.md file must
30
+ import from here, not roll its own regex.
31
+
32
+ ## Pattern
33
+
34
+ - Pure functions; no I/O. The caller reads the file; the parser parses the string.
35
+ - Returns chronological-recent-first arrays.
36
+ - Tolerant of bullet, numbered list, and table formats.
37
+ - Detects optional `[methodology: /mos:X]` and `[jtbd: <id>]` annotations.
38
+ - CJS, zero npm deps, Canon Part 8 LOCAL-only.
39
+
40
+ ## Canon parts
41
+
42
+ - **Part 3 (Tri-Context Decision Gate):** Decisions are the LOCAL context that
43
+ the gate reads. STATE.md is the durable record; this parser is the canonical
44
+ reader.
45
+ - **Part 7 (Reuse Before Build):** centralizing the parser keeps Phases 101-106
46
+ from reinventing it.
@@ -0,0 +1,215 @@
1
+ /*
2
+ * Copyright (c) 2026 Mindrian. BSL 1.1.
3
+ *
4
+ * Phase 100-02 Task 1 -- STATE.md Decisions section parser
5
+ * =========================================================
6
+ * Reusable shared parser for the `## Decisions` (or `### Decisions`)
7
+ * section that appears in both project-level STATE.md and per-room
8
+ * STATE.md files. Phase 100 ships this as a side-effect deliverable
9
+ * (Decision #15 / CONTEXT code_context) -- the JTBD classifier
10
+ * (lib/hmi/jtbd-classifier.cjs) reads recent decisions as Stratum 3
11
+ * input per CONTEXT D-04, and Phase 101+ phases consume the same
12
+ * parser without re-implementing.
13
+ *
14
+ * Public API:
15
+ * parseDecisions(stateMdText)
16
+ * -> Array<{
17
+ * date: string|null, // ISO or human-readable
18
+ * verb: string|null, // one of Canon Part 3 ten or 'Free-Text'
19
+ * summary: string, // bullet body sans annotations
20
+ * methodology_hook?: string, // e.g. '/mos:rs-fetch'
21
+ * jtbd?: string, // e.g. 'find-bottleneck'
22
+ * raw: string // original line content
23
+ * }>
24
+ *
25
+ * recentDecisions(stateMdText, n=3)
26
+ * -> Array<...same shape...>
27
+ *
28
+ * Behavior:
29
+ * - Pure function; no I/O. Caller reads the file.
30
+ * - Tolerant of bullet, numbered list, or table formats.
31
+ * - Tolerant of trailing/leading whitespace and blank lines.
32
+ * - Returns chronological-recent-first (top of section assumed newest;
33
+ * STATE.md convention is to prepend recent entries).
34
+ * - Detects optional `[methodology: /mos:X]` annotation.
35
+ * - Detects optional `[jtbd: <id>]` annotation.
36
+ * - Empty / missing section returns [].
37
+ *
38
+ * Canon Part 8 compliance:
39
+ * - LOCAL-only. No Brain query. No external I/O. Zero npm deps.
40
+ *
41
+ * License: BSL 1.1.
42
+ */
43
+
44
+ 'use strict';
45
+
46
+ // Canon Part 3 ten verbs (lowercase compare). Plus 'Free-Text' which
47
+ // is verb #10. We accept any leading verb token followed by ':' or '-'
48
+ // or ' '. Unknown verbs are treated as Free-Text.
49
+ const CANON_VERBS = [
50
+ 'run methodology',
51
+ 'reformulate',
52
+ 'spawn sub-agent',
53
+ 'navigate graph',
54
+ "devil's advocate",
55
+ 'devils advocate',
56
+ 'scenario plan',
57
+ 'synthesize',
58
+ 'bank opportunity',
59
+ 'defer',
60
+ 'free-text',
61
+ ];
62
+
63
+ // Match `[methodology: /mos:X]` or `[methodology:/mos:X]`.
64
+ const METHODOLOGY_RE = /\[methodology:\s*(\/mos:[a-z0-9_-]+|\/gsd:[a-z0-9_-]+)\s*\]/i;
65
+
66
+ // Match `[jtbd: <id>]` -- id is kebab-case.
67
+ const JTBD_RE = /\[jtbd:\s*([a-z0-9_-]+)\s*\]/i;
68
+
69
+ // Match leading date in ISO (YYYY-MM-DD) or simple human form (YYYY/MM/DD,
70
+ // "Apr 29 2026", etc.). We extract the most-leading match.
71
+ const ISO_DATE_RE = /^(\d{4}-\d{2}-\d{2})/;
72
+
73
+ // Bullet/numbered list line detector. Capture the body after the marker.
74
+ // - foo
75
+ // * foo
76
+ // 1. foo
77
+ // 2) foo
78
+ const BULLET_RE = /^\s*(?:[-*+]|\d+[.)])\s+(.+)$/;
79
+
80
+ // Table-row detector. Capture cells separated by `|`.
81
+ const TABLE_ROW_RE = /^\s*\|(.+)\|\s*$/;
82
+
83
+ function findSection(text, headingRegex) {
84
+ // Returns the body lines BETWEEN headingRegex and the next heading of equal
85
+ // or greater scope (## or ###).
86
+ const lines = text.split(/\r?\n/);
87
+ let inSection = false;
88
+ const out = [];
89
+ for (let i = 0; i < lines.length; i++) {
90
+ const line = lines[i];
91
+ if (!inSection) {
92
+ if (headingRegex.test(line)) {
93
+ inSection = true;
94
+ }
95
+ continue;
96
+ }
97
+ // Stop at next heading of any level (### Decisions usually sits inside ##
98
+ // Accumulated Context, so we stop at ## OR ### OR #).
99
+ if (/^#{1,6}\s+\S/.test(line)) {
100
+ break;
101
+ }
102
+ out.push(line);
103
+ }
104
+ return out;
105
+ }
106
+
107
+ function extractDate(body) {
108
+ const m = body.match(ISO_DATE_RE);
109
+ if (m) return m[1];
110
+ // Fallback: try "[Phase 71]" style as date-substitute. Treat as null date.
111
+ return null;
112
+ }
113
+
114
+ function extractVerb(body) {
115
+ const lower = body.toLowerCase().trim();
116
+ for (const verb of CANON_VERBS) {
117
+ // Verb appears at start, possibly after a date prefix that we already saw.
118
+ // We accept verb followed by space, colon, or end-of-segment.
119
+ if (lower.startsWith(verb + ' ') || lower.startsWith(verb + ':') || lower === verb) {
120
+ return verb === "devils advocate" ? "Devil's Advocate" : capitalizeVerb(verb);
121
+ }
122
+ }
123
+ return null;
124
+ }
125
+
126
+ function capitalizeVerb(v) {
127
+ return v.split(' ')
128
+ .map(w => w === "advocate" || w === "methodology" || w === "graph" || w === "plan" || w === "opportunity" || w === "agent" || w === "sub-agent"
129
+ ? w.charAt(0).toUpperCase() + w.slice(1)
130
+ : w.charAt(0).toUpperCase() + w.slice(1))
131
+ .join(' ');
132
+ }
133
+
134
+ function parseLine(rawBody) {
135
+ // rawBody is the line body AFTER bullet/number marker stripping.
136
+ const raw = rawBody.trim();
137
+ if (!raw) return null;
138
+
139
+ // Date
140
+ const date = extractDate(raw);
141
+
142
+ // Annotations
143
+ const mhMatch = raw.match(METHODOLOGY_RE);
144
+ const jtbdMatch = raw.match(JTBD_RE);
145
+ const methodology_hook = mhMatch ? mhMatch[1] : undefined;
146
+ const jtbd = jtbdMatch ? jtbdMatch[1] : undefined;
147
+
148
+ // Strip annotations from summary
149
+ let summary = raw
150
+ .replace(METHODOLOGY_RE, '')
151
+ .replace(JTBD_RE, '')
152
+ .replace(/\s{2,}/g, ' ')
153
+ .trim();
154
+
155
+ // Strip leading date from summary
156
+ if (date) summary = summary.replace(ISO_DATE_RE, '').trim();
157
+
158
+ // Strip leading "[Phase XX]:" prefix common in plugin STATE.md
159
+ summary = summary.replace(/^\[[^\]]+\]:\s*/, '').trim();
160
+
161
+ // Verb extraction (after stripping date + phase prefix)
162
+ const verb = extractVerb(summary) || 'Free-Text';
163
+
164
+ const out = { date, verb, summary, raw };
165
+ if (methodology_hook) out.methodology_hook = methodology_hook;
166
+ if (jtbd) out.jtbd = jtbd;
167
+ return out;
168
+ }
169
+
170
+ function parseDecisions(stateMdText) {
171
+ if (!stateMdText || typeof stateMdText !== 'string') return [];
172
+
173
+ // Tolerant headers: "## Decisions" or "### Decisions" (case-sensitive on
174
+ // word but allowing trailing whitespace).
175
+ const headingRe = /^#{2,3}\s+Decisions\s*$/;
176
+ const body = findSection(stateMdText, headingRe);
177
+
178
+ const out = [];
179
+ for (const line of body) {
180
+ // Bullet / numbered
181
+ const bulletMatch = line.match(BULLET_RE);
182
+ if (bulletMatch) {
183
+ const parsed = parseLine(bulletMatch[1]);
184
+ if (parsed) out.push(parsed);
185
+ continue;
186
+ }
187
+ // Table row -- skip header rows (contain only --- and |)
188
+ const tableMatch = line.match(TABLE_ROW_RE);
189
+ if (tableMatch) {
190
+ const inner = tableMatch[1];
191
+ // Skip header divider row like | --- | --- |
192
+ if (/^\s*[-:|\s]+\s*$/.test(inner)) continue;
193
+ // Take the LAST non-empty cell as summary, FIRST cell as potential date
194
+ const cells = inner.split('|').map(c => c.trim()).filter(c => c !== '');
195
+ if (cells.length === 0) continue;
196
+ const joined = cells.join(' ');
197
+ const parsed = parseLine(joined);
198
+ if (parsed) out.push(parsed);
199
+ continue;
200
+ }
201
+ // Plain prose lines: skip (not a decision row).
202
+ }
203
+ return out;
204
+ }
205
+
206
+ function recentDecisions(stateMdText, n) {
207
+ const all = parseDecisions(stateMdText);
208
+ const limit = (typeof n === 'number' && n > 0) ? n : 3;
209
+ return all.slice(0, limit);
210
+ }
211
+
212
+ module.exports = {
213
+ parseDecisions,
214
+ recentDecisions,
215
+ };
@@ -0,0 +1,38 @@
1
+ # lib/statusline/
2
+
3
+ ICM Layer 0 identity for the statusline visibility subsystem (per CLAUDE.md decision #15).
4
+
5
+ ## Purpose
6
+
7
+ Shared modules that support the MindrianOS statusline visibility surface across
8
+ CLI / Desktop / Cowork. Phase 106 (statusline-visibility-context-window-broadcast)
9
+ introduced this directory; the substrate is consumed by:
10
+
11
+ - scripts/context-monitor (CLI rich statusline renderer; Phase 106 D-02)
12
+ - scripts/statusline-fallback-echo.cjs (Desktop / Cowork prose echo; Phase 106 D-04)
13
+ - scripts/doctor.cjs class G (visibility detector; Phase 106 D-03)
14
+
15
+ ## Modules
16
+
17
+ - banner-suppression.cjs - shouldSuppress(touchFileContent, currentVersion, now).
18
+ 24h timing + version-bump invalidation contract for the one-time visibility
19
+ banner. Extracted from the Plan 106-03 inline contract. Pure function, no I/O.
20
+ - surface-detect.cjs - detectStatuslineSurface() returns 'CLI' / 'DESKTOP' /
21
+ 'COWORK'. Distinct from lib/mcp/surface-detect.cjs (which picks MCP transport
22
+ at server startup; lowercase strings + transport field).
23
+
24
+ ## Canon
25
+
26
+ - Part 3: statusline + fallback echo are LOCAL-context surfaces for the Decision Gate.
27
+ - Part 8: zero remote egress; reads are local files only. Telemetry stays local.
28
+
29
+ ## Tests
30
+
31
+ - tests/test-statusline-banner-suppression.cjs - 5 contract tests (Plan 106-03
32
+ fenced inline; Plan 106-04 swapped to require() the shared module).
33
+ - tests/test-surface-detect.cjs - 6 tests for the literal-string contract
34
+ (CLI / DESKTOP / COWORK + never-null) and the explicit env override.
35
+ - tests/test-fallback-echo-compose.cjs - 7 tests for echo composition + per
36
+ surface routing + banner suppression + graceful state-file absence.
37
+ - tests/test-fallback-echo-30day.cjs - 5 tests for the 30-day default-flip
38
+ + explicit MINDRIAN_STATUSLINE_FALLBACK_ECHO override behavior.
@@ -0,0 +1,50 @@
1
+ /*
2
+ * Copyright (c) 2026 Mindrian. BSL 1.1.
3
+ *
4
+ * Phase 106 Plan 04 -- statusline banner-suppression module.
5
+ *
6
+ * Extracted from the inline contract fenced at
7
+ * tests/test-statusline-banner-suppression.cjs:49-60 by Plan 106-03
8
+ * (see 106-03-SUMMARY.md "Plan 106-04 Contract Handoff" section).
9
+ *
10
+ * Touch-file path: ~/.mindrian/banner-state/statusline-visibility-warned.json
11
+ * Shape:
12
+ * {
13
+ * "status": "ok" | "warn" | "error",
14
+ * "last_check": "<ISO>", // populated when status='ok'
15
+ * "last_warned": "<ISO>", // populated when status='warn' or 'error'
16
+ * "installed_version": "<plugin.json version>"
17
+ * }
18
+ *
19
+ * shouldSuppress() returns true ONLY when:
20
+ * - touch-file content is non-null, AND
21
+ * - installed_version === current plugin version (version-bump invalidates), AND
22
+ * - status in {'warn', 'error'}, AND
23
+ * - last_warned is a parseable ISO timestamp, AND
24
+ * - last_warned is within 24h of `now`.
25
+ *
26
+ * Used by:
27
+ * - scripts/statusline-fallback-echo.cjs (D-04 fallback echo, 106-04)
28
+ * - tests/test-statusline-banner-suppression.cjs (5-test contract fence, 106-03)
29
+ *
30
+ * Canon Part 8: pure function, no I/O, no network, no side effects.
31
+ * Pure CJS, node built-ins only, zero npm deps (Phase 87 invariant).
32
+ */
33
+
34
+ 'use strict';
35
+
36
+ const TWENTY_FOUR_HOURS_MS = 24 * 3600 * 1000;
37
+
38
+ function shouldSuppress(touchFileContent, currentVersion, now) {
39
+ if (now === undefined) now = Date.now();
40
+ if (!touchFileContent) return false;
41
+ if (touchFileContent.installed_version !== currentVersion) return false;
42
+ if (touchFileContent.status !== 'warn' && touchFileContent.status !== 'error') return false;
43
+ if (!touchFileContent.last_warned) return false;
44
+ const warnedAt = Date.parse(touchFileContent.last_warned);
45
+ if (Number.isNaN(warnedAt)) return false;
46
+ const ageMs = now - warnedAt;
47
+ return ageMs < TWENTY_FOUR_HOURS_MS;
48
+ }
49
+
50
+ module.exports = { shouldSuppress, TWENTY_FOUR_HOURS_MS };