@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,266 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2026 Mindrian. BSL 1.1.
|
|
3
|
+
*
|
|
4
|
+
* Phase 88.1-04 -- statusline MINTO segment cache + format helpers
|
|
5
|
+
* =================================================================
|
|
6
|
+
* Pure, disk-backed, 5-second TTL cache for the Claude Code statusline
|
|
7
|
+
* MINTO segment. The statusline fires once per turn via scripts/context-monitor
|
|
8
|
+
* and must stay within a 300ms render budget (CONTEXT R1 mitigation).
|
|
9
|
+
*
|
|
10
|
+
* Cold path (no cache): readTriple + cache write. Target < 200ms on fresh WSL2.
|
|
11
|
+
* Warm path (cache hit): JSON parse + string format. Target < 10ms.
|
|
12
|
+
*
|
|
13
|
+
* The cache is LOCAL by construction per Canon Part 8: all writes land at
|
|
14
|
+
* roomRoot/.mindrian/statusline-cache.json and never egress. No Brain calls,
|
|
15
|
+
* no fetch, no external endpoints.
|
|
16
|
+
*
|
|
17
|
+
* Canon references:
|
|
18
|
+
* Part 2 UI glyph vocabulary -- classifyHealth emits check / warn / low / --
|
|
19
|
+
* for the MINTO reasoning_health_score (shared with
|
|
20
|
+
* lib/memory/triple-context-formatter.cjs + Phase 88.1-03 hook sysmsg
|
|
21
|
+
* retrofit so rendering is byte-identical across surfaces).
|
|
22
|
+
* Part 3 Tri-Context Decision Gate -- statusline renders LOCAL only.
|
|
23
|
+
* Never BRAIN (BRAIN payloads are generic methodology only), never SIGNAL
|
|
24
|
+
* (SIGNAL belongs to scheduled sweeps, not per-turn UI).
|
|
25
|
+
* Part 8 Graph Boundary -- all cached bytes describe this specific user's
|
|
26
|
+
* room state. They stay in this room. The boundary is architectural.
|
|
27
|
+
*
|
|
28
|
+
* Composes with:
|
|
29
|
+
* Phase 87-02 atomic write pattern (openSync('wx') + fsync + rename) for
|
|
30
|
+
* concurrent-safe cache persistence under rapid-statusline-fire.
|
|
31
|
+
* Phase 88-01 folder-memory.readTriple(sectionPath) is the read contract;
|
|
32
|
+
* cached 'triple' values are the shape readTriple returns.
|
|
33
|
+
* Phase 88-07 / 88.1-03 classifyHealth shared vocabulary so every L3
|
|
34
|
+
* surface (session-start sysmsg, guardian, statusline, /mos:status) uses
|
|
35
|
+
* the same glyph classifier.
|
|
36
|
+
*
|
|
37
|
+
* Pure CJS, node built-ins only, zero npm dependencies. Three-surface
|
|
38
|
+
* compatible by construction (CLI + Desktop MCP + Cowork).
|
|
39
|
+
*
|
|
40
|
+
* API:
|
|
41
|
+
* TTL_MS -- number, fixed at 5000ms.
|
|
42
|
+
* getCached(roomRoot, sectionPath) -> {value, fresh}
|
|
43
|
+
* setCached(roomRoot, sectionPath, triple) -> void (graceful on error)
|
|
44
|
+
* truncateGoverningThought(str) -> string (60-char budget)
|
|
45
|
+
* classifyHealth(score) -> 'check'|'warn'|'low'|'--'
|
|
46
|
+
*
|
|
47
|
+
* Cache file shape:
|
|
48
|
+
* { ts: number, sectionPath: string, triple: object }
|
|
49
|
+
*/
|
|
50
|
+
|
|
51
|
+
'use strict';
|
|
52
|
+
|
|
53
|
+
const fs = require('node:fs');
|
|
54
|
+
const path = require('node:path');
|
|
55
|
+
|
|
56
|
+
// ---------- Tunables (frozen invariants) ----------
|
|
57
|
+
|
|
58
|
+
// 5-second TTL per CONTEXT R1 mitigation. Short enough that users perceive
|
|
59
|
+
// freshness (a change to MINTO.md shows up on the next statusline tick after
|
|
60
|
+
// 5s), long enough that 12 statusline fires/min all hit the warm path.
|
|
61
|
+
const TTL_MS = 5000;
|
|
62
|
+
|
|
63
|
+
// 60-char budget for the governing_thought segment. Matches the description-
|
|
64
|
+
// discipline budget from Plan 88.1-01 so the statusline honors the same
|
|
65
|
+
// text-economy contract the rest of L3 uses.
|
|
66
|
+
const GOVERNING_THOUGHT_BUDGET = 60;
|
|
67
|
+
|
|
68
|
+
// Cache filename under roomRoot/.mindrian/. Named separately from the Phase
|
|
69
|
+
// 88-06 session-snapshot.json to avoid any cross-purpose coupling.
|
|
70
|
+
const CACHE_FILENAME = 'statusline-cache.json';
|
|
71
|
+
|
|
72
|
+
// ---------- Path helpers ----------
|
|
73
|
+
|
|
74
|
+
function cacheDir(roomRoot) {
|
|
75
|
+
return path.join(roomRoot, '.mindrian');
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
function cacheFile(roomRoot) {
|
|
79
|
+
return path.join(cacheDir(roomRoot), CACHE_FILENAME);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// ---------- classifyHealth (shared with triple-context-formatter) ----------
|
|
83
|
+
//
|
|
84
|
+
// Canon Part 2 glyph vocabulary: check / warn / low / --. Byte-identical to
|
|
85
|
+
// the classifier extracted in Phase 88.1-03 so the statusline, the hook
|
|
86
|
+
// systemMessage retrofit, and the TRIPLE_CONTEXT formatter all render the
|
|
87
|
+
// same glyph for the same score. See lib/memory/triple-context-formatter.cjs
|
|
88
|
+
// lines 80-101 for the canonical source; this copy mirrors it for zero-cost
|
|
89
|
+
// import from lib/core/ consumers that should not reach into lib/memory/.
|
|
90
|
+
//
|
|
91
|
+
// Signature: classifyHealth(score: number | null | undefined)
|
|
92
|
+
// -> 'check' | 'warn' | 'low' | '--'
|
|
93
|
+
|
|
94
|
+
function classifyHealth(score) {
|
|
95
|
+
if (typeof score !== 'number' || !Number.isFinite(score)) return '--';
|
|
96
|
+
if (score >= 0.7) return 'check';
|
|
97
|
+
if (score >= 0.4) return 'warn';
|
|
98
|
+
return 'low';
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
// ---------- truncateGoverningThought ----------
|
|
102
|
+
|
|
103
|
+
/**
|
|
104
|
+
* truncateGoverningThought
|
|
105
|
+
* Enforce the 60-char statusline budget on a governing_thought string.
|
|
106
|
+
* Short inputs (<= 60 chars) pass through byte-identical. Long inputs are
|
|
107
|
+
* truncated to 59 chars plus a Unicode ellipsis (U+2026) for a total of
|
|
108
|
+
* exactly 60 chars. Non-string / empty inputs return the empty string.
|
|
109
|
+
*/
|
|
110
|
+
function truncateGoverningThought(str) {
|
|
111
|
+
if (typeof str !== 'string') return '';
|
|
112
|
+
if (str.length <= GOVERNING_THOUGHT_BUDGET) return str;
|
|
113
|
+
// 59 chars + 1 ellipsis = exactly 60 chars, matching the budget.
|
|
114
|
+
return str.slice(0, GOVERNING_THOUGHT_BUDGET - 1) + '\u2026';
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
// ---------- Disk read ----------
|
|
118
|
+
|
|
119
|
+
function safeReadSync(p) {
|
|
120
|
+
try {
|
|
121
|
+
return fs.readFileSync(p, 'utf8');
|
|
122
|
+
} catch (_e) {
|
|
123
|
+
return null;
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
/**
|
|
128
|
+
* getCached(roomRoot, sectionPath) -> {value, fresh}
|
|
129
|
+
*
|
|
130
|
+
* Returns {value: null, fresh: false} when:
|
|
131
|
+
* - cache file does not exist
|
|
132
|
+
* - cache file is malformed JSON
|
|
133
|
+
* - cache entry belongs to a different sectionPath
|
|
134
|
+
* - cache entry is older than TTL_MS
|
|
135
|
+
*
|
|
136
|
+
* Returns {value: triple, fresh: true} on a warm hit. The statusline reads
|
|
137
|
+
* the triple fields directly (reasoning.governing_thought,
|
|
138
|
+
* reasoning.reasoning_health_score, reasoning.is_stale).
|
|
139
|
+
*
|
|
140
|
+
* Never throws. Graceful fallback on every error path.
|
|
141
|
+
*/
|
|
142
|
+
function getCached(roomRoot, sectionPath) {
|
|
143
|
+
const miss = { value: null, fresh: false };
|
|
144
|
+
if (!roomRoot || !sectionPath) return miss;
|
|
145
|
+
|
|
146
|
+
const file = cacheFile(roomRoot);
|
|
147
|
+
const raw = safeReadSync(file);
|
|
148
|
+
if (raw === null) return miss;
|
|
149
|
+
|
|
150
|
+
let parsed;
|
|
151
|
+
try {
|
|
152
|
+
parsed = JSON.parse(raw);
|
|
153
|
+
} catch (_e) {
|
|
154
|
+
return miss;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
if (!parsed || typeof parsed !== 'object') return miss;
|
|
158
|
+
if (parsed.sectionPath !== sectionPath) return miss;
|
|
159
|
+
if (typeof parsed.ts !== 'number' || !Number.isFinite(parsed.ts)) return miss;
|
|
160
|
+
|
|
161
|
+
const age = Date.now() - parsed.ts;
|
|
162
|
+
if (age < 0 || age >= TTL_MS) {
|
|
163
|
+
return { value: null, fresh: false };
|
|
164
|
+
}
|
|
165
|
+
if (!parsed.triple || typeof parsed.triple !== 'object') return miss;
|
|
166
|
+
|
|
167
|
+
return { value: parsed.triple, fresh: true };
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
// ---------- Disk write (atomic) ----------
|
|
171
|
+
|
|
172
|
+
/**
|
|
173
|
+
* setCached(roomRoot, sectionPath, triple) -> void
|
|
174
|
+
*
|
|
175
|
+
* Atomic write sequence (Phase 87-02 pattern):
|
|
176
|
+
* - mkdir -p roomRoot/.mindrian/ (idempotent)
|
|
177
|
+
* - openSync('wx') on a pid-scoped tmp file
|
|
178
|
+
* - writeSync + fsyncSync + closeSync
|
|
179
|
+
* - renameSync over the real cache file
|
|
180
|
+
*
|
|
181
|
+
* Graceful fallback: any filesystem error (EEXIST on tmp, EACCES on mkdir,
|
|
182
|
+
* ENOENT on rename, disk full, etc.) is swallowed silently. The cache is
|
|
183
|
+
* a performance optimization; a failed write must NEVER bubble up to
|
|
184
|
+
* context-monitor because the statusline has to render regardless.
|
|
185
|
+
*
|
|
186
|
+
* Never throws. This is the Part 8-by-construction guarantee: a pure
|
|
187
|
+
* persistence side-effect that cannot leak state anywhere except the
|
|
188
|
+
* LOCAL roomRoot/.mindrian/ directory.
|
|
189
|
+
*/
|
|
190
|
+
function setCached(roomRoot, sectionPath, triple) {
|
|
191
|
+
if (!roomRoot || !sectionPath) return;
|
|
192
|
+
|
|
193
|
+
const dir = cacheDir(roomRoot);
|
|
194
|
+
try {
|
|
195
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
196
|
+
} catch (_e) {
|
|
197
|
+
// If mkdir fails (e.g. .mindrian is a regular file, not a directory),
|
|
198
|
+
// bail out silently. Graceful-fallback invariant.
|
|
199
|
+
return;
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
const payload = {
|
|
203
|
+
ts: Date.now(),
|
|
204
|
+
sectionPath: sectionPath,
|
|
205
|
+
triple: triple,
|
|
206
|
+
};
|
|
207
|
+
|
|
208
|
+
let serialized;
|
|
209
|
+
try {
|
|
210
|
+
serialized = JSON.stringify(payload);
|
|
211
|
+
} catch (_e) {
|
|
212
|
+
// Cyclic references in triple -> serialization failure. Bail.
|
|
213
|
+
return;
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
const finalPath = cacheFile(roomRoot);
|
|
217
|
+
const tmpPath = finalPath + '.tmp.' + process.pid + '.statusline';
|
|
218
|
+
|
|
219
|
+
let fd;
|
|
220
|
+
try {
|
|
221
|
+
fd = fs.openSync(tmpPath, 'wx');
|
|
222
|
+
} catch (e) {
|
|
223
|
+
if (e && e.code === 'EEXIST') {
|
|
224
|
+
// Stale tmp from a prior crash owned by the same pid slot. Clean and
|
|
225
|
+
// retry once. Any second failure is swallowed.
|
|
226
|
+
try { fs.unlinkSync(tmpPath); } catch (_) {}
|
|
227
|
+
try {
|
|
228
|
+
fd = fs.openSync(tmpPath, 'wx');
|
|
229
|
+
} catch (_e2) {
|
|
230
|
+
return;
|
|
231
|
+
}
|
|
232
|
+
} else {
|
|
233
|
+
return;
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
try {
|
|
238
|
+
fs.writeSync(fd, serialized);
|
|
239
|
+
try { fs.fsyncSync(fd); } catch (_) { /* fsync best-effort */ }
|
|
240
|
+
} catch (_e) {
|
|
241
|
+
try { fs.closeSync(fd); } catch (_) {}
|
|
242
|
+
try { fs.unlinkSync(tmpPath); } catch (_) {}
|
|
243
|
+
return;
|
|
244
|
+
} finally {
|
|
245
|
+
try { fs.closeSync(fd); } catch (_) {}
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
try {
|
|
249
|
+
fs.renameSync(tmpPath, finalPath);
|
|
250
|
+
} catch (_e) {
|
|
251
|
+
// Rename failed. Clean up tmp and bail.
|
|
252
|
+
try { fs.unlinkSync(tmpPath); } catch (_) {}
|
|
253
|
+
return;
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
// ---------- Exports ----------
|
|
258
|
+
|
|
259
|
+
module.exports = {
|
|
260
|
+
TTL_MS: TTL_MS,
|
|
261
|
+
GOVERNING_THOUGHT_BUDGET: GOVERNING_THOUGHT_BUDGET,
|
|
262
|
+
getCached: getCached,
|
|
263
|
+
setCached: setCached,
|
|
264
|
+
truncateGoverningThought: truncateGoverningThought,
|
|
265
|
+
classifyHealth: classifyHealth,
|
|
266
|
+
};
|
|
@@ -0,0 +1,348 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2026 Mindrian. BSL 1.1.
|
|
3
|
+
*
|
|
4
|
+
* Phase 88.1-16 -- query-efficiency token estimator + aggregation math.
|
|
5
|
+
* =====================================================================
|
|
6
|
+
* Pure utility module consumed by two callers:
|
|
7
|
+
*
|
|
8
|
+
* 1. scripts/query-efficiency-telemetry.cjs (PostToolUse hook) uses
|
|
9
|
+
* estimateTokens(payload) to measure tokens_used per Read|Grep|Glob
|
|
10
|
+
* return, and estimateRoomTokens(roomRoot) to compute the
|
|
11
|
+
* tokens_naive_estimate baseline that defines the 57x efficiency claim.
|
|
12
|
+
*
|
|
13
|
+
* 2. scripts/scout-telemetry-aggregator.cjs (aggregation helper spawned
|
|
14
|
+
* by /mos:scout) uses aggregateEvents(events) to render the median
|
|
15
|
+
* ratio + top-5 commands + threshold-status summary.
|
|
16
|
+
*
|
|
17
|
+
* Canon:
|
|
18
|
+
* Part 6 Product-as-Venture -- the 57x token-efficiency claim is a
|
|
19
|
+
* venture claim; without this measurement arm it is marketing copy.
|
|
20
|
+
* This module is the measurement arm: the ratio it produces is the
|
|
21
|
+
* release-gate signal (>= 40x PASS; < 40x RETUNE) per CONTEXT
|
|
22
|
+
* criterion #15.
|
|
23
|
+
* Part 8 Graph Boundary -- LOCAL-by-construction. Zero http/https
|
|
24
|
+
* imports, zero child_process, zero network egress. Only fs reads
|
|
25
|
+
* (for estimateRoomTokens walking .md files). The caller hooks
|
|
26
|
+
* persist ONLY scalar integer counts to the JSONL; no user-artifact
|
|
27
|
+
* bytes reach the telemetry file.
|
|
28
|
+
*
|
|
29
|
+
* Design notes:
|
|
30
|
+
* - estimateTokens uses the chars/4 approximation established by
|
|
31
|
+
* lib/memory/triple-context-formatter.cjs (Phase 88-07). Consistency
|
|
32
|
+
* across the codebase matters: if the Phase 88-07 formatter reports
|
|
33
|
+
* "this emission is 3825 tokens", the telemetry hook must estimate
|
|
34
|
+
* with the same yardstick so ratios compare apples-to-apples.
|
|
35
|
+
*
|
|
36
|
+
* - estimateRoomTokens uses a module-level in-memory cache keyed by
|
|
37
|
+
* absolute room path. First call walks the tree summing .md file
|
|
38
|
+
* sizes; subsequent calls return the cached value. The cache is
|
|
39
|
+
* session-scoped (cleared when the Node process exits). A ledger
|
|
40
|
+
* cache is NOT persisted to disk because:
|
|
41
|
+
* (a) the estimator is fast enough (single walk of a room tree
|
|
42
|
+
* typically completes in < 50ms on commodity SSDs), and
|
|
43
|
+
* (b) on-disk caches introduce invalidation complexity that is
|
|
44
|
+
* unnecessary for a per-session approximation used as the
|
|
45
|
+
* denominator of a ratio. Staleness by up to one session is
|
|
46
|
+
* within the noise floor of the measurement itself.
|
|
47
|
+
*
|
|
48
|
+
* - validateEventShape enforces the 8-field contract specified in
|
|
49
|
+
* 88.1-16 PLAN <interfaces>. The hook uses this as a pre-write guard
|
|
50
|
+
* so a malformed event can never reach the JSONL.
|
|
51
|
+
*
|
|
52
|
+
* - aggregateEvents reduces a raw event stream to { median, mean, top5,
|
|
53
|
+
* count }. Top-5 is computed per-command (taking the MAX ratio
|
|
54
|
+
* observed for each command) so a single command cannot occupy
|
|
55
|
+
* multiple slots. This matches the CONTEXT-specified "top 5 commands
|
|
56
|
+
* by ratio" rendering.
|
|
57
|
+
*
|
|
58
|
+
* - classifyRatio maps a numeric ratio to { 'normal' | 'warn' } at the
|
|
59
|
+
* 10x threshold per PLAN <action>. Below-10x arms the hook's warn
|
|
60
|
+
* systemMessage; at-or-above 10x stays silent (no verification fatigue
|
|
61
|
+
* per Plan 03 reviewer R5).
|
|
62
|
+
*
|
|
63
|
+
* API:
|
|
64
|
+
* estimateTokens(str) -> integer >= 0
|
|
65
|
+
* estimateRoomTokens(roomRoot) -> integer >= 0 | null
|
|
66
|
+
* clearCache() -> void (test hygiene)
|
|
67
|
+
* validateEventShape(obj) -> { valid, missing:string[] }
|
|
68
|
+
* classifyRatio(ratio) -> 'normal' | 'warn'
|
|
69
|
+
* aggregateEvents(events) -> { count, median, mean, top5 }
|
|
70
|
+
* EVENT_REQUIRED_FIELDS -> readonly string[]
|
|
71
|
+
* BELOW_THRESHOLD_RATIO -> 10
|
|
72
|
+
*/
|
|
73
|
+
|
|
74
|
+
'use strict';
|
|
75
|
+
|
|
76
|
+
const fs = require('node:fs');
|
|
77
|
+
const path = require('node:path');
|
|
78
|
+
|
|
79
|
+
// ---------- Constants ----------
|
|
80
|
+
|
|
81
|
+
// Classification threshold for below-10x warn emission. Canon: anything
|
|
82
|
+
// <10x is a leakage signal (query is pulling more room context than it
|
|
83
|
+
// needs). This is the hook's advisory trigger, not the 57x release gate.
|
|
84
|
+
const BELOW_THRESHOLD_RATIO = 10;
|
|
85
|
+
|
|
86
|
+
// 8-field contract per 88.1-16 PLAN <interfaces>. validateEventShape
|
|
87
|
+
// enforces this on every event before it is appended to the JSONL.
|
|
88
|
+
const EVENT_REQUIRED_FIELDS = Object.freeze([
|
|
89
|
+
'event',
|
|
90
|
+
'ts',
|
|
91
|
+
'command',
|
|
92
|
+
'tool',
|
|
93
|
+
'tokens_used',
|
|
94
|
+
'tokens_naive_estimate',
|
|
95
|
+
'ratio',
|
|
96
|
+
'room_slug',
|
|
97
|
+
]);
|
|
98
|
+
|
|
99
|
+
// Max depth guard for the estimateRoomTokens walker. Rooms are expected
|
|
100
|
+
// to be shallow (<10 levels). A runaway symlink loop or surprise deep
|
|
101
|
+
// tree must not hang the hook.
|
|
102
|
+
const MAX_ROOM_DEPTH = 12;
|
|
103
|
+
|
|
104
|
+
// ---------- Module-level session cache ----------
|
|
105
|
+
//
|
|
106
|
+
// Key: absolute, resolved room path (realpath when possible).
|
|
107
|
+
// Value: cached token count (integer).
|
|
108
|
+
// Cleared via clearCache() for tests; otherwise persists for the life
|
|
109
|
+
// of the Node process (i.e. for the duration of a single hook fork or
|
|
110
|
+
// scout invocation).
|
|
111
|
+
const roomTokenCache = new Map();
|
|
112
|
+
|
|
113
|
+
// ---------- estimateTokens ----------
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* estimateTokens(str) -> integer >= 0
|
|
117
|
+
*
|
|
118
|
+
* Chars-over-4 estimator, matching Phase 88-07 triple-context-formatter.
|
|
119
|
+
* Graceful on null / undefined / non-string: returns 0. This matches the
|
|
120
|
+
* behavior established by formatter.estimateTokens so the two producers
|
|
121
|
+
* are interchangeable.
|
|
122
|
+
*/
|
|
123
|
+
function estimateTokens(str) {
|
|
124
|
+
if (typeof str !== 'string') return 0;
|
|
125
|
+
if (str.length === 0) return 0;
|
|
126
|
+
return Math.ceil(str.length / 4);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
// ---------- estimateRoomTokens ----------
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Walk the room tree summing the byte-length of every .md file, divide
|
|
133
|
+
* by 4 to approximate tokens. Skips non-.md files (binaries, images,
|
|
134
|
+
* generated JSON, etc.) so the denominator of the efficiency ratio
|
|
135
|
+
* reflects ONLY the text payload a naive whole-room ingestion would
|
|
136
|
+
* consume.
|
|
137
|
+
*
|
|
138
|
+
* Uses fs.readdirSync with withFileTypes so a single syscall per
|
|
139
|
+
* directory yields both names and types; avoids the stat-per-entry
|
|
140
|
+
* overhead that a readdir+stat loop would incur.
|
|
141
|
+
*
|
|
142
|
+
* Returns null on:
|
|
143
|
+
* - invalid input (non-string, empty string, null, undefined)
|
|
144
|
+
* - path does not exist
|
|
145
|
+
* - path is not a directory
|
|
146
|
+
* - any error during the walk (defensive; the hook must never throw)
|
|
147
|
+
*
|
|
148
|
+
* Caches the result in module-level roomTokenCache keyed by the resolved
|
|
149
|
+
* path. Second call returns cached value without re-walking; this is the
|
|
150
|
+
* property test 5 asserts.
|
|
151
|
+
*/
|
|
152
|
+
function estimateRoomTokens(roomRoot) {
|
|
153
|
+
if (typeof roomRoot !== 'string' || roomRoot.length === 0) return null;
|
|
154
|
+
|
|
155
|
+
// Cache hit?
|
|
156
|
+
if (roomTokenCache.has(roomRoot)) {
|
|
157
|
+
return roomTokenCache.get(roomRoot);
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
let stat;
|
|
161
|
+
try {
|
|
162
|
+
stat = fs.statSync(roomRoot);
|
|
163
|
+
} catch (_e) {
|
|
164
|
+
return null;
|
|
165
|
+
}
|
|
166
|
+
if (!stat || !stat.isDirectory()) return null;
|
|
167
|
+
|
|
168
|
+
let totalChars = 0;
|
|
169
|
+
|
|
170
|
+
function walk(dir, depth) {
|
|
171
|
+
if (depth > MAX_ROOM_DEPTH) return;
|
|
172
|
+
let entries;
|
|
173
|
+
try {
|
|
174
|
+
entries = fs.readdirSync(dir, { withFileTypes: true });
|
|
175
|
+
} catch (_e) {
|
|
176
|
+
return; // permission denied, vanished dir, etc.
|
|
177
|
+
}
|
|
178
|
+
for (const entry of entries) {
|
|
179
|
+
if (!entry) continue;
|
|
180
|
+
const name = entry.name;
|
|
181
|
+
// Skip hidden directories / files starting with '.' (e.g. .git,
|
|
182
|
+
// .mindrian, .obsidian). Counting them would over-estimate the
|
|
183
|
+
// naive baseline and dilute the ratio.
|
|
184
|
+
if (typeof name === 'string' && name.length > 0 && name[0] === '.') continue;
|
|
185
|
+
const full = path.join(dir, name);
|
|
186
|
+
if (entry.isDirectory()) {
|
|
187
|
+
walk(full, depth + 1);
|
|
188
|
+
continue;
|
|
189
|
+
}
|
|
190
|
+
if (!entry.isFile()) continue; // skip symlinks, sockets, devices
|
|
191
|
+
// Only count .md files.
|
|
192
|
+
if (!/\.md$/i.test(name)) continue;
|
|
193
|
+
try {
|
|
194
|
+
const s = fs.statSync(full);
|
|
195
|
+
if (s && typeof s.size === 'number') {
|
|
196
|
+
totalChars += s.size;
|
|
197
|
+
}
|
|
198
|
+
} catch (_e) { /* vanished file, skip */ }
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
try {
|
|
203
|
+
walk(roomRoot, 0);
|
|
204
|
+
} catch (_e) {
|
|
205
|
+
return null;
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
const tokens = Math.ceil(totalChars / 4);
|
|
209
|
+
roomTokenCache.set(roomRoot, tokens);
|
|
210
|
+
return tokens;
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
// ---------- clearCache ----------
|
|
214
|
+
|
|
215
|
+
/**
|
|
216
|
+
* Reset the session cache. Used by tests for deterministic between-case
|
|
217
|
+
* isolation; production callers typically do not need to invoke this
|
|
218
|
+
* (process lifetime bounds the cache naturally).
|
|
219
|
+
*/
|
|
220
|
+
function clearCache() {
|
|
221
|
+
roomTokenCache.clear();
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
// ---------- validateEventShape ----------
|
|
225
|
+
|
|
226
|
+
/**
|
|
227
|
+
* Return { valid, missing } for a telemetry event object. An event is
|
|
228
|
+
* valid iff every field in EVENT_REQUIRED_FIELDS is present AND
|
|
229
|
+
* non-null / non-undefined. The hook uses this as a write guard: a
|
|
230
|
+
* malformed event must never reach the JSONL (prevents aggregation
|
|
231
|
+
* NaN poisoning later).
|
|
232
|
+
*/
|
|
233
|
+
function validateEventShape(obj) {
|
|
234
|
+
if (!obj || typeof obj !== 'object' || Array.isArray(obj)) {
|
|
235
|
+
return { valid: false, missing: EVENT_REQUIRED_FIELDS.slice() };
|
|
236
|
+
}
|
|
237
|
+
const missing = [];
|
|
238
|
+
for (let i = 0; i < EVENT_REQUIRED_FIELDS.length; i += 1) {
|
|
239
|
+
const k = EVENT_REQUIRED_FIELDS[i];
|
|
240
|
+
const v = obj[k];
|
|
241
|
+
if (v === undefined || v === null) missing.push(k);
|
|
242
|
+
}
|
|
243
|
+
return { valid: missing.length === 0, missing: missing };
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
// ---------- classifyRatio ----------
|
|
247
|
+
|
|
248
|
+
/**
|
|
249
|
+
* Classify a ratio into 'normal' or 'warn' at the 10x threshold.
|
|
250
|
+
* Defensive: any non-finite or non-numeric input -> 'warn' (the ratio
|
|
251
|
+
* was never computed correctly, which is itself a signal the hook
|
|
252
|
+
* should surface).
|
|
253
|
+
*/
|
|
254
|
+
function classifyRatio(ratio) {
|
|
255
|
+
if (typeof ratio !== 'number' || !Number.isFinite(ratio)) return 'warn';
|
|
256
|
+
if (ratio >= BELOW_THRESHOLD_RATIO) return 'normal';
|
|
257
|
+
return 'warn';
|
|
258
|
+
}
|
|
259
|
+
|
|
260
|
+
// ---------- aggregateEvents ----------
|
|
261
|
+
|
|
262
|
+
/**
|
|
263
|
+
* Reduce an event stream to { count, median, mean, top5 }.
|
|
264
|
+
*
|
|
265
|
+
* - count: events.length.
|
|
266
|
+
* - median: midpoint of the sorted ratios. Even-length: average of the
|
|
267
|
+
* two middle values. Odd-length: the middle value.
|
|
268
|
+
* - mean: arithmetic mean of ratios, rounded to 1 decimal for display
|
|
269
|
+
* friendliness in the scout render.
|
|
270
|
+
* - top5: up to 5 entries, sorted descending by ratio, deduped by
|
|
271
|
+
* command (each command appears at most once, with its MAX
|
|
272
|
+
* observed ratio).
|
|
273
|
+
*
|
|
274
|
+
* Empty input -> { count:0, median:null, mean:null, top5:[] }.
|
|
275
|
+
*
|
|
276
|
+
* Filters out malformed events (missing or non-numeric ratio) so a
|
|
277
|
+
* corrupt JSONL line cannot poison the aggregate.
|
|
278
|
+
*/
|
|
279
|
+
function aggregateEvents(events) {
|
|
280
|
+
const out = { count: 0, median: null, mean: null, top5: [] };
|
|
281
|
+
if (!Array.isArray(events) || events.length === 0) return out;
|
|
282
|
+
|
|
283
|
+
// Collect only well-formed ratio values. A malformed event counts
|
|
284
|
+
// toward `count` (reported events) but NOT toward median/mean/top5.
|
|
285
|
+
const ratios = [];
|
|
286
|
+
const byCommand = new Map(); // command -> max ratio
|
|
287
|
+
for (let i = 0; i < events.length; i += 1) {
|
|
288
|
+
const e = events[i];
|
|
289
|
+
if (!e || typeof e !== 'object') continue;
|
|
290
|
+
const r = e.ratio;
|
|
291
|
+
if (typeof r !== 'number' || !Number.isFinite(r)) continue;
|
|
292
|
+
ratios.push(r);
|
|
293
|
+
const cmd = typeof e.command === 'string' && e.command.length > 0
|
|
294
|
+
? e.command
|
|
295
|
+
: '(unknown)';
|
|
296
|
+
const prev = byCommand.get(cmd);
|
|
297
|
+
if (prev === undefined || r > prev) byCommand.set(cmd, r);
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
out.count = events.length;
|
|
301
|
+
|
|
302
|
+
if (ratios.length === 0) return out;
|
|
303
|
+
|
|
304
|
+
// Median: sort asc, pick middle.
|
|
305
|
+
const sorted = ratios.slice().sort(function (a, b) { return a - b; });
|
|
306
|
+
const n = sorted.length;
|
|
307
|
+
if (n % 2 === 1) {
|
|
308
|
+
out.median = sorted[(n - 1) / 2];
|
|
309
|
+
} else {
|
|
310
|
+
const lo = sorted[(n / 2) - 1];
|
|
311
|
+
const hi = sorted[n / 2];
|
|
312
|
+
out.median = (lo + hi) / 2;
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
// Mean, 1-decimal rounded for display.
|
|
316
|
+
let sum = 0;
|
|
317
|
+
for (let i = 0; i < n; i += 1) sum += sorted[i];
|
|
318
|
+
const rawMean = sum / n;
|
|
319
|
+
out.mean = Math.round(rawMean * 10) / 10;
|
|
320
|
+
|
|
321
|
+
// Top-5 per-command, sorted desc by ratio. Ties broken by command
|
|
322
|
+
// name for determinism.
|
|
323
|
+
const pairs = [];
|
|
324
|
+
const keys = Array.from(byCommand.keys());
|
|
325
|
+
for (let i = 0; i < keys.length; i += 1) {
|
|
326
|
+
pairs.push({ command: keys[i], ratio: byCommand.get(keys[i]) });
|
|
327
|
+
}
|
|
328
|
+
pairs.sort(function (a, b) {
|
|
329
|
+
if (b.ratio !== a.ratio) return b.ratio - a.ratio;
|
|
330
|
+
return a.command < b.command ? -1 : a.command > b.command ? 1 : 0;
|
|
331
|
+
});
|
|
332
|
+
out.top5 = pairs.slice(0, 5);
|
|
333
|
+
|
|
334
|
+
return out;
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
// ---------- Exports ----------
|
|
338
|
+
|
|
339
|
+
module.exports = {
|
|
340
|
+
BELOW_THRESHOLD_RATIO: BELOW_THRESHOLD_RATIO,
|
|
341
|
+
EVENT_REQUIRED_FIELDS: EVENT_REQUIRED_FIELDS,
|
|
342
|
+
estimateTokens: estimateTokens,
|
|
343
|
+
estimateRoomTokens: estimateRoomTokens,
|
|
344
|
+
clearCache: clearCache,
|
|
345
|
+
validateEventShape: validateEventShape,
|
|
346
|
+
classifyRatio: classifyRatio,
|
|
347
|
+
aggregateEvents: aggregateEvents,
|
|
348
|
+
};
|