@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.
- package/.claude-plugin/plugin.json +21 -0
- package/.mcp.json +9 -0
- package/CHANGELOG.md +3333 -0
- package/LICENSE +123 -0
- package/README.md +673 -0
- package/agents/brain-query.md +80 -0
- package/agents/framework-runner.md +237 -0
- package/agents/grading.md +188 -0
- package/agents/investor.md +128 -0
- package/agents/larry-extended.md +135 -0
- package/agents/opportunity-scanner.md +91 -0
- package/agents/persona-analyst.md +132 -0
- package/agents/research.md +89 -0
- package/agents/reverse-salient-agent.md +27 -0
- package/bin/cli.js +142 -0
- package/bin/mindrian-mcp-server.cjs +182 -0
- package/bin/mindrian-tools.cjs +765 -0
- package/commands/act.md +439 -0
- package/commands/admin.md +404 -0
- package/commands/analyze-needs.md +42 -0
- package/commands/analyze-systems.md +39 -0
- package/commands/analyze-timing.md +42 -0
- package/commands/auto-explore.md +64 -0
- package/commands/beautiful-question.md +40 -0
- package/commands/brain-derive.md +78 -0
- package/commands/build-knowledge.md +42 -0
- package/commands/build-thesis.md +46 -0
- package/commands/causal.md +234 -0
- package/commands/challenge-assumptions.md +33 -0
- package/commands/compare-ventures.md +83 -0
- package/commands/dashboard.md +110 -0
- package/commands/deep-grade.md +82 -0
- package/commands/diagnose.md +58 -0
- package/commands/diagnostics.md +151 -0
- package/commands/doctor.md +151 -0
- package/commands/dominant-designs.md +40 -0
- package/commands/explain-decision.md +87 -0
- package/commands/explore-domains.md +42 -0
- package/commands/explore-futures.md +40 -0
- package/commands/explore-trends.md +42 -0
- package/commands/export.md +103 -0
- package/commands/file-meeting.md +724 -0
- package/commands/find-analogies.md +188 -0
- package/commands/find-bottlenecks.md +62 -0
- package/commands/find-connections.md +76 -0
- package/commands/funding.md +81 -0
- package/commands/grade.md +203 -0
- package/commands/graph.md +128 -0
- package/commands/hat-briefing.md +125 -0
- package/commands/heal.md +196 -0
- package/commands/help.md +399 -0
- package/commands/hmi-status.md +172 -0
- package/commands/jtbd.md +241 -0
- package/commands/leadership.md +73 -0
- package/commands/lean-canvas.md +40 -0
- package/commands/macro-trends.md +40 -0
- package/commands/map-unknowns.md +40 -0
- package/commands/memory.md +173 -0
- package/commands/models.md +175 -0
- package/commands/mos-reason.md +285 -0
- package/commands/mullins.md +120 -0
- package/commands/new-project.md +481 -0
- package/commands/onboard.md +434 -0
- package/commands/operator.md +149 -0
- package/commands/opportunities.md +144 -0
- package/commands/organize.md +497 -0
- package/commands/persona.md +198 -0
- package/commands/pipeline.md +112 -0
- package/commands/present.md +91 -0
- package/commands/publish.md +201 -0
- package/commands/query.md +124 -0
- package/commands/radar.md +72 -0
- package/commands/reanalyze.md +91 -0
- package/commands/research.md +196 -0
- package/commands/room.md +352 -0
- package/commands/rooms.md +598 -0
- package/commands/root-cause.md +40 -0
- package/commands/rs-experts.md +85 -0
- package/commands/rs-explain.md +100 -0
- package/commands/rs-fetch.md +94 -0
- package/commands/rs-thesis.md +85 -0
- package/commands/scenario-plan.md +40 -0
- package/commands/scheduled-tasks.md +285 -0
- package/commands/score-innovation.md +43 -0
- package/commands/scout.md +239 -0
- package/commands/setup.md +618 -0
- package/commands/snapshot.md +147 -0
- package/commands/speakers.md +84 -0
- package/commands/splash.md +28 -0
- package/commands/status.md +75 -0
- package/commands/structure-argument.md +42 -0
- package/commands/suggest-next.md +80 -0
- package/commands/systems-thinking.md +40 -0
- package/commands/think-hats.md +42 -0
- package/commands/update.md +181 -0
- package/commands/user-needs.md +40 -0
- package/commands/validate.md +40 -0
- package/commands/value-proposition.md +61 -0
- package/commands/vault.md +180 -0
- package/commands/visualize.md +52 -0
- package/commands/whitespace.md +507 -0
- package/commands/wiki.md +69 -0
- package/hooks/hooks.json +381 -0
- package/hooks/run-hook.cmd +64 -0
- package/lib/__init__.py +0 -0
- package/lib/__pycache__/__init__.cpython-312.pyc +0 -0
- package/lib/agents/auto-explore-agent.cjs +1043 -0
- package/lib/agents/reverse-salient-agent.cjs +679 -0
- package/lib/agents/tension-hook-agent.cjs +544 -0
- package/lib/brain/ROOM.md +44 -0
- package/lib/brain/chain-recommender.cjs +301 -0
- package/lib/chat/chat-context.js +185 -0
- package/lib/chat/chat-panel.js +721 -0
- package/lib/chat/fabric-chat.cjs +288 -0
- package/lib/chat/generative-tools.js +219 -0
- package/lib/conversation/ROOM.md +39 -0
- package/lib/conversation/classifier-rules.json +38 -0
- package/lib/conversation/classifier.cjs +264 -0
- package/lib/conversation/operator.cjs +287 -0
- package/lib/copy/115-spec-strings.cjs +55 -0
- package/lib/core/__init__.py +0 -0
- package/lib/core/__nav-stub.cjs +14 -0
- package/lib/core/__pycache__/__init__.cpython-312.pyc +0 -0
- package/lib/core/__pycache__/rs-math.cpython-312.pyc +0 -0
- package/lib/core/__pycache__/rs_cache.cpython-312.pyc +0 -0
- package/lib/core/__pycache__/rs_corpus.cpython-312.pyc +0 -0
- package/lib/core/__pycache__/rs_hybrid.cpython-312.pyc +0 -0
- package/lib/core/__pycache__/rs_math.cpython-312.pyc +0 -0
- package/lib/core/__pycache__/rs_rooms.cpython-312.pyc +0 -0
- package/lib/core/artifact-id.cjs +148 -0
- package/lib/core/asset-ops.cjs +151 -0
- package/lib/core/auto-commit-throttle.cjs +129 -0
- package/lib/core/bearer-token.cjs +199 -0
- package/lib/core/brain-client.cjs +865 -0
- package/lib/core/brain-derivation-prompts.cjs +326 -0
- package/lib/core/brain-derivation-queue.cjs +431 -0
- package/lib/core/brain-derivation.cjs +580 -0
- package/lib/core/brain-md-schema.cjs +528 -0
- package/lib/core/brain-md-staleness.cjs +357 -0
- package/lib/core/brain-response-sanitize.cjs +188 -0
- package/lib/core/bridge-writer.cjs +477 -0
- package/lib/core/chat-context-builder.cjs +253 -0
- package/lib/core/cross-room-aggregator.cjs +762 -0
- package/lib/core/daily-briefing.cjs +438 -0
- package/lib/core/decision-capture.cjs +618 -0
- package/lib/core/deep-links.cjs +82 -0
- package/lib/core/dispatch-optimizer.cjs +354 -0
- package/lib/core/dual-path-detector.cjs +84 -0
- package/lib/core/dual-path-detector.test.cjs +334 -0
- package/lib/core/exports-log.cjs +79 -0
- package/lib/core/feynman-minto-invariants.cjs +605 -0
- package/lib/core/folder-memory-async.cjs +338 -0
- package/lib/core/folder-memory-shared.cjs +890 -0
- package/lib/core/folder-memory.cjs +416 -0
- package/lib/core/framework-chain-composer.cjs +411 -0
- package/lib/core/frontmatter-schemas.cjs +330 -0
- package/lib/core/git-ops.cjs +141 -0
- package/lib/core/graph-ops.cjs +258 -0
- package/lib/core/hat-persistence.cjs +362 -0
- package/lib/core/index.cjs +60 -0
- package/lib/core/integration-registry.cjs +232 -0
- package/lib/core/intelligence-cascade.cjs +661 -0
- package/lib/core/lazygraph-ops.cjs +1057 -0
- package/lib/core/lru-cache.cjs +139 -0
- package/lib/core/mcp-profiles.cjs +182 -0
- package/lib/core/meeting-ops.cjs +54 -0
- package/lib/core/memory-ops.cjs +600 -0
- package/lib/core/migrations/ROOM.md +33 -0
- package/lib/core/migrations/phase-109-nodes-provenance.cjs +339 -0
- package/lib/core/migrations/phase-109-session-focus.cjs +99 -0
- package/lib/core/model-profiles.cjs +246 -0
- package/lib/core/mullins-scaffold.cjs +160 -0
- package/lib/core/nav-dial.cjs +316 -0
- package/lib/core/navigation/ROOM.md +15 -0
- package/lib/core/navigation/explanation.cjs +43 -0
- package/lib/core/navigation/focus.cjs +135 -0
- package/lib/core/navigation/ingestion.cjs +82 -0
- package/lib/core/navigation/insights.cjs +350 -0
- package/lib/core/navigation/memory-events.cjs +118 -0
- package/lib/core/navigation/neighborhood.cjs +78 -0
- package/lib/core/navigation/packet.cjs +182 -0
- package/lib/core/navigation/room-home.cjs +127 -0
- package/lib/core/navigation/transitions.cjs +82 -0
- package/lib/core/navigation-engine-shared.cjs +242 -0
- package/lib/core/navigation-engine.cjs +664 -0
- package/lib/core/navigation.cjs +60 -0
- package/lib/core/nl-graph-queries.cjs +164 -0
- package/lib/core/offer-presenter.cjs +406 -0
- package/lib/core/opportunity-extractor.cjs +183 -0
- package/lib/core/opportunity-ops.cjs +1371 -0
- package/lib/core/persona-ops.cjs +537 -0
- package/lib/core/persona-taxonomy.cjs +190 -0
- package/lib/core/platform-gates.cjs +120 -0
- package/lib/core/platform.cjs +257 -0
- package/lib/core/proactive-intelligence.cjs +528 -0
- package/lib/core/problem-type-router.cjs +315 -0
- package/lib/core/reasoning-ops.cjs +639 -0
- package/lib/core/reverse-salient-persona-suffix.cjs +115 -0
- package/lib/core/room-classifier-strict-mode.cjs +229 -0
- package/lib/core/room-db.cjs +127 -0
- package/lib/core/room-ops-async.cjs +92 -0
- package/lib/core/room-ops-shared.cjs +64 -0
- package/lib/core/room-ops-sync.cjs +70 -0
- package/lib/core/room-ops.cjs +32 -0
- package/lib/core/room-type-detector.cjs +386 -0
- package/lib/core/rs-brain-substrate-prompts.cjs +129 -0
- package/lib/core/rs-brain-substrate.cjs +570 -0
- package/lib/core/rs-breakthrough-scorer.cjs +255 -0
- package/lib/core/rs-canon-violations.cjs +82 -0
- package/lib/core/rs-chain-feeder.cjs +343 -0
- package/lib/core/rs-commercial-assessor.cjs +280 -0
- package/lib/core/rs-differential-scorer.cjs +376 -0
- package/lib/core/rs-domain-analyzer.cjs +385 -0
- package/lib/core/rs-egress-prompts.cjs +113 -0
- package/lib/core/rs-egress-telemetry.cjs +225 -0
- package/lib/core/rs-egress-violations.cjs +53 -0
- package/lib/core/rs-expert-mapper.cjs +467 -0
- package/lib/core/rs-fetcher-academic.cjs +697 -0
- package/lib/core/rs-fetcher-experts.cjs +314 -0
- package/lib/core/rs-fetcher-industry.cjs +731 -0
- package/lib/core/rs-fetcher-patents.cjs +564 -0
- package/lib/core/rs-innovation-classifier.cjs +194 -0
- package/lib/core/rs-mind-map.cjs +656 -0
- package/lib/core/rs-neo4j-writer.cjs +388 -0
- package/lib/core/rs-nl-to-query.cjs +425 -0
- package/lib/core/rs-pinecone-bridge.cjs +303 -0
- package/lib/core/rs-preprocessor.cjs +350 -0
- package/lib/core/rs-query-matrix.cjs +316 -0
- package/lib/core/rs-query-to-text.cjs +438 -0
- package/lib/core/rs-sqlite-mirror.cjs +443 -0
- package/lib/core/rs-thesis-generator.cjs +188 -0
- package/lib/core/rs_cache.py +479 -0
- package/lib/core/rs_corpus.py +468 -0
- package/lib/core/rs_hybrid.py +586 -0
- package/lib/core/rs_math.py +287 -0
- package/lib/core/rs_rooms.py +193 -0
- package/lib/core/scheduled-scanner.cjs +463 -0
- package/lib/core/scratchpad-ops.cjs +201 -0
- package/lib/core/section-8-trace-schema.cjs +138 -0
- package/lib/core/section-registry.cjs +111 -0
- package/lib/core/session-state.cjs +144 -0
- package/lib/core/shallow-doc-parser.cjs +174 -0
- package/lib/core/shallow-doc-parser.test.cjs +226 -0
- package/lib/core/skill-activation-router.cjs +284 -0
- package/lib/core/state-ops.cjs +46 -0
- package/lib/core/statusline-cache.cjs +266 -0
- package/lib/core/token-estimator.cjs +348 -0
- package/lib/core/user-archetype.cjs +239 -0
- package/lib/core/user-md-ops.cjs +524 -0
- package/lib/core/visual-ops.cjs +624 -0
- package/lib/core/write-lock.cjs +149 -0
- package/lib/graph/canvas-graph.js +467 -0
- package/lib/graph/constellation-config.cjs +299 -0
- package/lib/graph/graph-detail-panel.js +165 -0
- package/lib/hmi/ROOM.md +47 -0
- package/lib/hmi/across-session-memory.cjs +604 -0
- package/lib/hmi/cross-room-memory.cjs +575 -0
- package/lib/hmi/decoy-tier.cjs +395 -0
- package/lib/hmi/jtbd-classifier.cjs +219 -0
- package/lib/hmi/jtbd-state.cjs +199 -0
- package/lib/hmi/jtbd-taxonomy.json +392 -0
- package/lib/hmi/selector-dispatcher.cjs +546 -0
- package/lib/hmi/selector-telemetry.cjs +263 -0
- package/lib/hmi/shape-f0-renderer.cjs +139 -0
- package/lib/hmi/shape-f1-fallback.cjs +80 -0
- package/lib/hmi/shape-f1-renderer.cjs +138 -0
- package/lib/hmi/shape-f2-renderer.cjs +132 -0
- package/lib/hmi/shape-f3-renderer.cjs +66 -0
- package/lib/hmi/shape-f4-renderer.cjs +72 -0
- package/lib/hmi/shape-f5-renderer.cjs +155 -0
- package/lib/hmi/shape-f6-plan-review-renderer.cjs +312 -0
- package/lib/hmi/shape-f6-renderer.cjs +144 -0
- package/lib/hmi/shape-g-renderer.cjs +219 -0
- package/lib/hmi/shape-h-renderer.cjs +222 -0
- package/lib/hmi/tier-check.cjs +63 -0
- package/lib/import/PRECONDITIONS.md +41 -0
- package/lib/import/branding.cjs +210 -0
- package/lib/import/branding.test.cjs +235 -0
- package/lib/import/classifications-sync.cjs +104 -0
- package/lib/import/classifications-sync.test.cjs +129 -0
- package/lib/import/enricher.cjs +296 -0
- package/lib/import/enricher.test.cjs +273 -0
- package/lib/import/integration.test.cjs +376 -0
- package/lib/import/manifest.cjs +129 -0
- package/lib/import/manifest.schema.json +185 -0
- package/lib/import/manifest.test.cjs +123 -0
- package/lib/import/meeting-detector.cjs +92 -0
- package/lib/import/meeting-detector.test.cjs +100 -0
- package/lib/import/person-detector.cjs +229 -0
- package/lib/import/person-detector.test.cjs +149 -0
- package/lib/import/report.cjs +186 -0
- package/lib/import/report.test.cjs +186 -0
- package/lib/import/room-md-scaffolder.cjs +49 -0
- package/lib/import/router.cjs +224 -0
- package/lib/import/router.test.cjs +356 -0
- package/lib/import/run-all-tests.cjs +36 -0
- package/lib/import/smoke-test.cjs +213 -0
- package/lib/import/smoke-test.test.cjs +148 -0
- package/lib/import/test-fixtures/collision-vault/preexisting-room/STATE.md +8 -0
- package/lib/import/test-fixtures/collision-vault/preexisting-room/problem-definition/onboarding/onboarding.md +7 -0
- package/lib/import/test-fixtures/collision-vault/source/onboarding.md +5 -0
- package/lib/import/test-fixtures/obsidian-vault/.obsidian/workspace.json +1 -0
- package/lib/import/test-fixtures/obsidian-vault/notes/with-wikilinks.md +4 -0
- package/lib/import/test-fixtures/tiny-vault/notes/2026-01-15-team-sync.md +9 -0
- package/lib/import/test-fixtures/tiny-vault/notes/empty.md +3 -0
- package/lib/import/test-fixtures/tiny-vault/notes/onboarding.md +5 -0
- package/lib/import/test-fixtures/tiny-vault/notes/pricing.md +5 -0
- package/lib/import/test-fixtures/tiny-vault/notes/random.md +4 -0
- package/lib/import/undo.test.cjs +199 -0
- package/lib/import/vault-scanner.cjs +105 -0
- package/lib/import/vault-scanner.test.cjs +67 -0
- package/lib/mcp/app-html/dashboard.html +316 -0
- package/lib/mcp/app-html/graph.html +428 -0
- package/lib/mcp/app-html/mindrian-platform.html +1841 -0
- package/lib/mcp/app-html/wiki.html +383 -0
- package/lib/mcp/app-views.cjs +322 -0
- package/lib/mcp/brain-router.cjs +418 -0
- package/lib/mcp/capability-registry.cjs +62 -0
- package/lib/mcp/larry-context.cjs +46 -0
- package/lib/mcp/larry-server-instructions.md +114 -0
- package/lib/mcp/pipeline-state.cjs +275 -0
- package/lib/mcp/prompts.cjs +302 -0
- package/lib/mcp/resources.cjs +227 -0
- package/lib/mcp/session-catchup.cjs +327 -0
- package/lib/mcp/surface-detect.cjs +75 -0
- package/lib/mcp/tool-router.cjs +1034 -0
- package/lib/memory/aaak-compress.cjs +403 -0
- package/lib/memory/aaak-compress.test.cjs +288 -0
- package/lib/memory/async-artifact-auto-commit.test.cjs +223 -0
- package/lib/memory/bearer-token.test.cjs +315 -0
- package/lib/memory/brain-cache-lru.test.cjs +259 -0
- package/lib/memory/brain-client-query-shape.test.cjs +160 -0
- package/lib/memory/brain-derivation-graceful-degradation.test.cjs +1019 -0
- package/lib/memory/brain-derivation-queue.test.cjs +539 -0
- package/lib/memory/brain-derivation.test.cjs +634 -0
- package/lib/memory/brain-derive-command.test.cjs +534 -0
- package/lib/memory/brain-md-invariants-validator.test.cjs +704 -0
- package/lib/memory/brain-md-schema.test.cjs +467 -0
- package/lib/memory/brain-md-staleness.test.cjs +525 -0
- package/lib/memory/brain-server-resolution.test.cjs +314 -0
- package/lib/memory/chain-recommender.test.cjs +233 -0
- package/lib/memory/chat-context.test.cjs +128 -0
- package/lib/memory/command-registry.test.cjs +220 -0
- package/lib/memory/cross-room-aggregator.test.cjs +909 -0
- package/lib/memory/dashboard-server.test.cjs +256 -0
- package/lib/memory/debouncer-drain-at-prompt.test.cjs +389 -0
- package/lib/memory/decision-capture.test.cjs +632 -0
- package/lib/memory/decision-capture.worker.cjs +70 -0
- package/lib/memory/explain-decision-command.test.cjs +521 -0
- package/lib/memory/explain-decision-footer.test.cjs +316 -0
- package/lib/memory/explored-materials-store.cjs +392 -0
- package/lib/memory/feynman-minto-guardian.test.cjs +736 -0
- package/lib/memory/feynman-minto-invariants.test.cjs +511 -0
- package/lib/memory/feynman-prompts-drift.test.cjs +144 -0
- package/lib/memory/feynman-prompts.cjs +151 -0
- package/lib/memory/feynman-prompts.test.cjs +96 -0
- package/lib/memory/folder-memory-quadruple.test.cjs +548 -0
- package/lib/memory/folder-memory.test.cjs +503 -0
- package/lib/memory/framework-chain-composer.test.cjs +515 -0
- package/lib/memory/frontmatter-schema-validator.test.cjs +290 -0
- package/lib/memory/heal-command.test.cjs +604 -0
- package/lib/memory/index-artifact-transaction.test.cjs +333 -0
- package/lib/memory/lazygraph-rs-discoveries-view.test.cjs +122 -0
- package/lib/memory/mcp-input-validation.test.cjs +240 -0
- package/lib/memory/mcp-server-brain-deps.test.cjs +270 -0
- package/lib/memory/mcp-stack-fallback.test.cjs +433 -0
- package/lib/memory/minto-debouncer.test.cjs +407 -0
- package/lib/memory/minto-debouncer.worker.cjs +46 -0
- package/lib/memory/minto-migration-v88.test.cjs +265 -0
- package/lib/memory/minto-schema-v88.test.cjs +390 -0
- package/lib/memory/mos-status-renderer.test.cjs +631 -0
- package/lib/memory/narrative-schema.cjs +376 -0
- package/lib/memory/narrative-schema.test.cjs +209 -0
- package/lib/memory/nav-dial.test.cjs +414 -0
- package/lib/memory/navigation-engine-core.test.cjs +722 -0
- package/lib/memory/navigation-invariants.test.cjs +483 -0
- package/lib/memory/offer-presenter.test.cjs +554 -0
- package/lib/memory/on-stop-snapshot.test.cjs +404 -0
- package/lib/memory/pending-tension-store.cjs +373 -0
- package/lib/memory/post-compact-reinjection.test.cjs +854 -0
- package/lib/memory/post-write-triple.test.cjs +317 -0
- package/lib/memory/pre-compact-snapshot.test.cjs +495 -0
- package/lib/memory/problem-type-router.test.cjs +656 -0
- package/lib/memory/query-efficiency-telemetry.test.cjs +370 -0
- package/lib/memory/recompile-room-references.test.cjs +392 -0
- package/lib/memory/recompile-room-references.worker.cjs +42 -0
- package/lib/memory/record-decision-dual-write.test.cjs +454 -0
- package/lib/memory/room-classifier-strict-mode.test.cjs +417 -0
- package/lib/memory/room-minto-hook.test.cjs +398 -0
- package/lib/memory/rs-discovery-engine.test.cjs +323 -0
- package/lib/memory/run-feynman-tests.cjs +1247 -0
- package/lib/memory/security-trifecta.test.cjs +312 -0
- package/lib/memory/session-start-brain-staleness.test.cjs +363 -0
- package/lib/memory/session-start-triple-injection.test.cjs +514 -0
- package/lib/memory/sessionstart-banner-formatter.cjs +318 -0
- package/lib/memory/sessionstart-minto-banner.test.cjs +373 -0
- package/lib/memory/skill-activation-router.test.cjs +419 -0
- package/lib/memory/stamp-artifact-write.test.cjs +304 -0
- package/lib/memory/statusline-active-room.test.cjs +315 -0
- package/lib/memory/statusline-minto-segment.test.cjs +292 -0
- package/lib/memory/sync-async-entry-points.test.cjs +204 -0
- package/lib/memory/test-bridge-writer-enhanced.cjs +452 -0
- package/lib/memory/test-rs-brain-substrate-shape.cjs +529 -0
- package/lib/memory/test-rs-brain-substrate.cjs +636 -0
- package/lib/memory/test-rs-breakthrough-scorer.cjs +375 -0
- package/lib/memory/test-rs-canon-violations.cjs +218 -0
- package/lib/memory/test-rs-chain-feeder-core.cjs +344 -0
- package/lib/memory/test-rs-chain-feeder-skill-spawn.cjs +297 -0
- package/lib/memory/test-rs-commercial-assessor.cjs +385 -0
- package/lib/memory/test-rs-differential-scorer.cjs +480 -0
- package/lib/memory/test-rs-discovery-engine.cjs +603 -0
- package/lib/memory/test-rs-domain-analyzer.cjs +492 -0
- package/lib/memory/test-rs-egress-primitives.cjs +420 -0
- package/lib/memory/test-rs-expert-mapper.cjs +547 -0
- package/lib/memory/test-rs-explain-command.cjs +443 -0
- package/lib/memory/test-rs-fetcher-academic.cjs +848 -0
- package/lib/memory/test-rs-fetcher-experts.cjs +496 -0
- package/lib/memory/test-rs-fetcher-industry.cjs +702 -0
- package/lib/memory/test-rs-fetcher-patents.cjs +674 -0
- package/lib/memory/test-rs-innovation-classifier.cjs +301 -0
- package/lib/memory/test-rs-mind-map.cjs +646 -0
- package/lib/memory/test-rs-neo4j-writer.cjs +518 -0
- package/lib/memory/test-rs-nl-to-query.cjs +449 -0
- package/lib/memory/test-rs-pinecone-bridge.cjs +277 -0
- package/lib/memory/test-rs-preprocessor.cjs +433 -0
- package/lib/memory/test-rs-query-matrix.cjs +391 -0
- package/lib/memory/test-rs-query-to-text.cjs +551 -0
- package/lib/memory/test-rs-sqlite-mirror.cjs +649 -0
- package/lib/memory/test-rs-thesis-generator.cjs +360 -0
- package/lib/memory/triple-context-formatter.cjs +473 -0
- package/lib/memory/triple-context-formatter.test.cjs +442 -0
- package/lib/memory/user-md-persona.test.cjs +565 -0
- package/lib/memory/userpromptsubmit-integration.test.cjs +690 -0
- package/lib/memory/validators/README.md +157 -0
- package/lib/memory/validators/brain-md-invariants.cjs +475 -0
- package/lib/memory/validators/brain-substrate-invariants.cjs +285 -0
- package/lib/memory/validators/external-academic-invariants.cjs +249 -0
- package/lib/memory/validators/external-industry-invariants.cjs +271 -0
- package/lib/memory/validators/external-patents-invariants.cjs +266 -0
- package/lib/memory/validators/minto-invariants.cjs +62 -0
- package/lib/memory/validators/navigation-invariants.cjs +340 -0
- package/lib/memory/validators/queue-health.cjs +95 -0
- package/lib/memory/validators/snapshot-integrity.cjs +129 -0
- package/lib/memory/validators/stale-lifecycle.cjs +116 -0
- package/lib/memory/vault-section-minto-generator-atomic.test.cjs +556 -0
- package/lib/memory/vault-section-minto-generator-atomic.worker.cjs +73 -0
- package/lib/memory/write-lock-atomic.test.cjs +137 -0
- package/lib/memory/write-lock-atomic.worker.cjs +55 -0
- package/lib/parity/check-parity.cjs +83 -0
- package/lib/presentation/presentation-server.cjs +101 -0
- package/lib/presentation/presentation-watcher.cjs +123 -0
- package/lib/quickview/hub-server.cjs +719 -0
- package/lib/quickview/server.cjs +533 -0
- package/lib/render/JTBD-PALETTES.md +145 -0
- package/lib/render/ROOM.md +59 -0
- package/lib/render/render-v2.cjs +486 -0
- package/lib/render/render-v2.test.cjs +267 -0
- package/lib/render/render.cjs +65 -0
- package/lib/state/ROOM.md +46 -0
- package/lib/state/state-md-parser.cjs +215 -0
- package/lib/statusline/ROOM.md +38 -0
- package/lib/statusline/banner-suppression.cjs +50 -0
- package/lib/statusline/surface-detect.cjs +85 -0
- package/lib/update-bootstrap.sh.template +145 -0
- package/lib/vault/frontmatter-schema.cjs +297 -0
- package/lib/vault/room-scanner.cjs +352 -0
- package/lib/vault/wikilink-builder.cjs +231 -0
- package/lib/vault/wikilink-builder.test.cjs +182 -0
- package/lib/wiki/graph-links.cjs +281 -0
- package/lib/wiki/page-renderer.cjs +229 -0
- package/lib/wiki/wiki-chat.cjs +81 -0
- package/lib/wiki/wiki-layout.cjs +1459 -0
- package/lib/wiki/wiki-search.cjs +142 -0
- package/lib/wiki/wiki-server.cjs +678 -0
- package/lib/wiki/wiki-watcher.cjs +105 -0
- package/lib/workflow/ROOM.md +47 -0
- package/lib/workflow/command-resolver.cjs +155 -0
- package/lib/workflow/command-resolver.test.cjs +235 -0
- package/package.json +44 -0
- package/pipelines/analogy/01-decompose.md +80 -0
- package/pipelines/analogy/02-abstract.md +87 -0
- package/pipelines/analogy/03-search.md +135 -0
- package/pipelines/analogy/04-transfer.md +101 -0
- package/pipelines/analogy/05-validate.md +106 -0
- package/pipelines/analogy/CHAIN.md +56 -0
- package/pipelines/discovery/01-explore-domains.md +44 -0
- package/pipelines/discovery/02-think-hats.md +50 -0
- package/pipelines/discovery/03-analyze-needs.md +54 -0
- package/pipelines/discovery/CHAIN.md +37 -0
- package/pipelines/thesis/01-structure-argument.md +45 -0
- package/pipelines/thesis/02-challenge-assumptions.md +48 -0
- package/pipelines/thesis/03-build-thesis.md +54 -0
- package/pipelines/thesis/CHAIN.md +37 -0
- package/references/brain/causal-directives.md +91 -0
- package/references/brain/causal-enrichment.cypher +165 -0
- package/references/brain/command-triggers-schema.md +226 -0
- package/references/brain/graph-architecture.md +317 -0
- package/references/brain/query-patterns.md +460 -0
- package/references/brain/room-hierarchy-schema.md +218 -0
- package/references/brain/schema.md +76 -0
- package/references/capability-radar/capabilities-index.md +241 -0
- package/references/capability-radar/changelog-cache.md +81 -0
- package/references/causal/causal-schema.md +103 -0
- package/references/design/email-template-standard.md +155 -0
- package/references/design/graph-visualization-standard.md +178 -0
- package/references/document-generation.md +179 -0
- package/references/hsi/HSI-TOOLS-REFERENCE.md +222 -0
- package/references/import-config.md +141 -0
- package/references/integrations/detection-patterns.md +101 -0
- package/references/meeting/artifact-template.md +377 -0
- package/references/meeting/cross-meeting-intelligence.md +216 -0
- package/references/meeting/cross-relationship-patterns.md +202 -0
- package/references/meeting/live-join-interface.md +244 -0
- package/references/meeting/section-mapping.md +192 -0
- package/references/meeting/segment-classification.md +258 -0
- package/references/meeting/speaker-profile-template.md +219 -0
- package/references/meeting/summary-template.md +348 -0
- package/references/meeting/transcript-patterns.md +226 -0
- package/references/methodology/analyze-needs.md +135 -0
- package/references/methodology/analyze-systems.md +121 -0
- package/references/methodology/analyze-timing.md +149 -0
- package/references/methodology/beautiful-question.md +109 -0
- package/references/methodology/build-knowledge.md +161 -0
- package/references/methodology/build-thesis.md +237 -0
- package/references/methodology/challenge-assumptions.md +127 -0
- package/references/methodology/diagnose.md +169 -0
- package/references/methodology/dominant-designs.md +212 -0
- package/references/methodology/explore-domains.md +147 -0
- package/references/methodology/explore-futures.md +163 -0
- package/references/methodology/explore-trends.md +129 -0
- package/references/methodology/find-bottlenecks.md +131 -0
- package/references/methodology/grade.md +211 -0
- package/references/methodology/index.md +97 -0
- package/references/methodology/leadership.md +200 -0
- package/references/methodology/lean-canvas.md +116 -0
- package/references/methodology/macro-trends.md +192 -0
- package/references/methodology/map-unknowns.md +137 -0
- package/references/methodology/mullins-7-domains.md +104 -0
- package/references/methodology/problem-types.md +65 -0
- package/references/methodology/root-cause.md +178 -0
- package/references/methodology/sapphire-encoding.md +355 -0
- package/references/methodology/scenario-plan.md +178 -0
- package/references/methodology/score-innovation.md +154 -0
- package/references/methodology/structure-argument.md +158 -0
- package/references/methodology/systems-thinking.md +159 -0
- package/references/methodology/think-hats.md +147 -0
- package/references/methodology/triz-matrix.json +751 -0
- package/references/methodology/triz-principles.md +501 -0
- package/references/methodology/user-needs.md +199 -0
- package/references/methodology/validate.md +163 -0
- package/references/methodology/value-proposition.md +244 -0
- package/references/opportunities/funding-lifecycle.md +103 -0
- package/references/opportunities/grant-api-patterns.md +99 -0
- package/references/opportunities/opportunity-template.md +84 -0
- package/references/personality/assessment-philosophy.md +72 -0
- package/references/personality/lexicon.md +100 -0
- package/references/personality/persona-chains.md +56 -0
- package/references/personality/pws-lexicon-full.md +499 -0
- package/references/personality/voice-dna.md +156 -0
- package/references/personas/hat-perspectives.md +76 -0
- package/references/personas/persona-template.md +63 -0
- package/references/pipeline/act-output-contract.md +88 -0
- package/references/pipeline/chains-index.md +39 -0
- package/references/pws-profile-generation.md +79 -0
- package/references/reasoning/reasoning-schema.md +143 -0
- package/references/reasoning/reasoning-template.md +68 -0
- package/references/reasoning/run-template.md +38 -0
- package/references/research/RESEARCH_14_CLAUDE_CODE_SOURCE_ARCHITECTURE.md +209 -0
- package/references/research/RESEARCH_15_V1.8_OPTIMIZATION_JTBD.md +375 -0
- package/references/research/RESEARCH_16_NATIVE_FIRST_PLUGIN_ARCHITECTURE.md +575 -0
- package/references/research/RESEARCH_17_MCP_UI_FRAMEWORKS.md +272 -0
- package/references/taxonomy/TAXONOMY.md +192 -0
- package/references/templates/MINTO.md +36 -0
- package/references/user-research/2026-04-05-leah-lawrence-session.md +202 -0
- package/references/vault-kit/README.md +35 -0
- package/references/vault-kit/app.json +12 -0
- package/references/vault-kit/appearance.json +12 -0
- package/references/vault-kit/graph.json +35 -0
- package/references/vault-kit/snippets/mindrian-destijl.css +297 -0
- package/references/vault-kit/templates/new-artifact.md +37 -0
- package/references/vault-kit/templates/new-meeting-note.md +35 -0
- package/references/vault-kit/templates/new-team-profile.md +29 -0
- package/references/vault-kit/templates/new-xref.md +35 -0
- package/references/visual/symbol-system.md +151 -0
- package/skills/MOSDeckEngine/SKILL.md +325 -0
- package/skills/brain-connector/SKILL.md +114 -0
- package/skills/context-engine/SKILL.md +147 -0
- package/skills/conversation-mode/SKILL.md +102 -0
- package/skills/larry-personality/SKILL.md +219 -0
- package/skills/larry-personality/framework-chains.md +92 -0
- package/skills/larry-personality/mode-engine.md +185 -0
- package/skills/mullins-scaffold/SKILL.md +61 -0
- package/skills/mullins-scaffold/scaffold.json +146 -0
- package/skills/pws-methodology/SKILL.md +49 -0
- package/skills/room-passive/SKILL.md +165 -0
- package/skills/room-proactive/SKILL.md +250 -0
- package/skills/ui-system/SKILL.md +277 -0
|
@@ -0,0 +1,664 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
'use strict';
|
|
3
|
+
|
|
4
|
+
/*
|
|
5
|
+
* Copyright (c) 2026 Mindrian. BSL 1.1.
|
|
6
|
+
*
|
|
7
|
+
* Phase 91-00 -- Navigation Engine core (L5 Decision layer)
|
|
8
|
+
* ==========================================================
|
|
9
|
+
* Single export `decide(turn, context) -> decision` composes the five
|
|
10
|
+
* input signals (ICM scope + SQL relations + Feynman-MINTO reasoning +
|
|
11
|
+
* BRAIN.md derivations + intent/persona) through a structured decision
|
|
12
|
+
* function (NOT a weighted score per locked decision D-02) and returns
|
|
13
|
+
* a typed decision struct with full trace.
|
|
14
|
+
*
|
|
15
|
+
* Contract source of truth:
|
|
16
|
+
* .planning/research/navigation-engine-brain-interface.md (v1, frozen
|
|
17
|
+
* at Phase 90-09).
|
|
18
|
+
*
|
|
19
|
+
* Canon Part 8 posture (READ THIS FIRST):
|
|
20
|
+
* This module is a PURE LOCAL READER. It NEVER queries Brain. It
|
|
21
|
+
* NEVER reads BRAIN.md directly via fs.readFileSync; the only allowed
|
|
22
|
+
* read path is folder-memory.cjs readQuadruple. It NEVER constructs
|
|
23
|
+
* Cypher queries. It NEVER embeds BRAIN.md body strings in network
|
|
24
|
+
* payloads. The two PERMITTED brain-client touches are
|
|
25
|
+
* brain-client.isAvailable() (boolean, no network when cached) and
|
|
26
|
+
* brain-client.schema() (scalar, no user content leaves the local
|
|
27
|
+
* process). Both are optional; the engine works fully offline.
|
|
28
|
+
*
|
|
29
|
+
* Per Section 2.4: a single readQuadruple result MAY be cached for the
|
|
30
|
+
* duration of one decide() call. Cross-call caching is FORBIDDEN.
|
|
31
|
+
*
|
|
32
|
+
* Per locked decision D-02: this engine is NOT a weighted score across
|
|
33
|
+
* the five signals. Brain's section weights (0.35 + 0.20 + 0.15 + ...)
|
|
34
|
+
* stay INSIDE the BRAIN.md contribution block; the staleness multiplier
|
|
35
|
+
* scales that block as a whole. Inter-signal composition is rule-based
|
|
36
|
+
* with explicit precedence, not arithmetic.
|
|
37
|
+
*
|
|
38
|
+
* License: BSL 1.1.
|
|
39
|
+
*/
|
|
40
|
+
|
|
41
|
+
const shared = require('./navigation-engine-shared.cjs');
|
|
42
|
+
const folderMemory = require('./folder-memory.cjs'); // for readQuadruple
|
|
43
|
+
// brain-client is OPTIONAL: required only if downstream callers want to
|
|
44
|
+
// pass an explicit brainAvailable hint, but isAvailable() is also fine
|
|
45
|
+
// from the call site. We do NOT require() it here to keep the dependency
|
|
46
|
+
// graph minimal and to make grep-guards trivially passable.
|
|
47
|
+
|
|
48
|
+
const STALENESS_MULTIPLIERS = shared.STALENESS_MULTIPLIERS;
|
|
49
|
+
const SECTION_WEIGHTS = shared.SECTION_WEIGHTS;
|
|
50
|
+
const REQUIRED_SECTION_KEYS = shared.REQUIRED_SECTION_KEYS;
|
|
51
|
+
const OPTIONAL_SECTION_KEYS = shared.OPTIONAL_SECTION_KEYS;
|
|
52
|
+
const CANONICAL_VERBS = shared.CANONICAL_VERBS;
|
|
53
|
+
|
|
54
|
+
// Wicked-score escalation threshold per Canon Appendix E rule R4.
|
|
55
|
+
const WICKED_ESCALATION_THRESHOLD = 8;
|
|
56
|
+
|
|
57
|
+
// Confidence floor for Section 6 RECOMMENDED gate.
|
|
58
|
+
const RECOMMENDED_CONFIDENCE_FLOOR = 0.7;
|
|
59
|
+
|
|
60
|
+
// ---------- Pattern parsers (tolerant; never throw) ----------
|
|
61
|
+
|
|
62
|
+
/**
|
|
63
|
+
* Parse pattern_matches body for `confidence: <float>` candidates.
|
|
64
|
+
* Returns the highest confidence found in [0, 1], or null when no
|
|
65
|
+
* candidate is parseable. Tolerant of body whitespace, multi-line, and
|
|
66
|
+
* forward-compat extra fields.
|
|
67
|
+
*/
|
|
68
|
+
function extractHighestConfidence(body) {
|
|
69
|
+
if (typeof body !== 'string' || body.length === 0) return null;
|
|
70
|
+
const re = /confidence\s*:\s*([01](?:\.\d+)?|0?\.\d+)/gi;
|
|
71
|
+
let best = null;
|
|
72
|
+
let m;
|
|
73
|
+
while ((m = re.exec(body)) !== null) {
|
|
74
|
+
const v = parseFloat(m[1]);
|
|
75
|
+
if (!isNaN(v) && v >= 0.0 && v <= 1.0) {
|
|
76
|
+
if (best === null || v > best) best = v;
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
return best;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Extract the verb associated with the highest-confidence candidate in
|
|
84
|
+
* pattern_matches. Returns null when no candidate is parseable. Verbs
|
|
85
|
+
* MUST appear before the `(confidence: ...)` parenthetical, matching
|
|
86
|
+
* Canon Part 3 vocabulary.
|
|
87
|
+
*
|
|
88
|
+
* Body shape (per Plan 90-01 deriveSection emit):
|
|
89
|
+
* - Run Methodology (confidence: 0.85, source: SWOT)
|
|
90
|
+
* - Reformulate (confidence: 0.72, source: 5 Whys)
|
|
91
|
+
*/
|
|
92
|
+
function extractTopCandidateVerb(body) {
|
|
93
|
+
if (typeof body !== 'string' || body.length === 0) return null;
|
|
94
|
+
const lines = body.split(/\r?\n/);
|
|
95
|
+
let bestConf = -1;
|
|
96
|
+
let bestVerb = null;
|
|
97
|
+
for (const line of lines) {
|
|
98
|
+
// Match `<prefix> <verb> (confidence: 0.NN ...)`
|
|
99
|
+
const m = /^[\s\-*]*(.+?)\s*\(\s*confidence\s*:\s*([01](?:\.\d+)?|0?\.\d+)/i.exec(line);
|
|
100
|
+
if (!m) continue;
|
|
101
|
+
const verb = m[1].trim();
|
|
102
|
+
const conf = parseFloat(m[2]);
|
|
103
|
+
if (isNaN(conf)) continue;
|
|
104
|
+
// Resolve verb against canonical vocabulary (case-insensitive
|
|
105
|
+
// contains-match; closed set).
|
|
106
|
+
let resolved = null;
|
|
107
|
+
for (const v of CANONICAL_VERBS) {
|
|
108
|
+
if (verb.toLowerCase() === v.toLowerCase()) {
|
|
109
|
+
resolved = v;
|
|
110
|
+
break;
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
if (resolved !== null && conf > bestConf) {
|
|
114
|
+
bestConf = conf;
|
|
115
|
+
bestVerb = resolved;
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
return bestVerb;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* Parse wicked_indicators body for `wicked_score: <int>`. Returns 0 when
|
|
123
|
+
* no score is parseable.
|
|
124
|
+
*/
|
|
125
|
+
function extractWickedScore(body) {
|
|
126
|
+
if (typeof body !== 'string' || body.length === 0) return 0;
|
|
127
|
+
const m = /wicked_score\s*:\s*(\d+)/i.exec(body);
|
|
128
|
+
if (!m) return 0;
|
|
129
|
+
const v = parseInt(m[1], 10);
|
|
130
|
+
if (isNaN(v) || v < 0) return 0;
|
|
131
|
+
return v;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
// ---------- Trace builders ----------
|
|
135
|
+
|
|
136
|
+
function buildIcmScope(quadruple, turn) {
|
|
137
|
+
const room = quadruple && quadruple.room;
|
|
138
|
+
return {
|
|
139
|
+
section_path: turn && turn.sectionPath ? turn.sectionPath : null,
|
|
140
|
+
room_exists: room && room.exists ? true : false,
|
|
141
|
+
artifact_count: quadruple && quadruple.state ? (quadruple.state.artifact_count || 0) : 0,
|
|
142
|
+
};
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
function buildSqlSignals(_quadruple, _turn) {
|
|
146
|
+
// Phase 91-00 ships the engine surface; the SQL signal is consumed
|
|
147
|
+
// from a future plan that wires room.db edge counts. For this plan
|
|
148
|
+
// the slot is structural (so /mos:explain-decision can render it) but
|
|
149
|
+
// contributes no rules. Returning an object (not null) keeps the
|
|
150
|
+
// five-signal capture invariant met (Test 20).
|
|
151
|
+
return { edges_consulted: 0, contradictions_detected: 0 };
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
function buildMintoReasoning(quadruple) {
|
|
155
|
+
const r = quadruple && quadruple.reasoning;
|
|
156
|
+
if (!r) {
|
|
157
|
+
return { exists: false, reasoning_health_score: 0, governing_thought: null, is_stale: true };
|
|
158
|
+
}
|
|
159
|
+
return {
|
|
160
|
+
exists: r.exists === true,
|
|
161
|
+
reasoning_health_score:
|
|
162
|
+
typeof r.reasoning_health_score === 'number' ? r.reasoning_health_score : 0,
|
|
163
|
+
governing_thought: typeof r.governing_thought === 'string' ? r.governing_thought : null,
|
|
164
|
+
is_stale: r.is_stale === true,
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
function buildIntentPersona(context) {
|
|
169
|
+
const persona = context && context.userPersona ? context.userPersona : null;
|
|
170
|
+
const intent = context && context.intentSignal ? context.intentSignal : null;
|
|
171
|
+
return {
|
|
172
|
+
archetype: persona && persona.archetype ? persona.archetype : null,
|
|
173
|
+
problem_type: persona && persona.problem_type ? persona.problem_type : null,
|
|
174
|
+
venture_stage: persona && persona.venture_stage ? persona.venture_stage : null,
|
|
175
|
+
intent: intent && intent.intent ? intent.intent : null,
|
|
176
|
+
intent_confidence: intent && typeof intent.confidence === 'number' ? intent.confidence : null,
|
|
177
|
+
};
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
// ---------- Section consumption ----------
|
|
181
|
+
|
|
182
|
+
function consumedSections(brain) {
|
|
183
|
+
if (!brain || typeof brain !== 'object' || !brain.sections) return [];
|
|
184
|
+
const out = [];
|
|
185
|
+
// Required first (canonical order), then optional.
|
|
186
|
+
for (const k of REQUIRED_SECTION_KEYS) {
|
|
187
|
+
const v = brain.sections[k];
|
|
188
|
+
if (v !== null && v !== undefined && typeof v === 'object' && typeof v.body === 'string' && v.body.length > 0 && !/^\s*\(no signal\)\s*$/i.test(v.body)) {
|
|
189
|
+
out.push(k);
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
for (const k of OPTIONAL_SECTION_KEYS) {
|
|
193
|
+
const v = brain.sections[k];
|
|
194
|
+
if (v !== null && v !== undefined && typeof v === 'object' && typeof v.body === 'string' && v.body.length > 0 && !/^\s*\(no signal\)\s*$/i.test(v.body)) {
|
|
195
|
+
out.push(k);
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
return out;
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
// ---------- Five-signal composition (rule-based per D-02) ----------
|
|
202
|
+
|
|
203
|
+
/**
|
|
204
|
+
* Resolve the best fire_skill given Brain signal state and the wicked
|
|
205
|
+
* escalation rule. Returns null when no signal warrants firing.
|
|
206
|
+
*/
|
|
207
|
+
function resolveFireSkill(brain, weightApplied, tierMode) {
|
|
208
|
+
// Wicked escalation takes precedence over normal routing (Canon
|
|
209
|
+
// Appendix E rule R4).
|
|
210
|
+
const wickedSection = brain && brain.sections && brain.sections.wicked_indicators;
|
|
211
|
+
if (wickedSection && wickedSection.body) {
|
|
212
|
+
const score = extractWickedScore(wickedSection.body);
|
|
213
|
+
if (score >= WICKED_ESCALATION_THRESHOLD) {
|
|
214
|
+
return 'soft-systems';
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
// Mode A + non-zero weight + parseable verb -> fire that verb's skill.
|
|
218
|
+
if (tierMode === 'mode_a' && weightApplied > 0) {
|
|
219
|
+
const pm = brain && brain.sections && brain.sections.pattern_matches;
|
|
220
|
+
if (pm && pm.body) {
|
|
221
|
+
const verb = extractTopCandidateVerb(pm.body);
|
|
222
|
+
if (verb !== null) {
|
|
223
|
+
// Map canonical verb to skill family. v1 ships a minimal mapping
|
|
224
|
+
// sufficient to satisfy "fire_skill non-null when fresh" (Test 20).
|
|
225
|
+
return verbToSkillFamily(verb);
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
// Fresh Brain but no pattern_matches verb: fall back to context
|
|
229
|
+
// skill so Test 20 still has a non-null fire_skill.
|
|
230
|
+
if (weightApplied >= 0.9) {
|
|
231
|
+
return 'context-engine';
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
return null;
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
function verbToSkillFamily(verb) {
|
|
238
|
+
// Closed mapping. Future plans can extend the table; the engine
|
|
239
|
+
// never invents skill names from prose.
|
|
240
|
+
switch (verb) {
|
|
241
|
+
case 'Run Methodology': return 'methodology-router';
|
|
242
|
+
case 'Reformulate': return 'beautiful-question';
|
|
243
|
+
case 'Spawn Sub-Agent': return 'subagent-dispatcher';
|
|
244
|
+
case 'Navigate Graph': return 'graph-navigator';
|
|
245
|
+
case "Devil's Advocate": return 'red-team';
|
|
246
|
+
case 'Scenario Plan': return 'scenario-planner';
|
|
247
|
+
case 'Synthesize': return 'blue-hat';
|
|
248
|
+
case 'Bank Opportunity': return 'opportunity-bank';
|
|
249
|
+
case 'Defer': return 'defer';
|
|
250
|
+
case 'Free-Text': return 'larry-default';
|
|
251
|
+
default: return null;
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
/**
|
|
256
|
+
* Compose suppress_skills based on intent and persona signals. v1
|
|
257
|
+
* ships an empty array; future plans will populate from intent
|
|
258
|
+
* heuristics. Returning [] keeps Test 31 honoring the array invariant.
|
|
259
|
+
*/
|
|
260
|
+
function resolveSuppressSkills(_context, _brain) {
|
|
261
|
+
return [];
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
/**
|
|
265
|
+
* persona_updates: returns null when no threshold crossed; otherwise an
|
|
266
|
+
* object with allowed keys archetype/problem_type/venture_stage.
|
|
267
|
+
*
|
|
268
|
+
* v1 ships null (no auto-update). Plan 91-01 owns the durability layer.
|
|
269
|
+
*/
|
|
270
|
+
function resolvePersonaUpdates(_context, _brain) {
|
|
271
|
+
return null;
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
/**
|
|
275
|
+
* offer_next_step: returns null OR { command, reason }. Max 1 per turn
|
|
276
|
+
* is enforced trivially (we never return more than one).
|
|
277
|
+
*
|
|
278
|
+
* v1 ships null until Plan 91-04 wires the offer presentation rules.
|
|
279
|
+
*/
|
|
280
|
+
function resolveOfferNextStep(_context, _brain) {
|
|
281
|
+
return null;
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
// ---------- decide() main entry ----------
|
|
285
|
+
|
|
286
|
+
/**
|
|
287
|
+
* decide(turn, context) -> decision
|
|
288
|
+
*
|
|
289
|
+
* Inputs:
|
|
290
|
+
* turn = { userText, sectionPath, sessionId }
|
|
291
|
+
* context = {
|
|
292
|
+
* quadruple, // pre-read quadruple (caller responsibility)
|
|
293
|
+
* brainAvailable, // boolean from brain-client.isAvailable()
|
|
294
|
+
* userPersona, // { archetype, problem_type, venture_stage }
|
|
295
|
+
* intentSignal // { intent, confidence }
|
|
296
|
+
* }
|
|
297
|
+
*
|
|
298
|
+
* Returns: typed decision struct with complete trace.
|
|
299
|
+
*
|
|
300
|
+
* Never throws. Every internal failure falls through to a safe
|
|
301
|
+
* emptyDecision() with chosen_rationale noting the fault.
|
|
302
|
+
*
|
|
303
|
+
* Per-turn cache: a single quadruple lookup may be cached within this
|
|
304
|
+
* call only. This implementation expects context.quadruple to be
|
|
305
|
+
* pre-read (the standard Phase 91-02 hook integration pattern), so the
|
|
306
|
+
* cache is implicit. If context.quadruple is absent, decide() falls
|
|
307
|
+
* back to readQuadruple(turn.sectionPath) once and caches the result
|
|
308
|
+
* for the duration of THIS call only. Cross-call caching FORBIDDEN per
|
|
309
|
+
* Section 2.4.
|
|
310
|
+
*/
|
|
311
|
+
function decide(turn, context) {
|
|
312
|
+
// Safe defaults.
|
|
313
|
+
const ctx = context || {};
|
|
314
|
+
const t = turn || {};
|
|
315
|
+
const decision = shared.emptyDecision();
|
|
316
|
+
const trace = decision.decision_trace;
|
|
317
|
+
|
|
318
|
+
try {
|
|
319
|
+
// Per-turn cache scope: read quadruple if not provided. The local
|
|
320
|
+
// `quadruple` binding is the cache; it never escapes this function.
|
|
321
|
+
let quadruple = ctx.quadruple;
|
|
322
|
+
if (!quadruple) {
|
|
323
|
+
const sectionPath = t.sectionPath;
|
|
324
|
+
if (typeof sectionPath === 'string' && sectionPath.length > 0) {
|
|
325
|
+
quadruple = folderMemory.readQuadruple(sectionPath);
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
// If we still have no quadruple, return empty decision in tier_0.
|
|
330
|
+
if (!quadruple) {
|
|
331
|
+
trace.brain_md_tier_mode = 'tier_0';
|
|
332
|
+
trace.brain_md_staleness = 'absent';
|
|
333
|
+
trace.brain_md_weight_applied = 0.0;
|
|
334
|
+
trace.icm_scope = buildIcmScope(null, t);
|
|
335
|
+
trace.sql_signals = buildSqlSignals(null, t);
|
|
336
|
+
trace.minto_reasoning = buildMintoReasoning(null);
|
|
337
|
+
trace.intent_persona = buildIntentPersona(ctx);
|
|
338
|
+
trace.chosen_rationale = 'Tier 0 fallback: no quadruple available for sectionPath.';
|
|
339
|
+
return decision;
|
|
340
|
+
}
|
|
341
|
+
|
|
342
|
+
const brainAvailable = ctx.brainAvailable === true;
|
|
343
|
+
const brain = quadruple.brain || null;
|
|
344
|
+
|
|
345
|
+
// Section 5 tier-mode resolver (re-evaluated every turn; no cache).
|
|
346
|
+
const tierMode = shared.resolveTierMode(quadruple, brainAvailable);
|
|
347
|
+
trace.brain_md_tier_mode = tierMode;
|
|
348
|
+
|
|
349
|
+
// Section 4 staleness multiplier (BRAIN.md contribution scaler).
|
|
350
|
+
// The result is later assigned to trace.brain_md_weight_applied per
|
|
351
|
+
// Section 8.1. Multiplier source rows:
|
|
352
|
+
// null brain -> 0.0
|
|
353
|
+
// staleness:fresh -> 1.0
|
|
354
|
+
// stale_reason:brain_offline -> 0.9 (EXEMPT)
|
|
355
|
+
// stale_reason:age_exceeded -> 0.7
|
|
356
|
+
// stale_reason:brain_graph_version_mismatch -> 0.5
|
|
357
|
+
// stale_reason:governing_thought_changed -> 0.3
|
|
358
|
+
// stale_reason:derivation_timeout -> 0.2
|
|
359
|
+
// stale_reason:parse_failed -> 0.0
|
|
360
|
+
// staleness:unavailable -> 0.0
|
|
361
|
+
let weightApplied = shared.applyStalenessMultiplier(brain);
|
|
362
|
+
|
|
363
|
+
// Section 3.1 attribution guard. brain.author MUST equal 'brain';
|
|
364
|
+
// any other value indicates a Canon Part 2 attribution breach and
|
|
365
|
+
// demotes the BRAIN.md weight to 0.0 with a trace note. The
|
|
366
|
+
// resulting brain_md_weight_applied is observable in the decision
|
|
367
|
+
// trace by /mos:explain-decision (Plan 91-05).
|
|
368
|
+
let attributionBreach = false;
|
|
369
|
+
if (brain !== null && brain.author !== undefined && brain.author !== 'brain') {
|
|
370
|
+
attributionBreach = true;
|
|
371
|
+
weightApplied = 0.0; // brain_md_weight_applied demotion target
|
|
372
|
+
}
|
|
373
|
+
// Section 8.1 trace.brain_md_weight_applied final assignment.
|
|
374
|
+
trace.brain_md_weight_applied = weightApplied;
|
|
375
|
+
|
|
376
|
+
// Section 8 trace fields (always emitted, even with brain null).
|
|
377
|
+
if (brain === null) {
|
|
378
|
+
trace.brain_md_version = null;
|
|
379
|
+
trace.brain_md_staleness = 'absent';
|
|
380
|
+
trace.brain_md_stale_reason = null;
|
|
381
|
+
} else {
|
|
382
|
+
trace.brain_md_version = typeof brain.brain_graph_version === 'number'
|
|
383
|
+
? brain.brain_graph_version
|
|
384
|
+
: null;
|
|
385
|
+
trace.brain_md_staleness = typeof brain.staleness === 'string'
|
|
386
|
+
? brain.staleness
|
|
387
|
+
: 'absent';
|
|
388
|
+
trace.brain_md_stale_reason = typeof brain.stale_reason === 'string'
|
|
389
|
+
? brain.stale_reason
|
|
390
|
+
: null;
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
// Section 3.2 sections_consumed list (non-null bodies only).
|
|
394
|
+
trace.brain_md_sections_consumed = consumedSections(brain);
|
|
395
|
+
|
|
396
|
+
// Section 6 RECOMMENDED gate evaluation (record confidence even
|
|
397
|
+
// when gate fails; render marker only when ALL conditions hold).
|
|
398
|
+
let recommendedConfidence = null;
|
|
399
|
+
let recommendedRendered = false;
|
|
400
|
+
if (brain !== null && brain.sections) {
|
|
401
|
+
const pm = brain.sections.pattern_matches;
|
|
402
|
+
if (pm && typeof pm.body === 'string') {
|
|
403
|
+
recommendedConfidence = extractHighestConfidence(pm.body);
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
trace.brain_md_recommended_confidence = recommendedConfidence;
|
|
407
|
+
|
|
408
|
+
if (
|
|
409
|
+
tierMode === 'mode_a' &&
|
|
410
|
+
!attributionBreach &&
|
|
411
|
+
recommendedConfidence !== null &&
|
|
412
|
+
recommendedConfidence >= RECOMMENDED_CONFIDENCE_FLOOR
|
|
413
|
+
) {
|
|
414
|
+
// Section 6 condition 4: candidate verb must match a Canon Part 3
|
|
415
|
+
// vocabulary entry. extractTopCandidateVerb returns the resolved
|
|
416
|
+
// canonical verb or null.
|
|
417
|
+
const pm = brain.sections.pattern_matches;
|
|
418
|
+
const verb = extractTopCandidateVerb(pm.body);
|
|
419
|
+
if (verb !== null) {
|
|
420
|
+
recommendedRendered = true;
|
|
421
|
+
}
|
|
422
|
+
}
|
|
423
|
+
trace.brain_md_recommended_marker_rendered = recommendedRendered;
|
|
424
|
+
|
|
425
|
+
// Five-signal triangulation (structural; rule-based per D-02).
|
|
426
|
+
trace.icm_scope = buildIcmScope(quadruple, t);
|
|
427
|
+
trace.sql_signals = buildSqlSignals(quadruple, t);
|
|
428
|
+
trace.minto_reasoning = buildMintoReasoning(quadruple);
|
|
429
|
+
trace.intent_persona = buildIntentPersona(ctx);
|
|
430
|
+
|
|
431
|
+
// Compose chosen_rationale from active rules.
|
|
432
|
+
const rationale = [];
|
|
433
|
+
if (tierMode === 'tier_0') {
|
|
434
|
+
if (brain === null) {
|
|
435
|
+
rationale.push('Tier 0 fallback: BRAIN.md absent.');
|
|
436
|
+
} else if (brain.parse_failed === true) {
|
|
437
|
+
rationale.push('brain_parse_failed fell through to tier_0.');
|
|
438
|
+
} else if (brain.staleness === 'unavailable') {
|
|
439
|
+
rationale.push('Tier 0 fallback: BRAIN.md staleness unavailable.');
|
|
440
|
+
} else {
|
|
441
|
+
rationale.push('Tier 0 fallback: brain unreachable and not offline-exempt.');
|
|
442
|
+
}
|
|
443
|
+
} else if (tierMode === 'mode_b') {
|
|
444
|
+
rationale.push('Mode B local-only: brain unreachable; brain_offline exemption applied (weight 0.9).');
|
|
445
|
+
rationale.push('mode_b suppresses RECOMMENDED per Canon Part 3.');
|
|
446
|
+
} else {
|
|
447
|
+
// mode_a
|
|
448
|
+
if (recommendedRendered) {
|
|
449
|
+
rationale.push('Mode A: RECOMMENDED rendered (confidence >= 0.7).');
|
|
450
|
+
} else if (recommendedConfidence !== null) {
|
|
451
|
+
rationale.push('Mode A: pattern_matches confidence ' + recommendedConfidence + ' below 0.7 floor; RECOMMENDED suppressed.');
|
|
452
|
+
} else {
|
|
453
|
+
rationale.push('Mode A: no pattern_matches candidate available.');
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
if (attributionBreach) {
|
|
457
|
+
rationale.unshift('canon_part_2_attribution_breach: brain.author is not "brain"; weight demoted to 0.0.');
|
|
458
|
+
}
|
|
459
|
+
|
|
460
|
+
// Wicked escalation note (added to rationale when triggered).
|
|
461
|
+
const wickedSection = brain && brain.sections && brain.sections.wicked_indicators;
|
|
462
|
+
if (wickedSection && wickedSection.body) {
|
|
463
|
+
const score = extractWickedScore(wickedSection.body);
|
|
464
|
+
if (score >= WICKED_ESCALATION_THRESHOLD) {
|
|
465
|
+
rationale.push('wicked_escalation: wicked_score >= ' + WICKED_ESCALATION_THRESHOLD + ' triggers soft-systems family.');
|
|
466
|
+
}
|
|
467
|
+
}
|
|
468
|
+
|
|
469
|
+
trace.chosen_rationale = rationale.join(' ');
|
|
470
|
+
|
|
471
|
+
// Compose decision outputs (rule-based, not weighted score).
|
|
472
|
+
decision.fire_skill = resolveFireSkill(brain, weightApplied, tierMode);
|
|
473
|
+
decision.suppress_skills = resolveSuppressSkills(ctx, brain);
|
|
474
|
+
decision.persona_updates = resolvePersonaUpdates(ctx, brain);
|
|
475
|
+
decision.offer_next_step = resolveOfferNextStep(ctx, brain);
|
|
476
|
+
|
|
477
|
+
// Phase 91-07: problem-type routing contribution. Lazy-require so a
|
|
478
|
+
// missing module does not crash the engine -- problem-type routing
|
|
479
|
+
// BIASES the rationale (and may inform fire_skill when no
|
|
480
|
+
// higher-priority signal has set one), but is never load-bearing.
|
|
481
|
+
// Per locked decision D-08 + Canon Appendix E rule R4.
|
|
482
|
+
try {
|
|
483
|
+
if (brain !== null && brain.sections) {
|
|
484
|
+
const ptRouter = require('./problem-type-router.cjs');
|
|
485
|
+
const ptSection = brain.sections.problemtype_classification;
|
|
486
|
+
const wkSection = brain.sections.wicked_indicators;
|
|
487
|
+
const parsed = ptRouter.parseProblemTypeSection(ptSection);
|
|
488
|
+
let routing = null;
|
|
489
|
+
if (parsed !== null) {
|
|
490
|
+
routing = ptRouter.routeByProblemType(parsed.type, parsed.confidence);
|
|
491
|
+
}
|
|
492
|
+
const wickedEscalate = ptRouter.detectWickedEscalation(wkSection);
|
|
493
|
+
if (wickedEscalate) {
|
|
494
|
+
// Wicked override applies regardless of base type. If we have
|
|
495
|
+
// no base routing, synthesize a stub so applyWickedOverride
|
|
496
|
+
// has a baseline to overlay.
|
|
497
|
+
if (routing === null) {
|
|
498
|
+
routing = {
|
|
499
|
+
recommended_skills: [],
|
|
500
|
+
suppressed_skills: [],
|
|
501
|
+
reason: '',
|
|
502
|
+
wicked_override: false,
|
|
503
|
+
};
|
|
504
|
+
}
|
|
505
|
+
routing = ptRouter.applyWickedOverride(routing);
|
|
506
|
+
}
|
|
507
|
+
|
|
508
|
+
if (routing !== null && routing.reason && routing.reason.length > 0) {
|
|
509
|
+
// Append routing reason to chosen_rationale so /mos:explain-
|
|
510
|
+
// decision can surface the contribution.
|
|
511
|
+
const existing = trace.chosen_rationale || '';
|
|
512
|
+
const sep = existing.length > 0 ? ' ' : '';
|
|
513
|
+
trace.chosen_rationale = existing + sep + 'ProblemType routing: ' + routing.reason;
|
|
514
|
+
|
|
515
|
+
// If no higher-priority signal set fire_skill AND routing
|
|
516
|
+
// recommends a primary skill AND confidence is usable, set
|
|
517
|
+
// fire_skill to the primary recommendation. Wicked override
|
|
518
|
+
// always wins when present (its first skill is the soft-
|
|
519
|
+
// systems entry point).
|
|
520
|
+
const conf = parsed && typeof parsed.confidence === 'number' ? parsed.confidence : null;
|
|
521
|
+
const confOk = routing.wicked_override || (typeof conf === 'number' && conf >= 0.5);
|
|
522
|
+
if (
|
|
523
|
+
decision.fire_skill === null &&
|
|
524
|
+
confOk &&
|
|
525
|
+
Array.isArray(routing.recommended_skills) &&
|
|
526
|
+
routing.recommended_skills.length > 0
|
|
527
|
+
) {
|
|
528
|
+
decision.fire_skill = routing.recommended_skills[0];
|
|
529
|
+
}
|
|
530
|
+
}
|
|
531
|
+
}
|
|
532
|
+
} catch (_e) {
|
|
533
|
+
// Routing is biasing-only; never block the decision on a routing
|
|
534
|
+
// module fault. Leave decision + trace untouched.
|
|
535
|
+
}
|
|
536
|
+
|
|
537
|
+
// Phase 91-08: framework-chain-composer contribution. Lazy-require
|
|
538
|
+
// so a missing module never crashes the engine -- chain composition
|
|
539
|
+
// is one signal among five and biases offer_next_step when no higher-
|
|
540
|
+
// priority signal has populated it. Per locked decision D-09 + Canon
|
|
541
|
+
// Part 2 Engine 1 + Appendix E. Reads BRAIN.md framework_chain_
|
|
542
|
+
// predictions section (FEEDS_INTO edges authored by Phase 90-01).
|
|
543
|
+
// Canon Part 8: zero Brain queries at engine time -- composer reads
|
|
544
|
+
// only the LOCAL section body passed in via quadruple.
|
|
545
|
+
try {
|
|
546
|
+
if (brain !== null && brain.sections) {
|
|
547
|
+
const chainComposer = require('./framework-chain-composer.cjs');
|
|
548
|
+
const chainSection = brain.sections.framework_chain_predictions;
|
|
549
|
+
if (chainSection !== null && chainSection !== undefined) {
|
|
550
|
+
const edges = chainComposer.parseFrameworkChainSection(chainSection);
|
|
551
|
+
if (edges.length > 0) {
|
|
552
|
+
const completed = chainComposer.detectCompletedFramework(
|
|
553
|
+
null, // roomDir is not needed when reasoning carries the signal
|
|
554
|
+
t.sectionPath || null,
|
|
555
|
+
quadruple.reasoning || null
|
|
556
|
+
);
|
|
557
|
+
if (completed !== null) {
|
|
558
|
+
const proposal = chainComposer.proposeNextFramework(completed, edges);
|
|
559
|
+
if (proposal !== null) {
|
|
560
|
+
// User override detection: if lastTurnOffer was the chain
|
|
561
|
+
// proposal AND current turn invokes a DIFFERENT /mos:
|
|
562
|
+
// command, record REJECTED chain suggestion as graph data
|
|
563
|
+
// per Canon Part 4. The override check runs BEFORE we
|
|
564
|
+
// overwrite offer_next_step so the user's intent (in
|
|
565
|
+
// userText) gets recorded against the prior offer.
|
|
566
|
+
const lastOffer = ctx.lastTurnOffer;
|
|
567
|
+
const userText = typeof t.userText === 'string' ? t.userText : '';
|
|
568
|
+
if (
|
|
569
|
+
lastOffer &&
|
|
570
|
+
typeof lastOffer === 'object' &&
|
|
571
|
+
typeof lastOffer.command === 'string' &&
|
|
572
|
+
lastOffer.command.length > 0 &&
|
|
573
|
+
userText.length > 0
|
|
574
|
+
) {
|
|
575
|
+
// Match a /mos: command in the user text.
|
|
576
|
+
const userCmdMatch = /\/mos:[a-z][a-z0-9-]*/i.exec(userText);
|
|
577
|
+
if (userCmdMatch !== null) {
|
|
578
|
+
const userCmd = userCmdMatch[0];
|
|
579
|
+
if (userCmd !== lastOffer.command) {
|
|
580
|
+
// Override recorded as graph data per Canon Part 4.
|
|
581
|
+
const existing = trace.chosen_rationale || '';
|
|
582
|
+
const sep = existing.length > 0 ? ' ' : '';
|
|
583
|
+
trace.chosen_rationale =
|
|
584
|
+
existing + sep +
|
|
585
|
+
'REJECTED chain suggestion: user invoked ' + userCmd +
|
|
586
|
+
' instead of ' + lastOffer.command + '.';
|
|
587
|
+
trace.chain_override_recorded = true;
|
|
588
|
+
}
|
|
589
|
+
}
|
|
590
|
+
}
|
|
591
|
+
|
|
592
|
+
// Set offer_next_step ONLY when no higher-priority signal
|
|
593
|
+
// has already populated it. Chain composition is one
|
|
594
|
+
// signal among five; it never overwrites a stronger one.
|
|
595
|
+
if (decision.offer_next_step === null) {
|
|
596
|
+
decision.offer_next_step = {
|
|
597
|
+
command: proposal.command,
|
|
598
|
+
reason: proposal.reason,
|
|
599
|
+
};
|
|
600
|
+
// Surface chain-specific eligibility flag in trace so
|
|
601
|
+
// offer-presenter (Plan 91-04) can render the
|
|
602
|
+
// RECOMMENDED marker per Canon Part 3 Section 6 when
|
|
603
|
+
// Mode A holds AND chain confidence >= 0.7.
|
|
604
|
+
trace.chain_recommended_eligible = proposal.recommended_eligible === true;
|
|
605
|
+
// When the engine's RECOMMENDED gate already fired
|
|
606
|
+
// (e.g. via pattern_matches), the chain proposal piggy-
|
|
607
|
+
// backs on the existing marker. When pattern_matches
|
|
608
|
+
// had no candidate (the common case in Wave 3 fixtures)
|
|
609
|
+
// and Mode A holds and the chain proposal is eligible,
|
|
610
|
+
// promote the marker so the presenter renders it.
|
|
611
|
+
if (
|
|
612
|
+
tierMode === 'mode_a' &&
|
|
613
|
+
proposal.recommended_eligible === true &&
|
|
614
|
+
trace.brain_md_recommended_marker_rendered === false
|
|
615
|
+
) {
|
|
616
|
+
trace.brain_md_recommended_marker_rendered = true;
|
|
617
|
+
trace.brain_md_recommended_confidence =
|
|
618
|
+
typeof trace.brain_md_recommended_confidence === 'number' &&
|
|
619
|
+
trace.brain_md_recommended_confidence > proposal.confidence
|
|
620
|
+
? trace.brain_md_recommended_confidence
|
|
621
|
+
: proposal.confidence;
|
|
622
|
+
}
|
|
623
|
+
// Append chain rationale for /mos:explain-decision.
|
|
624
|
+
const existing2 = trace.chosen_rationale || '';
|
|
625
|
+
const sep2 = existing2.length > 0 ? ' ' : '';
|
|
626
|
+
trace.chosen_rationale =
|
|
627
|
+
existing2 + sep2 +
|
|
628
|
+
'Framework chain: ' + completed + ' FEEDS_INTO ' + proposal.next +
|
|
629
|
+
' (Brain confidence ' + proposal.confidence.toFixed(2) + ').';
|
|
630
|
+
}
|
|
631
|
+
}
|
|
632
|
+
}
|
|
633
|
+
}
|
|
634
|
+
}
|
|
635
|
+
}
|
|
636
|
+
} catch (_e) {
|
|
637
|
+
// Chain composition is biasing-only; never block the decision on
|
|
638
|
+
// a composer fault. Leave decision + trace untouched.
|
|
639
|
+
}
|
|
640
|
+
|
|
641
|
+
return decision;
|
|
642
|
+
} catch (err) {
|
|
643
|
+
// Graceful degradation: never throw out of decide().
|
|
644
|
+
const safe = shared.emptyDecision();
|
|
645
|
+
safe.decision_trace.chosen_rationale =
|
|
646
|
+
'engine_fault: ' + (err && err.message ? err.message : String(err));
|
|
647
|
+
safe.decision_trace.brain_md_tier_mode = 'tier_0';
|
|
648
|
+
safe.decision_trace.brain_md_staleness = 'absent';
|
|
649
|
+
return safe;
|
|
650
|
+
}
|
|
651
|
+
}
|
|
652
|
+
|
|
653
|
+
module.exports = {
|
|
654
|
+
decide: decide,
|
|
655
|
+
// Internal helpers exposed for downstream Phase 91 plans + testing:
|
|
656
|
+
resolveTierMode: shared.resolveTierMode,
|
|
657
|
+
applyStalenessMultiplier: shared.applyStalenessMultiplier,
|
|
658
|
+
evaluateRecommendedGate: function (decision, quadruple) {
|
|
659
|
+
// Convenience accessor matching plan frontmatter contract.
|
|
660
|
+
return decision &&
|
|
661
|
+
decision.decision_trace &&
|
|
662
|
+
decision.decision_trace.brain_md_recommended_marker_rendered === true;
|
|
663
|
+
},
|
|
664
|
+
};
|