@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,148 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MindrianOS Plugin -- Artifact ID & Pipeline Provenance
|
|
3
|
+
* Stable artifact ID computation and frontmatter injection.
|
|
4
|
+
* IDs are deterministic 12-char hex hashes from room+section+title+created.
|
|
5
|
+
*
|
|
6
|
+
* Exports: computeArtifactId, injectArtifactId, injectPipelineProvenance
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
'use strict';
|
|
10
|
+
|
|
11
|
+
const fs = require('fs');
|
|
12
|
+
const path = require('path');
|
|
13
|
+
const crypto = require('crypto');
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Compute a stable artifact ID from room, section, title, and creation date.
|
|
17
|
+
* @param {string} roomDir - Absolute path to room directory
|
|
18
|
+
* @param {string} section - Section name (e.g. "problem-definition")
|
|
19
|
+
* @param {string} title - Artifact title (first # heading or filename)
|
|
20
|
+
* @param {string} created - Creation date string (e.g. "2026-03-15")
|
|
21
|
+
* @returns {string} 12-char hex hash
|
|
22
|
+
*/
|
|
23
|
+
function computeArtifactId(roomDir, section, title, created) {
|
|
24
|
+
const roomName = path.basename(roomDir);
|
|
25
|
+
const input = roomName + ':' + section + ':' + title + ':' + created;
|
|
26
|
+
return crypto.createHash('sha256').update(input).digest('hex').slice(0, 12);
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Extract the first # heading from file content.
|
|
31
|
+
* @param {string} content - File content
|
|
32
|
+
* @param {string} filePath - Fallback basename
|
|
33
|
+
* @returns {string}
|
|
34
|
+
*/
|
|
35
|
+
function extractTitle(content, filePath) {
|
|
36
|
+
const match = content.match(/^# (.+)$/m);
|
|
37
|
+
return match ? match[1].trim() : path.basename(filePath, '.md');
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* Extract a frontmatter field value from content.
|
|
42
|
+
* @param {string} content - File content
|
|
43
|
+
* @param {string} field - Field name
|
|
44
|
+
* @returns {string}
|
|
45
|
+
*/
|
|
46
|
+
function extractField(content, field) {
|
|
47
|
+
const fmMatch = content.match(/^---\n([\s\S]*?)\n---/);
|
|
48
|
+
if (!fmMatch) return '';
|
|
49
|
+
const line = fmMatch[1].split('\n').find(l => l.startsWith(field + ':'));
|
|
50
|
+
if (!line) return '';
|
|
51
|
+
return line.slice(field.length + 1).trim().replace(/^["']|["']$/g, '');
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Inject artifact_id into frontmatter of a markdown file. Idempotent.
|
|
56
|
+
* @param {string} filePath - Absolute path to .md file
|
|
57
|
+
* @param {string} roomDir - Absolute path to room directory
|
|
58
|
+
* @returns {{artifact_id: string, injected: boolean} | null} null if no frontmatter or already has ID
|
|
59
|
+
*/
|
|
60
|
+
function injectArtifactId(filePath, roomDir) {
|
|
61
|
+
if (!fs.existsSync(filePath)) return null;
|
|
62
|
+
|
|
63
|
+
const content = fs.readFileSync(filePath, 'utf-8');
|
|
64
|
+
|
|
65
|
+
// Must have frontmatter
|
|
66
|
+
if (!content.startsWith('---\n')) return null;
|
|
67
|
+
|
|
68
|
+
// Already has artifact_id -- idempotent
|
|
69
|
+
const fmEnd = content.indexOf('\n---', 4);
|
|
70
|
+
if (fmEnd === -1) return null;
|
|
71
|
+
const frontmatter = content.slice(4, fmEnd);
|
|
72
|
+
if (frontmatter.includes('artifact_id:')) return null;
|
|
73
|
+
|
|
74
|
+
// Extract metadata
|
|
75
|
+
const section = path.basename(path.dirname(filePath));
|
|
76
|
+
const title = extractTitle(content, filePath);
|
|
77
|
+
const created = extractField(content, 'date') || extractField(content, 'created') || '';
|
|
78
|
+
|
|
79
|
+
const artifactId = computeArtifactId(roomDir, section, title, created);
|
|
80
|
+
|
|
81
|
+
// Insert artifact_id line after the first ---
|
|
82
|
+
const lines = content.split('\n');
|
|
83
|
+
// lines[0] is '---', insert after it
|
|
84
|
+
lines.splice(1, 0, `artifact_id: ${artifactId}`);
|
|
85
|
+
fs.writeFileSync(filePath, lines.join('\n'), 'utf-8');
|
|
86
|
+
|
|
87
|
+
return { artifact_id: artifactId, injected: true };
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
/**
|
|
91
|
+
* Inject pipeline provenance fields into frontmatter. Idempotent.
|
|
92
|
+
* @param {string} filePath - Absolute path to .md file
|
|
93
|
+
* @param {string} pipeline - Pipeline name (e.g. "meeting-filing")
|
|
94
|
+
* @param {string} stage - Pipeline stage (e.g. "3")
|
|
95
|
+
* @param {string[]} requires - What this stage requires
|
|
96
|
+
* @param {string[]} provides - What this stage provides
|
|
97
|
+
* @returns {{injected: boolean} | null}
|
|
98
|
+
*/
|
|
99
|
+
function injectPipelineProvenance(filePath, pipeline, stage, requires, provides) {
|
|
100
|
+
if (!fs.existsSync(filePath)) return null;
|
|
101
|
+
|
|
102
|
+
const content = fs.readFileSync(filePath, 'utf-8');
|
|
103
|
+
|
|
104
|
+
// Must have frontmatter
|
|
105
|
+
if (!content.startsWith('---\n')) return null;
|
|
106
|
+
|
|
107
|
+
// Already has pipeline -- idempotent
|
|
108
|
+
const fmEnd = content.indexOf('\n---', 4);
|
|
109
|
+
if (fmEnd === -1) return null;
|
|
110
|
+
const frontmatter = content.slice(4, fmEnd);
|
|
111
|
+
if (frontmatter.includes('pipeline:')) return null;
|
|
112
|
+
|
|
113
|
+
// Find insertion point: after artifact_id if present, otherwise after first ---
|
|
114
|
+
const lines = content.split('\n');
|
|
115
|
+
let insertIdx = 1; // default: right after first ---
|
|
116
|
+
for (let i = 1; i < lines.length; i++) {
|
|
117
|
+
if (lines[i] === '---') break;
|
|
118
|
+
if (lines[i].startsWith('artifact_id:')) {
|
|
119
|
+
insertIdx = i + 1;
|
|
120
|
+
break;
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
const reqYaml = requires && requires.length > 0
|
|
125
|
+
? '[' + requires.join(', ') + ']'
|
|
126
|
+
: '[]';
|
|
127
|
+
const provYaml = provides && provides.length > 0
|
|
128
|
+
? '[' + provides.join(', ') + ']'
|
|
129
|
+
: '[]';
|
|
130
|
+
|
|
131
|
+
const provenanceLines = [
|
|
132
|
+
`pipeline: ${pipeline}`,
|
|
133
|
+
`pipeline_stage: ${stage}`,
|
|
134
|
+
`pipeline_requires: ${reqYaml}`,
|
|
135
|
+
`pipeline_provides: ${provYaml}`,
|
|
136
|
+
];
|
|
137
|
+
|
|
138
|
+
lines.splice(insertIdx, 0, ...provenanceLines);
|
|
139
|
+
fs.writeFileSync(filePath, lines.join('\n'), 'utf-8');
|
|
140
|
+
|
|
141
|
+
return { injected: true };
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
module.exports = {
|
|
145
|
+
computeArtifactId,
|
|
146
|
+
injectArtifactId,
|
|
147
|
+
injectPipelineProvenance,
|
|
148
|
+
};
|
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MindrianOS Plugin -- Asset Operations
|
|
3
|
+
* Wraps scripts/file-asset via execSync. Pure Node.js built-ins only.
|
|
4
|
+
* Provides manifest reading for dashboard/wiki consumption.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
'use strict';
|
|
8
|
+
|
|
9
|
+
const { execSync } = require('child_process');
|
|
10
|
+
const fs = require('fs');
|
|
11
|
+
const path = require('path');
|
|
12
|
+
|
|
13
|
+
const SCRIPTS_DIR = path.resolve(__dirname, '../../scripts');
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* File a binary asset into a room with markdown wrapper + manifest update.
|
|
17
|
+
* @param {string} roomDir - Path to room directory
|
|
18
|
+
* @param {string} filePath - Path to the binary file to file
|
|
19
|
+
* @param {string} section - Target section name
|
|
20
|
+
* @param {object} [options] - Optional parameters
|
|
21
|
+
* @param {string} [options.meetingId] - Meeting ID for meeting-mode filing
|
|
22
|
+
* @returns {string} Output from file-asset script (e.g. "FILED:section:type:filename")
|
|
23
|
+
*/
|
|
24
|
+
function fileAsset(roomDir, filePath, section, options) {
|
|
25
|
+
const resolved = path.resolve(roomDir);
|
|
26
|
+
const resolvedFile = path.resolve(filePath);
|
|
27
|
+
const scriptPath = path.join(SCRIPTS_DIR, 'file-asset');
|
|
28
|
+
|
|
29
|
+
let cmd = `bash "${scriptPath}" "${resolved}" "${resolvedFile}" "${section}"`;
|
|
30
|
+
if (options && options.meetingId) {
|
|
31
|
+
cmd += ` --meeting "${options.meetingId}"`;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
try {
|
|
35
|
+
const result = execSync(cmd, {
|
|
36
|
+
timeout: 10000,
|
|
37
|
+
encoding: 'utf-8',
|
|
38
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
39
|
+
});
|
|
40
|
+
return result.trim();
|
|
41
|
+
} catch (e) {
|
|
42
|
+
throw new Error(`file-asset failed: ${e.message}`);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Read and parse ASSET_MANIFEST.md from room root.
|
|
48
|
+
* Returns array of asset objects. Empty array if no manifest exists.
|
|
49
|
+
* @param {string} roomDir - Path to room directory
|
|
50
|
+
* @returns {Array<{file: string, type: string, section: string, size: string, date: string, path: string}>}
|
|
51
|
+
*/
|
|
52
|
+
function readManifest(roomDir) {
|
|
53
|
+
const resolved = path.resolve(roomDir);
|
|
54
|
+
const manifestPath = path.join(resolved, 'ASSET_MANIFEST.md');
|
|
55
|
+
|
|
56
|
+
if (!fs.existsSync(manifestPath)) {
|
|
57
|
+
return [];
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
const content = fs.readFileSync(manifestPath, 'utf-8');
|
|
61
|
+
const lines = content.split('\n');
|
|
62
|
+
const assets = [];
|
|
63
|
+
|
|
64
|
+
// Skip header lines: title, empty, blockquote, empty, table header, separator
|
|
65
|
+
// Find the table by looking for lines starting with |
|
|
66
|
+
let inTable = false;
|
|
67
|
+
let headerSkipped = 0;
|
|
68
|
+
|
|
69
|
+
for (const line of lines) {
|
|
70
|
+
const trimmed = line.trim();
|
|
71
|
+
if (!trimmed.startsWith('|')) {
|
|
72
|
+
inTable = false;
|
|
73
|
+
headerSkipped = 0;
|
|
74
|
+
continue;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
if (!inTable) {
|
|
78
|
+
inTable = true;
|
|
79
|
+
headerSkipped = 0;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
headerSkipped++;
|
|
83
|
+
// Skip table header row and separator row
|
|
84
|
+
if (headerSkipped <= 2) continue;
|
|
85
|
+
|
|
86
|
+
// Parse pipe-delimited columns: | File | Type | Section | Size | Date |
|
|
87
|
+
const cols = trimmed.split('|').map(c => c.trim()).filter(c => c.length > 0);
|
|
88
|
+
if (cols.length < 5) continue;
|
|
89
|
+
|
|
90
|
+
// Extract link target from [text](path) pattern in File column
|
|
91
|
+
const fileCol = cols[0];
|
|
92
|
+
const linkMatch = fileCol.match(/\[([^\]]+)\]\(([^)]+)\)/);
|
|
93
|
+
const fileName = linkMatch ? linkMatch[1] : fileCol;
|
|
94
|
+
const filePath = linkMatch ? linkMatch[2] : '';
|
|
95
|
+
|
|
96
|
+
assets.push({
|
|
97
|
+
file: fileName,
|
|
98
|
+
type: cols[1],
|
|
99
|
+
section: cols[2],
|
|
100
|
+
size: cols[3],
|
|
101
|
+
date: cols[4],
|
|
102
|
+
path: filePath,
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
return assets;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
/**
|
|
110
|
+
* Regenerate ASSET_MANIFEST.md without filing a new asset.
|
|
111
|
+
* Useful for dashboard refresh.
|
|
112
|
+
* @param {string} roomDir - Path to room directory
|
|
113
|
+
* @returns {string} Output from file-asset --manifest-only
|
|
114
|
+
*/
|
|
115
|
+
function updateManifest(roomDir) {
|
|
116
|
+
const resolved = path.resolve(roomDir);
|
|
117
|
+
const scriptPath = path.join(SCRIPTS_DIR, 'file-asset');
|
|
118
|
+
|
|
119
|
+
try {
|
|
120
|
+
const result = execSync(`bash "${scriptPath}" "${resolved}" --manifest-only`, {
|
|
121
|
+
timeout: 10000,
|
|
122
|
+
encoding: 'utf-8',
|
|
123
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
124
|
+
});
|
|
125
|
+
return result.trim();
|
|
126
|
+
} catch (e) {
|
|
127
|
+
throw new Error(`file-asset --manifest-only failed: ${e.message}`);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Get assets filtered by section.
|
|
133
|
+
* @param {string} roomDir - Path to room directory
|
|
134
|
+
* @param {string} section - Section name to filter by
|
|
135
|
+
* @returns {Array<{file: string, type: string, section: string, size: string, date: string, path: string}>}
|
|
136
|
+
*/
|
|
137
|
+
function getAssetsBySection(roomDir, section) {
|
|
138
|
+
return readManifest(roomDir).filter(a => a.section === section);
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
/**
|
|
142
|
+
* Get assets filtered by type (pdf, image, video, audio, document, archive).
|
|
143
|
+
* @param {string} roomDir - Path to room directory
|
|
144
|
+
* @param {string} type - Asset type to filter by
|
|
145
|
+
* @returns {Array<{file: string, type: string, section: string, size: string, date: string, path: string}>}
|
|
146
|
+
*/
|
|
147
|
+
function getAssetsByType(roomDir, type) {
|
|
148
|
+
return readManifest(roomDir).filter(a => a.type === type);
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
module.exports = { fileAsset, readManifest, updateManifest, getAssetsBySection, getAssetsByType };
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2026 Mindrian. BSL 1.1.
|
|
3
|
+
*
|
|
4
|
+
* Phase 88.1-08 -- async artifact auto-commit throttle
|
|
5
|
+
* ======================================================
|
|
6
|
+
* Pure throttle bookkeeping for the async artifact auto-commit hook. The
|
|
7
|
+
* hook fires PostToolUse on Write|Edit|MultiEdit and must avoid commit
|
|
8
|
+
* storms under rapid-write sequences (MultiEdit, tool chains). This module
|
|
9
|
+
* is the decision layer: given a file path, a wall-clock timestamp, and a
|
|
10
|
+
* ledger of per-path last-commit timestamps, it answers:
|
|
11
|
+
* - should this write be throttled (drop the auto-commit this time)?
|
|
12
|
+
* - what is the ledger after recording a fresh commit for this path?
|
|
13
|
+
* - which stale entries can be pruned to keep the ledger bounded?
|
|
14
|
+
*
|
|
15
|
+
* Pure CJS, zero fs imports, zero network. The hook script wraps this
|
|
16
|
+
* module with openSync('wx') + rename atomic writes (Phase 87-02 pattern)
|
|
17
|
+
* for ledger persistence. This file is LOCAL-by-construction per Canon
|
|
18
|
+
* Part 8: it has no I/O surface at all.
|
|
19
|
+
*
|
|
20
|
+
* Canon references:
|
|
21
|
+
* Part 4 Every Choice Is Graph Data -- auto-commits preserve the
|
|
22
|
+
* provenance trail of every artifact write (filesystem IS the graph).
|
|
23
|
+
* Part 6 Product-as-Venture -- our own rooms get auto-commits too.
|
|
24
|
+
* Part 8 Graph Boundary -- auto-commit is LOCAL; this pure module has no
|
|
25
|
+
* exfiltration surface.
|
|
26
|
+
*
|
|
27
|
+
* API:
|
|
28
|
+
* THROTTLE_WINDOW_MS -- number, fixed at 5000ms.
|
|
29
|
+
* shouldThrottle(filePath, now, ledger) -> boolean
|
|
30
|
+
* updateLedger(filePath, now, ledger) -> new ledger object
|
|
31
|
+
* pruneOldEntries(ledger, now) -> new ledger object with
|
|
32
|
+
* entries older than 1 day
|
|
33
|
+
* dropped.
|
|
34
|
+
*
|
|
35
|
+
* Ledger shape:
|
|
36
|
+
* { [absolutePath:string]: lastCommitTsMs:number }
|
|
37
|
+
*/
|
|
38
|
+
|
|
39
|
+
'use strict';
|
|
40
|
+
|
|
41
|
+
// ---------- Tunables (frozen invariants) ----------
|
|
42
|
+
|
|
43
|
+
// 1 commit / 5 seconds per path. Matches PLAN spec and prevents commit
|
|
44
|
+
// storms during rapid-write sequences (MultiEdit, chained tools).
|
|
45
|
+
const THROTTLE_WINDOW_MS = 5000;
|
|
46
|
+
|
|
47
|
+
// 1 day retention on ledger entries. Anything older is irrelevant because
|
|
48
|
+
// the 5s throttle window is vastly smaller; old entries only exist because
|
|
49
|
+
// the hook has not re-fired recently. Pruning keeps the ledger bounded.
|
|
50
|
+
const LEDGER_RETENTION_MS = 24 * 60 * 60 * 1000; // 86400000
|
|
51
|
+
|
|
52
|
+
// ---------- Core functions ----------
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* shouldThrottle
|
|
56
|
+
* Decide whether an auto-commit for filePath at wall-clock now should be
|
|
57
|
+
* suppressed because a prior commit landed inside the throttle window.
|
|
58
|
+
*
|
|
59
|
+
* Semantics:
|
|
60
|
+
* - No ledger entry for filePath -> not throttled.
|
|
61
|
+
* - now - lastTs < WINDOW_MS -> throttled (drop the commit).
|
|
62
|
+
* - now - lastTs >= WINDOW_MS -> not throttled (allow commit).
|
|
63
|
+
*
|
|
64
|
+
* Boundary behavior: exactly WINDOW_MS old is allowed (matches the
|
|
65
|
+
* user-visible contract "1 commit per 5 seconds": the 5-second tick
|
|
66
|
+
* unlocks the next commit).
|
|
67
|
+
*/
|
|
68
|
+
function shouldThrottle(filePath, now, ledger) {
|
|
69
|
+
if (!ledger || typeof ledger !== 'object') return false;
|
|
70
|
+
const lastTs = ledger[filePath];
|
|
71
|
+
if (typeof lastTs !== 'number') return false;
|
|
72
|
+
const age = now - lastTs;
|
|
73
|
+
if (!Number.isFinite(age)) return false;
|
|
74
|
+
return age < THROTTLE_WINDOW_MS;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* updateLedger
|
|
79
|
+
* Return a NEW ledger object with filePath's timestamp set to now. The
|
|
80
|
+
* input ledger is never mutated (pure function). Pre-existing entries
|
|
81
|
+
* for other paths are preserved.
|
|
82
|
+
*/
|
|
83
|
+
function updateLedger(filePath, now, ledger) {
|
|
84
|
+
const base = (ledger && typeof ledger === 'object') ? ledger : {};
|
|
85
|
+
// Object spread yields a fresh object; assignment to filePath sets the
|
|
86
|
+
// new timestamp without mutating the input.
|
|
87
|
+
const next = Object.assign({}, base);
|
|
88
|
+
next[filePath] = now;
|
|
89
|
+
return next;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* pruneOldEntries
|
|
94
|
+
* Return a NEW ledger object containing only entries with timestamps
|
|
95
|
+
* within LEDGER_RETENTION_MS of now. Entries exactly at the boundary
|
|
96
|
+
* (now - ts === LEDGER_RETENTION_MS) are dropped -- this matches the
|
|
97
|
+
* "older than 1 day" wording in the PLAN: strictly older is pruned.
|
|
98
|
+
*
|
|
99
|
+
* Non-number or non-finite timestamps are treated as stale and pruned
|
|
100
|
+
* defensively so a corrupt ledger entry cannot survive forever.
|
|
101
|
+
*/
|
|
102
|
+
function pruneOldEntries(ledger, now) {
|
|
103
|
+
const out = {};
|
|
104
|
+
if (!ledger || typeof ledger !== 'object') return out;
|
|
105
|
+
const keys = Object.keys(ledger);
|
|
106
|
+
for (let i = 0; i < keys.length; i += 1) {
|
|
107
|
+
const k = keys[i];
|
|
108
|
+
const ts = ledger[k];
|
|
109
|
+
if (typeof ts !== 'number' || !Number.isFinite(ts)) continue;
|
|
110
|
+
const age = now - ts;
|
|
111
|
+
// Keep entries strictly newer than the retention window.
|
|
112
|
+
// age < LEDGER_RETENTION_MS -> keep.
|
|
113
|
+
// age >= LEDGER_RETENTION_MS -> prune.
|
|
114
|
+
if (age < LEDGER_RETENTION_MS) {
|
|
115
|
+
out[k] = ts;
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
return out;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// ---------- Exports ----------
|
|
122
|
+
|
|
123
|
+
module.exports = {
|
|
124
|
+
THROTTLE_WINDOW_MS: THROTTLE_WINDOW_MS,
|
|
125
|
+
LEDGER_RETENTION_MS: LEDGER_RETENTION_MS,
|
|
126
|
+
shouldThrottle: shouldThrottle,
|
|
127
|
+
updateLedger: updateLedger,
|
|
128
|
+
pruneOldEntries: pruneOldEntries,
|
|
129
|
+
};
|
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* bearer-token.cjs -- In-memory Bearer token + CSRF token session store for
|
|
5
|
+
* the live dashboard's BYO-API chat panel (Phase 87-09).
|
|
6
|
+
*
|
|
7
|
+
* BSL 1.1. Copyright (c) Mindrian 2026.
|
|
8
|
+
* (Business Source License 1.1; SPDX BUSL-1.1; see LICENSE.)
|
|
9
|
+
*
|
|
10
|
+
* Purpose
|
|
11
|
+
* Browser POSTs its Anthropic api_key to /api/auth/session ONCE. Server
|
|
12
|
+
* generates a 64-hex Bearer token + 32-hex CSRF token, binds both to the
|
|
13
|
+
* request's Origin, holds the api_key in memory keyed by the token for a
|
|
14
|
+
* 30-minute TTL, returns {token, expires_in, csrf_token} to the browser,
|
|
15
|
+
* and sets a Set-Cookie: mos_csrf=<csrf> for the OWASP double-submit
|
|
16
|
+
* cookie pattern. Subsequent /api/room/chat calls send
|
|
17
|
+
* `Authorization: Bearer <token>` + `X-CSRF-Token: <csrf>` (echoed from
|
|
18
|
+
* the mos_csrf cookie). The api_key NEVER flows through a request body
|
|
19
|
+
* after auth.
|
|
20
|
+
*
|
|
21
|
+
* Security model (R-87-09-CSRF):
|
|
22
|
+
* Gap 3 -- tokens are bound to their creating Origin. lookupToken(token,
|
|
23
|
+
* origin) returns the api_key only if record.origin === origin.
|
|
24
|
+
* Gap 6 -- every Bearer token has an associated csrfToken that the caller
|
|
25
|
+
* (serve-dashboard-live) compares against both the mos_csrf cookie AND the
|
|
26
|
+
* X-CSRF-Token header.
|
|
27
|
+
* The api_key is never logged, never written to disk, and is cleared on
|
|
28
|
+
* server shutdown via clearAll().
|
|
29
|
+
*
|
|
30
|
+
* Dependencies
|
|
31
|
+
* Node built-in `crypto` only. Zero new runtime deps.
|
|
32
|
+
*/
|
|
33
|
+
|
|
34
|
+
const crypto = require('node:crypto');
|
|
35
|
+
|
|
36
|
+
const DEFAULT_TTL_MS = 30 * 60 * 1000; // 30 minutes
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Internal token registry.
|
|
40
|
+
* Map<tokenHex, { apiKey, origin, csrfToken, expiresAt }>
|
|
41
|
+
*/
|
|
42
|
+
const tokens = new Map();
|
|
43
|
+
|
|
44
|
+
let sweepTimer = null;
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Create a new Bearer + CSRF token pair bound to the calling Origin.
|
|
48
|
+
* Returns { token, expiresIn, csrfToken }; does NOT log the api_key.
|
|
49
|
+
* @param {string} apiKey Anthropic api_key string (sk-...).
|
|
50
|
+
* @param {string} origin The request's Origin header value at creation time.
|
|
51
|
+
* Stored verbatim; lookupToken() later compares strict-equal.
|
|
52
|
+
* @param {number} [ttlMs=DEFAULT_TTL_MS]
|
|
53
|
+
* @returns {{token:string, expiresIn:number, csrfToken:string}}
|
|
54
|
+
*/
|
|
55
|
+
function createToken(apiKey, origin, ttlMs = DEFAULT_TTL_MS) {
|
|
56
|
+
if (!apiKey || typeof apiKey !== 'string') {
|
|
57
|
+
throw new Error('createToken: apiKey must be a non-empty string');
|
|
58
|
+
}
|
|
59
|
+
if (typeof origin !== 'string') {
|
|
60
|
+
throw new Error('createToken: origin must be a string');
|
|
61
|
+
}
|
|
62
|
+
const token = crypto.randomBytes(32).toString('hex'); // 64 hex chars
|
|
63
|
+
const csrfToken = crypto.randomBytes(16).toString('hex'); // 32 hex chars
|
|
64
|
+
tokens.set(token, {
|
|
65
|
+
apiKey,
|
|
66
|
+
origin,
|
|
67
|
+
csrfToken,
|
|
68
|
+
expiresAt: Date.now() + ttlMs,
|
|
69
|
+
});
|
|
70
|
+
return { token, expiresIn: Math.floor(ttlMs / 1000), csrfToken };
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
* Look up the api_key bound to a token, enforcing Origin binding.
|
|
75
|
+
* Returns the api_key only if the token exists, is unexpired, AND the
|
|
76
|
+
* requestOrigin matches the origin recorded at creation time
|
|
77
|
+
* (R-87-09-CSRF gap 3). Returns null otherwise. Expired entries are
|
|
78
|
+
* purged as a side effect.
|
|
79
|
+
* @param {string} token
|
|
80
|
+
* @param {string} requestOrigin
|
|
81
|
+
* @returns {string|null}
|
|
82
|
+
*/
|
|
83
|
+
function lookupToken(token, requestOrigin) {
|
|
84
|
+
if (!token || typeof token !== 'string') return null;
|
|
85
|
+
const entry = tokens.get(token);
|
|
86
|
+
if (!entry) return null;
|
|
87
|
+
if (entry.expiresAt <= Date.now()) {
|
|
88
|
+
tokens.delete(token);
|
|
89
|
+
return null;
|
|
90
|
+
}
|
|
91
|
+
if (entry.origin !== requestOrigin) {
|
|
92
|
+
// R-87-09-CSRF gap 3: a token created for Origin A must not work from
|
|
93
|
+
// Origin B, even if both origins are in the server's allowlist.
|
|
94
|
+
return null;
|
|
95
|
+
}
|
|
96
|
+
return entry.apiKey;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* Return the CSRF token bound to this Bearer token, for server-side
|
|
101
|
+
* double-submit verification (R-87-09-CSRF gap 6). Returns null if the
|
|
102
|
+
* token is unknown or expired.
|
|
103
|
+
* @param {string} token
|
|
104
|
+
* @returns {string|null}
|
|
105
|
+
*/
|
|
106
|
+
function lookupCsrfForToken(token) {
|
|
107
|
+
if (!token || typeof token !== 'string') return null;
|
|
108
|
+
const entry = tokens.get(token);
|
|
109
|
+
if (!entry) return null;
|
|
110
|
+
if (entry.expiresAt <= Date.now()) {
|
|
111
|
+
tokens.delete(token);
|
|
112
|
+
return null;
|
|
113
|
+
}
|
|
114
|
+
return entry.csrfToken;
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
/**
|
|
118
|
+
* Explicitly revoke a token (e.g. on user logout). Returns true if the
|
|
119
|
+
* entry existed, false otherwise.
|
|
120
|
+
* @param {string} token
|
|
121
|
+
* @returns {boolean}
|
|
122
|
+
*/
|
|
123
|
+
function revokeToken(token) {
|
|
124
|
+
return tokens.delete(token);
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
/**
|
|
128
|
+
* Iterate the registry and delete every entry whose expiresAt <= now.
|
|
129
|
+
* Returns the number of entries removed.
|
|
130
|
+
* @returns {number}
|
|
131
|
+
*/
|
|
132
|
+
function sweepExpired() {
|
|
133
|
+
const now = Date.now();
|
|
134
|
+
let removed = 0;
|
|
135
|
+
for (const [t, entry] of tokens.entries()) {
|
|
136
|
+
if (entry.expiresAt <= now) {
|
|
137
|
+
tokens.delete(t);
|
|
138
|
+
removed += 1;
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
return removed;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
/**
|
|
145
|
+
* Start a background interval timer that calls sweepExpired() every
|
|
146
|
+
* `intervalMs` milliseconds (default 60s). The timer is `unref()`ed so it
|
|
147
|
+
* never keeps the process alive on its own. Idempotent: a second call
|
|
148
|
+
* returns the existing handle.
|
|
149
|
+
* @param {number} [intervalMs=60000]
|
|
150
|
+
* @returns {NodeJS.Timeout}
|
|
151
|
+
*/
|
|
152
|
+
function startSweepInterval(intervalMs = 60000) {
|
|
153
|
+
if (sweepTimer) return sweepTimer;
|
|
154
|
+
sweepTimer = setInterval(sweepExpired, intervalMs);
|
|
155
|
+
if (sweepTimer && typeof sweepTimer.unref === 'function') sweepTimer.unref();
|
|
156
|
+
return sweepTimer;
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
/**
|
|
160
|
+
* Stop the background sweep timer. Safe to call when none is running.
|
|
161
|
+
*/
|
|
162
|
+
function stopSweepInterval() {
|
|
163
|
+
if (sweepTimer) {
|
|
164
|
+
clearInterval(sweepTimer);
|
|
165
|
+
sweepTimer = null;
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
/**
|
|
170
|
+
* Nuke every token entry and stop the sweep timer. Called on SIGINT /
|
|
171
|
+
* SIGTERM by serve-dashboard-live so no api_key survives a restart.
|
|
172
|
+
*/
|
|
173
|
+
function clearAll() {
|
|
174
|
+
tokens.clear();
|
|
175
|
+
stopSweepInterval();
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
/**
|
|
179
|
+
* Test-only introspection hook. Returns the current number of live
|
|
180
|
+
* entries. NOT part of the public contract; do not rely on it from
|
|
181
|
+
* production code paths.
|
|
182
|
+
* @returns {number}
|
|
183
|
+
*/
|
|
184
|
+
function _size() {
|
|
185
|
+
return tokens.size;
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
module.exports = {
|
|
189
|
+
createToken,
|
|
190
|
+
lookupToken,
|
|
191
|
+
lookupCsrfForToken,
|
|
192
|
+
revokeToken,
|
|
193
|
+
sweepExpired,
|
|
194
|
+
startSweepInterval,
|
|
195
|
+
stopSweepInterval,
|
|
196
|
+
clearAll,
|
|
197
|
+
_size,
|
|
198
|
+
DEFAULT_TTL_MS,
|
|
199
|
+
};
|