@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,312 @@
1
+ /*
2
+ * Copyright (c) 2026 Mindrian. BSL 1.1.
3
+ *
4
+ * Phase 88.2-06 -- Shape F.6 Plan Review Round renderer (UISEL-88.2-08 part 2).
5
+ *
6
+ * NAMING NOTE (R1 collision mitigation): this file is shape-f6-PLAN-REVIEW-renderer.cjs.
7
+ * lib/hmi/shape-f6-renderer.cjs is Phase 101-01 JTBD-aware Next Move and is byte-stable
8
+ * (sha256 baseline 1792535860abc791222bf0ecf59599d66e49ad9cc1606b3d8679fca2922150cf).
9
+ * The umbrella 'F' branch in selector-dispatcher.cjs continues to route to the Phase
10
+ * 101-01 module via JTBD logic; ONLY explicit requestedShape:'F.6' (string) routes
11
+ * to this plan-review renderer.
12
+ *
13
+ * Replaces the 88.2-00 Wave-0 stub. Implements F.6 per CONTEXT.md amendment
14
+ * 2026-04-29 + D-AMEND-01 (F.6 in scope of Phase 88.2 finish) + D-AMEND-04
15
+ * (open-vocab parent shape with optional personaContext suffix) + D-AMEND-05
16
+ * (Tier 2 persona-aware decoys via lib/hmi/decoy-tier.cjs dispatcher).
17
+ *
18
+ * F.6 invariants (CONTEXT.md spec body 2026-04-29):
19
+ * - Round size 15-30 questions (default 20). Real:decoy ratio 4:1 (16 + 4 at 20).
20
+ * - Real questions sourced from artifact's structural decomposition (tasks, gates,
21
+ * dependencies, acceptance criteria, success metrics).
22
+ * - Decoys NOT disclosed during round; debrief Shape A action report at round-end
23
+ * discloses (decoy-ethics contract, room/decisions/decision-decoy-ethics.md).
24
+ * - Each response writes ONE REVIEWED typed edge with properties
25
+ * { round_id, position, latency_ms, was_decoy, response, confidence_self_report }.
26
+ * Position-aware so debrief identifies decoys-caught vs decoys-missed.
27
+ * - Round can be paused via 'q' keystroke; resumes from same position; persists
28
+ * across session restarts (round-state graph node survives).
29
+ * - Cowork: parallel actor responses captured per actor_id (Cowork concurrency
30
+ * contract, room/decisions/decision-cowork-round-locking.md).
31
+ * - Double-line border (parent shape -- structural distinction from F.0 single-line).
32
+ *
33
+ * Three exports:
34
+ * 1. renderShapeF6PlanReview({ tier, round_id?, position, totalQuestions, claim,
35
+ * counts, header?, personaContext? })
36
+ * -> { zones, contract }
37
+ * Renders one question of N. The renderer is PURE (no FS reads, no DB writes);
38
+ * caller composes the round (decoy interleaving, real-question sourcing) by
39
+ * pairing this renderer with lib/hmi/decoy-tier.cjs and a per-position invocation.
40
+ *
41
+ * 2. buildReviewedEdge({ roomDir, round_id, position, latency_ms, was_decoy,
42
+ * response, confidence_self_report?, reason?, actor_id? })
43
+ * -> { ok, reason?, eventId? }
44
+ * Writes one REVIEWED edge to room.db via Phase 109 logEvent with eventType
45
+ * 'selector_response'. Closed-vocab response: confirm | reject | discuss |
46
+ * free-text. Graceful-fail envelope -- never throws.
47
+ *
48
+ * 3. emitRoundCompleted({ roomDir, round_id, real_count, decoy_count, tier })
49
+ * -> { ok, reason?, eventId? }
50
+ * Writes one f6_round_completed event at round close. Surfaces summary
51
+ * counts so the debrief Shape A action report can render without re-querying.
52
+ * Graceful-fail envelope -- never throws.
53
+ *
54
+ * Persona suffix (per D-AMEND-04 / 88.2-03 DISCRETION-AMEND-01 RESOLVED PATTERN):
55
+ * When personaContext is a non-empty string, the header gets ' ({personaContext} lens)'
56
+ * suffix. Cold-start (omitted/null/empty) preserves the persona-agnostic header
57
+ * byte-for-byte. The renderer stays PURE; the caller (dispatcher per 88.2-04) supplies
58
+ * the pre-computed string from readUserMd().role_blend per Phase 115 D-12.
59
+ *
60
+ * Glyph audit (12-glyph vocabulary per skills/ui-system/SKILL.md):
61
+ * ALLOWED: triangle (filled) / triangle (open) / square / dot / arrow.
62
+ * FORBIDDEN: rounded box corners, heavy horizontals/verticals, error/warn glyphs.
63
+ * The progress bar uses '=' and '-' ASCII (not heavy horizontal); count chips use plain text.
64
+ *
65
+ * Pure CJS, node built-ins only, zero deps (Phase 87 invariant).
66
+ * Canon Part 4: every response produces a typed REVIEWED edge -- no silent dismiss.
67
+ * Canon Part 8: all writes are LOCAL room.db; zero Brain egress.
68
+ */
69
+ 'use strict';
70
+
71
+ const F6_VERBS = ['Confirm', 'Reject', 'Discuss'];
72
+
73
+ // Closed-vocab response taxonomy. The renderer surfaces 3 verbs; the
74
+ // 'free-text' branch is reachable when a downstream prompt captures a
75
+ // reject reason (the reason becomes a property on the REVIEWED edge,
76
+ // not a separate response token).
77
+ const VALID_RESPONSES = ['confirm', 'reject', 'discuss', 'free-text'];
78
+
79
+ const RESPONSE_EVENT_TYPE = 'selector_response';
80
+ const ROUND_COMPLETED_EVENT_TYPE = 'f6_round_completed';
81
+
82
+ const MARKER_ROW = '▷'; // 12-glyph row marker (matches 88.2-01 + F.0)
83
+ const MARKER_PROGRESS = '▶'; // 12-glyph progress chevron
84
+ const DEFAULT_HEADER_PREFIX = '-- mindrianOS -- plan review --';
85
+
86
+ // Progress bar geometry: 20 cells, 5% each. Filled cells use '='; empty
87
+ // cells use '-'. ASCII only -- no heavy-horizontal box-drawing chars
88
+ // (forbidden in the 12-glyph audit; project-wide convention).
89
+ const PROGRESS_BAR_CELLS = 20;
90
+
91
+ /**
92
+ * Render one question of a F.6 plan-review round.
93
+ *
94
+ * Returns { zones, contract } where:
95
+ * - zones.header carries 'ROUND {position} of {totalQuestions}' + optional
96
+ * ' ({personaContext} lens)' suffix
97
+ * - zones.body carries the user-story claim + 3 verb rows + progress bar +
98
+ * count chips
99
+ * - contract.shape === 'F.6'; freeTextOffered === false; verbs frozen at
100
+ * ['Confirm', 'Reject', 'Discuss']; recommended === null in both modes
101
+ * (closed-vocab; no Brain confidence marker on F.6 itself)
102
+ */
103
+ function renderShapeF6PlanReview(input) {
104
+ const opts = (input && typeof input === 'object') ? input : {};
105
+ const tier = (typeof opts.tier === 'number') ? opts.tier : 0;
106
+ const mode = (tier >= 2) ? 'A' : 'B';
107
+ const position = (Number.isInteger(opts.position) && opts.position >= 1) ? opts.position : 1;
108
+ const totalQuestions = (Number.isInteger(opts.totalQuestions) && opts.totalQuestions >= 1)
109
+ ? opts.totalQuestions : 20;
110
+ const claim = (typeof opts.claim === 'string') ? opts.claim : '';
111
+ const counts = (opts.counts && typeof opts.counts === 'object') ? opts.counts : {};
112
+ const personaContext = (typeof opts.personaContext === 'string' && opts.personaContext.length > 0)
113
+ ? opts.personaContext : null;
114
+ const roundId = (typeof opts.round_id === 'string' && opts.round_id.length > 0)
115
+ ? opts.round_id : null;
116
+
117
+ // Header: caller override OR canonical 'ROUND N of M -- PLAN REVIEW' format.
118
+ const headerBase = (typeof opts.header === 'string' && opts.header.length > 0)
119
+ ? opts.header
120
+ : DEFAULT_HEADER_PREFIX + ' ROUND ' + position + ' of ' + totalQuestions;
121
+ const composedHeader = personaContext
122
+ ? (headerBase + ' (' + personaContext + ' lens)')
123
+ : headerBase;
124
+
125
+ // Body: user-story scaffold + 3 verb rows.
126
+ const verbRows = F6_VERBS.map((v, i) => MARKER_ROW + ' ' + (i + 1) + ') ' + v);
127
+
128
+ // Progress bar: 20 cells, 5% each. Filled = '=' / Empty = '-'.
129
+ const pct = Math.floor((position / totalQuestions) * 100);
130
+ const filledCells = Math.max(0, Math.min(PROGRESS_BAR_CELLS, Math.floor(pct / 5)));
131
+ let bar = '[';
132
+ for (let i = 0; i < PROGRESS_BAR_CELLS; i++) {
133
+ bar += (i < filledCells) ? '=' : '-';
134
+ }
135
+ bar += ']';
136
+
137
+ // Count chips: 'reject N discuss N confirm N' (plain text -- no decorative
138
+ // glyphs). The order matches the CONTEXT.md spec body sample box.
139
+ const cReject = Number.isInteger(counts.reject) ? counts.reject : 0;
140
+ const cDiscuss = Number.isInteger(counts.discuss) ? counts.discuss : 0;
141
+ const cConfirm = Number.isInteger(counts.confirm) ? counts.confirm : 0;
142
+ const countLine = 'reject ' + cReject + ' discuss ' + cDiscuss + ' confirm ' + cConfirm;
143
+
144
+ const claimBlock = claim
145
+ ? 'As a navigator, I confirm:\n "' + claim + '"\n\n'
146
+ : '';
147
+ const body = claimBlock +
148
+ verbRows.join('\n') + '\n\n' +
149
+ MARKER_PROGRESS + ' progress ' + bar + ' ' + pct + '%\n' +
150
+ countLine;
151
+
152
+ return {
153
+ zones: {
154
+ header: composedHeader,
155
+ body: body,
156
+ signals: '',
157
+ footer: null,
158
+ },
159
+ contract: {
160
+ shape: 'F.6',
161
+ keyboard: 'askuserquestion',
162
+ verbs: F6_VERBS.slice(), // defensive copy
163
+ freeTextOffered: false, // closed-vocab; reject reason captured via downstream prompt
164
+ mode: mode,
165
+ recommended: null, // F.6 carries no RECOMMENDED (closed-vocab parent shape)
166
+ border_style: 'double', // parent shape; distinct from F.0's single-line
167
+ personaContext: personaContext,
168
+ round_id: roundId,
169
+ position: position,
170
+ total_questions: totalQuestions,
171
+ },
172
+ };
173
+ }
174
+
175
+ /**
176
+ * Build and emit a REVIEWED typed edge to room.db via Phase 109.
177
+ * Graceful-fail envelope -- never throws.
178
+ *
179
+ * Closed-vocab response: confirm | reject | discuss | free-text.
180
+ * was_decoy is required boolean (per F.6 spec position-aware debrief).
181
+ * confidence_self_report is optional 1-5 integer.
182
+ * reason is optional free-text (carries reject reason when response === 'reject').
183
+ * actor_id is optional Cowork scalar handle (parallel-actor disambiguation).
184
+ */
185
+ function buildReviewedEdge(args) {
186
+ try {
187
+ const a = (args && typeof args === 'object') ? args : {};
188
+
189
+ if (typeof a.round_id !== 'string' || a.round_id.length === 0) {
190
+ return { ok: false, reason: 'invalid_round_id' };
191
+ }
192
+ if (!Number.isInteger(a.position) || a.position < 1) {
193
+ return { ok: false, reason: 'invalid_position' };
194
+ }
195
+ if (typeof a.was_decoy !== 'boolean') {
196
+ return { ok: false, reason: 'invalid_was_decoy' };
197
+ }
198
+ if (VALID_RESPONSES.indexOf(a.response) === -1) {
199
+ return { ok: false, reason: 'invalid_response' };
200
+ }
201
+ if (a.confidence_self_report !== undefined) {
202
+ const c = a.confidence_self_report;
203
+ if (!Number.isInteger(c) || c < 1 || c > 5) {
204
+ return { ok: false, reason: 'invalid_confidence' };
205
+ }
206
+ }
207
+
208
+ const { EVENT_TYPES, logEvent } = require('../core/navigation/memory-events.cjs');
209
+ if (!EVENT_TYPES.has(RESPONSE_EVENT_TYPE)) {
210
+ return { ok: false, reason: 'event_type_not_registered' };
211
+ }
212
+
213
+ // Project standard: node:sqlite DatabaseSync per lib/core/lazygraph-ops.cjs.
214
+ // better-sqlite3 documented as broken in lib/import/PRECONDITIONS.md.
215
+ let DatabaseSync;
216
+ try { ({ DatabaseSync } = require('node:sqlite')); }
217
+ catch (_e) { return { ok: false, reason: 'sqlite_unavailable' }; }
218
+
219
+ const fs = require('node:fs');
220
+ const path = require('node:path');
221
+ if (typeof a.roomDir !== 'string' || a.roomDir.length === 0) {
222
+ return { ok: false, reason: 'invalid_room_dir' };
223
+ }
224
+ const dbPath = path.join(a.roomDir, '.mindrian', 'room.db');
225
+ if (!fs.existsSync(dbPath)) {
226
+ return { ok: false, reason: 'db_not_initialized' };
227
+ }
228
+
229
+ const db = new DatabaseSync(dbPath);
230
+ try {
231
+ const props = {
232
+ round_id: a.round_id,
233
+ position: a.position,
234
+ latency_ms: (typeof a.latency_ms === 'number') ? a.latency_ms : null,
235
+ was_decoy: a.was_decoy,
236
+ response: a.response,
237
+ };
238
+ if (a.confidence_self_report !== undefined) props.confidence_self_report = a.confidence_self_report;
239
+ if (typeof a.reason === 'string' && a.reason.length > 0) props.reason = a.reason;
240
+ if (typeof a.actor_id === 'string' && a.actor_id.length > 0) props.actor_id = a.actor_id;
241
+ return logEvent(db, RESPONSE_EVENT_TYPE, props);
242
+ } finally {
243
+ db.close();
244
+ }
245
+ } catch (_e) {
246
+ return { ok: false, reason: 'edge_build_threw_caught' };
247
+ }
248
+ }
249
+
250
+ /**
251
+ * Emit the f6_round_completed event at round close. Carries summary counts
252
+ * + tier label so debrief Shape A action report can render without
253
+ * re-querying the per-response edges.
254
+ *
255
+ * Graceful-fail envelope -- never throws.
256
+ */
257
+ function emitRoundCompleted(args) {
258
+ try {
259
+ const a = (args && typeof args === 'object') ? args : {};
260
+ if (typeof a.round_id !== 'string' || a.round_id.length === 0) {
261
+ return { ok: false, reason: 'invalid_round_id' };
262
+ }
263
+
264
+ const { EVENT_TYPES, logEvent } = require('../core/navigation/memory-events.cjs');
265
+ if (!EVENT_TYPES.has(ROUND_COMPLETED_EVENT_TYPE)) {
266
+ return { ok: false, reason: 'event_type_not_registered' };
267
+ }
268
+
269
+ let DatabaseSync;
270
+ try { ({ DatabaseSync } = require('node:sqlite')); }
271
+ catch (_e) { return { ok: false, reason: 'sqlite_unavailable' }; }
272
+
273
+ const fs = require('node:fs');
274
+ const path = require('node:path');
275
+ if (typeof a.roomDir !== 'string' || a.roomDir.length === 0) {
276
+ return { ok: false, reason: 'invalid_room_dir' };
277
+ }
278
+ const dbPath = path.join(a.roomDir, '.mindrian', 'room.db');
279
+ if (!fs.existsSync(dbPath)) {
280
+ return { ok: false, reason: 'db_not_initialized' };
281
+ }
282
+
283
+ const db = new DatabaseSync(dbPath);
284
+ try {
285
+ const props = {
286
+ round_id: a.round_id,
287
+ real_count: (typeof a.real_count === 'number') ? a.real_count : null,
288
+ decoy_count: (typeof a.decoy_count === 'number') ? a.decoy_count : null,
289
+ tier: (typeof a.tier === 'string') ? a.tier : null,
290
+ ended_at: new Date().toISOString(),
291
+ };
292
+ return logEvent(db, ROUND_COMPLETED_EVENT_TYPE, props);
293
+ } finally {
294
+ db.close();
295
+ }
296
+ } catch (_e) {
297
+ return { ok: false, reason: 'round_completion_threw_caught' };
298
+ }
299
+ }
300
+
301
+ module.exports = {
302
+ renderShapeF6PlanReview: renderShapeF6PlanReview,
303
+ buildReviewedEdge: buildReviewedEdge,
304
+ emitRoundCompleted: emitRoundCompleted,
305
+ F6_VERBS: F6_VERBS.slice(),
306
+ VALID_RESPONSES: VALID_RESPONSES.slice(),
307
+ _internal: {
308
+ RESPONSE_EVENT_TYPE: RESPONSE_EVENT_TYPE,
309
+ ROUND_COMPLETED_EVENT_TYPE: ROUND_COMPLETED_EVENT_TYPE,
310
+ PROGRESS_BAR_CELLS: PROGRESS_BAR_CELLS,
311
+ },
312
+ };
@@ -0,0 +1,144 @@
1
+ /*
2
+ * Copyright (c) 2026 Mindrian. BSL 1.1.
3
+ *
4
+ * Phase 101-01 -- Shape F.6 (JTBD-aware Next Move) renderer.
5
+ *
6
+ * Renders the Phase 100 JTBD signal into a Shape F selector body whose
7
+ * verb vocabulary is drawn from `entries[i].next_move_verbs[]` of the
8
+ * canonical taxonomy. Free-Text is appended last (D-10 hardcoded).
9
+ * Mode A (tier >= 2) marks `recommendedVerb` with `▶`; Mode B suppresses
10
+ * the marker (D-05). Falls through to F.1 if the taxonomy entry is
11
+ * degenerate (verb set < 3 entries) or missing -- per D-04 + RESEARCH §8
12
+ * pitfall 4. Keyboard contract delegates to F.1 (D-01).
13
+ *
14
+ * Allowed glyphs in body (per Canon Part 3 + skills/ui-system/SKILL.md
15
+ * 12-glyph vocabulary): ▶ ▷ ■ • →. No box characters.
16
+ *
17
+ * Pure CJS, node built-ins only, zero deps (Phase 87 invariant).
18
+ *
19
+ * API:
20
+ * renderShapeF6({ jtbd, tier, recommendedVerb, header? })
21
+ * -> { zones, contract }
22
+ * | { error: string } // taxonomy missing
23
+ * | { fallthrough: true, reason: string } // degenerate verb set
24
+ */
25
+
26
+ 'use strict';
27
+
28
+ const fs = require('node:fs');
29
+ const path = require('node:path');
30
+
31
+ const TAXONOMY_PATH = path.join(__dirname, 'jtbd-taxonomy.json');
32
+ const FREE_TEXT = 'Free-Text';
33
+ const MIN_VERBS = 3; // D-04 + RESEARCH §8 pitfall 4: < 3 verbs = degenerate
34
+ const MARKER_RECOMMENDED = '▶';
35
+ const MARKER_ROW = '▷';
36
+
37
+ function loadTaxonomy() {
38
+ // Taxonomy-missing graceful branch (D-12, CONTEXT). Out-of-order execution
39
+ // (Phase 101 before Phase 100 Wave 1) returns 3-line error per Canon Part 3
40
+ // Rule 2 instead of crashing.
41
+ if (!fs.existsSync(TAXONOMY_PATH)) return null;
42
+ try {
43
+ const raw = fs.readFileSync(TAXONOMY_PATH, 'utf8');
44
+ return JSON.parse(raw);
45
+ } catch (e) {
46
+ return null;
47
+ }
48
+ }
49
+
50
+ function entriesOf(taxonomy) {
51
+ if (!taxonomy) return [];
52
+ if (Array.isArray(taxonomy.entries)) return taxonomy.entries;
53
+ if (Array.isArray(taxonomy.jtbds)) return taxonomy.jtbds;
54
+ return [];
55
+ }
56
+
57
+ function findEntry(taxonomy, jtbdId) {
58
+ const list = entriesOf(taxonomy);
59
+ if (!Array.isArray(list) || typeof jtbdId !== 'string') return null;
60
+ return list.find(e => e && e.id === jtbdId) || null;
61
+ }
62
+
63
+ function defaultHeader(jtbdId) {
64
+ // Zone 1 default per RESEARCH §2 step 6.
65
+ return '-- mindrianOS -- ' + jtbdId + ' -- next move --';
66
+ }
67
+
68
+ function renderShapeF6(input) {
69
+ const opts = (input && typeof input === 'object') ? input : {};
70
+ const jtbd = typeof opts.jtbd === 'string' ? opts.jtbd : null;
71
+ const tier = typeof opts.tier === 'number' ? opts.tier : 0;
72
+ const recommendedVerb = typeof opts.recommendedVerb === 'string' && opts.recommendedVerb.length > 0
73
+ ? opts.recommendedVerb : null;
74
+ const header = typeof opts.header === 'string' && opts.header.length > 0
75
+ ? opts.header : null;
76
+
77
+ // Step 0: taxonomy-missing graceful branch (Canon Part 3 Rule 2 + D-12).
78
+ const taxonomy = loadTaxonomy();
79
+ if (!taxonomy) {
80
+ return {
81
+ error:
82
+ 'x lib/hmi/jtbd-taxonomy.json not found\n' +
83
+ 'Why: Phase 100 Wave 1 must ship before F.6 can render\n' +
84
+ 'Fix: complete Phase 100 plans 100-01 first',
85
+ };
86
+ }
87
+
88
+ // Null-jtbd is dispatcher's job (D-02); F.6 expects non-null. Treat as
89
+ // fallthrough so the dispatcher routes to F.1.
90
+ if (!jtbd) {
91
+ return { fallthrough: true, reason: 'jtbd_null' };
92
+ }
93
+
94
+ const entry = findEntry(taxonomy, jtbd);
95
+ if (!entry) {
96
+ return { fallthrough: true, reason: 'jtbd_unknown' };
97
+ }
98
+
99
+ const rawVerbs = Array.isArray(entry.next_move_verbs) ? entry.next_move_verbs.slice() : [];
100
+
101
+ // Step 2: degenerate verb set (D-04 + RESEARCH §8 pitfall 4). Per the plan,
102
+ // count is on raw taxonomy verbs (Free-Text is appended later).
103
+ if (rawVerbs.length < MIN_VERBS) {
104
+ return { fallthrough: true, reason: 'degenerate_verb_set' };
105
+ }
106
+
107
+ // Step 3: Free-Text always last (D-10 hardcoded). Caller cannot omit.
108
+ // Defensive de-dup if taxonomy already lists Free-Text (current taxonomy
109
+ // does for the explore JTBD; explore is < 3 anyway, but be safe).
110
+ const verbsNoFree = rawVerbs.filter(v => v !== FREE_TEXT);
111
+ const verbs = verbsNoFree.concat([FREE_TEXT]);
112
+
113
+ // Step 4: Mode A (tier >= 2) renders recommendation; Mode B suppresses.
114
+ const mode = tier >= 2 ? 'A' : 'B';
115
+ const recInList = (mode === 'A' && recommendedVerb && verbs.indexOf(recommendedVerb) !== -1)
116
+ ? recommendedVerb : null;
117
+
118
+ // Step 6: render zones.
119
+ const zone1 = header || defaultHeader(jtbd);
120
+ const zone2Lines = verbs.map((verb, i) => {
121
+ const prefix = (verb === recInList) ? MARKER_RECOMMENDED : MARKER_ROW;
122
+ const num = String(i + 1);
123
+ return prefix + ' ' + num + '. ' + verb;
124
+ });
125
+
126
+ const zones = {
127
+ header: zone1,
128
+ body: zone2Lines.join('\n'),
129
+ signals: '',
130
+ footer: null, // F.x is itself the selector; no separate Zone 4 footer.
131
+ };
132
+
133
+ // Step 7: contract -- keyboard inheritance from F.1 (D-01).
134
+ const contract = {
135
+ keyboard: 'f1-inheritance',
136
+ verbs: verbs.slice(),
137
+ mode: mode,
138
+ recommended: recInList,
139
+ };
140
+
141
+ return { zones, contract };
142
+ }
143
+
144
+ module.exports = { renderShapeF6 };
@@ -0,0 +1,219 @@
1
+ /*
2
+ * Copyright (c) 2026 Mindrian. BSL 1.1.
3
+ *
4
+ * Phase 101-02 -- lib/hmi/shape-g-renderer.cjs
5
+ * Shape G (Comparison Matrix). Render-only in v1; interactive table picking
6
+ * is v1.15.x. Produces a 4-zone block whose Zone 2 body is a comparison
7
+ * matrix: rows = options, columns = criteria.
8
+ *
9
+ * Contract (RESEARCH section 3 + PLAN must_haves):
10
+ * - options.length >= 3 AND criteria.length >= 2 (D-08); else fallthrough to E
11
+ * - cells clamped to CELL_CLAMP (12) chars + ellipsis; criteria to CRITERION_CLAMP (10)
12
+ * - 4-zone wrapping with Zone 4 footer (>= 2 /mos: references)
13
+ * - 12-glyph compliance + ASCII pipe column separator (allowed for table use)
14
+ *
15
+ * Source uses String.fromCharCode(0x2500) for U+2500 so the file contains
16
+ * zero literal forbidden box-drawing chars per scripts/doctor.cjs scan.
17
+ *
18
+ * Canon Part 8 (LOCAL ONLY): pure function. License: BSL 1.1.
19
+ */
20
+
21
+ 'use strict';
22
+
23
+ // path + fs reserved for forward-compat (PLAN names them as imports).
24
+ require('node:path');
25
+ require('node:fs');
26
+
27
+ const G_FILLED_SQUARE = '■'; // U+25A0
28
+ const G_RIGHT_TRI_FILLED = '▶'; // U+25B6
29
+ const G_RIGHT_TRI_EMPTY = '▷'; // U+25B7
30
+ const G_DASH = String.fromCharCode(0x2500); // U+2500 (avoid literal in source)
31
+
32
+ const CRITERION_CLAMP = 10;
33
+ const CELL_CLAMP = 12;
34
+ const ELLIPSIS = '...';
35
+
36
+ const DEFAULT_FOOTER_VERBS = [
37
+ { glyph: G_RIGHT_TRI_FILLED, command: '/mos:scenario-plan', hint: 'Branch into futures' },
38
+ { glyph: G_RIGHT_TRI_EMPTY, command: '/mos:deep-grade', hint: 'Score each option' },
39
+ { glyph: G_RIGHT_TRI_EMPTY, command: '/mos:jtbd set decide-pursue', hint: 'Move to commit gate' },
40
+ ];
41
+
42
+ function clamp(s, maxLen) {
43
+ const str = (s === null || s === undefined) ? '' : String(s);
44
+ if (str.length <= maxLen) return str;
45
+ if (maxLen <= ELLIPSIS.length) return str.slice(0, maxLen);
46
+ return str.slice(0, maxLen - ELLIPSIS.length) + ELLIPSIS;
47
+ }
48
+
49
+ function pad(s, width) {
50
+ const str = (s === null || s === undefined) ? '' : String(s);
51
+ if (str.length >= width) return str;
52
+ return str + ' '.repeat(width - str.length);
53
+ }
54
+
55
+ function repeatChar(ch, width) {
56
+ return width <= 0 ? '' : ch.repeat(width);
57
+ }
58
+
59
+ function validateDimensions(options, criteria) {
60
+ if (!Array.isArray(options) || options.length < 3) {
61
+ return { fallthrough: true, fallthroughTo: 'E', reason: 'degenerate_matrix' };
62
+ }
63
+ if (!Array.isArray(criteria) || criteria.length < 2) {
64
+ return { fallthrough: true, fallthroughTo: 'E', reason: 'degenerate_matrix' };
65
+ }
66
+ return null;
67
+ }
68
+
69
+ function computeWidths(options, criteriaClamped, cellsClamped) {
70
+ let optionColWidth = 0;
71
+ for (const opt of options) {
72
+ const len = String(opt || '').length;
73
+ if (len > optionColWidth) optionColWidth = len;
74
+ }
75
+ const criterionWidths = new Array(criteriaClamped.length);
76
+ for (let c = 0; c < criteriaClamped.length; c++) {
77
+ let maxCell = criteriaClamped[c].length;
78
+ for (const opt of options) {
79
+ const cell = (cellsClamped[opt] && cellsClamped[opt][criteriaClamped[c]]) || '';
80
+ if (cell.length > maxCell) maxCell = cell.length;
81
+ }
82
+ criterionWidths[c] = maxCell;
83
+ }
84
+ return { optionColWidth: optionColWidth, criterionWidths: criterionWidths };
85
+ }
86
+
87
+ function renderHeader(options, criteria, opts) {
88
+ const room = (opts && opts.room) ? opts.room : 'mindrianOS';
89
+ const middle = opts && opts.title
90
+ ? opts.title
91
+ : (opts && opts.jtbd ? opts.jtbd : 'compare');
92
+ const dims = options.length + ' options x ' + criteria.length + ' criteria';
93
+ return '-- ' + room + ' -- ' + middle + ' -- ' + dims + ' --';
94
+ }
95
+
96
+ function renderBody(options, criteriaClamped, cellsClamped) {
97
+ const widths = computeWidths(options, criteriaClamped, cellsClamped);
98
+ const optW = widths.optionColWidth;
99
+ const critW = widths.criterionWidths;
100
+ const lines = [];
101
+ lines.push(G_FILLED_SQUARE + ' Comparison Matrix');
102
+
103
+ let headerRow = ' ' + pad('', optW);
104
+ for (let c = 0; c < criteriaClamped.length; c++) {
105
+ headerRow += ' | ' + pad(criteriaClamped[c], critW[c]);
106
+ }
107
+ lines.push(headerRow);
108
+
109
+ let sepRow = ' ' + repeatChar(G_DASH, optW);
110
+ for (let c = 0; c < criteriaClamped.length; c++) {
111
+ sepRow += ' ' + G_DASH + ' ' + repeatChar(G_DASH, critW[c]);
112
+ }
113
+ lines.push(sepRow);
114
+
115
+ for (const opt of options) {
116
+ let row = ' ' + pad(String(opt), optW);
117
+ for (let c = 0; c < criteriaClamped.length; c++) {
118
+ const cell = (cellsClamped[opt] && cellsClamped[opt][criteriaClamped[c]]) || '';
119
+ row += ' | ' + pad(cell, critW[c]);
120
+ }
121
+ lines.push(row);
122
+ }
123
+ return lines.join('\n');
124
+ }
125
+
126
+ /**
127
+ * Phase 101-05: Mode-aware footer marker selection.
128
+ *
129
+ * Per Canon Part 3 "Option generation tier-awareness":
130
+ * Mode A (tier >= 2) -> first verb may carry RECOMMENDED ▶ marker (Brain-derived).
131
+ * Mode B (tier === 1) -> all verbs use the empty triangle ▷ marker; ▶ NEVER renders.
132
+ *
133
+ * Defense-in-depth: this guard lives inside the renderer regardless of whether
134
+ * the dispatcher already gated the call. A buggy caller forgetting to pass mode
135
+ * still cannot leak a ▶ marker in Mode B.
136
+ */
137
+ function renderFooter(footerVerbs, mode) {
138
+ const verbs = (Array.isArray(footerVerbs) && footerVerbs.length >= 2)
139
+ ? footerVerbs
140
+ : DEFAULT_FOOTER_VERBS;
141
+ const isModeB = (mode === 'B');
142
+ const lines = [];
143
+ for (let i = 0; i < verbs.length; i++) {
144
+ const v = verbs[i];
145
+ const defaultGlyph = (i === 0 && !isModeB) ? G_RIGHT_TRI_FILLED : G_RIGHT_TRI_EMPTY;
146
+ // In Mode B, hard-override any caller-supplied ▶ glyph to ▷ (Canon Part 3).
147
+ const rawGlyph = v.glyph || defaultGlyph;
148
+ const glyph = isModeB ? G_RIGHT_TRI_EMPTY : rawGlyph;
149
+ const cmd = v.command || '/mos:status';
150
+ const hint = v.hint || '';
151
+ const hintSuffix = hint ? '# ' + hint : '';
152
+ lines.push(' ' + glyph + ' ' + pad(cmd, 28) + hintSuffix);
153
+ }
154
+ return lines.join('\n');
155
+ }
156
+
157
+ function renderShapeG(input) {
158
+ const opts = input || {};
159
+ const options = opts.options;
160
+ const criteria = opts.criteria;
161
+ const cells = opts.cells || {};
162
+
163
+ // Phase 101-05: derive mode from explicit `mode` arg or fall back to `tier`.
164
+ // Mode A (Brain reachable + tier >= 2) renders RECOMMENDED ▶ on first footer verb.
165
+ // Mode B (tier 1 / Brain unreachable) suppresses ▶ across the entire footer.
166
+ let mode;
167
+ if (opts.mode === 'A' || opts.mode === 'B') {
168
+ mode = opts.mode;
169
+ } else if (typeof opts.tier === 'number') {
170
+ mode = opts.tier >= 2 ? 'A' : 'B';
171
+ } else {
172
+ mode = 'A'; // back-compat: pre-101-05 callers default to Mode A behavior
173
+ }
174
+
175
+ const dimErr = validateDimensions(options, criteria);
176
+ if (dimErr) return dimErr;
177
+
178
+ const criteriaClamped = criteria.map((c) => clamp(c, CRITERION_CLAMP));
179
+ const cellsClamped = {};
180
+ for (const opt of options) {
181
+ cellsClamped[opt] = {};
182
+ const optCells = (cells[opt] && typeof cells[opt] === 'object') ? cells[opt] : {};
183
+ for (let c = 0; c < criteria.length; c++) {
184
+ const origKey = criteria[c];
185
+ const clampedKey = criteriaClamped[c];
186
+ const raw = optCells[origKey] !== undefined ? optCells[origKey] : optCells[clampedKey];
187
+ cellsClamped[opt][clampedKey] = clamp(raw, CELL_CLAMP);
188
+ }
189
+ }
190
+
191
+ return {
192
+ zones: {
193
+ header: renderHeader(options, criteria, opts),
194
+ body: renderBody(options, criteriaClamped, cellsClamped),
195
+ signals: '',
196
+ footer: renderFooter(opts.footerVerbs, mode),
197
+ },
198
+ contract: {
199
+ shape: 'G',
200
+ passthrough: false,
201
+ options_count: options.length,
202
+ criteria_count: criteria.length,
203
+ mode: mode,
204
+ },
205
+ };
206
+ }
207
+
208
+ module.exports = {
209
+ renderShapeG: renderShapeG,
210
+ _internal: {
211
+ clamp: clamp,
212
+ pad: pad,
213
+ validateDimensions: validateDimensions,
214
+ computeWidths: computeWidths,
215
+ CRITERION_CLAMP: CRITERION_CLAMP,
216
+ CELL_CLAMP: CELL_CLAMP,
217
+ ELLIPSIS: ELLIPSIS,
218
+ },
219
+ };