@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,395 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2026 Mindrian. BSL 1.1.
|
|
3
|
+
*
|
|
4
|
+
* Phase 88.2-06 -- decoy-tier dispatcher (UISEL-88.2-08 part 1).
|
|
5
|
+
*
|
|
6
|
+
* Replaces the 88.2-00 Wave-0 stub with the real implementation per
|
|
7
|
+
* RESEARCH.md DISCRETION-AMEND-02 (verbatim contract) + CONTEXT.md
|
|
8
|
+
* F.6 spec body 2026-04-29 + D-AMEND-05 (Tier 2 persona-aware decoys).
|
|
9
|
+
*
|
|
10
|
+
* Responsibility: given an artifact + an optional comprehension profile +
|
|
11
|
+
* an optional role_blend + Brain reachability, return a deterministically
|
|
12
|
+
* seeded set of calibration decoys for a F.6 plan-review round. The 5
|
|
13
|
+
* perturbation axes (wrong_owner / wrong_dependency / wrong_metric /
|
|
14
|
+
* wrong_scope / wrong_invariant) are applied round-robin across the round
|
|
15
|
+
* so every axis is exercised before any axis repeats.
|
|
16
|
+
*
|
|
17
|
+
* Tier auto-detection (rules verbatim per RESEARCH.md DISCRETION-AMEND-02):
|
|
18
|
+
* - Tier 2 IF profile.rounds_total >= 100 AND
|
|
19
|
+
* Array.isArray(profile.weak_spots) AND
|
|
20
|
+
* Array.isArray(profile.strong_spots) AND
|
|
21
|
+
* (profile.weak_spots.length + profile.strong_spots.length) >= 3 AND
|
|
22
|
+
* roleBlend AND Object.values(roleBlend).some(v => v >= 0.3)
|
|
23
|
+
* - ELSE IF brainAvailable === true OR process.env.MINDRIAN_BRAIN_KEY -> tier-1
|
|
24
|
+
* - ELSE -> tier-0
|
|
25
|
+
*
|
|
26
|
+
* Topic-coverage backstop: even if rounds_total >= 100, if the union of
|
|
27
|
+
* weak_spots + strong_spots holds < 3 entries, the dispatcher falls back
|
|
28
|
+
* to Tier 1 (or Tier 0 on cold-start) so a thinly-populated profile cannot
|
|
29
|
+
* trigger persona-aware decoys against an unfit corpus. This is the R3
|
|
30
|
+
* mitigation called out in RESEARCH.md (DISCRETION-AMEND-02 final paragraph).
|
|
31
|
+
*
|
|
32
|
+
* Tier 0 (cold-start / Brain unreachable):
|
|
33
|
+
* Algorithmic perturbation drawn from the artifact's own vocabulary.
|
|
34
|
+
* Each decoy applies one of 5 axes in round-robin order:
|
|
35
|
+
* a. wrong_owner : swap subject (Phase 89 -> Phase 90)
|
|
36
|
+
* b. wrong_dependency : swap referenced module (Pinecone -> Neo4j)
|
|
37
|
+
* c. wrong_metric : swap a number/threshold (>=0.7 -> >=0.9)
|
|
38
|
+
* d. wrong_scope : add/remove qualifier (public -> private)
|
|
39
|
+
* e. wrong_invariant : flip an "always" / "never" claim
|
|
40
|
+
*
|
|
41
|
+
* Tier 1 (Brain reachable, profile immature):
|
|
42
|
+
* Returns Tier-0-style perturbation tagged with tier:'tier-1'. The Brain
|
|
43
|
+
* teaching-graph integration (framework chaining rules -> plausible-but-
|
|
44
|
+
* wrong substitutions calibrated against student-grading data) is deferred
|
|
45
|
+
* to a Phase 91 follow-on; this phase ships the structural placeholder
|
|
46
|
+
* so downstream consumers can route on tier label without code change.
|
|
47
|
+
*
|
|
48
|
+
* Tier 2 (cross-session, persona-aware):
|
|
49
|
+
* Reads weak_spots + strong_spots from the comprehension profile. The
|
|
50
|
+
* bias mechanism preferentially exercises perturbation axes that map to
|
|
51
|
+
* weak topics; reduces (but does not eliminate) frequency on axes that
|
|
52
|
+
* map to strong topics. Profile is graph-local per Canon Part 8.
|
|
53
|
+
*
|
|
54
|
+
* Comprehension profile path (per RESEARCH.md):
|
|
55
|
+
* <roomDir>/.mindrian/comprehension-profile-<sha256(user_id):16>.json
|
|
56
|
+
* Schema (verbatim):
|
|
57
|
+
* {
|
|
58
|
+
* schema_version: 1,
|
|
59
|
+
* rounds_total: number,
|
|
60
|
+
* weak_spots: [{ topic, miss_rate, n }, ...],
|
|
61
|
+
* strong_spots: [{ topic, hit_rate, n }, ...],
|
|
62
|
+
* last_calibration_round: ISO-string,
|
|
63
|
+
* last_round_seed: 16-hex,
|
|
64
|
+
* preferences: { decoy_opt_out, tier_pin: null|'tier-0'|'tier-1'|'tier-2' },
|
|
65
|
+
* }
|
|
66
|
+
*
|
|
67
|
+
* Distribution seed: sha256(artifactPath + '|' + roundSize + '|' + Date.now()).slice(0, 16).
|
|
68
|
+
*
|
|
69
|
+
* Decoy-ethics contract (room/decisions/decision-decoy-ethics.md):
|
|
70
|
+
* - Decoys NOT disclosed during the round; debrief Shape A discloses at round end.
|
|
71
|
+
* - profile.preferences.decoy_opt_out === true bypasses decoy injection
|
|
72
|
+
* (returns { decoys: [], tier_reason: 'decoy_opt_out_user_preference' }).
|
|
73
|
+
*
|
|
74
|
+
* Pure CJS, node built-ins only, zero deps (Phase 87 invariant).
|
|
75
|
+
* Canon Part 8: zero Brain egress; profile read is local only; no network IO.
|
|
76
|
+
*/
|
|
77
|
+
'use strict';
|
|
78
|
+
|
|
79
|
+
const fs = require('node:fs');
|
|
80
|
+
const crypto = require('node:crypto');
|
|
81
|
+
|
|
82
|
+
const PERTURBATION_AXES = [
|
|
83
|
+
'wrong_owner',
|
|
84
|
+
'wrong_dependency',
|
|
85
|
+
'wrong_metric',
|
|
86
|
+
'wrong_scope',
|
|
87
|
+
'wrong_invariant',
|
|
88
|
+
];
|
|
89
|
+
|
|
90
|
+
// Topic-coverage backstop: profile weak + strong spots must total at least
|
|
91
|
+
// this many entries before Tier 2 persona-aware bias kicks in. Below this,
|
|
92
|
+
// the profile is too thin to safely steer decoy selection.
|
|
93
|
+
const TOPIC_COVERAGE_MIN = 3;
|
|
94
|
+
|
|
95
|
+
// Profile maturity gate: rounds_total below this falls back to Tier 1 or 0.
|
|
96
|
+
const PROFILE_MATURITY_MIN = 100;
|
|
97
|
+
|
|
98
|
+
// Persona signal threshold: at least one role_blend axis must clear this
|
|
99
|
+
// before Tier 2 activates. Below this, the navigator lacks a strong
|
|
100
|
+
// enough lens to bias decoy generation against.
|
|
101
|
+
const PERSONA_AXIS_MIN = 0.3;
|
|
102
|
+
|
|
103
|
+
/**
|
|
104
|
+
* Determine which tier the round will run at, given the inputs.
|
|
105
|
+
* Pure function -- no IO. The reason string is surfaced back to callers
|
|
106
|
+
* (and the renderer's debrief Shape A) so the choice is auditable.
|
|
107
|
+
*/
|
|
108
|
+
function detectTier(args) {
|
|
109
|
+
const a = (args && typeof args === 'object') ? args : {};
|
|
110
|
+
const tierOverride = a.tierOverride;
|
|
111
|
+
if (tierOverride === 'tier-0' || tierOverride === 'tier-1' || tierOverride === 'tier-2') {
|
|
112
|
+
return { tier: tierOverride, reason: 'override' };
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
const profile = a.profile || null;
|
|
116
|
+
const roleBlend = a.roleBlend || null;
|
|
117
|
+
// Caller-supplied brainAvailable wins (boolean explicit). Env var only
|
|
118
|
+
// observed when caller leaves the field undefined. This protects test
|
|
119
|
+
// harnesses (and CI runs that ship MINDRIAN_BRAIN_KEY for unrelated
|
|
120
|
+
// commands) from accidentally promoting cold-start scenarios to Tier 1.
|
|
121
|
+
let brainAvailable;
|
|
122
|
+
if (a.brainAvailable === true) brainAvailable = true;
|
|
123
|
+
else if (a.brainAvailable === false) brainAvailable = false;
|
|
124
|
+
else brainAvailable = process.env.MINDRIAN_BRAIN_KEY ? true : false;
|
|
125
|
+
|
|
126
|
+
// Tier 2 gate: profile maturity + topic coverage + persona signal.
|
|
127
|
+
// All three must hold; any miss falls through to the Tier 1 / Tier 0 ladder.
|
|
128
|
+
if (profile
|
|
129
|
+
&& typeof profile.rounds_total === 'number'
|
|
130
|
+
&& profile.rounds_total >= PROFILE_MATURITY_MIN
|
|
131
|
+
&& Array.isArray(profile.weak_spots)
|
|
132
|
+
&& Array.isArray(profile.strong_spots)
|
|
133
|
+
&& (profile.weak_spots.length + profile.strong_spots.length) >= TOPIC_COVERAGE_MIN
|
|
134
|
+
&& roleBlend
|
|
135
|
+
&& typeof roleBlend === 'object'
|
|
136
|
+
&& Object.values(roleBlend).some(v => typeof v === 'number' && v >= PERSONA_AXIS_MIN)) {
|
|
137
|
+
return { tier: 'tier-2', reason: 'profile_mature_persona_signal' };
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
// Topic-coverage backstop / persona-signal backstop: profile may be
|
|
141
|
+
// mature but coverage or signal too thin -- prefer Tier 1 over Tier 2.
|
|
142
|
+
if (brainAvailable) {
|
|
143
|
+
return { tier: 'tier-1', reason: 'brain_reachable_profile_immature' };
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
return { tier: 'tier-0', reason: 'brain_unreachable_or_cold_start' };
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
/**
|
|
150
|
+
* Pull other phase-IDs / module names / numbers from the artifact for
|
|
151
|
+
* plausible swaps. Used by perturbReal() to keep wrong-X variants drawn
|
|
152
|
+
* from the artifact's own vocabulary (per RESEARCH.md DISCRETION-AMEND-02
|
|
153
|
+
* "Perturbation MUST stay plausible: drawn from artifact's own vocabulary,
|
|
154
|
+
* still parses as grammatical user-story, detectable on careful read but
|
|
155
|
+
* invisible on skim").
|
|
156
|
+
*/
|
|
157
|
+
function extractVocab(line, fullArtifact) {
|
|
158
|
+
const phaseIdsAll = fullArtifact.match(/Phase\s+\d+(?:\.\d+)?/g) || [];
|
|
159
|
+
const moduleNamesAll = fullArtifact.match(/[a-z][a-z-]+\.cjs|[A-Z][a-zA-Z]+\b/g) || [];
|
|
160
|
+
const numbersAll = fullArtifact.match(/\d+(?:\.\d+)?/g) || [];
|
|
161
|
+
return {
|
|
162
|
+
phase_ids: Array.from(new Set(phaseIdsAll)),
|
|
163
|
+
module_names: Array.from(new Set(moduleNamesAll)),
|
|
164
|
+
numbers: Array.from(new Set(numbersAll)),
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
/**
|
|
169
|
+
* Parse the artifact for "load-bearing claims" -- bullet lines and lines
|
|
170
|
+
* with a colon, since those most often state user-story-style assertions
|
|
171
|
+
* in PLAN.md / CONTEXT.md / SUMMARY.md format.
|
|
172
|
+
*/
|
|
173
|
+
function loadArtifactClaims(artifactPath) {
|
|
174
|
+
try {
|
|
175
|
+
if (typeof artifactPath !== 'string' || artifactPath.length === 0) return [];
|
|
176
|
+
const raw = fs.readFileSync(artifactPath, 'utf8');
|
|
177
|
+
const lines = raw.split('\n').filter(L => /^[-*]\s|:\s/.test(L) && L.length > 30 && L.length < 280);
|
|
178
|
+
return lines.map(L => ({ text: L.trim(), vocabulary_pool: extractVocab(L, raw) }));
|
|
179
|
+
} catch (_e) {
|
|
180
|
+
return [];
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
/**
|
|
185
|
+
* Apply ONE perturbation axis to a claim. The output stays plausible:
|
|
186
|
+
* it must still parse as a grammatical user-story so a skim reader cannot
|
|
187
|
+
* easily separate it from the real claims. The 5 axes match the RESEARCH.md
|
|
188
|
+
* verbatim list and are applied round-robin by selectDecoys().
|
|
189
|
+
*/
|
|
190
|
+
function perturbReal(claim, axis) {
|
|
191
|
+
const v = (claim && claim.vocabulary_pool) || {};
|
|
192
|
+
const text = (claim && typeof claim.text === 'string') ? claim.text : '';
|
|
193
|
+
let perturbed = text;
|
|
194
|
+
switch (axis) {
|
|
195
|
+
case 'wrong_owner': {
|
|
196
|
+
if (v.phase_ids && v.phase_ids.length >= 2) {
|
|
197
|
+
const a = v.phase_ids[0];
|
|
198
|
+
const b = v.phase_ids[1];
|
|
199
|
+
if (text.indexOf(a) !== -1) perturbed = text.replace(a, b);
|
|
200
|
+
else if (text.indexOf(b) !== -1) perturbed = text.replace(b, a);
|
|
201
|
+
else perturbed = text + ' (re-attributed to ' + b + ')';
|
|
202
|
+
} else {
|
|
203
|
+
perturbed = text.replace(/\bPhase\s+\d+(?:\.\d+)?/, 'Phase 0');
|
|
204
|
+
}
|
|
205
|
+
break;
|
|
206
|
+
}
|
|
207
|
+
case 'wrong_dependency': {
|
|
208
|
+
if (v.module_names && v.module_names.length >= 2) {
|
|
209
|
+
const a = v.module_names[0];
|
|
210
|
+
const b = v.module_names[1];
|
|
211
|
+
if (text.indexOf(a) !== -1) perturbed = text.replace(a, b);
|
|
212
|
+
else if (text.indexOf(b) !== -1) perturbed = text.replace(b, a);
|
|
213
|
+
else perturbed = text + ' (via ' + b + ')';
|
|
214
|
+
} else {
|
|
215
|
+
perturbed = text.replace(/Pinecone/, 'Neo4j');
|
|
216
|
+
}
|
|
217
|
+
break;
|
|
218
|
+
}
|
|
219
|
+
case 'wrong_metric': {
|
|
220
|
+
if (v.numbers && v.numbers.length >= 1) {
|
|
221
|
+
const original = v.numbers[0];
|
|
222
|
+
const n = parseFloat(original);
|
|
223
|
+
if (Number.isFinite(n)) {
|
|
224
|
+
const swap = (n < 1) ? (n + 0.2).toFixed(1) : String(n + 5);
|
|
225
|
+
perturbed = text.replace(original, swap);
|
|
226
|
+
}
|
|
227
|
+
} else {
|
|
228
|
+
perturbed = text.replace(/\b\d+\b/, '99');
|
|
229
|
+
}
|
|
230
|
+
break;
|
|
231
|
+
}
|
|
232
|
+
case 'wrong_scope': {
|
|
233
|
+
if (text.indexOf('public') !== -1) perturbed = text.replace('public', 'private');
|
|
234
|
+
else if (text.indexOf('private') !== -1) perturbed = text.replace('private', 'public');
|
|
235
|
+
else if (text.indexOf('only ') !== -1) perturbed = text.replace('only ', '');
|
|
236
|
+
else perturbed = text + ' (under restricted scope)';
|
|
237
|
+
break;
|
|
238
|
+
}
|
|
239
|
+
case 'wrong_invariant': {
|
|
240
|
+
if (text.indexOf('always') !== -1) perturbed = text.replace('always', 'never');
|
|
241
|
+
else if (text.indexOf('never') !== -1) perturbed = text.replace('never', 'always');
|
|
242
|
+
else if (/\bis\b/.test(text)) perturbed = text.replace(/\bis\b/, 'is not');
|
|
243
|
+
else if (/\bwill\b/.test(text)) perturbed = text.replace(/\bwill\b/, 'will not');
|
|
244
|
+
else perturbed = text + ' (assumption inverted)';
|
|
245
|
+
break;
|
|
246
|
+
}
|
|
247
|
+
default:
|
|
248
|
+
perturbed = text;
|
|
249
|
+
}
|
|
250
|
+
return perturbed;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
/**
|
|
254
|
+
* Tier 2 axis bias: re-order PERTURBATION_AXES so axes that match weak_spots
|
|
255
|
+
* topics are exercised first, axes that match strong_spots topics are pushed
|
|
256
|
+
* later. The ordering is stable -- if no topic mappings hit, the canonical
|
|
257
|
+
* round-robin order is preserved (no surprises in the cold-start slice of
|
|
258
|
+
* a Tier 2 round). The mapping uses simple substring containment, not a
|
|
259
|
+
* graph lookup; the topic vocabulary is closed (framework + relationship
|
|
260
|
+
* handles per Canon Part 8).
|
|
261
|
+
*/
|
|
262
|
+
function tier2BiasedAxes(weakSpots, strongSpots) {
|
|
263
|
+
const weakTopics = (Array.isArray(weakSpots) ? weakSpots : [])
|
|
264
|
+
.map(s => (s && typeof s.topic === 'string') ? s.topic.toLowerCase() : '')
|
|
265
|
+
.filter(s => s.length > 0);
|
|
266
|
+
const strongTopics = (Array.isArray(strongSpots) ? strongSpots : [])
|
|
267
|
+
.map(s => (s && typeof s.topic === 'string') ? s.topic.toLowerCase() : '')
|
|
268
|
+
.filter(s => s.length > 0);
|
|
269
|
+
|
|
270
|
+
// Heuristic mapping: each axis has affinity keywords. Score axes by
|
|
271
|
+
// weak-topic match (positive) minus strong-topic match (negative).
|
|
272
|
+
const axisAffinity = {
|
|
273
|
+
wrong_owner: ['owner', 'phase', 'subject', 'attribution'],
|
|
274
|
+
wrong_dependency: ['dependency', 'module', 'pinecone', 'neo4j', 'cascade'],
|
|
275
|
+
wrong_metric: ['metric', 'threshold', 'number', 'score'],
|
|
276
|
+
wrong_scope: ['scope', 'public', 'private', 'metadata'],
|
|
277
|
+
wrong_invariant: ['invariant', 'always', 'never', 'edges'],
|
|
278
|
+
};
|
|
279
|
+
|
|
280
|
+
function score(axis) {
|
|
281
|
+
const keys = axisAffinity[axis] || [];
|
|
282
|
+
let s = 0;
|
|
283
|
+
for (const k of keys) {
|
|
284
|
+
for (const t of weakTopics) if (t.indexOf(k) !== -1) s += 1;
|
|
285
|
+
for (const t of strongTopics) if (t.indexOf(k) !== -1) s -= 1;
|
|
286
|
+
}
|
|
287
|
+
return s;
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
// Stable sort by score descending; ties keep canonical order.
|
|
291
|
+
const indexed = PERTURBATION_AXES.map((axis, i) => ({ axis, i, s: score(axis) }));
|
|
292
|
+
indexed.sort((a, b) => (b.s - a.s) || (a.i - b.i));
|
|
293
|
+
return indexed.map(e => e.axis);
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
/**
|
|
297
|
+
* Generate the decoy set for a F.6 plan-review round.
|
|
298
|
+
*
|
|
299
|
+
* Signature (verbatim per RESEARCH.md DISCRETION-AMEND-02):
|
|
300
|
+
* selectDecoys({ artifactPath, roundSize=20, realCount, profile,
|
|
301
|
+
* roleBlend, brainAvailable, tierOverride })
|
|
302
|
+
* -> { decoys, tier, tier_reason, distribution_seed }
|
|
303
|
+
*
|
|
304
|
+
* decoys: Array<{ position, content, was_decoy: true, perturbation_axis,
|
|
305
|
+
* source_real_index }>
|
|
306
|
+
* position: -1 placeholder; the renderer assigns interleaved 1..N positions
|
|
307
|
+
* when composing the round (not this module's responsibility).
|
|
308
|
+
*
|
|
309
|
+
* Never throws. On any read error or malformed artifact, returns an empty
|
|
310
|
+
* decoy set with the detected tier label preserved so the renderer can
|
|
311
|
+
* still surface the round-state node.
|
|
312
|
+
*/
|
|
313
|
+
function selectDecoys(args) {
|
|
314
|
+
const a = (args && typeof args === 'object') ? args : {};
|
|
315
|
+
const artifactPath = a.artifactPath;
|
|
316
|
+
const roundSize = (typeof a.roundSize === 'number' && a.roundSize >= 15 && a.roundSize <= 30)
|
|
317
|
+
? a.roundSize : 20;
|
|
318
|
+
const realCount = (typeof a.realCount === 'number' && a.realCount >= 0 && a.realCount <= roundSize)
|
|
319
|
+
? a.realCount : Math.ceil(roundSize * 0.8);
|
|
320
|
+
const decoyCount = roundSize - realCount;
|
|
321
|
+
const profile = a.profile || null;
|
|
322
|
+
const roleBlend = a.roleBlend || null;
|
|
323
|
+
|
|
324
|
+
// decoy_opt_out preference -> empty decoy set. Tier label retained for
|
|
325
|
+
// debrief telemetry; tier_reason explicitly states the user opt-out.
|
|
326
|
+
if (profile && profile.preferences && profile.preferences.decoy_opt_out === true) {
|
|
327
|
+
return {
|
|
328
|
+
decoys: [],
|
|
329
|
+
tier: 'tier-0',
|
|
330
|
+
tier_reason: 'decoy_opt_out_user_preference',
|
|
331
|
+
distribution_seed: '0000000000000000',
|
|
332
|
+
};
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
const detected = detectTier({
|
|
336
|
+
profile: profile,
|
|
337
|
+
roleBlend: roleBlend,
|
|
338
|
+
brainAvailable: a.brainAvailable,
|
|
339
|
+
tierOverride: a.tierOverride,
|
|
340
|
+
});
|
|
341
|
+
|
|
342
|
+
const claims = loadArtifactClaims(artifactPath);
|
|
343
|
+
|
|
344
|
+
// Distribution seed: deterministic-ish hash of inputs + wall-clock so
|
|
345
|
+
// a re-run on the same artifact produces a different round, but the
|
|
346
|
+
// shape is auditable (16 hex chars, never variable-length).
|
|
347
|
+
const seed = crypto.createHash('sha256')
|
|
348
|
+
.update(String(artifactPath || '') + '|' + String(roundSize) + '|' + String(Date.now()))
|
|
349
|
+
.digest('hex').slice(0, 16);
|
|
350
|
+
|
|
351
|
+
// Choose the axis ordering. Tier 2 biases by weak/strong spots; Tier 1
|
|
352
|
+
// and Tier 0 use the canonical round-robin order verbatim from PERTURBATION_AXES.
|
|
353
|
+
const axisOrder = (detected.tier === 'tier-2')
|
|
354
|
+
? tier2BiasedAxes(profile.weak_spots, profile.strong_spots)
|
|
355
|
+
: PERTURBATION_AXES.slice();
|
|
356
|
+
|
|
357
|
+
const decoys = [];
|
|
358
|
+
for (let i = 0; i < decoyCount; i++) {
|
|
359
|
+
const claim = claims[i % Math.max(1, claims.length)] || {
|
|
360
|
+
text: 'placeholder claim ' + i,
|
|
361
|
+
vocabulary_pool: {},
|
|
362
|
+
};
|
|
363
|
+
const axis = axisOrder[i % axisOrder.length];
|
|
364
|
+
const content = perturbReal(claim, axis);
|
|
365
|
+
decoys.push({
|
|
366
|
+
position: -1, // renderer assigns interleaved position
|
|
367
|
+
content: content,
|
|
368
|
+
was_decoy: true,
|
|
369
|
+
perturbation_axis: axis,
|
|
370
|
+
source_real_index: i,
|
|
371
|
+
});
|
|
372
|
+
}
|
|
373
|
+
|
|
374
|
+
return {
|
|
375
|
+
decoys: decoys,
|
|
376
|
+
tier: detected.tier,
|
|
377
|
+
tier_reason: detected.reason,
|
|
378
|
+
distribution_seed: seed,
|
|
379
|
+
};
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
module.exports = {
|
|
383
|
+
selectDecoys: selectDecoys,
|
|
384
|
+
_internal: {
|
|
385
|
+
detectTier: detectTier,
|
|
386
|
+
perturbReal: perturbReal,
|
|
387
|
+
loadArtifactClaims: loadArtifactClaims,
|
|
388
|
+
extractVocab: extractVocab,
|
|
389
|
+
tier2BiasedAxes: tier2BiasedAxes,
|
|
390
|
+
PERTURBATION_AXES: PERTURBATION_AXES.slice(),
|
|
391
|
+
TOPIC_COVERAGE_MIN: TOPIC_COVERAGE_MIN,
|
|
392
|
+
PROFILE_MATURITY_MIN: PROFILE_MATURITY_MIN,
|
|
393
|
+
PERSONA_AXIS_MIN: PERSONA_AXIS_MIN,
|
|
394
|
+
},
|
|
395
|
+
};
|
|
@@ -0,0 +1,219 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2026 Mindrian. BSL 1.1.
|
|
3
|
+
*
|
|
4
|
+
* Phase 100-02 Task 2 -- Heuristic JTBD Classifier
|
|
5
|
+
* =================================================
|
|
6
|
+
* Phase 100's central computational primitive. Reads the 13-entry
|
|
7
|
+
* taxonomy (lib/hmi/jtbd-taxonomy.json), the active operator
|
|
8
|
+
* (Phase 99 lib/conversation/operator.cjs), the user's latest
|
|
9
|
+
* message, and the last N decisions from STATE.md. Returns a typed
|
|
10
|
+
* JTBD assignment with confidence and an evidence chain.
|
|
11
|
+
*
|
|
12
|
+
* Heuristic only. NO LLM round-trip per CONTEXT D-03.
|
|
13
|
+
* LOCAL-only per Canon Part 8 -- zero remote-methodology /
|
|
14
|
+
* vector-db / model-context-protocol imports or references
|
|
15
|
+
* (boundary tokens omitted by name; see CONTEXT Part 8 for the
|
|
16
|
+
* full forbidden-substring list this file must not match).
|
|
17
|
+
*
|
|
18
|
+
* Public API:
|
|
19
|
+
* classify({ userMessage, room, operator, decisionsRecency, currentJtbd })
|
|
20
|
+
* -> { jtbd: string|null, confidence: number, evidence: string[] }
|
|
21
|
+
*
|
|
22
|
+
* Algorithm (per RESEARCH §3):
|
|
23
|
+
* Three input strata, weighted 0.5 / 0.3 / 0.2:
|
|
24
|
+
* Stratum 1 (tokens): lowercase userMessage substring-matched
|
|
25
|
+
* against each JTBD's cue list. Normalized
|
|
26
|
+
* min(matches, 3) / 3, multiplied by 0.5.
|
|
27
|
+
* Stratum 2 (operator): if active operator in JTBD's
|
|
28
|
+
* operator_affinity[], add 0.3 (flat).
|
|
29
|
+
* Stratum 3 (recency): count of last-3-decisions whose
|
|
30
|
+
* methodology_hook matches one of the JTBD's
|
|
31
|
+
* methodology_hooks[]. Normalized
|
|
32
|
+
* min(matches, 3) / 3, multiplied by 0.2.
|
|
33
|
+
*
|
|
34
|
+
* Hysteresis: +0.1 to currentJtbd if set (sticky bias).
|
|
35
|
+
* Threshold: 0.6 default; 0.8 if operator === 'JUST_TALK'.
|
|
36
|
+
*
|
|
37
|
+
* Below threshold -> { jtbd: null, confidence: top.score,
|
|
38
|
+
* evidence: ['below_threshold',
|
|
39
|
+
* `top:${top.id}@${top.score}`] }
|
|
40
|
+
* Above threshold -> { jtbd: top.id, confidence: top.score,
|
|
41
|
+
* evidence: [`tokens:${tokens}`,
|
|
42
|
+
* `operator:${operator||'none'}`,
|
|
43
|
+
* `recency:${recency}`] }
|
|
44
|
+
* No room -> { jtbd: null, confidence: 0,
|
|
45
|
+
* evidence: ['no_room'] }
|
|
46
|
+
* Classifier err -> { jtbd: null, confidence: 0,
|
|
47
|
+
* evidence: ['classifier-error', errMsg.slice(0,80)] }
|
|
48
|
+
*
|
|
49
|
+
* Performance:
|
|
50
|
+
* - Cue regexes compiled once on module load (cache map).
|
|
51
|
+
* - Taxonomy lazy-loaded via require (Node caches automatically).
|
|
52
|
+
* - Target: < 5ms warm on a 100-char message + 13-JTBD scoring.
|
|
53
|
+
*
|
|
54
|
+
* License: BSL 1.1.
|
|
55
|
+
*/
|
|
56
|
+
|
|
57
|
+
'use strict';
|
|
58
|
+
|
|
59
|
+
// ---------- Taxonomy + cue cache ----------
|
|
60
|
+
|
|
61
|
+
const TAXONOMY = require('./jtbd-taxonomy.json');
|
|
62
|
+
|
|
63
|
+
// Compile cue substrings (lowercased, unique) once per module load.
|
|
64
|
+
// We use plain substring includes() rather than RegExp; substring is
|
|
65
|
+
// faster, deterministic, and identical in semantics for cue cards.
|
|
66
|
+
const CUE_CACHE = new Map(); // jtbdId -> Array<lowerCase string>
|
|
67
|
+
const HOOKS_CACHE = new Map(); // jtbdId -> Set<methodology hook string>
|
|
68
|
+
const AFFINITY_CACHE = new Map(); // jtbdId -> Set<operator>
|
|
69
|
+
|
|
70
|
+
// Taxonomy ships JTBD entries under `entries` (canonical key per
|
|
71
|
+
// 100-01 jtbd-taxonomy.json). We accept legacy `jtbds` for forward
|
|
72
|
+
// tolerance but `entries` is authoritative.
|
|
73
|
+
function _taxonomyEntries() {
|
|
74
|
+
if (!TAXONOMY) return [];
|
|
75
|
+
if (Array.isArray(TAXONOMY.entries)) return TAXONOMY.entries;
|
|
76
|
+
if (Array.isArray(TAXONOMY.jtbds)) return TAXONOMY.jtbds;
|
|
77
|
+
return [];
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
(function buildCaches() {
|
|
81
|
+
const list = _taxonomyEntries();
|
|
82
|
+
if (list.length === 0) return;
|
|
83
|
+
for (const jtbd of list) {
|
|
84
|
+
if (!jtbd || typeof jtbd.id !== 'string') continue;
|
|
85
|
+
const cues = Array.isArray(jtbd.cues) ? jtbd.cues : [];
|
|
86
|
+
CUE_CACHE.set(jtbd.id, cues.map(c => String(c).toLowerCase()));
|
|
87
|
+
const hooks = Array.isArray(jtbd.methodology_hooks) ? jtbd.methodology_hooks : [];
|
|
88
|
+
HOOKS_CACHE.set(jtbd.id, new Set(hooks));
|
|
89
|
+
const aff = Array.isArray(jtbd.operator_affinity) ? jtbd.operator_affinity : [];
|
|
90
|
+
AFFINITY_CACHE.set(jtbd.id, new Set(aff));
|
|
91
|
+
}
|
|
92
|
+
})();
|
|
93
|
+
|
|
94
|
+
// ---------- Helpers ----------
|
|
95
|
+
|
|
96
|
+
function normalizeMatchCount(n) {
|
|
97
|
+
// min(n, 3) / 3 -- saturating normalization
|
|
98
|
+
if (!Number.isFinite(n) || n < 0) return 0;
|
|
99
|
+
return Math.min(n, 3) / 3;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
function countCueMatches(lowerMsg, cues) {
|
|
103
|
+
if (!cues || cues.length === 0) return 0;
|
|
104
|
+
let n = 0;
|
|
105
|
+
for (const cue of cues) {
|
|
106
|
+
if (lowerMsg.includes(cue)) n += 1;
|
|
107
|
+
}
|
|
108
|
+
return n;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
function countRecencyMatches(decisions, hooksSet) {
|
|
112
|
+
if (!Array.isArray(decisions) || decisions.length === 0) return 0;
|
|
113
|
+
if (!hooksSet || hooksSet.size === 0) return 0;
|
|
114
|
+
let n = 0;
|
|
115
|
+
for (const d of decisions) {
|
|
116
|
+
if (!d) continue;
|
|
117
|
+
if (typeof d.methodology_hook === 'string' && hooksSet.has(d.methodology_hook)) {
|
|
118
|
+
n += 1;
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
return n;
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
// ---------- classify ----------
|
|
125
|
+
|
|
126
|
+
function classify(input) {
|
|
127
|
+
// Defensive null/undefined guard
|
|
128
|
+
if (!input || typeof input !== 'object') {
|
|
129
|
+
return { jtbd: null, confidence: 0, evidence: ['classifier-error', 'no_input'] };
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
try {
|
|
133
|
+
const { userMessage, room, operator, decisionsRecency, currentJtbd } = input;
|
|
134
|
+
|
|
135
|
+
// No-room edge
|
|
136
|
+
if (!room) {
|
|
137
|
+
return { jtbd: null, confidence: 0, evidence: ['no_room'] };
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
// Lowercased message; tolerate undefined / non-string
|
|
141
|
+
const msg = (typeof userMessage === 'string') ? userMessage.toLowerCase() : '';
|
|
142
|
+
|
|
143
|
+
// Score each JTBD
|
|
144
|
+
const scores = {};
|
|
145
|
+
const tokenCounts = {};
|
|
146
|
+
const recencyCounts = {};
|
|
147
|
+
|
|
148
|
+
const taxonomyList = _taxonomyEntries();
|
|
149
|
+
if (taxonomyList.length === 0) {
|
|
150
|
+
return { jtbd: null, confidence: 0, evidence: ['classifier-error', 'taxonomy_missing'] };
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
for (const jtbd of taxonomyList) {
|
|
154
|
+
const id = jtbd.id;
|
|
155
|
+
const cues = CUE_CACHE.get(id) || [];
|
|
156
|
+
const hooks = HOOKS_CACHE.get(id) || new Set();
|
|
157
|
+
const aff = AFFINITY_CACHE.get(id) || new Set();
|
|
158
|
+
|
|
159
|
+
// Stratum 1 (tokens, weight 0.5)
|
|
160
|
+
const tCount = countCueMatches(msg, cues);
|
|
161
|
+
tokenCounts[id] = tCount;
|
|
162
|
+
let s = 0.5 * normalizeMatchCount(tCount);
|
|
163
|
+
|
|
164
|
+
// Stratum 2 (operator, flat 0.3)
|
|
165
|
+
if (operator && aff.has(operator)) {
|
|
166
|
+
s += 0.3;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
// Stratum 3 (recency, weight 0.2)
|
|
170
|
+
const rCount = countRecencyMatches(decisionsRecency, hooks);
|
|
171
|
+
recencyCounts[id] = rCount;
|
|
172
|
+
s += 0.2 * normalizeMatchCount(rCount);
|
|
173
|
+
|
|
174
|
+
scores[id] = s;
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
// Hysteresis -- sticky +0.1 toward currentJtbd if set
|
|
178
|
+
if (currentJtbd && Object.prototype.hasOwnProperty.call(scores, currentJtbd)) {
|
|
179
|
+
scores[currentJtbd] += 0.1;
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
// Pick top
|
|
183
|
+
let topId = null;
|
|
184
|
+
let topScore = -Infinity;
|
|
185
|
+
for (const id of Object.keys(scores)) {
|
|
186
|
+
if (scores[id] > topScore) {
|
|
187
|
+
topScore = scores[id];
|
|
188
|
+
topId = id;
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
// Threshold (raise to 0.8 in JUST_TALK)
|
|
193
|
+
const threshold = (operator === 'JUST_TALK') ? 0.8 : 0.6;
|
|
194
|
+
|
|
195
|
+
if (topId === null || topScore < threshold) {
|
|
196
|
+
const topRound = Number.isFinite(topScore) ? topScore.toFixed(2) : '0.00';
|
|
197
|
+
return {
|
|
198
|
+
jtbd: null,
|
|
199
|
+
confidence: Number.isFinite(topScore) ? topScore : 0,
|
|
200
|
+
evidence: ['below_threshold', `top:${topId || 'none'}@${topRound}`],
|
|
201
|
+
};
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
return {
|
|
205
|
+
jtbd: topId,
|
|
206
|
+
confidence: topScore,
|
|
207
|
+
evidence: [
|
|
208
|
+
`tokens:${tokenCounts[topId] || 0}`,
|
|
209
|
+
`operator:${operator || 'none'}`,
|
|
210
|
+
`recency:${recencyCounts[topId] || 0}`,
|
|
211
|
+
],
|
|
212
|
+
};
|
|
213
|
+
} catch (err) {
|
|
214
|
+
const m = (err && err.message) ? String(err.message).slice(0, 80) : 'unknown';
|
|
215
|
+
return { jtbd: null, confidence: 0, evidence: ['classifier-error', m] };
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
module.exports = { classify };
|