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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (590) hide show
  1. package/CHANGELOG.md +340 -13
  2. package/README.md +76 -579
  3. package/bin/cli.js +166 -40
  4. package/lib/core/active-plugin-root.cjs +207 -0
  5. package/package.json +7 -13
  6. package/.claude-plugin/plugin.json +0 -21
  7. package/.mcp.json +0 -9
  8. package/agents/brain-query.md +0 -80
  9. package/agents/framework-runner.md +0 -237
  10. package/agents/grading.md +0 -188
  11. package/agents/investor.md +0 -128
  12. package/agents/larry-extended.md +0 -135
  13. package/agents/opportunity-scanner.md +0 -91
  14. package/agents/persona-analyst.md +0 -132
  15. package/agents/research.md +0 -89
  16. package/agents/reverse-salient-agent.md +0 -27
  17. package/bin/mindrian-mcp-server.cjs +0 -182
  18. package/bin/mindrian-tools.cjs +0 -765
  19. package/commands/act.md +0 -433
  20. package/commands/admin.md +0 -404
  21. package/commands/analyze-needs.md +0 -36
  22. package/commands/analyze-systems.md +0 -33
  23. package/commands/analyze-timing.md +0 -36
  24. package/commands/auto-explore.md +0 -64
  25. package/commands/beautiful-question.md +0 -34
  26. package/commands/brain-derive.md +0 -78
  27. package/commands/build-knowledge.md +0 -36
  28. package/commands/build-thesis.md +0 -40
  29. package/commands/causal.md +0 -228
  30. package/commands/challenge-assumptions.md +0 -33
  31. package/commands/compare-ventures.md +0 -77
  32. package/commands/dashboard.md +0 -110
  33. package/commands/deep-grade.md +0 -76
  34. package/commands/diagnose.md +0 -52
  35. package/commands/diagnostics.md +0 -145
  36. package/commands/doctor.md +0 -151
  37. package/commands/dominant-designs.md +0 -34
  38. package/commands/explain-decision.md +0 -87
  39. package/commands/explore-domains.md +0 -36
  40. package/commands/explore-futures.md +0 -34
  41. package/commands/explore-trends.md +0 -36
  42. package/commands/export.md +0 -103
  43. package/commands/file-meeting.md +0 -724
  44. package/commands/find-analogies.md +0 -182
  45. package/commands/find-bottlenecks.md +0 -56
  46. package/commands/find-connections.md +0 -70
  47. package/commands/funding.md +0 -81
  48. package/commands/grade.md +0 -197
  49. package/commands/graph.md +0 -128
  50. package/commands/hat-briefing.md +0 -119
  51. package/commands/heal.md +0 -196
  52. package/commands/help.md +0 -399
  53. package/commands/hmi-status.md +0 -172
  54. package/commands/jtbd.md +0 -241
  55. package/commands/leadership.md +0 -73
  56. package/commands/lean-canvas.md +0 -34
  57. package/commands/macro-trends.md +0 -34
  58. package/commands/map-unknowns.md +0 -34
  59. package/commands/memory.md +0 -173
  60. package/commands/models.md +0 -175
  61. package/commands/mos-reason.md +0 -279
  62. package/commands/mullins.md +0 -114
  63. package/commands/new-project.md +0 -481
  64. package/commands/onboard.md +0 -434
  65. package/commands/operator.md +0 -149
  66. package/commands/opportunities.md +0 -144
  67. package/commands/organize.md +0 -497
  68. package/commands/persona.md +0 -192
  69. package/commands/pipeline.md +0 -106
  70. package/commands/present.md +0 -91
  71. package/commands/publish.md +0 -201
  72. package/commands/query.md +0 -124
  73. package/commands/radar.md +0 -72
  74. package/commands/reanalyze.md +0 -91
  75. package/commands/research.md +0 -190
  76. package/commands/room.md +0 -352
  77. package/commands/rooms.md +0 -598
  78. package/commands/root-cause.md +0 -34
  79. package/commands/rs-experts.md +0 -79
  80. package/commands/rs-explain.md +0 -94
  81. package/commands/rs-fetch.md +0 -88
  82. package/commands/rs-thesis.md +0 -79
  83. package/commands/scenario-plan.md +0 -34
  84. package/commands/scheduled-tasks.md +0 -285
  85. package/commands/score-innovation.md +0 -37
  86. package/commands/scout.md +0 -239
  87. package/commands/setup.md +0 -618
  88. package/commands/snapshot.md +0 -147
  89. package/commands/speakers.md +0 -84
  90. package/commands/splash.md +0 -28
  91. package/commands/status.md +0 -75
  92. package/commands/structure-argument.md +0 -36
  93. package/commands/suggest-next.md +0 -74
  94. package/commands/systems-thinking.md +0 -34
  95. package/commands/think-hats.md +0 -36
  96. package/commands/update.md +0 -181
  97. package/commands/user-needs.md +0 -34
  98. package/commands/validate.md +0 -34
  99. package/commands/value-proposition.md +0 -55
  100. package/commands/vault.md +0 -180
  101. package/commands/visualize.md +0 -52
  102. package/commands/whitespace.md +0 -501
  103. package/commands/wiki.md +0 -69
  104. package/hooks/hooks.json +0 -381
  105. package/hooks/run-hook.cmd +0 -64
  106. package/lib/__init__.py +0 -0
  107. package/lib/__pycache__/__init__.cpython-312.pyc +0 -0
  108. package/lib/agents/auto-explore-agent.cjs +0 -1043
  109. package/lib/agents/reverse-salient-agent.cjs +0 -679
  110. package/lib/agents/tension-hook-agent.cjs +0 -544
  111. package/lib/chat/chat-context.js +0 -185
  112. package/lib/chat/chat-panel.js +0 -721
  113. package/lib/chat/fabric-chat.cjs +0 -288
  114. package/lib/chat/generative-tools.js +0 -219
  115. package/lib/conversation/ROOM.md +0 -39
  116. package/lib/conversation/classifier-rules.json +0 -38
  117. package/lib/conversation/classifier.cjs +0 -264
  118. package/lib/conversation/operator.cjs +0 -287
  119. package/lib/copy/115-spec-strings.cjs +0 -55
  120. package/lib/core/__init__.py +0 -0
  121. package/lib/core/__nav-stub.cjs +0 -14
  122. package/lib/core/__pycache__/__init__.cpython-312.pyc +0 -0
  123. package/lib/core/__pycache__/rs-math.cpython-312.pyc +0 -0
  124. package/lib/core/__pycache__/rs_cache.cpython-312.pyc +0 -0
  125. package/lib/core/__pycache__/rs_corpus.cpython-312.pyc +0 -0
  126. package/lib/core/__pycache__/rs_hybrid.cpython-312.pyc +0 -0
  127. package/lib/core/__pycache__/rs_math.cpython-312.pyc +0 -0
  128. package/lib/core/__pycache__/rs_rooms.cpython-312.pyc +0 -0
  129. package/lib/core/artifact-id.cjs +0 -148
  130. package/lib/core/asset-ops.cjs +0 -151
  131. package/lib/core/auto-commit-throttle.cjs +0 -129
  132. package/lib/core/bearer-token.cjs +0 -199
  133. package/lib/core/brain-client.cjs +0 -865
  134. package/lib/core/brain-derivation-prompts.cjs +0 -326
  135. package/lib/core/brain-derivation-queue.cjs +0 -431
  136. package/lib/core/brain-derivation.cjs +0 -580
  137. package/lib/core/brain-md-schema.cjs +0 -528
  138. package/lib/core/brain-md-staleness.cjs +0 -357
  139. package/lib/core/brain-response-sanitize.cjs +0 -188
  140. package/lib/core/bridge-writer.cjs +0 -477
  141. package/lib/core/chat-context-builder.cjs +0 -253
  142. package/lib/core/cross-room-aggregator.cjs +0 -762
  143. package/lib/core/daily-briefing.cjs +0 -438
  144. package/lib/core/decision-capture.cjs +0 -618
  145. package/lib/core/deep-links.cjs +0 -82
  146. package/lib/core/dispatch-optimizer.cjs +0 -354
  147. package/lib/core/dual-path-detector.cjs +0 -84
  148. package/lib/core/dual-path-detector.test.cjs +0 -334
  149. package/lib/core/exports-log.cjs +0 -79
  150. package/lib/core/feynman-minto-invariants.cjs +0 -605
  151. package/lib/core/folder-memory-async.cjs +0 -338
  152. package/lib/core/folder-memory-shared.cjs +0 -890
  153. package/lib/core/folder-memory.cjs +0 -416
  154. package/lib/core/framework-chain-composer.cjs +0 -411
  155. package/lib/core/frontmatter-schemas.cjs +0 -330
  156. package/lib/core/git-ops.cjs +0 -141
  157. package/lib/core/graph-ops.cjs +0 -258
  158. package/lib/core/hat-persistence.cjs +0 -362
  159. package/lib/core/index.cjs +0 -60
  160. package/lib/core/integration-registry.cjs +0 -232
  161. package/lib/core/intelligence-cascade.cjs +0 -661
  162. package/lib/core/lazygraph-ops.cjs +0 -1057
  163. package/lib/core/lru-cache.cjs +0 -139
  164. package/lib/core/mcp-profiles.cjs +0 -182
  165. package/lib/core/meeting-ops.cjs +0 -54
  166. package/lib/core/memory-ops.cjs +0 -600
  167. package/lib/core/migrations/ROOM.md +0 -33
  168. package/lib/core/migrations/phase-109-nodes-provenance.cjs +0 -339
  169. package/lib/core/migrations/phase-109-session-focus.cjs +0 -99
  170. package/lib/core/model-profiles.cjs +0 -246
  171. package/lib/core/mullins-scaffold.cjs +0 -160
  172. package/lib/core/nav-dial.cjs +0 -316
  173. package/lib/core/navigation/ROOM.md +0 -15
  174. package/lib/core/navigation/explanation.cjs +0 -43
  175. package/lib/core/navigation/focus.cjs +0 -135
  176. package/lib/core/navigation/ingestion.cjs +0 -82
  177. package/lib/core/navigation/insights.cjs +0 -350
  178. package/lib/core/navigation/memory-events.cjs +0 -118
  179. package/lib/core/navigation/neighborhood.cjs +0 -78
  180. package/lib/core/navigation/packet.cjs +0 -182
  181. package/lib/core/navigation/room-home.cjs +0 -127
  182. package/lib/core/navigation/transitions.cjs +0 -82
  183. package/lib/core/navigation-engine-shared.cjs +0 -242
  184. package/lib/core/navigation-engine.cjs +0 -664
  185. package/lib/core/navigation.cjs +0 -60
  186. package/lib/core/nl-graph-queries.cjs +0 -164
  187. package/lib/core/offer-presenter.cjs +0 -406
  188. package/lib/core/opportunity-extractor.cjs +0 -183
  189. package/lib/core/opportunity-ops.cjs +0 -1371
  190. package/lib/core/persona-ops.cjs +0 -537
  191. package/lib/core/persona-taxonomy.cjs +0 -190
  192. package/lib/core/platform-gates.cjs +0 -120
  193. package/lib/core/platform.cjs +0 -257
  194. package/lib/core/proactive-intelligence.cjs +0 -528
  195. package/lib/core/problem-type-router.cjs +0 -315
  196. package/lib/core/reasoning-ops.cjs +0 -639
  197. package/lib/core/reverse-salient-persona-suffix.cjs +0 -115
  198. package/lib/core/room-classifier-strict-mode.cjs +0 -229
  199. package/lib/core/room-db.cjs +0 -127
  200. package/lib/core/room-ops-async.cjs +0 -92
  201. package/lib/core/room-ops-shared.cjs +0 -64
  202. package/lib/core/room-ops-sync.cjs +0 -70
  203. package/lib/core/room-ops.cjs +0 -32
  204. package/lib/core/room-type-detector.cjs +0 -386
  205. package/lib/core/rs-brain-substrate-prompts.cjs +0 -129
  206. package/lib/core/rs-brain-substrate.cjs +0 -570
  207. package/lib/core/rs-breakthrough-scorer.cjs +0 -255
  208. package/lib/core/rs-canon-violations.cjs +0 -82
  209. package/lib/core/rs-chain-feeder.cjs +0 -343
  210. package/lib/core/rs-commercial-assessor.cjs +0 -280
  211. package/lib/core/rs-differential-scorer.cjs +0 -376
  212. package/lib/core/rs-domain-analyzer.cjs +0 -385
  213. package/lib/core/rs-egress-prompts.cjs +0 -113
  214. package/lib/core/rs-egress-telemetry.cjs +0 -225
  215. package/lib/core/rs-egress-violations.cjs +0 -53
  216. package/lib/core/rs-expert-mapper.cjs +0 -467
  217. package/lib/core/rs-fetcher-academic.cjs +0 -697
  218. package/lib/core/rs-fetcher-experts.cjs +0 -314
  219. package/lib/core/rs-fetcher-industry.cjs +0 -731
  220. package/lib/core/rs-fetcher-patents.cjs +0 -564
  221. package/lib/core/rs-innovation-classifier.cjs +0 -194
  222. package/lib/core/rs-mind-map.cjs +0 -656
  223. package/lib/core/rs-neo4j-writer.cjs +0 -388
  224. package/lib/core/rs-nl-to-query.cjs +0 -425
  225. package/lib/core/rs-pinecone-bridge.cjs +0 -303
  226. package/lib/core/rs-preprocessor.cjs +0 -350
  227. package/lib/core/rs-query-matrix.cjs +0 -316
  228. package/lib/core/rs-query-to-text.cjs +0 -438
  229. package/lib/core/rs-sqlite-mirror.cjs +0 -443
  230. package/lib/core/rs-thesis-generator.cjs +0 -188
  231. package/lib/core/rs_cache.py +0 -479
  232. package/lib/core/rs_corpus.py +0 -468
  233. package/lib/core/rs_hybrid.py +0 -586
  234. package/lib/core/rs_math.py +0 -287
  235. package/lib/core/rs_rooms.py +0 -193
  236. package/lib/core/scheduled-scanner.cjs +0 -463
  237. package/lib/core/scratchpad-ops.cjs +0 -201
  238. package/lib/core/section-8-trace-schema.cjs +0 -138
  239. package/lib/core/section-registry.cjs +0 -111
  240. package/lib/core/session-state.cjs +0 -144
  241. package/lib/core/shallow-doc-parser.cjs +0 -174
  242. package/lib/core/shallow-doc-parser.test.cjs +0 -226
  243. package/lib/core/skill-activation-router.cjs +0 -284
  244. package/lib/core/state-ops.cjs +0 -46
  245. package/lib/core/statusline-cache.cjs +0 -266
  246. package/lib/core/token-estimator.cjs +0 -348
  247. package/lib/core/user-archetype.cjs +0 -239
  248. package/lib/core/user-md-ops.cjs +0 -524
  249. package/lib/core/visual-ops.cjs +0 -624
  250. package/lib/core/write-lock.cjs +0 -149
  251. package/lib/graph/canvas-graph.js +0 -467
  252. package/lib/graph/constellation-config.cjs +0 -299
  253. package/lib/graph/graph-detail-panel.js +0 -165
  254. package/lib/hmi/ROOM.md +0 -47
  255. package/lib/hmi/across-session-memory.cjs +0 -604
  256. package/lib/hmi/cross-room-memory.cjs +0 -575
  257. package/lib/hmi/decoy-tier.cjs +0 -395
  258. package/lib/hmi/jtbd-classifier.cjs +0 -219
  259. package/lib/hmi/jtbd-state.cjs +0 -199
  260. package/lib/hmi/jtbd-taxonomy.json +0 -392
  261. package/lib/hmi/selector-dispatcher.cjs +0 -546
  262. package/lib/hmi/selector-telemetry.cjs +0 -263
  263. package/lib/hmi/shape-f0-renderer.cjs +0 -139
  264. package/lib/hmi/shape-f1-fallback.cjs +0 -80
  265. package/lib/hmi/shape-f1-renderer.cjs +0 -138
  266. package/lib/hmi/shape-f2-renderer.cjs +0 -132
  267. package/lib/hmi/shape-f3-renderer.cjs +0 -66
  268. package/lib/hmi/shape-f4-renderer.cjs +0 -72
  269. package/lib/hmi/shape-f5-renderer.cjs +0 -155
  270. package/lib/hmi/shape-f6-plan-review-renderer.cjs +0 -312
  271. package/lib/hmi/shape-f6-renderer.cjs +0 -144
  272. package/lib/hmi/shape-g-renderer.cjs +0 -219
  273. package/lib/hmi/shape-h-renderer.cjs +0 -222
  274. package/lib/hmi/tier-check.cjs +0 -63
  275. package/lib/import/PRECONDITIONS.md +0 -41
  276. package/lib/import/branding.cjs +0 -210
  277. package/lib/import/branding.test.cjs +0 -235
  278. package/lib/import/classifications-sync.cjs +0 -104
  279. package/lib/import/classifications-sync.test.cjs +0 -129
  280. package/lib/import/enricher.cjs +0 -296
  281. package/lib/import/enricher.test.cjs +0 -273
  282. package/lib/import/integration.test.cjs +0 -376
  283. package/lib/import/manifest.cjs +0 -129
  284. package/lib/import/manifest.schema.json +0 -185
  285. package/lib/import/manifest.test.cjs +0 -123
  286. package/lib/import/meeting-detector.cjs +0 -92
  287. package/lib/import/meeting-detector.test.cjs +0 -100
  288. package/lib/import/person-detector.cjs +0 -229
  289. package/lib/import/person-detector.test.cjs +0 -149
  290. package/lib/import/report.cjs +0 -186
  291. package/lib/import/report.test.cjs +0 -186
  292. package/lib/import/room-md-scaffolder.cjs +0 -49
  293. package/lib/import/router.cjs +0 -224
  294. package/lib/import/router.test.cjs +0 -356
  295. package/lib/import/run-all-tests.cjs +0 -36
  296. package/lib/import/smoke-test.cjs +0 -213
  297. package/lib/import/smoke-test.test.cjs +0 -148
  298. package/lib/import/test-fixtures/collision-vault/preexisting-room/STATE.md +0 -8
  299. package/lib/import/test-fixtures/collision-vault/preexisting-room/problem-definition/onboarding/onboarding.md +0 -7
  300. package/lib/import/test-fixtures/collision-vault/source/onboarding.md +0 -5
  301. package/lib/import/test-fixtures/obsidian-vault/.obsidian/workspace.json +0 -1
  302. package/lib/import/test-fixtures/obsidian-vault/notes/with-wikilinks.md +0 -4
  303. package/lib/import/test-fixtures/tiny-vault/notes/2026-01-15-team-sync.md +0 -9
  304. package/lib/import/test-fixtures/tiny-vault/notes/empty.md +0 -3
  305. package/lib/import/test-fixtures/tiny-vault/notes/onboarding.md +0 -5
  306. package/lib/import/test-fixtures/tiny-vault/notes/pricing.md +0 -5
  307. package/lib/import/test-fixtures/tiny-vault/notes/random.md +0 -4
  308. package/lib/import/undo.test.cjs +0 -199
  309. package/lib/import/vault-scanner.cjs +0 -105
  310. package/lib/import/vault-scanner.test.cjs +0 -67
  311. package/lib/mcp/app-html/dashboard.html +0 -316
  312. package/lib/mcp/app-html/graph.html +0 -428
  313. package/lib/mcp/app-html/mindrian-platform.html +0 -1841
  314. package/lib/mcp/app-html/wiki.html +0 -383
  315. package/lib/mcp/app-views.cjs +0 -322
  316. package/lib/mcp/brain-router.cjs +0 -418
  317. package/lib/mcp/capability-registry.cjs +0 -62
  318. package/lib/mcp/larry-context.cjs +0 -46
  319. package/lib/mcp/larry-server-instructions.md +0 -114
  320. package/lib/mcp/pipeline-state.cjs +0 -275
  321. package/lib/mcp/prompts.cjs +0 -302
  322. package/lib/mcp/resources.cjs +0 -227
  323. package/lib/mcp/session-catchup.cjs +0 -327
  324. package/lib/mcp/surface-detect.cjs +0 -75
  325. package/lib/mcp/tool-router.cjs +0 -1034
  326. package/lib/memory/aaak-compress.cjs +0 -403
  327. package/lib/memory/aaak-compress.test.cjs +0 -288
  328. package/lib/memory/async-artifact-auto-commit.test.cjs +0 -223
  329. package/lib/memory/bearer-token.test.cjs +0 -315
  330. package/lib/memory/brain-cache-lru.test.cjs +0 -259
  331. package/lib/memory/brain-client-query-shape.test.cjs +0 -160
  332. package/lib/memory/brain-derivation-graceful-degradation.test.cjs +0 -1019
  333. package/lib/memory/brain-derivation-queue.test.cjs +0 -539
  334. package/lib/memory/brain-derivation.test.cjs +0 -634
  335. package/lib/memory/brain-derive-command.test.cjs +0 -534
  336. package/lib/memory/brain-md-invariants-validator.test.cjs +0 -704
  337. package/lib/memory/brain-md-schema.test.cjs +0 -467
  338. package/lib/memory/brain-md-staleness.test.cjs +0 -525
  339. package/lib/memory/brain-server-resolution.test.cjs +0 -314
  340. package/lib/memory/chat-context.test.cjs +0 -128
  341. package/lib/memory/cross-room-aggregator.test.cjs +0 -909
  342. package/lib/memory/dashboard-server.test.cjs +0 -256
  343. package/lib/memory/debouncer-drain-at-prompt.test.cjs +0 -389
  344. package/lib/memory/decision-capture.test.cjs +0 -632
  345. package/lib/memory/decision-capture.worker.cjs +0 -70
  346. package/lib/memory/explain-decision-command.test.cjs +0 -521
  347. package/lib/memory/explain-decision-footer.test.cjs +0 -316
  348. package/lib/memory/explored-materials-store.cjs +0 -392
  349. package/lib/memory/feynman-minto-guardian.test.cjs +0 -736
  350. package/lib/memory/feynman-minto-invariants.test.cjs +0 -511
  351. package/lib/memory/feynman-prompts-drift.test.cjs +0 -144
  352. package/lib/memory/feynman-prompts.cjs +0 -151
  353. package/lib/memory/feynman-prompts.test.cjs +0 -96
  354. package/lib/memory/folder-memory-quadruple.test.cjs +0 -548
  355. package/lib/memory/folder-memory.test.cjs +0 -503
  356. package/lib/memory/framework-chain-composer.test.cjs +0 -515
  357. package/lib/memory/frontmatter-schema-validator.test.cjs +0 -290
  358. package/lib/memory/heal-command.test.cjs +0 -604
  359. package/lib/memory/index-artifact-transaction.test.cjs +0 -333
  360. package/lib/memory/lazygraph-rs-discoveries-view.test.cjs +0 -122
  361. package/lib/memory/mcp-input-validation.test.cjs +0 -240
  362. package/lib/memory/mcp-server-brain-deps.test.cjs +0 -270
  363. package/lib/memory/mcp-stack-fallback.test.cjs +0 -433
  364. package/lib/memory/minto-debouncer.test.cjs +0 -407
  365. package/lib/memory/minto-debouncer.worker.cjs +0 -46
  366. package/lib/memory/minto-migration-v88.test.cjs +0 -265
  367. package/lib/memory/minto-schema-v88.test.cjs +0 -390
  368. package/lib/memory/mos-status-renderer.test.cjs +0 -631
  369. package/lib/memory/narrative-schema.cjs +0 -376
  370. package/lib/memory/narrative-schema.test.cjs +0 -209
  371. package/lib/memory/nav-dial.test.cjs +0 -414
  372. package/lib/memory/navigation-engine-core.test.cjs +0 -722
  373. package/lib/memory/navigation-invariants.test.cjs +0 -483
  374. package/lib/memory/offer-presenter.test.cjs +0 -554
  375. package/lib/memory/on-stop-snapshot.test.cjs +0 -404
  376. package/lib/memory/pending-tension-store.cjs +0 -373
  377. package/lib/memory/post-compact-reinjection.test.cjs +0 -854
  378. package/lib/memory/post-write-triple.test.cjs +0 -317
  379. package/lib/memory/pre-compact-snapshot.test.cjs +0 -495
  380. package/lib/memory/problem-type-router.test.cjs +0 -656
  381. package/lib/memory/query-efficiency-telemetry.test.cjs +0 -370
  382. package/lib/memory/recompile-room-references.test.cjs +0 -392
  383. package/lib/memory/recompile-room-references.worker.cjs +0 -42
  384. package/lib/memory/record-decision-dual-write.test.cjs +0 -454
  385. package/lib/memory/room-classifier-strict-mode.test.cjs +0 -417
  386. package/lib/memory/room-minto-hook.test.cjs +0 -398
  387. package/lib/memory/rs-discovery-engine.test.cjs +0 -323
  388. package/lib/memory/run-feynman-tests.cjs +0 -1239
  389. package/lib/memory/security-trifecta.test.cjs +0 -312
  390. package/lib/memory/session-start-brain-staleness.test.cjs +0 -363
  391. package/lib/memory/session-start-triple-injection.test.cjs +0 -514
  392. package/lib/memory/sessionstart-banner-formatter.cjs +0 -318
  393. package/lib/memory/sessionstart-minto-banner.test.cjs +0 -373
  394. package/lib/memory/skill-activation-router.test.cjs +0 -419
  395. package/lib/memory/stamp-artifact-write.test.cjs +0 -304
  396. package/lib/memory/statusline-active-room.test.cjs +0 -315
  397. package/lib/memory/statusline-minto-segment.test.cjs +0 -292
  398. package/lib/memory/sync-async-entry-points.test.cjs +0 -204
  399. package/lib/memory/test-bridge-writer-enhanced.cjs +0 -452
  400. package/lib/memory/test-rs-brain-substrate-shape.cjs +0 -529
  401. package/lib/memory/test-rs-brain-substrate.cjs +0 -636
  402. package/lib/memory/test-rs-breakthrough-scorer.cjs +0 -375
  403. package/lib/memory/test-rs-canon-violations.cjs +0 -218
  404. package/lib/memory/test-rs-chain-feeder-core.cjs +0 -344
  405. package/lib/memory/test-rs-chain-feeder-skill-spawn.cjs +0 -297
  406. package/lib/memory/test-rs-commercial-assessor.cjs +0 -385
  407. package/lib/memory/test-rs-differential-scorer.cjs +0 -480
  408. package/lib/memory/test-rs-discovery-engine.cjs +0 -603
  409. package/lib/memory/test-rs-domain-analyzer.cjs +0 -492
  410. package/lib/memory/test-rs-egress-primitives.cjs +0 -420
  411. package/lib/memory/test-rs-expert-mapper.cjs +0 -547
  412. package/lib/memory/test-rs-explain-command.cjs +0 -443
  413. package/lib/memory/test-rs-fetcher-academic.cjs +0 -848
  414. package/lib/memory/test-rs-fetcher-experts.cjs +0 -496
  415. package/lib/memory/test-rs-fetcher-industry.cjs +0 -702
  416. package/lib/memory/test-rs-fetcher-patents.cjs +0 -674
  417. package/lib/memory/test-rs-innovation-classifier.cjs +0 -301
  418. package/lib/memory/test-rs-mind-map.cjs +0 -646
  419. package/lib/memory/test-rs-neo4j-writer.cjs +0 -518
  420. package/lib/memory/test-rs-nl-to-query.cjs +0 -449
  421. package/lib/memory/test-rs-pinecone-bridge.cjs +0 -277
  422. package/lib/memory/test-rs-preprocessor.cjs +0 -433
  423. package/lib/memory/test-rs-query-matrix.cjs +0 -391
  424. package/lib/memory/test-rs-query-to-text.cjs +0 -551
  425. package/lib/memory/test-rs-sqlite-mirror.cjs +0 -649
  426. package/lib/memory/test-rs-thesis-generator.cjs +0 -360
  427. package/lib/memory/triple-context-formatter.cjs +0 -473
  428. package/lib/memory/triple-context-formatter.test.cjs +0 -442
  429. package/lib/memory/user-md-persona.test.cjs +0 -565
  430. package/lib/memory/userpromptsubmit-integration.test.cjs +0 -690
  431. package/lib/memory/validators/README.md +0 -157
  432. package/lib/memory/validators/brain-md-invariants.cjs +0 -475
  433. package/lib/memory/validators/brain-substrate-invariants.cjs +0 -285
  434. package/lib/memory/validators/external-academic-invariants.cjs +0 -249
  435. package/lib/memory/validators/external-industry-invariants.cjs +0 -271
  436. package/lib/memory/validators/external-patents-invariants.cjs +0 -266
  437. package/lib/memory/validators/minto-invariants.cjs +0 -62
  438. package/lib/memory/validators/navigation-invariants.cjs +0 -340
  439. package/lib/memory/validators/queue-health.cjs +0 -95
  440. package/lib/memory/validators/snapshot-integrity.cjs +0 -129
  441. package/lib/memory/validators/stale-lifecycle.cjs +0 -116
  442. package/lib/memory/vault-section-minto-generator-atomic.test.cjs +0 -556
  443. package/lib/memory/vault-section-minto-generator-atomic.worker.cjs +0 -73
  444. package/lib/memory/write-lock-atomic.test.cjs +0 -137
  445. package/lib/memory/write-lock-atomic.worker.cjs +0 -55
  446. package/lib/parity/check-parity.cjs +0 -83
  447. package/lib/presentation/presentation-server.cjs +0 -101
  448. package/lib/presentation/presentation-watcher.cjs +0 -123
  449. package/lib/quickview/hub-server.cjs +0 -719
  450. package/lib/quickview/server.cjs +0 -533
  451. package/lib/render/JTBD-PALETTES.md +0 -145
  452. package/lib/render/ROOM.md +0 -59
  453. package/lib/render/render-v2.cjs +0 -486
  454. package/lib/render/render-v2.test.cjs +0 -267
  455. package/lib/render/render.cjs +0 -65
  456. package/lib/state/ROOM.md +0 -46
  457. package/lib/state/state-md-parser.cjs +0 -215
  458. package/lib/statusline/ROOM.md +0 -38
  459. package/lib/statusline/banner-suppression.cjs +0 -50
  460. package/lib/statusline/surface-detect.cjs +0 -85
  461. package/lib/update-bootstrap.sh.template +0 -145
  462. package/lib/vault/frontmatter-schema.cjs +0 -297
  463. package/lib/vault/room-scanner.cjs +0 -352
  464. package/lib/vault/wikilink-builder.cjs +0 -231
  465. package/lib/vault/wikilink-builder.test.cjs +0 -182
  466. package/lib/wiki/graph-links.cjs +0 -281
  467. package/lib/wiki/page-renderer.cjs +0 -229
  468. package/lib/wiki/wiki-chat.cjs +0 -81
  469. package/lib/wiki/wiki-layout.cjs +0 -1459
  470. package/lib/wiki/wiki-search.cjs +0 -142
  471. package/lib/wiki/wiki-server.cjs +0 -678
  472. package/lib/wiki/wiki-watcher.cjs +0 -105
  473. package/pipelines/analogy/01-decompose.md +0 -80
  474. package/pipelines/analogy/02-abstract.md +0 -87
  475. package/pipelines/analogy/03-search.md +0 -135
  476. package/pipelines/analogy/04-transfer.md +0 -101
  477. package/pipelines/analogy/05-validate.md +0 -106
  478. package/pipelines/analogy/CHAIN.md +0 -56
  479. package/pipelines/discovery/01-explore-domains.md +0 -44
  480. package/pipelines/discovery/02-think-hats.md +0 -50
  481. package/pipelines/discovery/03-analyze-needs.md +0 -54
  482. package/pipelines/discovery/CHAIN.md +0 -37
  483. package/pipelines/thesis/01-structure-argument.md +0 -45
  484. package/pipelines/thesis/02-challenge-assumptions.md +0 -48
  485. package/pipelines/thesis/03-build-thesis.md +0 -54
  486. package/pipelines/thesis/CHAIN.md +0 -37
  487. package/references/brain/causal-directives.md +0 -91
  488. package/references/brain/causal-enrichment.cypher +0 -165
  489. package/references/brain/command-triggers-schema.md +0 -226
  490. package/references/brain/graph-architecture.md +0 -317
  491. package/references/brain/query-patterns.md +0 -460
  492. package/references/brain/room-hierarchy-schema.md +0 -218
  493. package/references/brain/schema.md +0 -76
  494. package/references/capability-radar/capabilities-index.md +0 -241
  495. package/references/capability-radar/changelog-cache.md +0 -81
  496. package/references/causal/causal-schema.md +0 -103
  497. package/references/design/email-template-standard.md +0 -155
  498. package/references/design/graph-visualization-standard.md +0 -178
  499. package/references/document-generation.md +0 -179
  500. package/references/hsi/HSI-TOOLS-REFERENCE.md +0 -222
  501. package/references/import-config.md +0 -141
  502. package/references/integrations/detection-patterns.md +0 -101
  503. package/references/meeting/artifact-template.md +0 -377
  504. package/references/meeting/cross-meeting-intelligence.md +0 -216
  505. package/references/meeting/cross-relationship-patterns.md +0 -202
  506. package/references/meeting/live-join-interface.md +0 -244
  507. package/references/meeting/section-mapping.md +0 -192
  508. package/references/meeting/segment-classification.md +0 -258
  509. package/references/meeting/speaker-profile-template.md +0 -219
  510. package/references/meeting/summary-template.md +0 -348
  511. package/references/meeting/transcript-patterns.md +0 -226
  512. package/references/methodology/analyze-needs.md +0 -135
  513. package/references/methodology/analyze-systems.md +0 -121
  514. package/references/methodology/analyze-timing.md +0 -149
  515. package/references/methodology/beautiful-question.md +0 -109
  516. package/references/methodology/build-knowledge.md +0 -161
  517. package/references/methodology/build-thesis.md +0 -237
  518. package/references/methodology/challenge-assumptions.md +0 -127
  519. package/references/methodology/diagnose.md +0 -169
  520. package/references/methodology/dominant-designs.md +0 -212
  521. package/references/methodology/explore-domains.md +0 -147
  522. package/references/methodology/explore-futures.md +0 -163
  523. package/references/methodology/explore-trends.md +0 -129
  524. package/references/methodology/find-bottlenecks.md +0 -131
  525. package/references/methodology/grade.md +0 -211
  526. package/references/methodology/index.md +0 -97
  527. package/references/methodology/leadership.md +0 -200
  528. package/references/methodology/lean-canvas.md +0 -116
  529. package/references/methodology/macro-trends.md +0 -192
  530. package/references/methodology/map-unknowns.md +0 -137
  531. package/references/methodology/mullins-7-domains.md +0 -104
  532. package/references/methodology/problem-types.md +0 -65
  533. package/references/methodology/root-cause.md +0 -178
  534. package/references/methodology/sapphire-encoding.md +0 -355
  535. package/references/methodology/scenario-plan.md +0 -178
  536. package/references/methodology/score-innovation.md +0 -154
  537. package/references/methodology/structure-argument.md +0 -158
  538. package/references/methodology/systems-thinking.md +0 -159
  539. package/references/methodology/think-hats.md +0 -147
  540. package/references/methodology/triz-matrix.json +0 -751
  541. package/references/methodology/triz-principles.md +0 -501
  542. package/references/methodology/user-needs.md +0 -199
  543. package/references/methodology/validate.md +0 -163
  544. package/references/methodology/value-proposition.md +0 -244
  545. package/references/opportunities/funding-lifecycle.md +0 -103
  546. package/references/opportunities/grant-api-patterns.md +0 -99
  547. package/references/opportunities/opportunity-template.md +0 -84
  548. package/references/personality/assessment-philosophy.md +0 -72
  549. package/references/personality/lexicon.md +0 -100
  550. package/references/personality/persona-chains.md +0 -56
  551. package/references/personality/pws-lexicon-full.md +0 -499
  552. package/references/personality/voice-dna.md +0 -156
  553. package/references/personas/hat-perspectives.md +0 -76
  554. package/references/personas/persona-template.md +0 -63
  555. package/references/pipeline/act-output-contract.md +0 -88
  556. package/references/pipeline/chains-index.md +0 -39
  557. package/references/pws-profile-generation.md +0 -79
  558. package/references/reasoning/reasoning-schema.md +0 -143
  559. package/references/reasoning/reasoning-template.md +0 -68
  560. package/references/reasoning/run-template.md +0 -38
  561. package/references/research/RESEARCH_14_CLAUDE_CODE_SOURCE_ARCHITECTURE.md +0 -209
  562. package/references/research/RESEARCH_15_V1.8_OPTIMIZATION_JTBD.md +0 -375
  563. package/references/research/RESEARCH_16_NATIVE_FIRST_PLUGIN_ARCHITECTURE.md +0 -575
  564. package/references/research/RESEARCH_17_MCP_UI_FRAMEWORKS.md +0 -272
  565. package/references/taxonomy/TAXONOMY.md +0 -192
  566. package/references/templates/MINTO.md +0 -36
  567. package/references/user-research/2026-04-05-leah-lawrence-session.md +0 -202
  568. package/references/vault-kit/README.md +0 -35
  569. package/references/vault-kit/app.json +0 -12
  570. package/references/vault-kit/appearance.json +0 -12
  571. package/references/vault-kit/graph.json +0 -35
  572. package/references/vault-kit/snippets/mindrian-destijl.css +0 -297
  573. package/references/vault-kit/templates/new-artifact.md +0 -37
  574. package/references/vault-kit/templates/new-meeting-note.md +0 -35
  575. package/references/vault-kit/templates/new-team-profile.md +0 -29
  576. package/references/vault-kit/templates/new-xref.md +0 -35
  577. package/references/visual/symbol-system.md +0 -151
  578. package/skills/MOSDeckEngine/SKILL.md +0 -325
  579. package/skills/brain-connector/SKILL.md +0 -114
  580. package/skills/context-engine/SKILL.md +0 -147
  581. package/skills/conversation-mode/SKILL.md +0 -102
  582. package/skills/larry-personality/SKILL.md +0 -219
  583. package/skills/larry-personality/framework-chains.md +0 -92
  584. package/skills/larry-personality/mode-engine.md +0 -185
  585. package/skills/mullins-scaffold/SKILL.md +0 -61
  586. package/skills/mullins-scaffold/scaffold.json +0 -146
  587. package/skills/pws-methodology/SKILL.md +0 -49
  588. package/skills/room-passive/SKILL.md +0 -165
  589. package/skills/room-proactive/SKILL.md +0 -250
  590. package/skills/ui-system/SKILL.md +0 -277
@@ -1,266 +0,0 @@
1
- /*
2
- * Copyright (c) 2026 Mindrian. BSL 1.1.
3
- *
4
- * Phase 88.1-04 -- statusline MINTO segment cache + format helpers
5
- * =================================================================
6
- * Pure, disk-backed, 5-second TTL cache for the Claude Code statusline
7
- * MINTO segment. The statusline fires once per turn via scripts/context-monitor
8
- * and must stay within a 300ms render budget (CONTEXT R1 mitigation).
9
- *
10
- * Cold path (no cache): readTriple + cache write. Target < 200ms on fresh WSL2.
11
- * Warm path (cache hit): JSON parse + string format. Target < 10ms.
12
- *
13
- * The cache is LOCAL by construction per Canon Part 8: all writes land at
14
- * roomRoot/.mindrian/statusline-cache.json and never egress. No Brain calls,
15
- * no fetch, no external endpoints.
16
- *
17
- * Canon references:
18
- * Part 2 UI glyph vocabulary -- classifyHealth emits check / warn / low / --
19
- * for the MINTO reasoning_health_score (shared with
20
- * lib/memory/triple-context-formatter.cjs + Phase 88.1-03 hook sysmsg
21
- * retrofit so rendering is byte-identical across surfaces).
22
- * Part 3 Tri-Context Decision Gate -- statusline renders LOCAL only.
23
- * Never BRAIN (BRAIN payloads are generic methodology only), never SIGNAL
24
- * (SIGNAL belongs to scheduled sweeps, not per-turn UI).
25
- * Part 8 Graph Boundary -- all cached bytes describe this specific user's
26
- * room state. They stay in this room. The boundary is architectural.
27
- *
28
- * Composes with:
29
- * Phase 87-02 atomic write pattern (openSync('wx') + fsync + rename) for
30
- * concurrent-safe cache persistence under rapid-statusline-fire.
31
- * Phase 88-01 folder-memory.readTriple(sectionPath) is the read contract;
32
- * cached 'triple' values are the shape readTriple returns.
33
- * Phase 88-07 / 88.1-03 classifyHealth shared vocabulary so every L3
34
- * surface (session-start sysmsg, guardian, statusline, /mos:status) uses
35
- * the same glyph classifier.
36
- *
37
- * Pure CJS, node built-ins only, zero npm dependencies. Three-surface
38
- * compatible by construction (CLI + Desktop MCP + Cowork).
39
- *
40
- * API:
41
- * TTL_MS -- number, fixed at 5000ms.
42
- * getCached(roomRoot, sectionPath) -> {value, fresh}
43
- * setCached(roomRoot, sectionPath, triple) -> void (graceful on error)
44
- * truncateGoverningThought(str) -> string (60-char budget)
45
- * classifyHealth(score) -> 'check'|'warn'|'low'|'--'
46
- *
47
- * Cache file shape:
48
- * { ts: number, sectionPath: string, triple: object }
49
- */
50
-
51
- 'use strict';
52
-
53
- const fs = require('node:fs');
54
- const path = require('node:path');
55
-
56
- // ---------- Tunables (frozen invariants) ----------
57
-
58
- // 5-second TTL per CONTEXT R1 mitigation. Short enough that users perceive
59
- // freshness (a change to MINTO.md shows up on the next statusline tick after
60
- // 5s), long enough that 12 statusline fires/min all hit the warm path.
61
- const TTL_MS = 5000;
62
-
63
- // 60-char budget for the governing_thought segment. Matches the description-
64
- // discipline budget from Plan 88.1-01 so the statusline honors the same
65
- // text-economy contract the rest of L3 uses.
66
- const GOVERNING_THOUGHT_BUDGET = 60;
67
-
68
- // Cache filename under roomRoot/.mindrian/. Named separately from the Phase
69
- // 88-06 session-snapshot.json to avoid any cross-purpose coupling.
70
- const CACHE_FILENAME = 'statusline-cache.json';
71
-
72
- // ---------- Path helpers ----------
73
-
74
- function cacheDir(roomRoot) {
75
- return path.join(roomRoot, '.mindrian');
76
- }
77
-
78
- function cacheFile(roomRoot) {
79
- return path.join(cacheDir(roomRoot), CACHE_FILENAME);
80
- }
81
-
82
- // ---------- classifyHealth (shared with triple-context-formatter) ----------
83
- //
84
- // Canon Part 2 glyph vocabulary: check / warn / low / --. Byte-identical to
85
- // the classifier extracted in Phase 88.1-03 so the statusline, the hook
86
- // systemMessage retrofit, and the TRIPLE_CONTEXT formatter all render the
87
- // same glyph for the same score. See lib/memory/triple-context-formatter.cjs
88
- // lines 80-101 for the canonical source; this copy mirrors it for zero-cost
89
- // import from lib/core/ consumers that should not reach into lib/memory/.
90
- //
91
- // Signature: classifyHealth(score: number | null | undefined)
92
- // -> 'check' | 'warn' | 'low' | '--'
93
-
94
- function classifyHealth(score) {
95
- if (typeof score !== 'number' || !Number.isFinite(score)) return '--';
96
- if (score >= 0.7) return 'check';
97
- if (score >= 0.4) return 'warn';
98
- return 'low';
99
- }
100
-
101
- // ---------- truncateGoverningThought ----------
102
-
103
- /**
104
- * truncateGoverningThought
105
- * Enforce the 60-char statusline budget on a governing_thought string.
106
- * Short inputs (<= 60 chars) pass through byte-identical. Long inputs are
107
- * truncated to 59 chars plus a Unicode ellipsis (U+2026) for a total of
108
- * exactly 60 chars. Non-string / empty inputs return the empty string.
109
- */
110
- function truncateGoverningThought(str) {
111
- if (typeof str !== 'string') return '';
112
- if (str.length <= GOVERNING_THOUGHT_BUDGET) return str;
113
- // 59 chars + 1 ellipsis = exactly 60 chars, matching the budget.
114
- return str.slice(0, GOVERNING_THOUGHT_BUDGET - 1) + '\u2026';
115
- }
116
-
117
- // ---------- Disk read ----------
118
-
119
- function safeReadSync(p) {
120
- try {
121
- return fs.readFileSync(p, 'utf8');
122
- } catch (_e) {
123
- return null;
124
- }
125
- }
126
-
127
- /**
128
- * getCached(roomRoot, sectionPath) -> {value, fresh}
129
- *
130
- * Returns {value: null, fresh: false} when:
131
- * - cache file does not exist
132
- * - cache file is malformed JSON
133
- * - cache entry belongs to a different sectionPath
134
- * - cache entry is older than TTL_MS
135
- *
136
- * Returns {value: triple, fresh: true} on a warm hit. The statusline reads
137
- * the triple fields directly (reasoning.governing_thought,
138
- * reasoning.reasoning_health_score, reasoning.is_stale).
139
- *
140
- * Never throws. Graceful fallback on every error path.
141
- */
142
- function getCached(roomRoot, sectionPath) {
143
- const miss = { value: null, fresh: false };
144
- if (!roomRoot || !sectionPath) return miss;
145
-
146
- const file = cacheFile(roomRoot);
147
- const raw = safeReadSync(file);
148
- if (raw === null) return miss;
149
-
150
- let parsed;
151
- try {
152
- parsed = JSON.parse(raw);
153
- } catch (_e) {
154
- return miss;
155
- }
156
-
157
- if (!parsed || typeof parsed !== 'object') return miss;
158
- if (parsed.sectionPath !== sectionPath) return miss;
159
- if (typeof parsed.ts !== 'number' || !Number.isFinite(parsed.ts)) return miss;
160
-
161
- const age = Date.now() - parsed.ts;
162
- if (age < 0 || age >= TTL_MS) {
163
- return { value: null, fresh: false };
164
- }
165
- if (!parsed.triple || typeof parsed.triple !== 'object') return miss;
166
-
167
- return { value: parsed.triple, fresh: true };
168
- }
169
-
170
- // ---------- Disk write (atomic) ----------
171
-
172
- /**
173
- * setCached(roomRoot, sectionPath, triple) -> void
174
- *
175
- * Atomic write sequence (Phase 87-02 pattern):
176
- * - mkdir -p roomRoot/.mindrian/ (idempotent)
177
- * - openSync('wx') on a pid-scoped tmp file
178
- * - writeSync + fsyncSync + closeSync
179
- * - renameSync over the real cache file
180
- *
181
- * Graceful fallback: any filesystem error (EEXIST on tmp, EACCES on mkdir,
182
- * ENOENT on rename, disk full, etc.) is swallowed silently. The cache is
183
- * a performance optimization; a failed write must NEVER bubble up to
184
- * context-monitor because the statusline has to render regardless.
185
- *
186
- * Never throws. This is the Part 8-by-construction guarantee: a pure
187
- * persistence side-effect that cannot leak state anywhere except the
188
- * LOCAL roomRoot/.mindrian/ directory.
189
- */
190
- function setCached(roomRoot, sectionPath, triple) {
191
- if (!roomRoot || !sectionPath) return;
192
-
193
- const dir = cacheDir(roomRoot);
194
- try {
195
- fs.mkdirSync(dir, { recursive: true });
196
- } catch (_e) {
197
- // If mkdir fails (e.g. .mindrian is a regular file, not a directory),
198
- // bail out silently. Graceful-fallback invariant.
199
- return;
200
- }
201
-
202
- const payload = {
203
- ts: Date.now(),
204
- sectionPath: sectionPath,
205
- triple: triple,
206
- };
207
-
208
- let serialized;
209
- try {
210
- serialized = JSON.stringify(payload);
211
- } catch (_e) {
212
- // Cyclic references in triple -> serialization failure. Bail.
213
- return;
214
- }
215
-
216
- const finalPath = cacheFile(roomRoot);
217
- const tmpPath = finalPath + '.tmp.' + process.pid + '.statusline';
218
-
219
- let fd;
220
- try {
221
- fd = fs.openSync(tmpPath, 'wx');
222
- } catch (e) {
223
- if (e && e.code === 'EEXIST') {
224
- // Stale tmp from a prior crash owned by the same pid slot. Clean and
225
- // retry once. Any second failure is swallowed.
226
- try { fs.unlinkSync(tmpPath); } catch (_) {}
227
- try {
228
- fd = fs.openSync(tmpPath, 'wx');
229
- } catch (_e2) {
230
- return;
231
- }
232
- } else {
233
- return;
234
- }
235
- }
236
-
237
- try {
238
- fs.writeSync(fd, serialized);
239
- try { fs.fsyncSync(fd); } catch (_) { /* fsync best-effort */ }
240
- } catch (_e) {
241
- try { fs.closeSync(fd); } catch (_) {}
242
- try { fs.unlinkSync(tmpPath); } catch (_) {}
243
- return;
244
- } finally {
245
- try { fs.closeSync(fd); } catch (_) {}
246
- }
247
-
248
- try {
249
- fs.renameSync(tmpPath, finalPath);
250
- } catch (_e) {
251
- // Rename failed. Clean up tmp and bail.
252
- try { fs.unlinkSync(tmpPath); } catch (_) {}
253
- return;
254
- }
255
- }
256
-
257
- // ---------- Exports ----------
258
-
259
- module.exports = {
260
- TTL_MS: TTL_MS,
261
- GOVERNING_THOUGHT_BUDGET: GOVERNING_THOUGHT_BUDGET,
262
- getCached: getCached,
263
- setCached: setCached,
264
- truncateGoverningThought: truncateGoverningThought,
265
- classifyHealth: classifyHealth,
266
- };
@@ -1,348 +0,0 @@
1
- /*
2
- * Copyright (c) 2026 Mindrian. BSL 1.1.
3
- *
4
- * Phase 88.1-16 -- query-efficiency token estimator + aggregation math.
5
- * =====================================================================
6
- * Pure utility module consumed by two callers:
7
- *
8
- * 1. scripts/query-efficiency-telemetry.cjs (PostToolUse hook) uses
9
- * estimateTokens(payload) to measure tokens_used per Read|Grep|Glob
10
- * return, and estimateRoomTokens(roomRoot) to compute the
11
- * tokens_naive_estimate baseline that defines the 57x efficiency claim.
12
- *
13
- * 2. scripts/scout-telemetry-aggregator.cjs (aggregation helper spawned
14
- * by /mos:scout) uses aggregateEvents(events) to render the median
15
- * ratio + top-5 commands + threshold-status summary.
16
- *
17
- * Canon:
18
- * Part 6 Product-as-Venture -- the 57x token-efficiency claim is a
19
- * venture claim; without this measurement arm it is marketing copy.
20
- * This module is the measurement arm: the ratio it produces is the
21
- * release-gate signal (>= 40x PASS; < 40x RETUNE) per CONTEXT
22
- * criterion #15.
23
- * Part 8 Graph Boundary -- LOCAL-by-construction. Zero http/https
24
- * imports, zero child_process, zero network egress. Only fs reads
25
- * (for estimateRoomTokens walking .md files). The caller hooks
26
- * persist ONLY scalar integer counts to the JSONL; no user-artifact
27
- * bytes reach the telemetry file.
28
- *
29
- * Design notes:
30
- * - estimateTokens uses the chars/4 approximation established by
31
- * lib/memory/triple-context-formatter.cjs (Phase 88-07). Consistency
32
- * across the codebase matters: if the Phase 88-07 formatter reports
33
- * "this emission is 3825 tokens", the telemetry hook must estimate
34
- * with the same yardstick so ratios compare apples-to-apples.
35
- *
36
- * - estimateRoomTokens uses a module-level in-memory cache keyed by
37
- * absolute room path. First call walks the tree summing .md file
38
- * sizes; subsequent calls return the cached value. The cache is
39
- * session-scoped (cleared when the Node process exits). A ledger
40
- * cache is NOT persisted to disk because:
41
- * (a) the estimator is fast enough (single walk of a room tree
42
- * typically completes in < 50ms on commodity SSDs), and
43
- * (b) on-disk caches introduce invalidation complexity that is
44
- * unnecessary for a per-session approximation used as the
45
- * denominator of a ratio. Staleness by up to one session is
46
- * within the noise floor of the measurement itself.
47
- *
48
- * - validateEventShape enforces the 8-field contract specified in
49
- * 88.1-16 PLAN <interfaces>. The hook uses this as a pre-write guard
50
- * so a malformed event can never reach the JSONL.
51
- *
52
- * - aggregateEvents reduces a raw event stream to { median, mean, top5,
53
- * count }. Top-5 is computed per-command (taking the MAX ratio
54
- * observed for each command) so a single command cannot occupy
55
- * multiple slots. This matches the CONTEXT-specified "top 5 commands
56
- * by ratio" rendering.
57
- *
58
- * - classifyRatio maps a numeric ratio to { 'normal' | 'warn' } at the
59
- * 10x threshold per PLAN <action>. Below-10x arms the hook's warn
60
- * systemMessage; at-or-above 10x stays silent (no verification fatigue
61
- * per Plan 03 reviewer R5).
62
- *
63
- * API:
64
- * estimateTokens(str) -> integer >= 0
65
- * estimateRoomTokens(roomRoot) -> integer >= 0 | null
66
- * clearCache() -> void (test hygiene)
67
- * validateEventShape(obj) -> { valid, missing:string[] }
68
- * classifyRatio(ratio) -> 'normal' | 'warn'
69
- * aggregateEvents(events) -> { count, median, mean, top5 }
70
- * EVENT_REQUIRED_FIELDS -> readonly string[]
71
- * BELOW_THRESHOLD_RATIO -> 10
72
- */
73
-
74
- 'use strict';
75
-
76
- const fs = require('node:fs');
77
- const path = require('node:path');
78
-
79
- // ---------- Constants ----------
80
-
81
- // Classification threshold for below-10x warn emission. Canon: anything
82
- // <10x is a leakage signal (query is pulling more room context than it
83
- // needs). This is the hook's advisory trigger, not the 57x release gate.
84
- const BELOW_THRESHOLD_RATIO = 10;
85
-
86
- // 8-field contract per 88.1-16 PLAN <interfaces>. validateEventShape
87
- // enforces this on every event before it is appended to the JSONL.
88
- const EVENT_REQUIRED_FIELDS = Object.freeze([
89
- 'event',
90
- 'ts',
91
- 'command',
92
- 'tool',
93
- 'tokens_used',
94
- 'tokens_naive_estimate',
95
- 'ratio',
96
- 'room_slug',
97
- ]);
98
-
99
- // Max depth guard for the estimateRoomTokens walker. Rooms are expected
100
- // to be shallow (<10 levels). A runaway symlink loop or surprise deep
101
- // tree must not hang the hook.
102
- const MAX_ROOM_DEPTH = 12;
103
-
104
- // ---------- Module-level session cache ----------
105
- //
106
- // Key: absolute, resolved room path (realpath when possible).
107
- // Value: cached token count (integer).
108
- // Cleared via clearCache() for tests; otherwise persists for the life
109
- // of the Node process (i.e. for the duration of a single hook fork or
110
- // scout invocation).
111
- const roomTokenCache = new Map();
112
-
113
- // ---------- estimateTokens ----------
114
-
115
- /**
116
- * estimateTokens(str) -> integer >= 0
117
- *
118
- * Chars-over-4 estimator, matching Phase 88-07 triple-context-formatter.
119
- * Graceful on null / undefined / non-string: returns 0. This matches the
120
- * behavior established by formatter.estimateTokens so the two producers
121
- * are interchangeable.
122
- */
123
- function estimateTokens(str) {
124
- if (typeof str !== 'string') return 0;
125
- if (str.length === 0) return 0;
126
- return Math.ceil(str.length / 4);
127
- }
128
-
129
- // ---------- estimateRoomTokens ----------
130
-
131
- /**
132
- * Walk the room tree summing the byte-length of every .md file, divide
133
- * by 4 to approximate tokens. Skips non-.md files (binaries, images,
134
- * generated JSON, etc.) so the denominator of the efficiency ratio
135
- * reflects ONLY the text payload a naive whole-room ingestion would
136
- * consume.
137
- *
138
- * Uses fs.readdirSync with withFileTypes so a single syscall per
139
- * directory yields both names and types; avoids the stat-per-entry
140
- * overhead that a readdir+stat loop would incur.
141
- *
142
- * Returns null on:
143
- * - invalid input (non-string, empty string, null, undefined)
144
- * - path does not exist
145
- * - path is not a directory
146
- * - any error during the walk (defensive; the hook must never throw)
147
- *
148
- * Caches the result in module-level roomTokenCache keyed by the resolved
149
- * path. Second call returns cached value without re-walking; this is the
150
- * property test 5 asserts.
151
- */
152
- function estimateRoomTokens(roomRoot) {
153
- if (typeof roomRoot !== 'string' || roomRoot.length === 0) return null;
154
-
155
- // Cache hit?
156
- if (roomTokenCache.has(roomRoot)) {
157
- return roomTokenCache.get(roomRoot);
158
- }
159
-
160
- let stat;
161
- try {
162
- stat = fs.statSync(roomRoot);
163
- } catch (_e) {
164
- return null;
165
- }
166
- if (!stat || !stat.isDirectory()) return null;
167
-
168
- let totalChars = 0;
169
-
170
- function walk(dir, depth) {
171
- if (depth > MAX_ROOM_DEPTH) return;
172
- let entries;
173
- try {
174
- entries = fs.readdirSync(dir, { withFileTypes: true });
175
- } catch (_e) {
176
- return; // permission denied, vanished dir, etc.
177
- }
178
- for (const entry of entries) {
179
- if (!entry) continue;
180
- const name = entry.name;
181
- // Skip hidden directories / files starting with '.' (e.g. .git,
182
- // .mindrian, .obsidian). Counting them would over-estimate the
183
- // naive baseline and dilute the ratio.
184
- if (typeof name === 'string' && name.length > 0 && name[0] === '.') continue;
185
- const full = path.join(dir, name);
186
- if (entry.isDirectory()) {
187
- walk(full, depth + 1);
188
- continue;
189
- }
190
- if (!entry.isFile()) continue; // skip symlinks, sockets, devices
191
- // Only count .md files.
192
- if (!/\.md$/i.test(name)) continue;
193
- try {
194
- const s = fs.statSync(full);
195
- if (s && typeof s.size === 'number') {
196
- totalChars += s.size;
197
- }
198
- } catch (_e) { /* vanished file, skip */ }
199
- }
200
- }
201
-
202
- try {
203
- walk(roomRoot, 0);
204
- } catch (_e) {
205
- return null;
206
- }
207
-
208
- const tokens = Math.ceil(totalChars / 4);
209
- roomTokenCache.set(roomRoot, tokens);
210
- return tokens;
211
- }
212
-
213
- // ---------- clearCache ----------
214
-
215
- /**
216
- * Reset the session cache. Used by tests for deterministic between-case
217
- * isolation; production callers typically do not need to invoke this
218
- * (process lifetime bounds the cache naturally).
219
- */
220
- function clearCache() {
221
- roomTokenCache.clear();
222
- }
223
-
224
- // ---------- validateEventShape ----------
225
-
226
- /**
227
- * Return { valid, missing } for a telemetry event object. An event is
228
- * valid iff every field in EVENT_REQUIRED_FIELDS is present AND
229
- * non-null / non-undefined. The hook uses this as a write guard: a
230
- * malformed event must never reach the JSONL (prevents aggregation
231
- * NaN poisoning later).
232
- */
233
- function validateEventShape(obj) {
234
- if (!obj || typeof obj !== 'object' || Array.isArray(obj)) {
235
- return { valid: false, missing: EVENT_REQUIRED_FIELDS.slice() };
236
- }
237
- const missing = [];
238
- for (let i = 0; i < EVENT_REQUIRED_FIELDS.length; i += 1) {
239
- const k = EVENT_REQUIRED_FIELDS[i];
240
- const v = obj[k];
241
- if (v === undefined || v === null) missing.push(k);
242
- }
243
- return { valid: missing.length === 0, missing: missing };
244
- }
245
-
246
- // ---------- classifyRatio ----------
247
-
248
- /**
249
- * Classify a ratio into 'normal' or 'warn' at the 10x threshold.
250
- * Defensive: any non-finite or non-numeric input -> 'warn' (the ratio
251
- * was never computed correctly, which is itself a signal the hook
252
- * should surface).
253
- */
254
- function classifyRatio(ratio) {
255
- if (typeof ratio !== 'number' || !Number.isFinite(ratio)) return 'warn';
256
- if (ratio >= BELOW_THRESHOLD_RATIO) return 'normal';
257
- return 'warn';
258
- }
259
-
260
- // ---------- aggregateEvents ----------
261
-
262
- /**
263
- * Reduce an event stream to { count, median, mean, top5 }.
264
- *
265
- * - count: events.length.
266
- * - median: midpoint of the sorted ratios. Even-length: average of the
267
- * two middle values. Odd-length: the middle value.
268
- * - mean: arithmetic mean of ratios, rounded to 1 decimal for display
269
- * friendliness in the scout render.
270
- * - top5: up to 5 entries, sorted descending by ratio, deduped by
271
- * command (each command appears at most once, with its MAX
272
- * observed ratio).
273
- *
274
- * Empty input -> { count:0, median:null, mean:null, top5:[] }.
275
- *
276
- * Filters out malformed events (missing or non-numeric ratio) so a
277
- * corrupt JSONL line cannot poison the aggregate.
278
- */
279
- function aggregateEvents(events) {
280
- const out = { count: 0, median: null, mean: null, top5: [] };
281
- if (!Array.isArray(events) || events.length === 0) return out;
282
-
283
- // Collect only well-formed ratio values. A malformed event counts
284
- // toward `count` (reported events) but NOT toward median/mean/top5.
285
- const ratios = [];
286
- const byCommand = new Map(); // command -> max ratio
287
- for (let i = 0; i < events.length; i += 1) {
288
- const e = events[i];
289
- if (!e || typeof e !== 'object') continue;
290
- const r = e.ratio;
291
- if (typeof r !== 'number' || !Number.isFinite(r)) continue;
292
- ratios.push(r);
293
- const cmd = typeof e.command === 'string' && e.command.length > 0
294
- ? e.command
295
- : '(unknown)';
296
- const prev = byCommand.get(cmd);
297
- if (prev === undefined || r > prev) byCommand.set(cmd, r);
298
- }
299
-
300
- out.count = events.length;
301
-
302
- if (ratios.length === 0) return out;
303
-
304
- // Median: sort asc, pick middle.
305
- const sorted = ratios.slice().sort(function (a, b) { return a - b; });
306
- const n = sorted.length;
307
- if (n % 2 === 1) {
308
- out.median = sorted[(n - 1) / 2];
309
- } else {
310
- const lo = sorted[(n / 2) - 1];
311
- const hi = sorted[n / 2];
312
- out.median = (lo + hi) / 2;
313
- }
314
-
315
- // Mean, 1-decimal rounded for display.
316
- let sum = 0;
317
- for (let i = 0; i < n; i += 1) sum += sorted[i];
318
- const rawMean = sum / n;
319
- out.mean = Math.round(rawMean * 10) / 10;
320
-
321
- // Top-5 per-command, sorted desc by ratio. Ties broken by command
322
- // name for determinism.
323
- const pairs = [];
324
- const keys = Array.from(byCommand.keys());
325
- for (let i = 0; i < keys.length; i += 1) {
326
- pairs.push({ command: keys[i], ratio: byCommand.get(keys[i]) });
327
- }
328
- pairs.sort(function (a, b) {
329
- if (b.ratio !== a.ratio) return b.ratio - a.ratio;
330
- return a.command < b.command ? -1 : a.command > b.command ? 1 : 0;
331
- });
332
- out.top5 = pairs.slice(0, 5);
333
-
334
- return out;
335
- }
336
-
337
- // ---------- Exports ----------
338
-
339
- module.exports = {
340
- BELOW_THRESHOLD_RATIO: BELOW_THRESHOLD_RATIO,
341
- EVENT_REQUIRED_FIELDS: EVENT_REQUIRED_FIELDS,
342
- estimateTokens: estimateTokens,
343
- estimateRoomTokens: estimateRoomTokens,
344
- clearCache: clearCache,
345
- validateEventShape: validateEventShape,
346
- classifyRatio: classifyRatio,
347
- aggregateEvents: aggregateEvents,
348
- };