@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,1034 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MindrianOS MCP Hierarchical Tool Router
|
|
3
|
+
*
|
|
4
|
+
* Registers 9 high-level MCP tools that dispatch to all 64 CLI commands.
|
|
5
|
+
* Hierarchical design keeps total tool definition under 7000 tokens
|
|
6
|
+
* (vs 30-60K for flat 64-tool registration).
|
|
7
|
+
*
|
|
8
|
+
* Router tools:
|
|
9
|
+
* 1. room_state - room health, state, recommendations (5 commands)
|
|
10
|
+
* 2. room_content - projects, opportunities, funding, personas (15 commands)
|
|
11
|
+
* 3. room_graph - knowledge graph, Minto reasoning, visualization (13 commands)
|
|
12
|
+
* 4. methodology - PWS innovation frameworks (14 commands)
|
|
13
|
+
* 5. analysis - systems analysis and trend exploration (13 commands)
|
|
14
|
+
* 6. intelligence - connections, grading, research (7 commands)
|
|
15
|
+
* 7. meeting - meeting filing, pipeline, speakers (3 commands)
|
|
16
|
+
* 8. export - dashboards, wikis, presentations, snapshots (7 commands)
|
|
17
|
+
* 9. orchestration - autonomous execution, rooms, scout, admin (20 commands)
|
|
18
|
+
*
|
|
19
|
+
* Total: 64 unique CLI commands across 9 router tools (verified via ALL_TOOL_COMMANDS).
|
|
20
|
+
*
|
|
21
|
+
* Suggested Next: Every tool response includes a ## Suggested Next section
|
|
22
|
+
* with tool_name, args, and rationale for LLM-orchestrated pipeline chaining (MCP-05).
|
|
23
|
+
*
|
|
24
|
+
* Intelligence Cascade: Write-tools call runCascade after operations
|
|
25
|
+
* for surface parity between CLI hooks and MCP handlers.
|
|
26
|
+
*/
|
|
27
|
+
|
|
28
|
+
'use strict';
|
|
29
|
+
|
|
30
|
+
const path = require('path');
|
|
31
|
+
const { z } = require('zod');
|
|
32
|
+
const { safeReadFile } = require('../core/index.cjs');
|
|
33
|
+
const pipelineState = require('./pipeline-state.cjs');
|
|
34
|
+
|
|
35
|
+
// ---------------------------------------------------------------------------
|
|
36
|
+
// 87-05: MCP input validation primitives (CASCADE-03 + CASCADE-05)
|
|
37
|
+
// ---------------------------------------------------------------------------
|
|
38
|
+
// Three coordinated guards protect the MCP tool surface:
|
|
39
|
+
// 1. SECTION_RE -- Zod regex whitelist for section parameters. No traversal
|
|
40
|
+
// tokens, no whitespace, no Unicode control codes, no caps.
|
|
41
|
+
// 2. safeResolveSection() -- path.resolve guard that rejects any input that
|
|
42
|
+
// escapes roomDir after resolution. Paired with SECTION_RE so a malformed
|
|
43
|
+
// section never reaches fs.readFile/writeFile.
|
|
44
|
+
// 3. opportunitySchema -- explicit Zod validation of the file-opportunity
|
|
45
|
+
// JSON payload (previously accepted raw).
|
|
46
|
+
//
|
|
47
|
+
// These primitives are exported via module.exports._test for unit tests in
|
|
48
|
+
// lib/memory/mcp-input-validation.test.cjs.
|
|
49
|
+
// ---------------------------------------------------------------------------
|
|
50
|
+
|
|
51
|
+
const SECTION_RE = /^[a-z0-9-]+$/;
|
|
52
|
+
const sectionOptional = z.string().regex(SECTION_RE, 'section must match [a-z0-9-]+').optional();
|
|
53
|
+
const sectionRequired = z.string().regex(SECTION_RE, 'section must match [a-z0-9-]+');
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Resolve `section` relative to `roomDir` and refuse any path that escapes
|
|
57
|
+
* roomDir. Complements the SECTION_RE regex: even if a caller bypasses Zod
|
|
58
|
+
* (internal handler, test harness, future refactor), path traversal is still
|
|
59
|
+
* blocked at the I/O boundary.
|
|
60
|
+
*
|
|
61
|
+
* @param {string} roomDir - Absolute room directory path.
|
|
62
|
+
* @param {string|null|undefined} section - Section name.
|
|
63
|
+
* @returns {string} Resolved absolute path inside roomDir.
|
|
64
|
+
* @throws {Error} If resolved path escapes roomDir.
|
|
65
|
+
*/
|
|
66
|
+
function safeResolveSection(roomDir, section) {
|
|
67
|
+
const base = path.resolve(roomDir);
|
|
68
|
+
if (section === null || section === undefined || section === '') return base;
|
|
69
|
+
const resolved = path.resolve(base, section);
|
|
70
|
+
if (resolved !== base && !resolved.startsWith(base + path.sep)) {
|
|
71
|
+
throw new Error(`section path traversal rejected: ${section}`);
|
|
72
|
+
}
|
|
73
|
+
return resolved;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
/**
|
|
77
|
+
* Zod schema for the file-opportunity payload. `passthrough()` preserves
|
|
78
|
+
* caller-supplied fields (source_url, opportunity_id, relevance_reasoning,
|
|
79
|
+
* etc.) that opportunity-ops.cjs reads dynamically, while still enforcing
|
|
80
|
+
* the human-facing title and bounding numeric score/amount fields.
|
|
81
|
+
*/
|
|
82
|
+
const opportunitySchema = z.object({
|
|
83
|
+
title: z.string().min(1).max(500),
|
|
84
|
+
program: z.string().max(500).optional(),
|
|
85
|
+
funder: z.string().max(500).optional(),
|
|
86
|
+
source: z.string().max(200).optional(),
|
|
87
|
+
source_url: z.string().max(2000).optional(),
|
|
88
|
+
opportunity_id: z.string().max(200).optional(),
|
|
89
|
+
domain: z.string().max(200).optional(),
|
|
90
|
+
deadline: z.string().max(50).optional(),
|
|
91
|
+
amount: z.number().nonnegative().optional(),
|
|
92
|
+
amount_floor: z.number().nonnegative().optional(),
|
|
93
|
+
amount_ceiling: z.number().nonnegative().optional(),
|
|
94
|
+
relevance_score: z.number().min(0).max(1).optional(),
|
|
95
|
+
relevance_reasoning: z.string().max(5000).optional(),
|
|
96
|
+
confidence: z.number().min(0).max(1).optional(),
|
|
97
|
+
}).passthrough();
|
|
98
|
+
|
|
99
|
+
// ---------------------------------------------------------------------------
|
|
100
|
+
// Command groupings - authoritative mapping from 52-RESEARCH.md
|
|
101
|
+
// ---------------------------------------------------------------------------
|
|
102
|
+
|
|
103
|
+
// Router 1: room_state (5 commands)
|
|
104
|
+
const ROOM_STATE_COMMANDS = [
|
|
105
|
+
'status', 'analyze', 'compute-state', 'get-state', 'suggest-next'
|
|
106
|
+
];
|
|
107
|
+
|
|
108
|
+
// Router 2: room_content (16 commands)
|
|
109
|
+
const ROOM_CONTENT_COMMANDS = [
|
|
110
|
+
'new-project', 'setup', 'update', 'help', 'detect-integrations',
|
|
111
|
+
'scan-opportunities', 'list-opportunities', 'file-opportunity',
|
|
112
|
+
'list-funding', 'create-funding', 'update-funding-stage',
|
|
113
|
+
'generate-personas', 'list-personas', 'invoke-persona', 'analyze-perspectives',
|
|
114
|
+
'organize'
|
|
115
|
+
];
|
|
116
|
+
|
|
117
|
+
// Router 3: room_graph (13 commands)
|
|
118
|
+
const ROOM_GRAPH_COMMANDS = [
|
|
119
|
+
'graph-index', 'graph-rebuild', 'graph-query', 'graph-stats',
|
|
120
|
+
'reasoning-get', 'reasoning-generate', 'reasoning-verify',
|
|
121
|
+
'reasoning-run', 'reasoning-list', 'reasoning-frontmatter',
|
|
122
|
+
'visualize-room', 'visualize-graph', 'visualize-chain'
|
|
123
|
+
];
|
|
124
|
+
|
|
125
|
+
// Router 4: methodology (14 commands)
|
|
126
|
+
const METHODOLOGY_COMMANDS = [
|
|
127
|
+
'lean-canvas', 'think-hats', 'structure-argument', 'beautiful-question',
|
|
128
|
+
'build-knowledge', 'challenge-assumptions', 'validate', 'map-unknowns',
|
|
129
|
+
'diagnose', 'score-innovation', 'explore-domains', 'analyze-needs',
|
|
130
|
+
'user-needs', 'find-analogies'
|
|
131
|
+
];
|
|
132
|
+
|
|
133
|
+
// Router 5: analysis (13 commands)
|
|
134
|
+
const ANALYSIS_COMMANDS = [
|
|
135
|
+
'analyze-systems', 'analyze-timing', 'find-bottlenecks', 'root-cause',
|
|
136
|
+
'systems-thinking', 'macro-trends', 'explore-trends', 'explore-futures',
|
|
137
|
+
'dominant-designs', 'scenario-plan',
|
|
138
|
+
'causal-extract', 'causal-trace', 'causal-predict'
|
|
139
|
+
];
|
|
140
|
+
|
|
141
|
+
// Router 6: intelligence (8 commands)
|
|
142
|
+
const INTELLIGENCE_COMMANDS = [
|
|
143
|
+
'find-connections', 'build-thesis', 'compare-ventures', 'research',
|
|
144
|
+
'deep-grade', 'grade', 'leadership', 'whitespace'
|
|
145
|
+
];
|
|
146
|
+
|
|
147
|
+
// Router 7: meeting (3 commands)
|
|
148
|
+
const MEETING_COMMANDS = [
|
|
149
|
+
'file-meeting', 'pipeline', 'speakers'
|
|
150
|
+
];
|
|
151
|
+
|
|
152
|
+
// Router 8: export (7 commands)
|
|
153
|
+
const EXPORT_COMMANDS = [
|
|
154
|
+
'export', 'radar', 'dashboard', 'wiki', 'present', 'publish', 'snapshot'
|
|
155
|
+
];
|
|
156
|
+
|
|
157
|
+
// Router 9: orchestration (22 commands)
|
|
158
|
+
const ORCHESTRATION_COMMANDS = [
|
|
159
|
+
'act', 'act-chain', 'act-swarm', 'act-dry-run',
|
|
160
|
+
'rooms-list', 'rooms-new', 'rooms-open', 'rooms-close', 'rooms-archive', 'rooms-where',
|
|
161
|
+
'scout', 'scout-health', 'scout-deadlines', 'scout-competitors', 'scout-hsi', 'scout-snapshot',
|
|
162
|
+
'reanalyze', 'onboard', 'models', 'admin',
|
|
163
|
+
'hat-briefing', 'scheduled-tasks'
|
|
164
|
+
];
|
|
165
|
+
|
|
166
|
+
/**
|
|
167
|
+
* Flat array of all 64 CLI command names for parity checking.
|
|
168
|
+
* Maps 1:1 to the 66 .md files in commands/ minus splash (banner-only)
|
|
169
|
+
* and funding (covered by room_content sub-commands) = 64 routed commands.
|
|
170
|
+
*
|
|
171
|
+
* CLI command names are used (not router sub-command names) for parity validation.
|
|
172
|
+
* Some CLI commands map to multiple router sub-commands (e.g., "room" -> status/analyze/etc).
|
|
173
|
+
*/
|
|
174
|
+
const ALL_TOOL_COMMANDS = [
|
|
175
|
+
// room_state (CLI names)
|
|
176
|
+
'status', 'room', 'suggest-next',
|
|
177
|
+
// room_content (CLI names)
|
|
178
|
+
'new-project', 'setup', 'update', 'help',
|
|
179
|
+
'opportunities', 'persona',
|
|
180
|
+
// room_graph (CLI names)
|
|
181
|
+
'graph', 'query', 'reason', 'visualize',
|
|
182
|
+
// methodology
|
|
183
|
+
...METHODOLOGY_COMMANDS,
|
|
184
|
+
// analysis - CLI names for causal subcommands
|
|
185
|
+
'analyze-systems', 'analyze-timing', 'find-bottlenecks', 'root-cause',
|
|
186
|
+
'systems-thinking', 'macro-trends', 'explore-trends', 'explore-futures',
|
|
187
|
+
'dominant-designs', 'scenario-plan', 'causal',
|
|
188
|
+
// intelligence
|
|
189
|
+
...INTELLIGENCE_COMMANDS,
|
|
190
|
+
// meeting - CLI names
|
|
191
|
+
'file-meeting', 'pipeline', 'speakers',
|
|
192
|
+
// export - CLI names
|
|
193
|
+
'export', 'radar', 'dashboard', 'wiki', 'present', 'publish', 'snapshot',
|
|
194
|
+
// orchestration - CLI names
|
|
195
|
+
'act', 'rooms', 'scout', 'reanalyze', 'onboard', 'models', 'admin',
|
|
196
|
+
'hat-briefing', 'scheduled-tasks'
|
|
197
|
+
];
|
|
198
|
+
|
|
199
|
+
// Write-tools that trigger intelligence cascade after operations
|
|
200
|
+
const WRITE_TOOLS = new Set([
|
|
201
|
+
'file-opportunity', 'create-funding', 'update-funding-stage',
|
|
202
|
+
'generate-personas', 'invoke-persona', 'new-project', 'setup', 'update',
|
|
203
|
+
'graph-index', 'graph-rebuild', 'reasoning-generate',
|
|
204
|
+
'file-meeting'
|
|
205
|
+
]);
|
|
206
|
+
|
|
207
|
+
// ---------------------------------------------------------------------------
|
|
208
|
+
// Helpers
|
|
209
|
+
// ---------------------------------------------------------------------------
|
|
210
|
+
|
|
211
|
+
/**
|
|
212
|
+
* Build a text response in MCP tool format.
|
|
213
|
+
* @param {string} text - Response text
|
|
214
|
+
* @param {boolean} [isError=false] - Whether this is an error response
|
|
215
|
+
*/
|
|
216
|
+
function textResponse(text, isError) {
|
|
217
|
+
const result = { content: [{ type: 'text', text }] };
|
|
218
|
+
if (isError) result.isError = true;
|
|
219
|
+
return result;
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
/**
|
|
223
|
+
* Format a Suggested Next section for pipeline chaining (MCP-05).
|
|
224
|
+
* @param {string} tool - MCP tool name (e.g., 'room_state')
|
|
225
|
+
* @param {Object} args - Tool arguments
|
|
226
|
+
* @param {string} rationale - Why this is the logical next step
|
|
227
|
+
* @returns {string} Formatted markdown section
|
|
228
|
+
*/
|
|
229
|
+
function formatSuggestedNext(tool, args, rationale) {
|
|
230
|
+
return `\n\n## Suggested Next\n\n**Tool:** \`${tool}\`\n**Args:** \`${JSON.stringify(args)}\`\n**Rationale:** ${rationale}`;
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
/**
|
|
234
|
+
* Load a methodology/command reference file from references/methodology/.
|
|
235
|
+
* Falls back to the command's CLI reference in commands/.
|
|
236
|
+
*/
|
|
237
|
+
function loadReference(pluginRoot, command) {
|
|
238
|
+
// Try methodology references first
|
|
239
|
+
const methodRef = safeReadFile(path.join(pluginRoot, 'references', 'methodology', `${command}.md`));
|
|
240
|
+
if (methodRef) return methodRef;
|
|
241
|
+
|
|
242
|
+
// Fall back to CLI command reference
|
|
243
|
+
const cmdRef = safeReadFile(path.join(pluginRoot, 'commands', `${command}.md`));
|
|
244
|
+
if (cmdRef) return cmdRef;
|
|
245
|
+
|
|
246
|
+
return null;
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
/**
|
|
250
|
+
* Load current room state (STATE.md contents). Returns empty string if unavailable.
|
|
251
|
+
*/
|
|
252
|
+
function loadRoomState(roomDir) {
|
|
253
|
+
const stateOps = require('../core/state-ops.cjs');
|
|
254
|
+
try {
|
|
255
|
+
return stateOps.getState(roomDir) || '';
|
|
256
|
+
} catch (_e) {
|
|
257
|
+
return '';
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
/**
|
|
262
|
+
* Build combined context: reference + room state + user focus.
|
|
263
|
+
*/
|
|
264
|
+
function buildContext(pluginRoot, roomDir, command, userContext) {
|
|
265
|
+
const ref = loadReference(pluginRoot, command);
|
|
266
|
+
const state = loadRoomState(roomDir);
|
|
267
|
+
|
|
268
|
+
const parts = [`## Command: ${command}`];
|
|
269
|
+
if (state) parts.push(`\n### Room State\n${state}`);
|
|
270
|
+
if (ref) parts.push(`\n### Reference\n${ref}`);
|
|
271
|
+
if (userContext) parts.push(`\n### Focus\n${userContext}`);
|
|
272
|
+
if (!ref) parts.push(`\n> Note: No reference file found for "${command}". Available commands in this group will still work.`);
|
|
273
|
+
|
|
274
|
+
return parts.join('\n');
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
/**
|
|
278
|
+
* Fire intelligence cascade after a write operation.
|
|
279
|
+
* Non-blocking - failures never break the tool response.
|
|
280
|
+
* @param {string} roomDir
|
|
281
|
+
* @param {string} command - The command that triggered the write
|
|
282
|
+
* @param {string} [section] - Section context
|
|
283
|
+
* @param {Object} [result] - Result from the write operation (may contain filePath)
|
|
284
|
+
*/
|
|
285
|
+
async function fireCascade(roomDir, command, section, result) {
|
|
286
|
+
try {
|
|
287
|
+
const { runCascade } = require('../core/intelligence-cascade.cjs');
|
|
288
|
+
const filePath = (result && result.filePath) || '';
|
|
289
|
+
await runCascade(roomDir, { trigger: 'mcp-tool', filePath, section: section || '' });
|
|
290
|
+
} catch (_e) {
|
|
291
|
+
// Cascade failures must never break tool responses
|
|
292
|
+
}
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
// ---------------------------------------------------------------------------
|
|
296
|
+
// Router registration
|
|
297
|
+
// ---------------------------------------------------------------------------
|
|
298
|
+
|
|
299
|
+
/**
|
|
300
|
+
* Register all 9 hierarchical router tools on the MCP server.
|
|
301
|
+
*
|
|
302
|
+
* @param {import('@modelcontextprotocol/sdk/server/mcp.js').McpServer} server
|
|
303
|
+
* @param {string} roomDir - Absolute path to the Data Room
|
|
304
|
+
* @param {string} pluginRoot - Absolute path to the plugin root
|
|
305
|
+
* @param {{ compact: string, full: string }} larryContext - Larry personality context
|
|
306
|
+
*/
|
|
307
|
+
function registerRouterTools(server, roomDir, pluginRoot, larryContext) {
|
|
308
|
+
const compact = (larryContext && larryContext.compact) || '';
|
|
309
|
+
|
|
310
|
+
// -------------------------------------------------------------------------
|
|
311
|
+
// 1. room_state - room health, state, and recommendations (5 commands)
|
|
312
|
+
// -------------------------------------------------------------------------
|
|
313
|
+
server.tool(
|
|
314
|
+
'room_state',
|
|
315
|
+
`Check room health, state, and get framework recommendations. ${compact.slice(0, 80)}`,
|
|
316
|
+
{
|
|
317
|
+
command: z.enum(ROOM_STATE_COMMANDS)
|
|
318
|
+
.describe('Room state operation to perform'),
|
|
319
|
+
section: sectionOptional
|
|
320
|
+
.describe('Section name for section-specific operations'),
|
|
321
|
+
},
|
|
322
|
+
async ({ command, section }) => {
|
|
323
|
+
// Phase 87-04: MCP path MUST use the async entry point so analyzeRoom's
|
|
324
|
+
// subprocess call does not block the handler loop. Importing bare
|
|
325
|
+
// './room-ops.cjs' would trip the MOS_DEP_ROOM_OPS_LEGACY warning.
|
|
326
|
+
const roomOps = require('../core/room-ops-async.cjs');
|
|
327
|
+
const stateOps = require('../core/state-ops.cjs');
|
|
328
|
+
|
|
329
|
+
switch (command) {
|
|
330
|
+
case 'status': {
|
|
331
|
+
const state = stateOps.getState(roomDir);
|
|
332
|
+
const response = state || 'No room initialized. Run new-project to create one.';
|
|
333
|
+
return textResponse(response + formatSuggestedNext('room_state', { command: 'analyze' }, 'Room status retrieved - run full analysis to detect patterns and gaps'));
|
|
334
|
+
}
|
|
335
|
+
case 'analyze': {
|
|
336
|
+
const analysis = await roomOps.analyzeRoom(roomDir);
|
|
337
|
+
return textResponse(analysis + formatSuggestedNext('room_graph', { command: 'graph-stats' }, 'Room analysis complete - check graph for relationship patterns'));
|
|
338
|
+
}
|
|
339
|
+
case 'compute-state': {
|
|
340
|
+
const computed = stateOps.computeState(roomDir);
|
|
341
|
+
return textResponse(computed + formatSuggestedNext('room_state', { command: 'suggest-next' }, 'State updated - get framework recommendation for current stage'));
|
|
342
|
+
}
|
|
343
|
+
case 'get-state': {
|
|
344
|
+
const state = stateOps.getState(roomDir);
|
|
345
|
+
return textResponse((state || 'No STATE.md found in room.') + formatSuggestedNext('room_state', { command: 'analyze' }, 'State retrieved - run analysis to detect intelligence patterns'));
|
|
346
|
+
}
|
|
347
|
+
case 'suggest-next': {
|
|
348
|
+
const ref = loadReference(pluginRoot, 'suggest-next');
|
|
349
|
+
const state = loadRoomState(roomDir);
|
|
350
|
+
const parts = ['## suggest-next'];
|
|
351
|
+
if (state) parts.push(`\n### Current Room State\n${state}`);
|
|
352
|
+
if (ref) parts.push(`\n### Reference\n${ref}`);
|
|
353
|
+
else parts.push('\n> Reference file for "suggest-next" not found.');
|
|
354
|
+
return textResponse(parts.join('\n') + formatSuggestedNext('methodology', { command: 'diagnose' }, 'Framework recommendation provided - diagnose room to identify starting point'));
|
|
355
|
+
}
|
|
356
|
+
default:
|
|
357
|
+
return textResponse(`Unknown room_state command: ${command}`, true);
|
|
358
|
+
}
|
|
359
|
+
}
|
|
360
|
+
);
|
|
361
|
+
|
|
362
|
+
// -------------------------------------------------------------------------
|
|
363
|
+
// 2. room_content - projects, opportunities, funding, personas (15 commands)
|
|
364
|
+
// -------------------------------------------------------------------------
|
|
365
|
+
server.tool(
|
|
366
|
+
'room_content',
|
|
367
|
+
'Manage room content: projects, opportunities, funding, personas, integrations.',
|
|
368
|
+
{
|
|
369
|
+
command: z.enum(ROOM_CONTENT_COMMANDS)
|
|
370
|
+
.describe('Room content operation to perform'),
|
|
371
|
+
section: sectionOptional
|
|
372
|
+
.describe('Section name or JSON data for operations'),
|
|
373
|
+
context: z.string().optional()
|
|
374
|
+
.describe('Additional context for the operation'),
|
|
375
|
+
},
|
|
376
|
+
async ({ command, section, context }) => {
|
|
377
|
+
const defaultNext = formatSuggestedNext('room_state', { command: 'analyze' }, 'Content updated - analyze room to detect new intelligence patterns');
|
|
378
|
+
|
|
379
|
+
switch (command) {
|
|
380
|
+
case 'new-project':
|
|
381
|
+
case 'setup':
|
|
382
|
+
case 'update':
|
|
383
|
+
case 'help': {
|
|
384
|
+
const ref = loadReference(pluginRoot, command);
|
|
385
|
+
const state = loadRoomState(roomDir);
|
|
386
|
+
const parts = [`## ${command}`];
|
|
387
|
+
if (state) parts.push(`\n### Current Room State\n${state}`);
|
|
388
|
+
if (ref) parts.push(`\n### Reference\n${ref}`);
|
|
389
|
+
if (context) parts.push(`\n### Focus\n${context}`);
|
|
390
|
+
else parts.push(`\n> Reference file for "${command}" not found.`);
|
|
391
|
+
const response = parts.join('\n');
|
|
392
|
+
// Fire cascade for write operations
|
|
393
|
+
if (WRITE_TOOLS.has(command)) {
|
|
394
|
+
await fireCascade(roomDir, command, section);
|
|
395
|
+
}
|
|
396
|
+
return textResponse(response + defaultNext);
|
|
397
|
+
}
|
|
398
|
+
case 'detect-integrations': {
|
|
399
|
+
const integrationRegistry = require('../core/integration-registry.cjs');
|
|
400
|
+
const integResult = integrationRegistry.detectIntegrations({ workDir: roomDir });
|
|
401
|
+
return textResponse(JSON.stringify(integResult, null, 2) + formatSuggestedNext('room_content', { command: 'setup' }, 'Integrations detected - run setup to configure discovered services'));
|
|
402
|
+
}
|
|
403
|
+
case 'scan-opportunities': {
|
|
404
|
+
const opportunityOps = require('../core/opportunity-ops.cjs');
|
|
405
|
+
const scanResult = await opportunityOps.scanOpportunities(roomDir);
|
|
406
|
+
return textResponse(JSON.stringify(scanResult, null, 2) + formatSuggestedNext('room_content', { command: 'list-opportunities' }, 'Scan complete - list discovered opportunities for review'));
|
|
407
|
+
}
|
|
408
|
+
case 'list-opportunities': {
|
|
409
|
+
const opportunityOps = require('../core/opportunity-ops.cjs');
|
|
410
|
+
const listResult = opportunityOps.listOpportunities(roomDir);
|
|
411
|
+
return textResponse(JSON.stringify(listResult, null, 2) + formatSuggestedNext('room_content', { command: 'file-opportunity' }, 'Opportunities listed - file the most promising one to your room'));
|
|
412
|
+
}
|
|
413
|
+
case 'file-opportunity': {
|
|
414
|
+
const opportunityOps = require('../core/opportunity-ops.cjs');
|
|
415
|
+
let oppData;
|
|
416
|
+
try {
|
|
417
|
+
oppData = section ? JSON.parse(section) : {};
|
|
418
|
+
} catch (_e) {
|
|
419
|
+
return textResponse('Invalid JSON in section parameter for file-opportunity', true);
|
|
420
|
+
}
|
|
421
|
+
// 87-05: validate opportunity payload structure before filing (CASCADE-05)
|
|
422
|
+
let validatedOpp;
|
|
423
|
+
try {
|
|
424
|
+
validatedOpp = opportunitySchema.parse(oppData);
|
|
425
|
+
} catch (zodErr) {
|
|
426
|
+
return textResponse('Invalid opportunity payload: ' + (zodErr && zodErr.message ? zodErr.message : String(zodErr)), true);
|
|
427
|
+
}
|
|
428
|
+
const fileResult = opportunityOps.fileOpportunity(roomDir, validatedOpp);
|
|
429
|
+
await fireCascade(roomDir, command, section, fileResult);
|
|
430
|
+
return textResponse(JSON.stringify(fileResult, null, 2) + formatSuggestedNext('room_state', { command: 'analyze' }, 'Opportunity filed - analyze room to detect cross-section patterns'));
|
|
431
|
+
}
|
|
432
|
+
case 'list-funding': {
|
|
433
|
+
const opportunityOps = require('../core/opportunity-ops.cjs');
|
|
434
|
+
const fundResult = opportunityOps.listFunding(roomDir);
|
|
435
|
+
return textResponse(JSON.stringify(fundResult, null, 2) + formatSuggestedNext('room_content', { command: 'create-funding' }, 'Funding listed - create a new funding tracker if needed'));
|
|
436
|
+
}
|
|
437
|
+
case 'create-funding': {
|
|
438
|
+
const opportunityOps = require('../core/opportunity-ops.cjs');
|
|
439
|
+
let fundData;
|
|
440
|
+
try {
|
|
441
|
+
fundData = section ? JSON.parse(section) : {};
|
|
442
|
+
} catch (_e) {
|
|
443
|
+
return textResponse('Invalid JSON in section parameter for create-funding', true);
|
|
444
|
+
}
|
|
445
|
+
if (!fundData.slug || !fundData.source) {
|
|
446
|
+
return textResponse('create-funding requires { slug, source } in section parameter', true);
|
|
447
|
+
}
|
|
448
|
+
const createResult = opportunityOps.createFunding(roomDir, fundData.slug, fundData.source);
|
|
449
|
+
await fireCascade(roomDir, command, section, createResult);
|
|
450
|
+
return textResponse(JSON.stringify(createResult, null, 2) + formatSuggestedNext('room_content', { command: 'update-funding-stage' }, 'Funding created - update its stage as progress is made'));
|
|
451
|
+
}
|
|
452
|
+
case 'update-funding-stage': {
|
|
453
|
+
const opportunityOps = require('../core/opportunity-ops.cjs');
|
|
454
|
+
let stageData;
|
|
455
|
+
try {
|
|
456
|
+
stageData = section ? JSON.parse(section) : {};
|
|
457
|
+
} catch (_e) {
|
|
458
|
+
return textResponse('Invalid JSON in section parameter for update-funding-stage', true);
|
|
459
|
+
}
|
|
460
|
+
if (!stageData.slug || !stageData.stage) {
|
|
461
|
+
return textResponse('update-funding-stage requires { slug, stage, note? } in section parameter', true);
|
|
462
|
+
}
|
|
463
|
+
const updateResult = opportunityOps.updateFundingStage(roomDir, stageData.slug, stageData.stage, stageData.note || '');
|
|
464
|
+
await fireCascade(roomDir, command, section, updateResult);
|
|
465
|
+
return textResponse(JSON.stringify(updateResult, null, 2) + formatSuggestedNext('room_state', { command: 'status' }, 'Funding stage updated - check room status for pipeline overview'));
|
|
466
|
+
}
|
|
467
|
+
case 'generate-personas': {
|
|
468
|
+
const personaOps = require('../core/persona-ops.cjs');
|
|
469
|
+
const genResult = personaOps.generatePersonas(roomDir);
|
|
470
|
+
await fireCascade(roomDir, command, section, genResult);
|
|
471
|
+
return textResponse(JSON.stringify(genResult, null, 2) + formatSuggestedNext('room_content', { command: 'list-personas' }, 'Personas generated - list them to see available perspectives'));
|
|
472
|
+
}
|
|
473
|
+
case 'list-personas': {
|
|
474
|
+
const personaOps = require('../core/persona-ops.cjs');
|
|
475
|
+
const personas = personaOps.listPersonas(roomDir);
|
|
476
|
+
return textResponse(JSON.stringify(personas, null, 2) + formatSuggestedNext('room_content', { command: 'invoke-persona' }, 'Personas available - invoke one for a specific perspective on your venture'));
|
|
477
|
+
}
|
|
478
|
+
case 'invoke-persona': {
|
|
479
|
+
const personaOps = require('../core/persona-ops.cjs');
|
|
480
|
+
let invokeParams;
|
|
481
|
+
try {
|
|
482
|
+
invokeParams = section ? JSON.parse(section) : {};
|
|
483
|
+
} catch (_e) {
|
|
484
|
+
invokeParams = { hat: section };
|
|
485
|
+
}
|
|
486
|
+
if (!invokeParams.hat) {
|
|
487
|
+
return textResponse('invoke-persona requires a hat color. Pass { "hat": "black", "artifact": "path" } in section parameter.', true);
|
|
488
|
+
}
|
|
489
|
+
const invokeResult = personaOps.invokePersona(roomDir, invokeParams.hat, invokeParams.artifact || null);
|
|
490
|
+
await fireCascade(roomDir, command, section, invokeResult);
|
|
491
|
+
return textResponse(JSON.stringify(invokeResult, null, 2) + formatSuggestedNext('room_content', { command: 'analyze-perspectives' }, 'Persona invoked - run full perspective analysis for all hats'));
|
|
492
|
+
}
|
|
493
|
+
case 'analyze-perspectives': {
|
|
494
|
+
const personaOps = require('../core/persona-ops.cjs');
|
|
495
|
+
let analyzeParams;
|
|
496
|
+
try {
|
|
497
|
+
analyzeParams = section ? JSON.parse(section) : {};
|
|
498
|
+
} catch (_e) {
|
|
499
|
+
analyzeParams = { artifact: section };
|
|
500
|
+
}
|
|
501
|
+
const analyzeResult = personaOps.analyzeAllPerspectives(roomDir, analyzeParams.artifact || null);
|
|
502
|
+
return textResponse(JSON.stringify(analyzeResult, null, 2) + formatSuggestedNext('room_state', { command: 'analyze' }, 'All perspectives analyzed - check room for new cross-section patterns'));
|
|
503
|
+
}
|
|
504
|
+
default:
|
|
505
|
+
return textResponse(`Unknown room_content command: ${command}`, true);
|
|
506
|
+
}
|
|
507
|
+
}
|
|
508
|
+
);
|
|
509
|
+
|
|
510
|
+
// -------------------------------------------------------------------------
|
|
511
|
+
// 3. room_graph - knowledge graph, Minto reasoning, visualization (13 cmds)
|
|
512
|
+
// -------------------------------------------------------------------------
|
|
513
|
+
server.tool(
|
|
514
|
+
'room_graph',
|
|
515
|
+
'Knowledge graph operations, Minto reasoning, and room visualization.',
|
|
516
|
+
{
|
|
517
|
+
command: z.enum(ROOM_GRAPH_COMMANDS)
|
|
518
|
+
.describe('Graph/reasoning/visualization operation'),
|
|
519
|
+
section: sectionOptional
|
|
520
|
+
.describe('Section name, file path, SQL query, or JSON params'),
|
|
521
|
+
query: z.string().optional()
|
|
522
|
+
.describe('SQL query for graph-query command'),
|
|
523
|
+
},
|
|
524
|
+
async ({ command, section, query }) => {
|
|
525
|
+
switch (command) {
|
|
526
|
+
case 'graph-index': {
|
|
527
|
+
const graphOps = require('../core/graph-ops.cjs');
|
|
528
|
+
if (!section) {
|
|
529
|
+
return textResponse('graph-index requires a file path in the section parameter', true);
|
|
530
|
+
}
|
|
531
|
+
const indexResult = await graphOps.indexArtifact(roomDir, section);
|
|
532
|
+
await fireCascade(roomDir, command, section, indexResult);
|
|
533
|
+
return textResponse(JSON.stringify(indexResult, null, 2) + formatSuggestedNext('room_graph', { command: 'graph-stats' }, 'Artifact indexed - check graph stats for new relationship patterns'));
|
|
534
|
+
}
|
|
535
|
+
case 'graph-rebuild': {
|
|
536
|
+
const graphOps = require('../core/graph-ops.cjs');
|
|
537
|
+
const rebuildResult = await graphOps.rebuildGraph(roomDir);
|
|
538
|
+
await fireCascade(roomDir, command, section, rebuildResult);
|
|
539
|
+
return textResponse(JSON.stringify(rebuildResult, null, 2) + formatSuggestedNext('room_graph', { command: 'graph-stats' }, 'Graph rebuilt - review stats to verify relationship coverage'));
|
|
540
|
+
}
|
|
541
|
+
case 'graph-query': {
|
|
542
|
+
const graphOps = require('../core/graph-ops.cjs');
|
|
543
|
+
const queryStr = query || section;
|
|
544
|
+
if (!queryStr) {
|
|
545
|
+
return textResponse(
|
|
546
|
+
'graph-query requires a SQL query in the query or section parameter.\n\n' +
|
|
547
|
+
'## SQLite LazyGraph Schema Reference\n' +
|
|
548
|
+
'Tables:\n' +
|
|
549
|
+
'- nodes(id TEXT PK, type TEXT, properties TEXT JSON)\n' +
|
|
550
|
+
' Node types: Artifact, Section, CausalClaim, WhitespaceZone\n' +
|
|
551
|
+
' Artifact properties: title, section, methodology, created, content_hash\n' +
|
|
552
|
+
' Section properties: name, label\n\n' +
|
|
553
|
+
'- edges(source TEXT, target TEXT, type TEXT, properties TEXT JSON)\n' +
|
|
554
|
+
' PK: (source, target, type)\n\n' +
|
|
555
|
+
'Edge types:\n' +
|
|
556
|
+
'- INFORMS (Artifact -> Artifact) - cross-section references via [[wikilinks]]\n' +
|
|
557
|
+
'- CONTRADICTS (Artifact -> Artifact, confidence) - conflicting claims\n' +
|
|
558
|
+
'- CONVERGES (Artifact -> Artifact, term) - themes in 3+ sections\n' +
|
|
559
|
+
'- ENABLES (Artifact -> Artifact) - unblocks another artifact\n' +
|
|
560
|
+
'- INVALIDATES (Artifact -> Artifact) - makes another claim stale\n' +
|
|
561
|
+
'- BELONGS_TO (Artifact -> Section) - section membership\n\n' +
|
|
562
|
+
'Example: SELECT n1.id, n2.id FROM edges e JOIN nodes n1 ON n1.id=e.source JOIN nodes n2 ON n2.id=e.target WHERE e.type=\'CONTRADICTS\'',
|
|
563
|
+
true
|
|
564
|
+
);
|
|
565
|
+
}
|
|
566
|
+
const queryResult = await graphOps.queryGraph(roomDir, queryStr);
|
|
567
|
+
return textResponse(JSON.stringify(queryResult, null, 2) + formatSuggestedNext('room_graph', { command: 'visualize-graph' }, 'Query results returned - visualize the graph to see structural patterns'));
|
|
568
|
+
}
|
|
569
|
+
case 'graph-stats': {
|
|
570
|
+
const graphOps = require('../core/graph-ops.cjs');
|
|
571
|
+
const statsResult = await graphOps.graphStats(roomDir);
|
|
572
|
+
return textResponse(JSON.stringify(statsResult, null, 2) + formatSuggestedNext('room_state', { command: 'analyze' }, 'Graph stats reviewed - analyze room for actionable intelligence'));
|
|
573
|
+
}
|
|
574
|
+
case 'reasoning-get': {
|
|
575
|
+
const reasoningOps = require('../core/reasoning-ops.cjs');
|
|
576
|
+
if (!section) return textResponse('reasoning-get requires a section name in the section parameter', true);
|
|
577
|
+
const getResult = reasoningOps.getReasoning(roomDir, section);
|
|
578
|
+
return textResponse(JSON.stringify(getResult, null, 2) + formatSuggestedNext('room_graph', { command: 'reasoning-verify', section }, 'Reasoning retrieved - verify its structural integrity'));
|
|
579
|
+
}
|
|
580
|
+
case 'reasoning-generate': {
|
|
581
|
+
const reasoningOps = require('../core/reasoning-ops.cjs');
|
|
582
|
+
const genResult = reasoningOps.generateReasoning(roomDir, section || null);
|
|
583
|
+
await fireCascade(roomDir, command, section, genResult);
|
|
584
|
+
return textResponse(JSON.stringify(genResult, null, 2) + formatSuggestedNext('room_graph', { command: 'graph-index' }, 'Reasoning filed - index for cross-reference discovery'));
|
|
585
|
+
}
|
|
586
|
+
case 'reasoning-verify': {
|
|
587
|
+
const reasoningOps = require('../core/reasoning-ops.cjs');
|
|
588
|
+
if (!section) return textResponse('reasoning-verify requires a section name in the section parameter', true);
|
|
589
|
+
const verifyResult = reasoningOps.verifyReasoning(roomDir, section);
|
|
590
|
+
return textResponse(JSON.stringify(verifyResult, null, 2) + formatSuggestedNext('room_graph', { command: 'reasoning-run', section }, 'Reasoning verified - create a reasoning run to track execution'));
|
|
591
|
+
}
|
|
592
|
+
case 'reasoning-run': {
|
|
593
|
+
const reasoningOps = require('../core/reasoning-ops.cjs');
|
|
594
|
+
if (!section) return textResponse('reasoning-run requires a section name in the section parameter', true);
|
|
595
|
+
const runResult = reasoningOps.createRun(roomDir, section);
|
|
596
|
+
return textResponse(JSON.stringify(runResult, null, 2) + formatSuggestedNext('room_state', { command: 'analyze' }, 'Reasoning run created - analyze room for updated intelligence'));
|
|
597
|
+
}
|
|
598
|
+
case 'reasoning-list': {
|
|
599
|
+
const reasoningOps = require('../core/reasoning-ops.cjs');
|
|
600
|
+
const listResult = reasoningOps.listReasoning(roomDir);
|
|
601
|
+
return textResponse(JSON.stringify(listResult, null, 2) + formatSuggestedNext('room_graph', { command: 'reasoning-get' }, 'Reasoning files listed - get a specific one for detailed review'));
|
|
602
|
+
}
|
|
603
|
+
case 'reasoning-frontmatter': {
|
|
604
|
+
const reasoningOps = require('../core/reasoning-ops.cjs');
|
|
605
|
+
if (!section) return textResponse('reasoning-frontmatter requires a section name or JSON in the section parameter', true);
|
|
606
|
+
let fmParams;
|
|
607
|
+
try {
|
|
608
|
+
fmParams = JSON.parse(section);
|
|
609
|
+
} catch (_e) {
|
|
610
|
+
const fmResult = reasoningOps.getReasoningFrontmatter(roomDir, section);
|
|
611
|
+
return textResponse(JSON.stringify(fmResult, null, 2) + formatSuggestedNext('room_graph', { command: 'reasoning-verify', section }, 'Frontmatter retrieved - verify reasoning structure'));
|
|
612
|
+
}
|
|
613
|
+
const { action, section: fmSection, field, value } = fmParams;
|
|
614
|
+
let fmResult;
|
|
615
|
+
switch (action) {
|
|
616
|
+
case 'set':
|
|
617
|
+
fmResult = reasoningOps.setReasoningFrontmatter(roomDir, fmSection, field, value);
|
|
618
|
+
break;
|
|
619
|
+
case 'merge':
|
|
620
|
+
fmResult = reasoningOps.mergeReasoningFrontmatter(roomDir, fmSection, fmParams);
|
|
621
|
+
break;
|
|
622
|
+
case 'get':
|
|
623
|
+
default:
|
|
624
|
+
fmResult = reasoningOps.getReasoningFrontmatter(roomDir, fmSection, field || null);
|
|
625
|
+
break;
|
|
626
|
+
}
|
|
627
|
+
return textResponse(JSON.stringify(fmResult, null, 2) + formatSuggestedNext('room_graph', { command: 'reasoning-verify', section: fmSection }, 'Frontmatter updated - verify reasoning integrity'));
|
|
628
|
+
}
|
|
629
|
+
case 'visualize-room': {
|
|
630
|
+
const visualOps = require('../core/visual-ops.cjs');
|
|
631
|
+
// Phase 87-04: async entry point; MCP handler is already async so
|
|
632
|
+
// awaiting listSections is free. See 87-04 for sync/async split.
|
|
633
|
+
const roomOpsViz = require('../core/room-ops-async.cjs');
|
|
634
|
+
const stateOpsViz = require('../core/state-ops.cjs');
|
|
635
|
+
let sections = [];
|
|
636
|
+
try {
|
|
637
|
+
const sectionData = await roomOpsViz.listSections(roomDir);
|
|
638
|
+
let stage = 'discovery';
|
|
639
|
+
try {
|
|
640
|
+
const stContent = stateOpsViz.getState(roomDir);
|
|
641
|
+
if (stContent) {
|
|
642
|
+
const sm = stContent.match(/venture_stage:\s*(.+)/);
|
|
643
|
+
if (sm) stage = sm[1].trim();
|
|
644
|
+
}
|
|
645
|
+
} catch (_e) {}
|
|
646
|
+
if (sectionData && sectionData.sections) {
|
|
647
|
+
sections = sectionData.sections.map(s => ({
|
|
648
|
+
name: s.name || s,
|
|
649
|
+
entryCount: s.entryCount || s.entries || 0,
|
|
650
|
+
stage: stage,
|
|
651
|
+
edges: s.edges || []
|
|
652
|
+
}));
|
|
653
|
+
}
|
|
654
|
+
} catch (_e) {}
|
|
655
|
+
if (sections.length === 0) {
|
|
656
|
+
sections = [{ name: 'No room data', entryCount: 0, stage: 'discovery', edges: [] }];
|
|
657
|
+
}
|
|
658
|
+
const roomMermaid = visualOps.generateMermaidRoom(sections);
|
|
659
|
+
return textResponse(visualOps.generateMermaidBlock(roomMermaid) + formatSuggestedNext('room_graph', { command: 'visualize-graph' }, 'Room structure visualized - see the knowledge graph relationships'));
|
|
660
|
+
}
|
|
661
|
+
case 'visualize-graph': {
|
|
662
|
+
const visualOps = require('../core/visual-ops.cjs');
|
|
663
|
+
const graphOpsViz = require('../core/graph-ops.cjs');
|
|
664
|
+
let nodes = [];
|
|
665
|
+
let edges = [];
|
|
666
|
+
try {
|
|
667
|
+
const stats = await graphOpsViz.graphStats(roomDir);
|
|
668
|
+
if (stats && stats.nodes) { nodes = stats.nodes; edges = stats.edges || []; }
|
|
669
|
+
} catch (_e) {}
|
|
670
|
+
if (nodes.length === 0) {
|
|
671
|
+
nodes = [{ id: 'empty', type: 'Section', label: 'No graph data' }];
|
|
672
|
+
}
|
|
673
|
+
const graphMermaid = visualOps.generateMermaidGraph(nodes, edges);
|
|
674
|
+
return textResponse(visualOps.generateMermaidBlock(graphMermaid) + formatSuggestedNext('room_graph', { command: 'graph-query' }, 'Graph visualized - query for specific relationship patterns'));
|
|
675
|
+
}
|
|
676
|
+
case 'visualize-chain': {
|
|
677
|
+
const visualOps = require('../core/visual-ops.cjs');
|
|
678
|
+
const chainSteps = [
|
|
679
|
+
{ name: 'Diagnose', framework: 'diagnose', status: 'pending' },
|
|
680
|
+
{ name: 'Framework', framework: '', status: 'pending' },
|
|
681
|
+
{ name: 'Apply', framework: '', status: 'pending' },
|
|
682
|
+
{ name: 'File', framework: '', status: 'pending' },
|
|
683
|
+
{ name: 'Cross-ref', framework: '', status: 'pending' },
|
|
684
|
+
{ name: 'Graph Update', framework: '', status: 'pending' }
|
|
685
|
+
];
|
|
686
|
+
const rfs = require('fs');
|
|
687
|
+
const rpath = require('path');
|
|
688
|
+
if (section) {
|
|
689
|
+
try {
|
|
690
|
+
// 87-05: safeResolveSection guards against path traversal even if
|
|
691
|
+
// a malformed section string bypassed the Zod regex at the MCP edge.
|
|
692
|
+
const sectionDir = safeResolveSection(roomDir, section);
|
|
693
|
+
const reasonDir = rpath.join(sectionDir, '.reasoning');
|
|
694
|
+
if (rfs.existsSync(reasonDir)) {
|
|
695
|
+
const runs = rfs.readdirSync(reasonDir).filter(f => f.endsWith('.md')).sort().reverse();
|
|
696
|
+
if (runs.length > 0) {
|
|
697
|
+
const content = rfs.readFileSync(rpath.join(reasonDir, runs[0]), 'utf-8');
|
|
698
|
+
const stepMatches = content.match(/##\s+Step\s+\d+[^#]*/g) || [];
|
|
699
|
+
if (stepMatches.length > 0) {
|
|
700
|
+
chainSteps.length = 0;
|
|
701
|
+
stepMatches.forEach((blk, i) => {
|
|
702
|
+
const nm = blk.match(/##\s+Step\s+\d+:\s*(.+)/);
|
|
703
|
+
const fw = blk.match(/framework:\s*(.+)/i);
|
|
704
|
+
const st = blk.match(/status:\s*(.+)/i);
|
|
705
|
+
chainSteps.push({
|
|
706
|
+
name: nm ? nm[1].trim() : `Step ${i + 1}`,
|
|
707
|
+
framework: fw ? fw[1].trim() : '',
|
|
708
|
+
status: st ? st[1].trim() : 'pending'
|
|
709
|
+
});
|
|
710
|
+
});
|
|
711
|
+
}
|
|
712
|
+
}
|
|
713
|
+
}
|
|
714
|
+
} catch (_e) {}
|
|
715
|
+
}
|
|
716
|
+
const chainMermaid = visualOps.generateMermaidChain(chainSteps);
|
|
717
|
+
return textResponse(visualOps.generateMermaidBlock(chainMermaid) + formatSuggestedNext('room_state', { command: 'suggest-next' }, 'Chain visualized - get framework recommendation for next step'));
|
|
718
|
+
}
|
|
719
|
+
// I-1 fix: removed dead cases (detect-integrations, suggest-next, new-project,
|
|
720
|
+
// setup, update, help) that are unreachable -- not in ROOM_GRAPH_COMMANDS z.enum.
|
|
721
|
+
// These commands are handled by room_state and orchestration routers.
|
|
722
|
+
default:
|
|
723
|
+
return textResponse(`Unknown room_graph command: ${command}`, true);
|
|
724
|
+
}
|
|
725
|
+
}
|
|
726
|
+
);
|
|
727
|
+
|
|
728
|
+
// -------------------------------------------------------------------------
|
|
729
|
+
// 4. methodology - PWS innovation frameworks (14 commands)
|
|
730
|
+
// -------------------------------------------------------------------------
|
|
731
|
+
server.tool(
|
|
732
|
+
'methodology',
|
|
733
|
+
'Run a PWS innovation methodology framework. Larry guides you through the selected framework and files results to your Data Room.',
|
|
734
|
+
{
|
|
735
|
+
command: z.enum(METHODOLOGY_COMMANDS)
|
|
736
|
+
.describe('Which methodology to run'),
|
|
737
|
+
context: z.string().optional()
|
|
738
|
+
.describe('Focus area or venture aspect'),
|
|
739
|
+
section: sectionOptional
|
|
740
|
+
.describe('Section to focus on'),
|
|
741
|
+
},
|
|
742
|
+
async ({ command, context, section }) => {
|
|
743
|
+
const response = buildContext(pluginRoot, roomDir, command, context);
|
|
744
|
+
|
|
745
|
+
// Record step in pipeline state and generate Pipeline Context header
|
|
746
|
+
const outputPath = section
|
|
747
|
+
? `${section}/${command}-output.md`
|
|
748
|
+
: `methodology/${command}-output.md`;
|
|
749
|
+
const updatedState = pipelineState.recordStep(roomDir, command, outputPath);
|
|
750
|
+
const pipelineCtx = pipelineState.formatPipelineContext(roomDir, command, outputPath);
|
|
751
|
+
|
|
752
|
+
// Use pipeline-aware suggested next if available, otherwise default
|
|
753
|
+
let suggestedNext;
|
|
754
|
+
if (updatedState.suggested_next) {
|
|
755
|
+
// Determine which router group the next tool belongs to
|
|
756
|
+
const nextTool = updatedState.suggested_next;
|
|
757
|
+
const routerGroup = ANALYSIS_COMMANDS.includes(nextTool) ? 'analysis'
|
|
758
|
+
: METHODOLOGY_COMMANDS.includes(nextTool) ? 'methodology'
|
|
759
|
+
: INTELLIGENCE_COMMANDS.includes(nextTool) ? 'intelligence'
|
|
760
|
+
: 'room_state';
|
|
761
|
+
const nextArgs = routerGroup === 'room_state'
|
|
762
|
+
? { command: 'analyze' }
|
|
763
|
+
: { command: nextTool };
|
|
764
|
+
suggestedNext = formatSuggestedNext(routerGroup, nextArgs, `Pipeline step ${updatedState.chain_position + 2} of ${updatedState.chain.length} - run ${nextTool} next`);
|
|
765
|
+
} else {
|
|
766
|
+
suggestedNext = formatSuggestedNext('room_state', { command: 'analyze' }, 'Methodology complete - analyze room to detect new patterns from this framework');
|
|
767
|
+
}
|
|
768
|
+
|
|
769
|
+
return textResponse(response + pipelineCtx + suggestedNext);
|
|
770
|
+
}
|
|
771
|
+
);
|
|
772
|
+
|
|
773
|
+
// -------------------------------------------------------------------------
|
|
774
|
+
// 5. analysis - systems analysis and trend exploration (13 commands)
|
|
775
|
+
// -------------------------------------------------------------------------
|
|
776
|
+
server.tool(
|
|
777
|
+
'analysis',
|
|
778
|
+
'Systems analysis, trend exploration, causal reasoning, and scenario planning.',
|
|
779
|
+
{
|
|
780
|
+
command: z.enum(ANALYSIS_COMMANDS)
|
|
781
|
+
.describe('Which analysis to run'),
|
|
782
|
+
context: z.string().optional()
|
|
783
|
+
.describe('Focus area or system boundary'),
|
|
784
|
+
section: sectionOptional
|
|
785
|
+
.describe('Section to focus on'),
|
|
786
|
+
},
|
|
787
|
+
async ({ command, context, section }) => {
|
|
788
|
+
// Record step in pipeline state
|
|
789
|
+
const outputPath = section
|
|
790
|
+
? `${section}/${command}-output.md`
|
|
791
|
+
: `analysis/${command}-output.md`;
|
|
792
|
+
const updatedState = pipelineState.recordStep(roomDir, command, outputPath);
|
|
793
|
+
const pipelineCtx = pipelineState.formatPipelineContext(roomDir, command, outputPath);
|
|
794
|
+
|
|
795
|
+
// Pipeline-aware suggested next: prefer chain ordering over hardcoded patterns
|
|
796
|
+
let suggestedNext;
|
|
797
|
+
if (updatedState.suggested_next) {
|
|
798
|
+
const nextTool = updatedState.suggested_next;
|
|
799
|
+
const routerGroup = ANALYSIS_COMMANDS.includes(nextTool) ? 'analysis'
|
|
800
|
+
: METHODOLOGY_COMMANDS.includes(nextTool) ? 'methodology'
|
|
801
|
+
: INTELLIGENCE_COMMANDS.includes(nextTool) ? 'intelligence'
|
|
802
|
+
: 'room_state';
|
|
803
|
+
const nextArgs = routerGroup === 'room_state'
|
|
804
|
+
? { command: 'analyze' }
|
|
805
|
+
: { command: nextTool };
|
|
806
|
+
suggestedNext = formatSuggestedNext(routerGroup, nextArgs, `Pipeline step ${updatedState.chain_position + 2} of ${updatedState.chain.length} - run ${nextTool} next`);
|
|
807
|
+
} else if (command === 'scenario-plan') {
|
|
808
|
+
suggestedNext = formatSuggestedNext('analysis', { command: 'root-cause' }, 'Scenarios mapped - identify root causes of key risks');
|
|
809
|
+
} else if (command === 'root-cause') {
|
|
810
|
+
suggestedNext = formatSuggestedNext('analysis', { command: 'causal-trace' }, 'Root causes identified - trace causal chains for deeper understanding');
|
|
811
|
+
} else if (command === 'causal-extract') {
|
|
812
|
+
suggestedNext = formatSuggestedNext('analysis', { command: 'causal-trace' }, 'Causal factors extracted - trace their chain of influence');
|
|
813
|
+
} else if (command === 'causal-trace') {
|
|
814
|
+
suggestedNext = formatSuggestedNext('analysis', { command: 'causal-predict' }, 'Causal chains traced - predict outcomes based on these chains');
|
|
815
|
+
} else if (command === 'causal-predict') {
|
|
816
|
+
suggestedNext = formatSuggestedNext('room_state', { command: 'analyze' }, 'Predictions generated - analyze room to integrate causal intelligence');
|
|
817
|
+
} else {
|
|
818
|
+
suggestedNext = formatSuggestedNext('room_state', { command: 'analyze' }, 'Analysis complete - check room for new intelligence patterns');
|
|
819
|
+
}
|
|
820
|
+
const response = buildContext(pluginRoot, roomDir, command, context);
|
|
821
|
+
return textResponse(response + pipelineCtx + suggestedNext);
|
|
822
|
+
}
|
|
823
|
+
);
|
|
824
|
+
|
|
825
|
+
// -------------------------------------------------------------------------
|
|
826
|
+
// 6. intelligence - connections, grading, research (7 commands)
|
|
827
|
+
// -------------------------------------------------------------------------
|
|
828
|
+
server.tool(
|
|
829
|
+
'intelligence',
|
|
830
|
+
'Run intelligence operations: find connections, grade ventures, build thesis, research. Larry evaluates with calibrated assessment.',
|
|
831
|
+
{
|
|
832
|
+
command: z.enum(INTELLIGENCE_COMMANDS)
|
|
833
|
+
.describe('Which intelligence operation to run'),
|
|
834
|
+
context: z.string().optional()
|
|
835
|
+
.describe('Focus area or evaluation target'),
|
|
836
|
+
},
|
|
837
|
+
async ({ command, context }) => {
|
|
838
|
+
let extra = '';
|
|
839
|
+
if (command === 'grade' || command === 'deep-grade') {
|
|
840
|
+
const assessment = safeReadFile(path.join(pluginRoot, 'references', 'personality', 'assessment-philosophy.md'));
|
|
841
|
+
if (assessment) extra = `\n### Assessment Philosophy\n${assessment}`;
|
|
842
|
+
}
|
|
843
|
+
let suggestedNext;
|
|
844
|
+
if (command === 'grade') {
|
|
845
|
+
suggestedNext = formatSuggestedNext('intelligence', { command: 'deep-grade' }, 'Quick grade done - run deep grade for detailed component feedback');
|
|
846
|
+
} else if (command === 'deep-grade') {
|
|
847
|
+
suggestedNext = formatSuggestedNext('room_state', { command: 'analyze' }, 'Deep grade complete - analyze room to see grading impact on intelligence');
|
|
848
|
+
} else {
|
|
849
|
+
suggestedNext = formatSuggestedNext('room_state', { command: 'analyze' }, 'Intelligence operation complete - analyze room for new patterns');
|
|
850
|
+
}
|
|
851
|
+
return textResponse(buildContext(pluginRoot, roomDir, command, context) + extra + suggestedNext);
|
|
852
|
+
}
|
|
853
|
+
);
|
|
854
|
+
|
|
855
|
+
// -------------------------------------------------------------------------
|
|
856
|
+
// 7. meeting - meeting filing, pipeline, speakers (3 commands)
|
|
857
|
+
// -------------------------------------------------------------------------
|
|
858
|
+
server.tool(
|
|
859
|
+
'meeting',
|
|
860
|
+
'Meeting filing, intelligence pipeline, and speaker identification.',
|
|
861
|
+
{
|
|
862
|
+
command: z.enum(MEETING_COMMANDS)
|
|
863
|
+
.describe('Meeting operation'),
|
|
864
|
+
context: z.string().optional()
|
|
865
|
+
.describe('Meeting transcript, date, or context'),
|
|
866
|
+
},
|
|
867
|
+
async ({ command, context }) => {
|
|
868
|
+
const state = loadRoomState(roomDir);
|
|
869
|
+
|
|
870
|
+
if (command === 'file-meeting') {
|
|
871
|
+
const meetingRef = safeReadFile(path.join(pluginRoot, 'references', 'meeting', 'filing-protocol.md'));
|
|
872
|
+
const parts = ['## File Meeting'];
|
|
873
|
+
if (state) parts.push(`\n### Room State\n${state}`);
|
|
874
|
+
if (meetingRef) parts.push(`\n### Filing Protocol\n${meetingRef}`);
|
|
875
|
+
if (context) parts.push(`\n### Transcript/Context\n${context}`);
|
|
876
|
+
if (!context) parts.push('\n> Provide a meeting transcript to file.');
|
|
877
|
+
const response = parts.join('\n');
|
|
878
|
+
await fireCascade(roomDir, command, null, { filePath: '' });
|
|
879
|
+
return textResponse(response + formatSuggestedNext('room_state', { command: 'analyze' }, 'Meeting filed - analyze room for new intelligence from this conversation'));
|
|
880
|
+
}
|
|
881
|
+
|
|
882
|
+
if (command === 'pipeline') {
|
|
883
|
+
const pipelineRef = safeReadFile(path.join(pluginRoot, 'references', 'pipeline', 'index.md'))
|
|
884
|
+
|| safeReadFile(path.join(pluginRoot, 'commands', 'pipeline.md'));
|
|
885
|
+
const parts = ['## Pipeline'];
|
|
886
|
+
if (state) parts.push(`\n### Room State\n${state}`);
|
|
887
|
+
if (pipelineRef) parts.push(`\n### Reference\n${pipelineRef}`);
|
|
888
|
+
if (context) parts.push(`\n### Context\n${context}`);
|
|
889
|
+
return textResponse(parts.join('\n') + formatSuggestedNext('meeting', { command: 'file-meeting' }, 'Pipeline reviewed - file a meeting to feed the intelligence pipeline'));
|
|
890
|
+
}
|
|
891
|
+
|
|
892
|
+
if (command === 'speakers') {
|
|
893
|
+
const speakersRef = loadReference(pluginRoot, 'speakers');
|
|
894
|
+
const parts = ['## Speakers'];
|
|
895
|
+
if (state) parts.push(`\n### Room State\n${state}`);
|
|
896
|
+
if (speakersRef) parts.push(`\n### Reference\n${speakersRef}`);
|
|
897
|
+
if (context) parts.push(`\n### Context\n${context}`);
|
|
898
|
+
else parts.push('\n> Provide meeting text or speaker context.');
|
|
899
|
+
return textResponse(parts.join('\n') + formatSuggestedNext('meeting', { command: 'file-meeting' }, 'Speakers identified - file the meeting to route segments to sections'));
|
|
900
|
+
}
|
|
901
|
+
|
|
902
|
+
return textResponse(`Unknown meeting command: ${command}`, true);
|
|
903
|
+
}
|
|
904
|
+
);
|
|
905
|
+
|
|
906
|
+
// -------------------------------------------------------------------------
|
|
907
|
+
// 8. export - dashboards, wikis, presentations, snapshots (7 commands)
|
|
908
|
+
// -------------------------------------------------------------------------
|
|
909
|
+
server.tool(
|
|
910
|
+
'export',
|
|
911
|
+
'Generate dashboards, wikis, presentations, snapshots, and radar charts from room data.',
|
|
912
|
+
{
|
|
913
|
+
command: z.enum(EXPORT_COMMANDS)
|
|
914
|
+
.describe('Export operation'),
|
|
915
|
+
context: z.string().optional()
|
|
916
|
+
.describe('Export options or context'),
|
|
917
|
+
format: z.string().optional()
|
|
918
|
+
.describe('Export format (e.g., html, pdf, md)'),
|
|
919
|
+
},
|
|
920
|
+
async ({ command, context, format }) => {
|
|
921
|
+
const ref = loadReference(pluginRoot, command);
|
|
922
|
+
const state = loadRoomState(roomDir);
|
|
923
|
+
const parts = [`## ${command}`];
|
|
924
|
+
if (state) parts.push(`\n### Room State\n${state}`);
|
|
925
|
+
if (ref) parts.push(`\n### Reference\n${ref}`);
|
|
926
|
+
if (context) parts.push(`\n### Context\n${context}`);
|
|
927
|
+
if (format) parts.push(`\n### Format\n${format}`);
|
|
928
|
+
|
|
929
|
+
let suggestedNext;
|
|
930
|
+
if (command === 'snapshot') {
|
|
931
|
+
suggestedNext = formatSuggestedNext('export', { command: 'publish' }, 'Snapshot generated - publish to hosting for sharing');
|
|
932
|
+
} else if (command === 'dashboard') {
|
|
933
|
+
suggestedNext = formatSuggestedNext('export', { command: 'wiki' }, 'Dashboard generated - create wiki for detailed article navigation');
|
|
934
|
+
} else {
|
|
935
|
+
suggestedNext = formatSuggestedNext('room_state', { command: 'status' }, 'Export complete - check room status for overview');
|
|
936
|
+
}
|
|
937
|
+
return textResponse(parts.join('\n') + suggestedNext);
|
|
938
|
+
}
|
|
939
|
+
);
|
|
940
|
+
|
|
941
|
+
// -------------------------------------------------------------------------
|
|
942
|
+
// 9. orchestration - autonomous execution, rooms, scout, admin (20 commands)
|
|
943
|
+
// -------------------------------------------------------------------------
|
|
944
|
+
server.tool(
|
|
945
|
+
'orchestration',
|
|
946
|
+
'Autonomous execution, multi-room management, scout intelligence, and system administration.',
|
|
947
|
+
{
|
|
948
|
+
command: z.enum(ORCHESTRATION_COMMANDS)
|
|
949
|
+
.describe('Orchestration operation'),
|
|
950
|
+
context: z.string().optional()
|
|
951
|
+
.describe('Operation context or parameters'),
|
|
952
|
+
room: z.string().optional()
|
|
953
|
+
.describe('Room name or path for room operations'),
|
|
954
|
+
},
|
|
955
|
+
async ({ command, context, room }) => {
|
|
956
|
+
// Brain-driven act* commands use brain-router
|
|
957
|
+
if (command === 'act' || command === 'act-chain' || command === 'act-dry-run' || command === 'act-swarm') {
|
|
958
|
+
const brainRouter = require('./brain-router.cjs');
|
|
959
|
+
const rec = await brainRouter.recommend(roomDir, { intent: context || 'general' });
|
|
960
|
+
|
|
961
|
+
if (command === 'act') {
|
|
962
|
+
try { pipelineState.initChain(roomDir, rec.chain, rec.source || 'local'); } catch (_e) {}
|
|
963
|
+
const parts = [
|
|
964
|
+
`## Framework Recommendation`,
|
|
965
|
+
`\n**Chain:** ${rec.chain.join(' -> ')}`,
|
|
966
|
+
`**Confidence:** ${(rec.confidence * 100).toFixed(0)}%`,
|
|
967
|
+
`**Source:** ${rec.source}`,
|
|
968
|
+
`**Reasoning:** ${rec.reasoning}`,
|
|
969
|
+
];
|
|
970
|
+
const ref = loadReference(pluginRoot, rec.chain[0]);
|
|
971
|
+
if (ref) parts.push(`\n### First Framework Reference\n${ref}`);
|
|
972
|
+
return textResponse(parts.join('\n') + formatSuggestedNext('methodology', { command: rec.chain[0], context: rec.reasoning }, `Run ${rec.chain[0]} as recommended`));
|
|
973
|
+
}
|
|
974
|
+
if (command === 'act-chain') {
|
|
975
|
+
try { pipelineState.initChain(roomDir, rec.chain, rec.source || 'local'); } catch (_e) {}
|
|
976
|
+
const chainSteps = rec.chain.map((m, i) => `${i + 1}. **${m}**`).join('\n');
|
|
977
|
+
return textResponse(`## Framework Chain\n\n**Full Sequence:**\n${chainSteps}\n\n**Confidence:** ${(rec.confidence * 100).toFixed(0)}%\n**Source:** ${rec.source}\n**Reasoning:** ${rec.reasoning}` + formatSuggestedNext('methodology', { command: rec.chain[0] }, 'Start chain with first framework'));
|
|
978
|
+
}
|
|
979
|
+
if (command === 'act-dry-run') {
|
|
980
|
+
const validation = brainRouter.validateChain(roomDir, rec.chain);
|
|
981
|
+
return textResponse(`## Dry Run\n\n**Chain:** ${rec.chain.join(' -> ')}\n**Confidence:** ${(rec.confidence * 100).toFixed(0)}%\n**Source:** ${rec.source}\n**Valid:** ${validation.valid ? 'Yes' : 'No - ' + validation.reason}\n\n> Dry run. No frameworks executed.`);
|
|
982
|
+
}
|
|
983
|
+
if (command === 'act-swarm') {
|
|
984
|
+
try { pipelineState.initChain(roomDir, rec.chain, rec.source || 'local'); } catch (_e) {}
|
|
985
|
+
const ref = loadReference(pluginRoot, 'act');
|
|
986
|
+
const parts = [
|
|
987
|
+
`## Autonomous Swarm Execution`,
|
|
988
|
+
`\n**Chain:** ${rec.chain.join(' -> ')}`,
|
|
989
|
+
`**Confidence:** ${(rec.confidence * 100).toFixed(0)}%`,
|
|
990
|
+
`**Source:** ${rec.source}`,
|
|
991
|
+
`**Reasoning:** ${rec.reasoning}`,
|
|
992
|
+
`\n> Swarm mode: all frameworks in the chain will execute sequentially with automatic artifact filing.`,
|
|
993
|
+
];
|
|
994
|
+
if (ref) parts.push(`\n### Reference\n${ref}`);
|
|
995
|
+
return textResponse(parts.join('\n') + formatSuggestedNext('room_state', { command: 'status' }, 'Swarm complete - check room status'));
|
|
996
|
+
}
|
|
997
|
+
}
|
|
998
|
+
|
|
999
|
+
// All other orchestration commands use reference-based pattern
|
|
1000
|
+
const ref = loadReference(pluginRoot, command);
|
|
1001
|
+
const state = loadRoomState(roomDir);
|
|
1002
|
+
const parts = [`## ${command}`];
|
|
1003
|
+
if (state) parts.push(`\n### Room State\n${state}`);
|
|
1004
|
+
if (ref) parts.push(`\n### Reference\n${ref}`);
|
|
1005
|
+
if (context) parts.push(`\n### Context\n${context}`);
|
|
1006
|
+
if (room) parts.push(`\n### Target Room\n${room}`);
|
|
1007
|
+
if (!ref) parts.push(`\n> Reference file for "${command}" not found.`);
|
|
1008
|
+
|
|
1009
|
+
let suggestedNext;
|
|
1010
|
+
if (command.startsWith('rooms-')) {
|
|
1011
|
+
suggestedNext = formatSuggestedNext('room_state', { command: 'status' }, 'Room operation complete - check status');
|
|
1012
|
+
} else if (command.startsWith('scout')) {
|
|
1013
|
+
suggestedNext = formatSuggestedNext('room_state', { command: 'analyze' }, 'Scout intelligence gathered - analyze room');
|
|
1014
|
+
} else {
|
|
1015
|
+
suggestedNext = formatSuggestedNext('room_state', { command: 'status' }, 'Operation complete - check room status');
|
|
1016
|
+
}
|
|
1017
|
+
return textResponse(parts.join('\n') + suggestedNext);
|
|
1018
|
+
}
|
|
1019
|
+
);
|
|
1020
|
+
|
|
1021
|
+
// (Duplicate orchestration router from 52-03 merge removed -- brain routing now integrated above)
|
|
1022
|
+
}
|
|
1023
|
+
|
|
1024
|
+
module.exports = { registerRouterTools, ALL_TOOL_COMMANDS };
|
|
1025
|
+
|
|
1026
|
+
// 87-05: Export validation primitives for unit tests
|
|
1027
|
+
// (kept out of the registerRouterTools surface area).
|
|
1028
|
+
module.exports._test = {
|
|
1029
|
+
SECTION_RE,
|
|
1030
|
+
sectionOptional,
|
|
1031
|
+
sectionRequired,
|
|
1032
|
+
safeResolveSection,
|
|
1033
|
+
opportunitySchema,
|
|
1034
|
+
};
|