@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,865 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Brain HTTP Client — calls mindrian-brain.onrender.com
|
|
5
|
+
*
|
|
6
|
+
* Replaces direct MCP tool calls (mcp__neo4j-brain__*, mcp__pinecone-brain__*)
|
|
7
|
+
* with a single HTTP API that handles Neo4j + Pinecone behind one key.
|
|
8
|
+
*
|
|
9
|
+
* Falls back gracefully:
|
|
10
|
+
* 1. If MINDRIAN_BRAIN_KEY is set → calls Brain API
|
|
11
|
+
* 2. If Brain API returns Pinecone quota error → retries with Neo4j-only
|
|
12
|
+
* 3. If no key → returns null (Tier 0, no Brain)
|
|
13
|
+
*
|
|
14
|
+
* Usage in commands/skills:
|
|
15
|
+
* const brain = require('./brain-client.cjs');
|
|
16
|
+
* const result = await brain.query('MATCH (f:Framework) RETURN f.name LIMIT 5');
|
|
17
|
+
* const result = await brain.search('innovation framework');
|
|
18
|
+
* const schema = await brain.schema();
|
|
19
|
+
*/
|
|
20
|
+
|
|
21
|
+
const BRAIN_URL = process.env.MINDRIAN_BRAIN_URL || 'https://mindrian-brain.onrender.com';
|
|
22
|
+
|
|
23
|
+
// Phase 87-07 (CASCADE-06): Brain session cache with 5-minute TTL.
|
|
24
|
+
// Every callTool() previously re-ran the `initialize` handshake (~1 network
|
|
25
|
+
// round-trip). With a long-lived MCP server this is wasted work -- sessions
|
|
26
|
+
// live longer than the ~60s transport timeout. Cache the initialized
|
|
27
|
+
// sessionId (keyed by api-key-hash) for 5 minutes.
|
|
28
|
+
//
|
|
29
|
+
// R-87-07-RACE (audit): two concurrent callTool() invocations with the same
|
|
30
|
+
// api_key previously both saw a cache miss, both initialized, and the second
|
|
31
|
+
// overwrote the first -- one of the two initialize handshakes was wasted.
|
|
32
|
+
// Fix: cache the init *Promise*, not the resolved session. The first caller
|
|
33
|
+
// stores { promise: initSession(apiKey), expiresAt }; concurrent callers
|
|
34
|
+
// within the TTL `await entry.promise`. On rejection we remove the entry so
|
|
35
|
+
// the next caller re-initializes fresh.
|
|
36
|
+
//
|
|
37
|
+
// Hash: sha256 truncated to 16 hex chars (64 bits of key space, zero realistic
|
|
38
|
+
// collision). A cheaper non-crypto hash was considered but its narrower int
|
|
39
|
+
// space has non-zero collision probability once the design extends across
|
|
40
|
+
// users; sha256 is effectively free at these volumes and eliminates the
|
|
41
|
+
// concern entirely (R-87-07-RACE).
|
|
42
|
+
const crypto = require('node:crypto');
|
|
43
|
+
const SESSION_TTL_MS = 5 * 60 * 1000; // 5 minutes
|
|
44
|
+
/** @type {Map<string, {promise: Promise<string>, expiresAt: number}>} */
|
|
45
|
+
const sessionCache = new Map();
|
|
46
|
+
|
|
47
|
+
function _hashKey(key) {
|
|
48
|
+
return crypto.createHash('sha256').update(String(key)).digest('hex').slice(0, 16);
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* SEC-01: Sanitize any user-origin string before interpolation into a
|
|
53
|
+
* Cypher query. Whitelist from 87-CONTEXT.md lines 121-127:
|
|
54
|
+
* [a-zA-Z0-9 ._-]
|
|
55
|
+
* Every other char (including `"`, `'`, backtick, newline, `{`, `}`, `$`,
|
|
56
|
+
* `\`, `;`, `/`, `*`) is stripped. Null/undefined return ''. Non-strings
|
|
57
|
+
* are coerced via String() defensively so the caller never crashes.
|
|
58
|
+
*
|
|
59
|
+
* This replaces the legacy single-quote-escape pattern that only
|
|
60
|
+
* escaped one metacharacter (double-quote) and was trivially bypassable
|
|
61
|
+
* via backticks, newlines, `${...}` expansions, or Cypher comments.
|
|
62
|
+
*
|
|
63
|
+
* @param {*} value
|
|
64
|
+
* @returns {string}
|
|
65
|
+
*/
|
|
66
|
+
function sanitizeCypherInput(value) {
|
|
67
|
+
if (value === null || value === undefined) return '';
|
|
68
|
+
if (typeof value !== 'string') {
|
|
69
|
+
try { value = String(value); } catch (_e) { return ''; }
|
|
70
|
+
}
|
|
71
|
+
return value.replace(/[^a-zA-Z0-9 ._-]/g, '');
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
/**
|
|
75
|
+
* SEC-02: Refuse to load a Brain API key from a .env file whose permissions
|
|
76
|
+
* expose it to group or world readers. Unix semantics only -- on Windows
|
|
77
|
+
* POSIX mode bits are not meaningful for NTFS ACLs, so we return true and
|
|
78
|
+
* warn once per process.
|
|
79
|
+
*
|
|
80
|
+
* mode & 0o077 !== 0 => any group/world bit is set => reject
|
|
81
|
+
* mode 0o600 (-rw-------) and 0o400 (-r--------) pass; 0o644, 0o664 fail.
|
|
82
|
+
*
|
|
83
|
+
* On stat failure we return false (no key beats a key we cannot verify).
|
|
84
|
+
*
|
|
85
|
+
* @param {string} envPath
|
|
86
|
+
* @returns {boolean}
|
|
87
|
+
*/
|
|
88
|
+
function checkFilePermissions(envPath) {
|
|
89
|
+
try {
|
|
90
|
+
const fs = require('fs');
|
|
91
|
+
if (process.platform === 'win32') {
|
|
92
|
+
if (!checkFilePermissions._warned) {
|
|
93
|
+
process.stderr.write(
|
|
94
|
+
'[mindrian-os] Note: API key file permission check is Linux/macOS only; '
|
|
95
|
+
+ 'on Windows rely on NTFS ACLs.\n'
|
|
96
|
+
);
|
|
97
|
+
checkFilePermissions._warned = true;
|
|
98
|
+
}
|
|
99
|
+
return true;
|
|
100
|
+
}
|
|
101
|
+
const stat = fs.statSync(envPath);
|
|
102
|
+
if ((stat.mode & 0o077) !== 0) {
|
|
103
|
+
process.stderr.write(
|
|
104
|
+
`[mindrian-os] Refusing to load API key from ${envPath}: `
|
|
105
|
+
+ `permissions too open (must be 0600). chmod 600 ${envPath}\n`
|
|
106
|
+
);
|
|
107
|
+
return false;
|
|
108
|
+
}
|
|
109
|
+
return true;
|
|
110
|
+
} catch (_e) {
|
|
111
|
+
return false;
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
checkFilePermissions._warned = false;
|
|
115
|
+
|
|
116
|
+
/**
|
|
117
|
+
* Get the Brain API key from environment.
|
|
118
|
+
* Checks: MINDRIAN_BRAIN_KEY, then falls back to reading .env in CWD.
|
|
119
|
+
* Every .env candidate path is gated by checkFilePermissions (SEC-02).
|
|
120
|
+
*/
|
|
121
|
+
function getApiKey() {
|
|
122
|
+
if (process.env.MINDRIAN_BRAIN_KEY) {
|
|
123
|
+
return process.env.MINDRIAN_BRAIN_KEY;
|
|
124
|
+
}
|
|
125
|
+
// Try reading .env from CWD
|
|
126
|
+
try {
|
|
127
|
+
const fs = require('fs');
|
|
128
|
+
const path = require('path');
|
|
129
|
+
const envPath = path.join(process.cwd(), '.env');
|
|
130
|
+
if (fs.existsSync(envPath) && checkFilePermissions(envPath)) {
|
|
131
|
+
const content = fs.readFileSync(envPath, 'utf8');
|
|
132
|
+
const match = content.match(/MINDRIAN_BRAIN_KEY=(.+)/);
|
|
133
|
+
if (match) return match[1].trim();
|
|
134
|
+
}
|
|
135
|
+
} catch (e) {}
|
|
136
|
+
// Fallback: try reading ~/.mindrian.env (global backup)
|
|
137
|
+
try {
|
|
138
|
+
const fs = require('fs');
|
|
139
|
+
const path = require('path');
|
|
140
|
+
const globalEnvPath = path.join(require('os').homedir(), '.mindrian.env');
|
|
141
|
+
if (fs.existsSync(globalEnvPath) && checkFilePermissions(globalEnvPath)) {
|
|
142
|
+
const content = fs.readFileSync(globalEnvPath, 'utf8');
|
|
143
|
+
const match = content.match(/MINDRIAN_BRAIN_KEY=(.+)/);
|
|
144
|
+
if (match) return match[1].trim();
|
|
145
|
+
}
|
|
146
|
+
} catch (e) {}
|
|
147
|
+
return null;
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
/**
|
|
151
|
+
* Check if Brain is available (key exists).
|
|
152
|
+
*/
|
|
153
|
+
function isAvailable() {
|
|
154
|
+
return !!getApiKey();
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
/**
|
|
158
|
+
* Phase 87-07: ensure we have a valid initialized Brain session for the given
|
|
159
|
+
* api key, reusing the cached one if non-expired. Uses the pending-promise
|
|
160
|
+
* pattern so concurrent callers share a single in-flight init (R-87-07-RACE).
|
|
161
|
+
*
|
|
162
|
+
* Returns the resolved session marker (an opaque string -- the Brain Streamable
|
|
163
|
+
* HTTP transport does not require us to echo a sessionId on subsequent requests
|
|
164
|
+
* inside the same cache window, but awaiting this promise proves the key is
|
|
165
|
+
* valid against the Brain endpoint exactly once per TTL window).
|
|
166
|
+
*
|
|
167
|
+
* On any init rejection (network error, 401, etc.) the cache entry is removed
|
|
168
|
+
* in the .catch() tail so the next caller retries fresh rather than inheriting
|
|
169
|
+
* a poisoned promise.
|
|
170
|
+
*
|
|
171
|
+
* Sentinel `{ error: 'invalid_key' }` is returned *through* the promise (not
|
|
172
|
+
* thrown) so callers treat 401 identically to the pre-cache flow.
|
|
173
|
+
*
|
|
174
|
+
* @param {string} apiKey
|
|
175
|
+
* @returns {Promise<string|{error:string,message:string}|null>}
|
|
176
|
+
*/
|
|
177
|
+
async function _ensureSession(apiKey) {
|
|
178
|
+
const keyHash = _hashKey(apiKey);
|
|
179
|
+
const cached = sessionCache.get(keyHash);
|
|
180
|
+
if (cached && cached.expiresAt > Date.now()) {
|
|
181
|
+
// Cache hit. Works whether the promise is still pending (concurrent init
|
|
182
|
+
// in flight) or already resolved (TTL reuse). Awaiting a resolved promise
|
|
183
|
+
// is a microtask no-op, so the fast path stays fast.
|
|
184
|
+
return cached.promise;
|
|
185
|
+
}
|
|
186
|
+
// Cache miss. Build the promise FIRST, install it in the cache BEFORE the
|
|
187
|
+
// first real await, so concurrent callers within the same event-loop tick
|
|
188
|
+
// see the same in-flight promise (R-87-07-RACE pending-promise pattern).
|
|
189
|
+
const promise = (async () => {
|
|
190
|
+
const initRes = await fetch(`${BRAIN_URL}/mcp`, {
|
|
191
|
+
method: 'POST',
|
|
192
|
+
headers: {
|
|
193
|
+
'Content-Type': 'application/json',
|
|
194
|
+
'Accept': 'application/json, text/event-stream',
|
|
195
|
+
'Authorization': `Bearer ${apiKey}`,
|
|
196
|
+
},
|
|
197
|
+
body: JSON.stringify({
|
|
198
|
+
jsonrpc: '2.0',
|
|
199
|
+
id: 1,
|
|
200
|
+
method: 'initialize',
|
|
201
|
+
params: {
|
|
202
|
+
protocolVersion: '2024-11-05',
|
|
203
|
+
capabilities: {},
|
|
204
|
+
clientInfo: { name: 'mindrian-cli', version: '1.0.0' },
|
|
205
|
+
},
|
|
206
|
+
}),
|
|
207
|
+
});
|
|
208
|
+
if (!initRes.ok) {
|
|
209
|
+
if (initRes.status === 401) {
|
|
210
|
+
return { error: 'invalid_key', message: 'Brain API key is invalid.' };
|
|
211
|
+
}
|
|
212
|
+
// Any other non-OK status becomes a throw so the cache entry is purged
|
|
213
|
+
// by the .catch() below and the next caller retries.
|
|
214
|
+
throw new Error(`Brain init HTTP ${initRes.status}`);
|
|
215
|
+
}
|
|
216
|
+
// Opaque session marker. Subsequent tools/call requests don't need to
|
|
217
|
+
// echo this back -- the transport is stateless at the HTTP level. What
|
|
218
|
+
// matters is that we validated the key is live within this TTL window.
|
|
219
|
+
return 'validated-' + Date.now();
|
|
220
|
+
})();
|
|
221
|
+
sessionCache.set(keyHash, { promise, expiresAt: Date.now() + SESSION_TTL_MS });
|
|
222
|
+
// On reject, purge the entry so the next caller initializes fresh. Swallow
|
|
223
|
+
// here (we re-throw in the awaiter below) so Node doesn't see an
|
|
224
|
+
// unhandledRejection on the cache handle itself.
|
|
225
|
+
promise.catch(() => { sessionCache.delete(keyHash); });
|
|
226
|
+
return promise;
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
/**
|
|
230
|
+
* Call a Brain MCP tool via HTTP.
|
|
231
|
+
* @param {string} toolName - e.g., 'brain_query', 'brain_search', 'brain_schema'
|
|
232
|
+
* @param {object} args - tool arguments
|
|
233
|
+
* @returns {object|null} - result or null if unavailable
|
|
234
|
+
*/
|
|
235
|
+
async function callTool(toolName, args) {
|
|
236
|
+
const key = getApiKey();
|
|
237
|
+
if (!key) return null;
|
|
238
|
+
|
|
239
|
+
try {
|
|
240
|
+
// Phase 87-07: reuse cached Brain session (5-min TTL) instead of
|
|
241
|
+
// re-running initialize on every callTool. Concurrent callers share
|
|
242
|
+
// the in-flight promise via the pending-promise pattern.
|
|
243
|
+
const session = await _ensureSession(key);
|
|
244
|
+
if (session && typeof session === 'object' && session.error === 'invalid_key') {
|
|
245
|
+
return session;
|
|
246
|
+
}
|
|
247
|
+
if (!session) return null;
|
|
248
|
+
|
|
249
|
+
// Call the tool
|
|
250
|
+
const toolRes = await fetch(`${BRAIN_URL}/mcp`, {
|
|
251
|
+
method: 'POST',
|
|
252
|
+
headers: {
|
|
253
|
+
'Content-Type': 'application/json',
|
|
254
|
+
'Accept': 'application/json, text/event-stream',
|
|
255
|
+
'Authorization': `Bearer ${key}`,
|
|
256
|
+
},
|
|
257
|
+
body: JSON.stringify({
|
|
258
|
+
jsonrpc: '2.0',
|
|
259
|
+
id: 2,
|
|
260
|
+
method: 'tools/call',
|
|
261
|
+
params: { name: toolName, arguments: args },
|
|
262
|
+
}),
|
|
263
|
+
});
|
|
264
|
+
|
|
265
|
+
if (!toolRes.ok) return null;
|
|
266
|
+
|
|
267
|
+
const text = await toolRes.text();
|
|
268
|
+
// Parse SSE response
|
|
269
|
+
const dataLine = text.split('\n').find(l => l.startsWith('data: '));
|
|
270
|
+
if (!dataLine) return null;
|
|
271
|
+
|
|
272
|
+
const parsed = JSON.parse(dataLine.slice(6));
|
|
273
|
+
if (parsed.result && parsed.result.content) {
|
|
274
|
+
const textContent = parsed.result.content.find(c => c.type === 'text');
|
|
275
|
+
if (textContent) {
|
|
276
|
+
try {
|
|
277
|
+
return JSON.parse(textContent.text);
|
|
278
|
+
} catch (e) {
|
|
279
|
+
return { text: textContent.text };
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
return parsed.result || null;
|
|
284
|
+
} catch (err) {
|
|
285
|
+
// Network error, timeout, etc.
|
|
286
|
+
return null;
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
/**
|
|
291
|
+
* Query Neo4j via Brain (Cypher query).
|
|
292
|
+
* This does NOT use Pinecone, no embedding quota consumed.
|
|
293
|
+
*
|
|
294
|
+
* NOTE (Finding I, v1.10.9 hotfix 2026-04-15): the Brain MCP brain_query
|
|
295
|
+
* tool expects the parameter name `cypher`, not `query`. Previously this
|
|
296
|
+
* function sent { query: cypher } which tripped an MCP input validation
|
|
297
|
+
* error (code -32602, path ["cypher"], "Required"). Downstream scripts
|
|
298
|
+
* like fetch-brain-baseline.cjs and compute-whitespace-gaps.py then
|
|
299
|
+
* silently fell through to empty-baseline mode even though Brain was
|
|
300
|
+
* fully reachable and the key was valid. Witnessed against the live
|
|
301
|
+
* iia-deeptech-centers room on 2026-04-15. brain_search uses `query`
|
|
302
|
+
* which is why Pinecone semantic search kept working and masked this.
|
|
303
|
+
*
|
|
304
|
+
* NOTE (2026-05-11, graph-on-graph P0): `query` now accepts an optional
|
|
305
|
+
* second argument `params` and forwards it to the `brain_query` MCP tool
|
|
306
|
+
* as { cypher, params }. The Brain tool declares `params:
|
|
307
|
+
* z.record(z.any()).optional()`, so a parameterized Cypher gets its
|
|
308
|
+
* bindings through cleanly. `params` MUST be a generic-handles-only object
|
|
309
|
+
* — framework names, phase identifiers, problem types per Canon Part 8 —
|
|
310
|
+
* NEVER user content (artifact bodies, meeting text, personal identifiers,
|
|
311
|
+
* proprietary numbers). Previously the second arg was silently dropped, so
|
|
312
|
+
* callers (rs-explain-command.cjs, rs-thesis-command.cjs, rs-nl-to-query)
|
|
313
|
+
* that generated parameterized Cypher had their bindings disappear or were
|
|
314
|
+
* pushed toward unsafe string interpolation. A param-less call still sends
|
|
315
|
+
* only { cypher } and behaves exactly as before.
|
|
316
|
+
*
|
|
317
|
+
* NOTE (2026-05-11, graph-on-graph P0 cont.): RESULT-SHAPE NORMALIZATION.
|
|
318
|
+
* The Brain MCP `brain_query` tool serializes its result as
|
|
319
|
+
* `JSON.stringify(records)` where `records` is a BARE ARRAY of row objects.
|
|
320
|
+
* `callTool` returns that array directly (or `{ text: 'Error: ...' }` on a
|
|
321
|
+
* Cypher error, or `null` when the Brain is unreachable / no API key).
|
|
322
|
+
* Consumers across the codebase — brain-router.cjs, brain-derivation.cjs's
|
|
323
|
+
* `renderRecords`, rs-chain-feeder.cjs, rs-experts-command.cjs,
|
|
324
|
+
* rs-explain-command.cjs, rs-thesis-command.cjs — all read `result.records`,
|
|
325
|
+
* so the bare-array shape silently dropped every row. `query` therefore now
|
|
326
|
+
* ALWAYS returns `{ records: [...] }` on a successful brain_query; an
|
|
327
|
+
* unreachable Brain / missing key still returns `null`; a Cypher-error
|
|
328
|
+
* response (`{ text: 'Error: ...' }` or `{ error: ... }`) passes through
|
|
329
|
+
* unchanged so callers that inspect the failure can still see it; any other
|
|
330
|
+
* unexpected shape collapses to `{ records: [] }` so callers never crash.
|
|
331
|
+
* `search`, `smartSearch`, `schema`, `stats`, `write`, `callTool` are
|
|
332
|
+
* deliberately untouched — only `query` is normalized.
|
|
333
|
+
*/
|
|
334
|
+
async function query(cypher, params) {
|
|
335
|
+
const args = { cypher: cypher };
|
|
336
|
+
if (params && typeof params === 'object' && Object.keys(params).length > 0) {
|
|
337
|
+
args.params = params;
|
|
338
|
+
}
|
|
339
|
+
const result = await callTool('brain_query', args);
|
|
340
|
+
if (result == null) return null; // unreachable / no API key
|
|
341
|
+
if (Array.isArray(result)) return { records: result }; // the normal brain_query shape
|
|
342
|
+
if (result && Array.isArray(result.records)) return result; // already normalized (defensive)
|
|
343
|
+
if (result && (result.error || result.text)) return result; // error / message passthrough
|
|
344
|
+
return { records: [] }; // unexpected shape -> empty, never crash
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
/**
|
|
348
|
+
* Search Pinecone via Brain (semantic search).
|
|
349
|
+
* If quota exhausted, returns error with fallback suggestion.
|
|
350
|
+
*/
|
|
351
|
+
async function search(queryText, options = {}) {
|
|
352
|
+
const result = await callTool('brain_search', {
|
|
353
|
+
query: queryText,
|
|
354
|
+
namespace: options.namespace || undefined,
|
|
355
|
+
topK: options.topK || 5,
|
|
356
|
+
});
|
|
357
|
+
|
|
358
|
+
// Check for Pinecone quota exhaustion
|
|
359
|
+
if (result && result.text && result.text.includes('RESOURCE_EXHAUSTED')) {
|
|
360
|
+
return {
|
|
361
|
+
error: 'pinecone_quota_exhausted',
|
|
362
|
+
message: 'Pinecone embedding quota exhausted for this month. Using Neo4j Cypher fallback.',
|
|
363
|
+
fallback: 'neo4j',
|
|
364
|
+
};
|
|
365
|
+
}
|
|
366
|
+
|
|
367
|
+
return result;
|
|
368
|
+
}
|
|
369
|
+
|
|
370
|
+
/**
|
|
371
|
+
* Search with automatic fallback: Pinecone first, Neo4j Cypher if quota exhausted.
|
|
372
|
+
*/
|
|
373
|
+
async function smartSearch(queryText, options = {}) {
|
|
374
|
+
// Try Pinecone first
|
|
375
|
+
const pineconeResult = await search(queryText, options);
|
|
376
|
+
|
|
377
|
+
if (pineconeResult && pineconeResult.error === 'pinecone_quota_exhausted') {
|
|
378
|
+
// Fallback to Neo4j full-text search
|
|
379
|
+
const cypher = `
|
|
380
|
+
CALL db.index.fulltext.queryNodes("framework_search", $query)
|
|
381
|
+
YIELD node, score
|
|
382
|
+
RETURN node.name AS name, node.description AS description, score
|
|
383
|
+
LIMIT ${options.topK || 5}
|
|
384
|
+
`;
|
|
385
|
+
const neo4jResult = await query(cypher.replace('$query', `"${sanitizeCypherInput(queryText)}"`));
|
|
386
|
+
if (neo4jResult) {
|
|
387
|
+
neo4jResult._source = 'neo4j_fallback';
|
|
388
|
+
neo4jResult._note = 'Pinecone quota exhausted. Results from Neo4j Cypher fulltext search.';
|
|
389
|
+
}
|
|
390
|
+
return neo4jResult;
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
return pineconeResult;
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
/**
|
|
397
|
+
* Get Neo4j schema.
|
|
398
|
+
*/
|
|
399
|
+
async function schema() {
|
|
400
|
+
return callTool('brain_schema', {});
|
|
401
|
+
}
|
|
402
|
+
|
|
403
|
+
/**
|
|
404
|
+
* Get Pinecone stats.
|
|
405
|
+
*/
|
|
406
|
+
async function stats() {
|
|
407
|
+
return callTool('brain_stats', {});
|
|
408
|
+
}
|
|
409
|
+
|
|
410
|
+
/**
|
|
411
|
+
* Enrich local graph with causal edges from Brain's teaching graph.
|
|
412
|
+
*
|
|
413
|
+
* Queries the Brain Neo4j for causal framework chains relevant to the
|
|
414
|
+
* given problem type or section keywords. Returns structured causal data
|
|
415
|
+
* suitable for writing to local SQLite graph as CAUSES/ROOT_CAUSE_OF edges.
|
|
416
|
+
*
|
|
417
|
+
* @param {string} problemType - Room problem type (e.g., 'market-validation')
|
|
418
|
+
* @param {string[]} sectionKeywords - Keywords from room sections for context
|
|
419
|
+
* @param {object} [options] - Optional config
|
|
420
|
+
* @param {number} [options.maxChainDepth=3] - Maximum causal chain depth
|
|
421
|
+
* @param {number} [options.minConfidence=0.5] - Minimum confidence threshold
|
|
422
|
+
* @returns {Promise<{ causes: Array, rootCauses: Array } | null>}
|
|
423
|
+
* causes: [{ from, to, mechanism, confidence, framework }]
|
|
424
|
+
* rootCauses: [{ from, to, chainLength, intermediateCauses, confidence }]
|
|
425
|
+
*/
|
|
426
|
+
async function enrichCausalEdges(problemType, sectionKeywords, options = {}) {
|
|
427
|
+
if (!isAvailable()) return null;
|
|
428
|
+
|
|
429
|
+
// SEC-01 defence-in-depth: coerce + bound numeric interpolants so a hostile
|
|
430
|
+
// non-number (e.g. an object with .toString() side-effects) cannot reach
|
|
431
|
+
// the Cypher string.
|
|
432
|
+
const maxDepth = Math.max(1, Math.min(10, Number(options.maxChainDepth) || 3));
|
|
433
|
+
const minConf = Math.max(0, Math.min(1, Number(options.minConfidence) || 0.5));
|
|
434
|
+
const keywordFilter = sectionKeywords && sectionKeywords.length > 0
|
|
435
|
+
? sectionKeywords.map(k => `"${sanitizeCypherInput(k)}"`).join(', ')
|
|
436
|
+
: '';
|
|
437
|
+
|
|
438
|
+
// Query 1: Direct causal relationships from framework chains
|
|
439
|
+
const causesCypher = `
|
|
440
|
+
MATCH (f1:Framework)-[r:ADDRESSES_PROBLEM_TYPE]->(pt:ProblemType)
|
|
441
|
+
WHERE pt.name CONTAINS "${sanitizeCypherInput(problemType || '')}"
|
|
442
|
+
WITH f1
|
|
443
|
+
MATCH (f1)-[co:CO_OCCURS]->(f2:Framework)
|
|
444
|
+
WHERE co.weight >= ${minConf}
|
|
445
|
+
RETURN f1.name AS cause_framework,
|
|
446
|
+
f2.name AS effect_framework,
|
|
447
|
+
co.weight AS confidence,
|
|
448
|
+
f1.description AS mechanism
|
|
449
|
+
LIMIT 20
|
|
450
|
+
`;
|
|
451
|
+
|
|
452
|
+
// Query 2: Root cause chains (multi-hop framework dependencies)
|
|
453
|
+
const rootCauseCypher = `
|
|
454
|
+
MATCH path = (root:Framework)-[:CO_OCCURS*1..${maxDepth}]->(leaf:Framework)
|
|
455
|
+
WHERE root <> leaf
|
|
456
|
+
${keywordFilter ? `AND ANY(k IN [${keywordFilter}] WHERE root.name CONTAINS k OR root.description CONTAINS k)` : ''}
|
|
457
|
+
WITH root, leaf, path, length(path) AS depth
|
|
458
|
+
WHERE depth >= 2
|
|
459
|
+
RETURN root.name AS root_cause,
|
|
460
|
+
leaf.name AS symptom,
|
|
461
|
+
depth AS chain_length,
|
|
462
|
+
[n IN nodes(path) | n.name] AS chain_nodes
|
|
463
|
+
LIMIT 10
|
|
464
|
+
`;
|
|
465
|
+
|
|
466
|
+
try {
|
|
467
|
+
const [causesResult, rootCausesResult] = await Promise.all([
|
|
468
|
+
query(causesCypher),
|
|
469
|
+
query(rootCauseCypher),
|
|
470
|
+
]);
|
|
471
|
+
|
|
472
|
+
const causes = [];
|
|
473
|
+
const rootCauses = [];
|
|
474
|
+
|
|
475
|
+
// Parse causes
|
|
476
|
+
if (causesResult && Array.isArray(causesResult.records)) {
|
|
477
|
+
for (const rec of causesResult.records) {
|
|
478
|
+
causes.push({
|
|
479
|
+
from: rec.cause_framework || rec[0],
|
|
480
|
+
to: rec.effect_framework || rec[1],
|
|
481
|
+
mechanism: rec.mechanism || rec[3] || '',
|
|
482
|
+
confidence: parseFloat(rec.confidence || rec[2] || 0),
|
|
483
|
+
framework: rec.cause_framework || rec[0] || '',
|
|
484
|
+
});
|
|
485
|
+
}
|
|
486
|
+
}
|
|
487
|
+
|
|
488
|
+
// Parse root causes
|
|
489
|
+
if (rootCausesResult && Array.isArray(rootCausesResult.records)) {
|
|
490
|
+
for (const rec of rootCausesResult.records) {
|
|
491
|
+
rootCauses.push({
|
|
492
|
+
from: rec.root_cause || rec[0],
|
|
493
|
+
to: rec.symptom || rec[1],
|
|
494
|
+
chainLength: parseInt(rec.chain_length || rec[2] || 1, 10),
|
|
495
|
+
intermediateCauses: rec.chain_nodes || rec[3] || [],
|
|
496
|
+
confidence: 1.0 / (parseInt(rec.chain_length || rec[2] || 1, 10) + 1),
|
|
497
|
+
});
|
|
498
|
+
}
|
|
499
|
+
}
|
|
500
|
+
|
|
501
|
+
return { causes, rootCauses };
|
|
502
|
+
} catch (err) {
|
|
503
|
+
// Brain query failed -- return null for graceful degradation
|
|
504
|
+
return null;
|
|
505
|
+
}
|
|
506
|
+
}
|
|
507
|
+
|
|
508
|
+
/**
|
|
509
|
+
* Hat-aware framework recommendation.
|
|
510
|
+
*
|
|
511
|
+
* Reads persistent hat states and adjusts Brain framework queries:
|
|
512
|
+
* - Black Hat concerns boost risk-related frameworks (Risk Matrix, SWOT threats)
|
|
513
|
+
* - Yellow Hat opportunities boost HSI scoring and opportunity frameworks
|
|
514
|
+
* - Blue Hat methodology notes avoid repeating ineffective frameworks
|
|
515
|
+
*
|
|
516
|
+
* @param {string} roomDir - Absolute path to room directory
|
|
517
|
+
* @param {string} problemType - Room problem type
|
|
518
|
+
* @param {object} [options] - Optional config
|
|
519
|
+
* @param {number} [options.topK=5] - Number of frameworks to return
|
|
520
|
+
* @returns {Promise<{ frameworks: Array, hat_influence: object } | null>}
|
|
521
|
+
*/
|
|
522
|
+
async function hatAwareRecommend(roomDir, problemType, options = {}) {
|
|
523
|
+
if (!isAvailable()) return null;
|
|
524
|
+
|
|
525
|
+
// Lazy-require to avoid circular dependency at module load time
|
|
526
|
+
const { loadAllHatStates } = require('./hat-persistence.cjs');
|
|
527
|
+
const hatStates = loadAllHatStates(roomDir);
|
|
528
|
+
// SEC-01 defence-in-depth: bound topK numeric interpolation.
|
|
529
|
+
const topK = Math.max(1, Math.min(100, Number(options.topK) || 5));
|
|
530
|
+
|
|
531
|
+
const hatInfluence = {
|
|
532
|
+
risk_boost: false,
|
|
533
|
+
opportunity_boost: false,
|
|
534
|
+
avoid_frameworks: [],
|
|
535
|
+
};
|
|
536
|
+
|
|
537
|
+
// Black Hat: if concerns exist, boost risk-related frameworks
|
|
538
|
+
const blackConcerns = hatStates.black.top_concerns || [];
|
|
539
|
+
const riskBoost = blackConcerns.length > 0;
|
|
540
|
+
hatInfluence.risk_boost = riskBoost;
|
|
541
|
+
|
|
542
|
+
// Yellow Hat: if opportunities exist, boost HSI/opportunity frameworks
|
|
543
|
+
const yellowOpps = hatStates.yellow.top_opportunities || [];
|
|
544
|
+
const oppBoost = yellowOpps.length > 0;
|
|
545
|
+
hatInfluence.opportunity_boost = oppBoost;
|
|
546
|
+
|
|
547
|
+
// Blue Hat: methodology notes may flag ineffective frameworks to avoid
|
|
548
|
+
const blueNotes = hatStates.blue.methodology_notes || [];
|
|
549
|
+
const avoidPatterns = blueNotes
|
|
550
|
+
.filter(n => /ineffective|didn't work|not useful|skip|avoid/i.test(n))
|
|
551
|
+
.map(n => {
|
|
552
|
+
// Extract framework name from notes like "SWOT was ineffective for this stage"
|
|
553
|
+
const match = n.match(/^(\w[\w\s]+?)\s+(?:was|is|were|proved)\s/i);
|
|
554
|
+
return match ? match[1].trim() : null;
|
|
555
|
+
})
|
|
556
|
+
.filter(Boolean);
|
|
557
|
+
hatInfluence.avoid_frameworks = avoidPatterns;
|
|
558
|
+
|
|
559
|
+
// Build Cypher query with hat-influenced scoring
|
|
560
|
+
const safeProblemType = sanitizeCypherInput(problemType || '');
|
|
561
|
+
const avoidClause = avoidPatterns.length > 0
|
|
562
|
+
? `AND NOT ANY(avoid IN [${avoidPatterns.map(a => `"${sanitizeCypherInput(a)}"`).join(', ')}] WHERE f.name CONTAINS avoid)`
|
|
563
|
+
: '';
|
|
564
|
+
|
|
565
|
+
// Query: frameworks for problem type, with hat-influenced ordering
|
|
566
|
+
const cypher = `
|
|
567
|
+
MATCH (f:Framework)-[:ADDRESSES_PROBLEM_TYPE]->(pt:ProblemType)
|
|
568
|
+
WHERE pt.name CONTAINS "${safeProblemType}"
|
|
569
|
+
${avoidClause}
|
|
570
|
+
WITH f
|
|
571
|
+
OPTIONAL MATCH (f)-[co:CO_OCCURS]->(f2:Framework)
|
|
572
|
+
WITH f, count(co) AS connections
|
|
573
|
+
RETURN f.name AS name,
|
|
574
|
+
f.description AS description,
|
|
575
|
+
connections,
|
|
576
|
+
CASE
|
|
577
|
+
WHEN ${riskBoost ? 'true' : 'false'} AND (f.name CONTAINS 'Risk' OR f.name CONTAINS 'SWOT' OR f.name CONTAINS 'Failure') THEN connections + 10
|
|
578
|
+
WHEN ${oppBoost ? 'true' : 'false'} AND (f.name CONTAINS 'HSI' OR f.name CONTAINS 'Opportunity' OR f.name CONTAINS 'Innovation') THEN connections + 10
|
|
579
|
+
ELSE connections
|
|
580
|
+
END AS hat_score
|
|
581
|
+
ORDER BY hat_score DESC
|
|
582
|
+
LIMIT ${topK}
|
|
583
|
+
`;
|
|
584
|
+
|
|
585
|
+
try {
|
|
586
|
+
const result = await query(cypher);
|
|
587
|
+
const frameworks = [];
|
|
588
|
+
|
|
589
|
+
if (result && Array.isArray(result.records)) {
|
|
590
|
+
for (const rec of result.records) {
|
|
591
|
+
frameworks.push({
|
|
592
|
+
name: rec.name || rec[0],
|
|
593
|
+
description: rec.description || rec[1],
|
|
594
|
+
connections: parseInt(rec.connections || rec[2] || 0, 10),
|
|
595
|
+
hat_score: parseInt(rec.hat_score || rec[3] || 0, 10),
|
|
596
|
+
});
|
|
597
|
+
}
|
|
598
|
+
}
|
|
599
|
+
|
|
600
|
+
return {
|
|
601
|
+
frameworks,
|
|
602
|
+
hat_influence: hatInfluence,
|
|
603
|
+
black_concerns: blackConcerns.slice(0, 3),
|
|
604
|
+
yellow_opportunities: yellowOpps.slice(0, 3),
|
|
605
|
+
blue_avoid: avoidPatterns,
|
|
606
|
+
};
|
|
607
|
+
} catch (err) {
|
|
608
|
+
return null;
|
|
609
|
+
}
|
|
610
|
+
}
|
|
611
|
+
|
|
612
|
+
/**
|
|
613
|
+
* Suggest validation steps for a banked opportunity using Brain framework chains.
|
|
614
|
+
*
|
|
615
|
+
* Queries Brain Neo4j for frameworks that ADDRESSES_PROBLEM_TYPE matching the
|
|
616
|
+
* opportunity's domain/problem, then follows FEEDS_INTO chains to build a
|
|
617
|
+
* suggested validation sequence.
|
|
618
|
+
*
|
|
619
|
+
* @param {Object} opportunity - Opportunity object with at minimum: problem, domain, knight_position
|
|
620
|
+
* @param {Object} [options] - Optional config
|
|
621
|
+
* @param {number} [options.maxSteps=5] - Maximum validation steps to return
|
|
622
|
+
* @param {number} [options.chainDepth=3] - Maximum FEEDS_INTO chain depth
|
|
623
|
+
* @returns {Promise<{ steps: Array<{framework: string, reason: string, order: number}>, chain_source: string } | null>}
|
|
624
|
+
* Returns null if Brain unavailable (Tier 0 graceful degradation)
|
|
625
|
+
*/
|
|
626
|
+
async function suggestValidationSteps(opportunity, options = {}) {
|
|
627
|
+
if (!isAvailable()) return null;
|
|
628
|
+
if (!opportunity || !opportunity.problem) return null;
|
|
629
|
+
|
|
630
|
+
const maxSteps = options.maxSteps || 5;
|
|
631
|
+
const chainDepth = options.chainDepth || 3;
|
|
632
|
+
const safeProblem = sanitizeCypherInput(opportunity.problem || '').substring(0, 200);
|
|
633
|
+
const safeDomain = sanitizeCypherInput(opportunity.domain || '').substring(0, 100);
|
|
634
|
+
const safeKnight = opportunity.knight_position || 'uncertainty';
|
|
635
|
+
|
|
636
|
+
// Query 1: Find frameworks that address this problem type / domain
|
|
637
|
+
const matchCypher = `
|
|
638
|
+
MATCH (f:Framework)-[:ADDRESSES_PROBLEM_TYPE]->(pt:ProblemType)
|
|
639
|
+
WHERE pt.name CONTAINS "${safeDomain}"
|
|
640
|
+
OR pt.description CONTAINS "${safeDomain}"
|
|
641
|
+
RETURN f.name AS name, f.description AS description
|
|
642
|
+
LIMIT 10
|
|
643
|
+
`;
|
|
644
|
+
|
|
645
|
+
// Query 2: Follow FEEDS_INTO chains from matched frameworks
|
|
646
|
+
const chainCypher = `
|
|
647
|
+
MATCH (f:Framework)-[:ADDRESSES_PROBLEM_TYPE]->(pt:ProblemType)
|
|
648
|
+
WHERE pt.name CONTAINS "${safeDomain}"
|
|
649
|
+
OR pt.description CONTAINS "${safeDomain}"
|
|
650
|
+
WITH f LIMIT 3
|
|
651
|
+
MATCH path = (f)-[:FEEDS_INTO*1..${chainDepth}]->(next:Framework)
|
|
652
|
+
RETURN f.name AS start_framework,
|
|
653
|
+
next.name AS next_framework,
|
|
654
|
+
next.description AS next_description,
|
|
655
|
+
length(path) AS depth
|
|
656
|
+
ORDER BY depth ASC
|
|
657
|
+
LIMIT ${maxSteps * 2}
|
|
658
|
+
`;
|
|
659
|
+
|
|
660
|
+
try {
|
|
661
|
+
const [matchResult, chainResult] = await Promise.all([
|
|
662
|
+
query(matchCypher),
|
|
663
|
+
query(chainCypher),
|
|
664
|
+
]);
|
|
665
|
+
|
|
666
|
+
const steps = [];
|
|
667
|
+
const seen = new Set();
|
|
668
|
+
|
|
669
|
+
// First: add the entry-point frameworks
|
|
670
|
+
if (matchResult && Array.isArray(matchResult.records)) {
|
|
671
|
+
for (const rec of matchResult.records) {
|
|
672
|
+
const name = rec.name || rec[0];
|
|
673
|
+
if (name && !seen.has(name) && steps.length < maxSteps) {
|
|
674
|
+
seen.add(name);
|
|
675
|
+
steps.push({
|
|
676
|
+
framework: name,
|
|
677
|
+
reason: safeKnight === 'uncertainty'
|
|
678
|
+
? `Explore this ${safeDomain} uncertainty with ${name}`
|
|
679
|
+
: `Validate this ${safeDomain} risk using ${name}`,
|
|
680
|
+
order: steps.length + 1,
|
|
681
|
+
});
|
|
682
|
+
}
|
|
683
|
+
}
|
|
684
|
+
}
|
|
685
|
+
|
|
686
|
+
// Then: add FEEDS_INTO chain steps
|
|
687
|
+
if (chainResult && Array.isArray(chainResult.records)) {
|
|
688
|
+
for (const rec of chainResult.records) {
|
|
689
|
+
const name = rec.next_framework || rec[1];
|
|
690
|
+
const desc = rec.next_description || rec[2] || '';
|
|
691
|
+
if (name && !seen.has(name) && steps.length < maxSteps) {
|
|
692
|
+
seen.add(name);
|
|
693
|
+
steps.push({
|
|
694
|
+
framework: name,
|
|
695
|
+
reason: desc ? `Then apply ${name}: ${desc.substring(0, 120)}` : `Then apply ${name} (follows from chain)`,
|
|
696
|
+
order: steps.length + 1,
|
|
697
|
+
});
|
|
698
|
+
}
|
|
699
|
+
}
|
|
700
|
+
}
|
|
701
|
+
|
|
702
|
+
if (steps.length === 0) return null;
|
|
703
|
+
|
|
704
|
+
return {
|
|
705
|
+
steps,
|
|
706
|
+
chain_source: 'brain_feeds_into',
|
|
707
|
+
};
|
|
708
|
+
} catch (err) {
|
|
709
|
+
// Brain query failed -- graceful degradation
|
|
710
|
+
return null;
|
|
711
|
+
}
|
|
712
|
+
}
|
|
713
|
+
|
|
714
|
+
/**
|
|
715
|
+
* Write Cypher to Neo4j via Brain (write operations).
|
|
716
|
+
* Used by sync-rooms-brain for creating Room/RoomGroup nodes and edges.
|
|
717
|
+
* Returns null if Brain is unavailable -- never throws.
|
|
718
|
+
*
|
|
719
|
+
* NOTE (Finding I sibling, v1.10.9 hotfix 2026-04-15): same param-name
|
|
720
|
+
* mismatch as brain_query had. Brain MCP brain_write expects `cypher`,
|
|
721
|
+
* not `query`. This function had the mirror bug since inception but
|
|
722
|
+
* never fired in production because sync-rooms-brain is rarely invoked
|
|
723
|
+
* against the live Brain. Caught by the plan-checker audit for Phase 85.
|
|
724
|
+
*
|
|
725
|
+
* @param {string} cypher - Cypher write query
|
|
726
|
+
* @returns {Promise<object|null>}
|
|
727
|
+
*/
|
|
728
|
+
async function write(cypher) {
|
|
729
|
+
return callTool('brain_write', { cypher: cypher });
|
|
730
|
+
}
|
|
731
|
+
|
|
732
|
+
/**
|
|
733
|
+
* Tier 0 fallback: hardcoded persona framework chains.
|
|
734
|
+
* Used when Brain is unavailable or query returns no results.
|
|
735
|
+
*
|
|
736
|
+
* @param {string} persona - 'tto', 'researcher', or 'business'
|
|
737
|
+
* @returns {{ persona: string, chain: Array<{framework: string, description: string, order: number}>, source: 'tier0' }}
|
|
738
|
+
*/
|
|
739
|
+
function getTier0Chain(persona) {
|
|
740
|
+
const chains = {
|
|
741
|
+
tto: [
|
|
742
|
+
{ framework: 'Domain Exploration', description: 'What domains could this technology touch?', order: 1 },
|
|
743
|
+
{ framework: 'Problem Definition', description: 'What specific problems does it solve?', order: 2 },
|
|
744
|
+
{ framework: 'JTBD Analysis', description: 'Who needs this solved and what progress do they want?', order: 3 },
|
|
745
|
+
{ framework: 'Value Proposition', description: 'What is the value and how to deliver it?', order: 4 },
|
|
746
|
+
],
|
|
747
|
+
researcher: [
|
|
748
|
+
{ framework: 'Problem Exploration', description: 'What problem does the research address?', order: 1 },
|
|
749
|
+
{ framework: 'JTBD Analysis', description: 'Who cares about this problem?', order: 2 },
|
|
750
|
+
{ framework: 'Value Proposition', description: 'What would a solution look like?', order: 3 },
|
|
751
|
+
{ framework: 'Lean Canvas', description: 'How to deliver and sustain this?', order: 4 },
|
|
752
|
+
],
|
|
753
|
+
business: [
|
|
754
|
+
{ framework: 'Opportunity Recognition', description: 'What opportunity exists in the market?', order: 1 },
|
|
755
|
+
{ framework: 'Market Analysis', description: 'How big is this market?', order: 2 },
|
|
756
|
+
{ framework: 'Problem Definition', description: 'What specific problem for whom?', order: 3 },
|
|
757
|
+
{ framework: 'Competitive Analysis', description: 'Who else is trying and what is your edge?', order: 4 },
|
|
758
|
+
],
|
|
759
|
+
};
|
|
760
|
+
|
|
761
|
+
return {
|
|
762
|
+
persona: persona,
|
|
763
|
+
chain: chains[persona] || chains.researcher,
|
|
764
|
+
source: 'tier0',
|
|
765
|
+
};
|
|
766
|
+
}
|
|
767
|
+
|
|
768
|
+
/**
|
|
769
|
+
* Get ordered framework chain for a persona.
|
|
770
|
+
*
|
|
771
|
+
* Brain-connected: queries FEEDS_INTO edges for dynamic chains.
|
|
772
|
+
* Brain-unavailable: returns Tier 0 hardcoded chains from getTier0Chain().
|
|
773
|
+
*
|
|
774
|
+
* Per CONV-02: Persona-aware chain routing.
|
|
775
|
+
* Per CONV-03: Brain framework chain selection with Tier 0 fallback.
|
|
776
|
+
*
|
|
777
|
+
* @param {string} persona - 'tto', 'researcher', or 'business' (case-insensitive)
|
|
778
|
+
* @returns {Promise<{ persona: string, chain: Array<{framework: string, description: string, order: number}>, source: 'brain'|'tier0' }>}
|
|
779
|
+
*/
|
|
780
|
+
async function getFrameworkChain(persona) {
|
|
781
|
+
const personaLower = (persona || '').toLowerCase();
|
|
782
|
+
|
|
783
|
+
// Persona-to-entry-framework mapping
|
|
784
|
+
const entryMap = {
|
|
785
|
+
tto: 'Domain Exploration',
|
|
786
|
+
researcher: 'Problem Exploration',
|
|
787
|
+
business: 'Opportunity Recognition',
|
|
788
|
+
};
|
|
789
|
+
|
|
790
|
+
const entryFramework = entryMap[personaLower];
|
|
791
|
+
if (!entryFramework) {
|
|
792
|
+
// Unknown persona - fall back to Tier 0 default (researcher chain)
|
|
793
|
+
return getTier0Chain('researcher');
|
|
794
|
+
}
|
|
795
|
+
|
|
796
|
+
// Try Brain first
|
|
797
|
+
if (isAvailable()) {
|
|
798
|
+
try {
|
|
799
|
+
const safeEntry = sanitizeCypherInput(entryFramework);
|
|
800
|
+
const cypher = `
|
|
801
|
+
MATCH path = (start:Framework)-[:FEEDS_INTO*1..4]->(next:Framework)
|
|
802
|
+
WHERE start.name CONTAINS "${safeEntry}"
|
|
803
|
+
RETURN start.name AS start_name,
|
|
804
|
+
[n IN nodes(path) | n.name] AS chain,
|
|
805
|
+
[n IN nodes(path) | n.description] AS descriptions,
|
|
806
|
+
length(path) AS depth
|
|
807
|
+
ORDER BY depth ASC
|
|
808
|
+
LIMIT 5
|
|
809
|
+
`;
|
|
810
|
+
const result = await query(cypher);
|
|
811
|
+
|
|
812
|
+
if (result && Array.isArray(result.records) && result.records.length > 0) {
|
|
813
|
+
// Build ordered chain from longest path
|
|
814
|
+
const longestPath = result.records[result.records.length - 1];
|
|
815
|
+
const chain = longestPath.chain || longestPath[1] || [];
|
|
816
|
+
const descriptions = longestPath.descriptions || longestPath[2] || [];
|
|
817
|
+
|
|
818
|
+
if (chain.length > 0) {
|
|
819
|
+
return {
|
|
820
|
+
persona: personaLower,
|
|
821
|
+
chain: chain.map((name, i) => ({
|
|
822
|
+
framework: name,
|
|
823
|
+
description: descriptions[i] || '',
|
|
824
|
+
order: i + 1,
|
|
825
|
+
})),
|
|
826
|
+
source: 'brain',
|
|
827
|
+
};
|
|
828
|
+
}
|
|
829
|
+
}
|
|
830
|
+
} catch (err) {
|
|
831
|
+
// Brain query failed - fall through to Tier 0
|
|
832
|
+
}
|
|
833
|
+
}
|
|
834
|
+
|
|
835
|
+
// Tier 0 fallback
|
|
836
|
+
return getTier0Chain(personaLower);
|
|
837
|
+
}
|
|
838
|
+
|
|
839
|
+
module.exports = {
|
|
840
|
+
isAvailable,
|
|
841
|
+
getApiKey,
|
|
842
|
+
callTool,
|
|
843
|
+
query,
|
|
844
|
+
write,
|
|
845
|
+
search,
|
|
846
|
+
smartSearch,
|
|
847
|
+
schema,
|
|
848
|
+
stats,
|
|
849
|
+
enrichCausalEdges,
|
|
850
|
+
hatAwareRecommend,
|
|
851
|
+
suggestValidationSteps,
|
|
852
|
+
getFrameworkChain,
|
|
853
|
+
// SEC-01/SEC-02 + CASCADE-06 test surface: not part of the public API.
|
|
854
|
+
// See lib/memory/security-trifecta.test.cjs + brain-cache-lru.test.cjs.
|
|
855
|
+
// Helpers are small and pure. sessionCache + _ensureSession + _hashKey +
|
|
856
|
+
// SESSION_TTL_MS are exposed for Phase 87-07 cache-behavior tests.
|
|
857
|
+
_test: {
|
|
858
|
+
sanitizeCypherInput,
|
|
859
|
+
checkFilePermissions,
|
|
860
|
+
sessionCache,
|
|
861
|
+
SESSION_TTL_MS,
|
|
862
|
+
_hashKey,
|
|
863
|
+
_ensureSession,
|
|
864
|
+
},
|
|
865
|
+
};
|