@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,731 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2026 Mindrian. BSL 1.1.
|
|
3
|
+
* Phase 89.2 Plan 04 -- Industry external-egress fetcher.
|
|
4
|
+
* Phase 94 Plan 05 amendment (2026-04-28): paid -> native -> cache
|
|
5
|
+
* fallback chain via opts.tavily / opts.webSearch / opts.cacheReader
|
|
6
|
+
* injection seams. Envelope return shape is now
|
|
7
|
+
* {tier, source, results, signals, telemetry}
|
|
8
|
+
* with `signals` preserved for backward compat with rs-discovery-
|
|
9
|
+
* engine line 380 industry.signals consumer. tier is one of
|
|
10
|
+
* 'paid' | 'native' | 'cache'
|
|
11
|
+
* source is one of
|
|
12
|
+
* 'tavily' | 'websearch' | 'cache'
|
|
13
|
+
* Per Canon Part 4 each tier transition becomes graph data; per
|
|
14
|
+
* Canon Part 7 the fallback reuses Anthropic native WebSearch
|
|
15
|
+
* instead of building a new client; per Canon Part 8 WebSearch
|
|
16
|
+
* carries user-typed query bytes only (public SIGNAL channel) and
|
|
17
|
+
* the existing brain-client chokepoint is untouched.
|
|
18
|
+
*
|
|
19
|
+
* Single source: Tavily orchestration. Per CONTEXT.md: "Crunchbase +
|
|
20
|
+
* corporate R&D + startup trackers (Tavily-orchestrated)". Tavily is
|
|
21
|
+
* the search-orchestration tier; we do NOT re-implement Crunchbase
|
|
22
|
+
* direct. Each user query expands to 3 refined sub-queries that target
|
|
23
|
+
* startup-database content via Tavily's general web crawl.
|
|
24
|
+
*
|
|
25
|
+
* Single chokepoint: buildIndustryQuery(query, opts) is the ONLY function
|
|
26
|
+
* that constructs an outbound URL + body. Two layers of Canon Part 8 audit
|
|
27
|
+
* fire BEFORE the request is built:
|
|
28
|
+
* Layer 1: auditQueryString(query, 'industry') on the user query
|
|
29
|
+
* Layer 2: auditQueryString(refined, 'industry') on each refined sub-query
|
|
30
|
+
*
|
|
31
|
+
* The two-layer audit is load-bearing: a clean user query can still be
|
|
32
|
+
* combined with an opts.refinement_template_override that smuggles a
|
|
33
|
+
* forbidden pattern into the refined sub-query. Layer 2 catches that.
|
|
34
|
+
* Both Test 10 (meeting-via-override) and Test 11 (phone-via-override)
|
|
35
|
+
* exercise the second layer.
|
|
36
|
+
*
|
|
37
|
+
* If TAVILY_API_KEY env var is absent: graceful degradation. fetchIndustry
|
|
38
|
+
* returns {signals: [], telemetry: [{source:'tavily', status:'api_key_missing'}]};
|
|
39
|
+
* never throws. Mirrors the api-key-missing pattern from rs-fetcher-academic.
|
|
40
|
+
*
|
|
41
|
+
* Per-source rate-limit graceful degradation per Phase 88.6-03 pattern
|
|
42
|
+
* (mirrors lib/core/rs-fetcher-patents.cjs byte-for-byte):
|
|
43
|
+
* 429 / 503 -> recordTelemetry(status='rate_limited') + return empty
|
|
44
|
+
* timeout -> recordTelemetry(status='timeout') + continue
|
|
45
|
+
* parse err -> recordTelemetry(status='api_error') + continue
|
|
46
|
+
* budget==0 -> skip Tavily (synthetic 'budget_exhausted' on returned
|
|
47
|
+
* telemetry; not in v1 ALLOWED_STATUSES so not persisted)
|
|
48
|
+
*
|
|
49
|
+
* NEVER throws on rate-limit. ONLY throws on Canon Part 8 violation.
|
|
50
|
+
*
|
|
51
|
+
* Network: native Node 18+ global fetch. AbortController gives a per-request
|
|
52
|
+
* 10s default timeout. NO node-fetch, NO axios, NO additional npm dep.
|
|
53
|
+
*
|
|
54
|
+
* Output shape per signal (after dedupe):
|
|
55
|
+
* { company, signal, source, url, fetched_at }
|
|
56
|
+
*
|
|
57
|
+
* Pure CJS, zero npm deps, node built-ins only beyond the three rs-egress-*
|
|
58
|
+
* primitives shipped in Wave 1 (Plan 89.2-01).
|
|
59
|
+
*/
|
|
60
|
+
'use strict';
|
|
61
|
+
|
|
62
|
+
const crypto = require('node:crypto');
|
|
63
|
+
|
|
64
|
+
const { ExternalEgressViolation } = require('./rs-egress-violations.cjs');
|
|
65
|
+
const { auditQueryString } = require('./rs-egress-prompts.cjs');
|
|
66
|
+
const {
|
|
67
|
+
recordTelemetry,
|
|
68
|
+
computeRemainingBudget,
|
|
69
|
+
DEFAULT_BUDGETS,
|
|
70
|
+
} = require('./rs-egress-telemetry.cjs');
|
|
71
|
+
|
|
72
|
+
// ---------- Frozen invariants ----------
|
|
73
|
+
|
|
74
|
+
const SOURCES = Object.freeze(['tavily']);
|
|
75
|
+
|
|
76
|
+
// Tavily is env-key-gated. If TAVILY_API_KEY absent: graceful degradation.
|
|
77
|
+
const SOURCE_ENV_VARS = Object.freeze({
|
|
78
|
+
tavily: 'TAVILY_API_KEY',
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
const DEFAULT_TIMEOUT_MS = 10000;
|
|
82
|
+
|
|
83
|
+
const USER_AGENT = 'MindrianOS-Plugin/1.11.0 (https://github.com/jsagir/mindrian-os-plugin)';
|
|
84
|
+
|
|
85
|
+
// Tavily Search API endpoint. POST with JSON body carrying api_key + query
|
|
86
|
+
// + max_results + search_depth.
|
|
87
|
+
const ENDPOINT_TAVILY = 'https://api.tavily.com/search';
|
|
88
|
+
|
|
89
|
+
// Frozen-order refinement templates. Three sub-queries per user query so
|
|
90
|
+
// Tavily's web crawl gets variation while staying anchored on the user's
|
|
91
|
+
// intent. Order is deterministic so same input -> byte-identical fetch
|
|
92
|
+
// sequence (Test 2 dedup determinism fence).
|
|
93
|
+
const REFINEMENT_TEMPLATES = Object.freeze([
|
|
94
|
+
'{query} startup OR company OR venture site:crunchbase.com OR site:pitchbook.com',
|
|
95
|
+
'{query} corporate research lab OR R&D announcement',
|
|
96
|
+
'{query} early stage funding OR seed round OR series A OR series B',
|
|
97
|
+
]);
|
|
98
|
+
|
|
99
|
+
const MAX_RESULTS_PER_SUBQUERY = 5;
|
|
100
|
+
|
|
101
|
+
// signal.signal field is capped at 200 chars per CONTEXT.md spec.
|
|
102
|
+
const SIGNAL_MAX_CHARS = 200;
|
|
103
|
+
|
|
104
|
+
// ---------- buildIndustryQuery (THE chokepoint) ----------
|
|
105
|
+
//
|
|
106
|
+
// This is the ONLY function in the module that constructs an outbound URL.
|
|
107
|
+
// Every dispatcher invokes it. auditQueryString runs at TWO layers:
|
|
108
|
+
// Layer 1: on the user query (clean? no forbidden pattern in the input?)
|
|
109
|
+
// Layer 2: on each refined sub-query (clean? no forbidden pattern after
|
|
110
|
+
// template substitution? defends against opts-override smuggling)
|
|
111
|
+
//
|
|
112
|
+
// Returns one of:
|
|
113
|
+
// { skip: true, reason: 'api_key_missing' } when TAVILY_API_KEY absent
|
|
114
|
+
// { skip: false, url, method, headers, refined_subqueries[] } when ready to fetch
|
|
115
|
+
//
|
|
116
|
+
// Throws:
|
|
117
|
+
// TypeError if query is not a non-empty string
|
|
118
|
+
// ExternalEgressViolation if user query OR any refined sub-query matches
|
|
119
|
+
// FORBIDDEN_PATTERNS
|
|
120
|
+
|
|
121
|
+
function buildIndustryQuery(query, opts) {
|
|
122
|
+
if (typeof query !== 'string' || query.length === 0) {
|
|
123
|
+
throw new TypeError('buildIndustryQuery: query must be a non-empty string');
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
// Canon Part 8 layer 1: throws on adversarial USER QUERY before any
|
|
127
|
+
// refined sub-query is computed. Test 8 + Test 9 fences.
|
|
128
|
+
auditQueryString(query, 'industry');
|
|
129
|
+
|
|
130
|
+
// Env-var gate. TAVILY_API_KEY absent -> graceful skip.
|
|
131
|
+
const envVar = SOURCE_ENV_VARS.tavily;
|
|
132
|
+
if (envVar && !process.env[envVar]) {
|
|
133
|
+
return { skip: true, reason: 'api_key_missing' };
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
opts = opts || {};
|
|
137
|
+
// The opts.refinement_template_override is allowed for testing the
|
|
138
|
+
// second-layer audit; in prod always REFINEMENT_TEMPLATES. The override
|
|
139
|
+
// path is what makes the two-layer defense necessary -- a clean user
|
|
140
|
+
// query can still produce a refined sub-query that contains a forbidden
|
|
141
|
+
// pattern when combined with an adversarial template.
|
|
142
|
+
const templates = (typeof opts.refinement_template_override === 'string'
|
|
143
|
+
&& opts.refinement_template_override.length > 0)
|
|
144
|
+
? [opts.refinement_template_override]
|
|
145
|
+
: REFINEMENT_TEMPLATES;
|
|
146
|
+
|
|
147
|
+
const refined = [];
|
|
148
|
+
for (const tmpl of templates) {
|
|
149
|
+
if (typeof tmpl !== 'string') {
|
|
150
|
+
throw new TypeError('buildIndustryQuery: refinement template must be a string');
|
|
151
|
+
}
|
|
152
|
+
const sub = tmpl.replace('{query}', query);
|
|
153
|
+
|
|
154
|
+
// Canon Part 8 layer 2: throws on adversarial REFINED SUB-QUERY before
|
|
155
|
+
// the URL/body is constructed. Test 10 (meeting-via-override) + Test 11
|
|
156
|
+
// (phone-via-override) fences.
|
|
157
|
+
auditQueryString(sub, 'industry');
|
|
158
|
+
|
|
159
|
+
refined.push(sub);
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
const headers = {
|
|
163
|
+
'User-Agent': USER_AGENT,
|
|
164
|
+
'Accept': 'application/json',
|
|
165
|
+
'Content-Type': 'application/json',
|
|
166
|
+
};
|
|
167
|
+
|
|
168
|
+
return {
|
|
169
|
+
skip: false,
|
|
170
|
+
url: ENDPOINT_TAVILY,
|
|
171
|
+
method: 'POST',
|
|
172
|
+
headers: headers,
|
|
173
|
+
source: 'tavily',
|
|
174
|
+
refined_subqueries: refined,
|
|
175
|
+
max_results_per_subquery: (typeof opts.maxResultsPerSubquery === 'number'
|
|
176
|
+
&& opts.maxResultsPerSubquery > 0)
|
|
177
|
+
? opts.maxResultsPerSubquery
|
|
178
|
+
: MAX_RESULTS_PER_SUBQUERY,
|
|
179
|
+
};
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
// ---------- buildTavilyBody ----------
|
|
183
|
+
//
|
|
184
|
+
// Helper for the per-sub-query POST body. The API key is read from env at
|
|
185
|
+
// call time so the test suite's setupScopedHome can mutate it between
|
|
186
|
+
// scenarios without restarting the module.
|
|
187
|
+
|
|
188
|
+
function buildTavilyBody(refinedSubquery, maxResults) {
|
|
189
|
+
return JSON.stringify({
|
|
190
|
+
api_key: process.env.TAVILY_API_KEY,
|
|
191
|
+
query: refinedSubquery,
|
|
192
|
+
max_results: maxResults,
|
|
193
|
+
search_depth: 'advanced',
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
// ---------- fetchWithTimeout (the ONE native fetch call site) ----------
|
|
198
|
+
//
|
|
199
|
+
// Per the chokepoint exclusivity rule: every dispatcher routes through
|
|
200
|
+
// this helper. This is the only place that touches global.fetch.
|
|
201
|
+
// Returns the raw Response on success; throws on network error or timeout.
|
|
202
|
+
|
|
203
|
+
async function fetchWithTimeout(req, opts) {
|
|
204
|
+
const timeoutMs = (opts && typeof opts.timeoutMs === 'number') ? opts.timeoutMs : DEFAULT_TIMEOUT_MS;
|
|
205
|
+
const controller = new AbortController();
|
|
206
|
+
const t = setTimeout(function () { controller.abort(); }, timeoutMs);
|
|
207
|
+
try {
|
|
208
|
+
const init = {
|
|
209
|
+
method: req.method,
|
|
210
|
+
headers: req.headers,
|
|
211
|
+
signal: controller.signal,
|
|
212
|
+
};
|
|
213
|
+
if (typeof req.body === 'string') {
|
|
214
|
+
init.body = req.body;
|
|
215
|
+
}
|
|
216
|
+
const res = await fetch(req.url, init);
|
|
217
|
+
return res;
|
|
218
|
+
} finally {
|
|
219
|
+
clearTimeout(t);
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
// ---------- parseRateLimit ----------
|
|
224
|
+
//
|
|
225
|
+
// Pulls a remaining-budget signal out of common HTTP headers. Returns
|
|
226
|
+
// undefined if absent.
|
|
227
|
+
|
|
228
|
+
function parseRateLimit(headers) {
|
|
229
|
+
if (!headers) return undefined;
|
|
230
|
+
const probes = ['x-ratelimit-remaining', 'X-RateLimit-Remaining', 'ratelimit-remaining'];
|
|
231
|
+
for (const k of probes) {
|
|
232
|
+
let v;
|
|
233
|
+
if (typeof headers.get === 'function') {
|
|
234
|
+
v = headers.get(k);
|
|
235
|
+
} else if (Object.prototype.hasOwnProperty.call(headers, k)) {
|
|
236
|
+
v = headers[k];
|
|
237
|
+
}
|
|
238
|
+
if (v !== undefined && v !== null) {
|
|
239
|
+
const n = parseInt(String(v), 10);
|
|
240
|
+
if (!Number.isNaN(n)) return n;
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
return undefined;
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
// ---------- extractCompany ----------
|
|
247
|
+
//
|
|
248
|
+
// Deterministic company-name extraction from Tavily result. Strategy:
|
|
249
|
+
// 1. If url has a non-trivial second-level domain, derive a Title-Cased
|
|
250
|
+
// company-like token from it (e.g. example-corp.com -> Example-Corp).
|
|
251
|
+
// 2. Else fall back to the first capitalized token from the title.
|
|
252
|
+
// 3. Else return 'Unknown' so the field is never empty (Test 1 fence).
|
|
253
|
+
//
|
|
254
|
+
// NO LLM, NO heuristics that depend on global state. Pure function.
|
|
255
|
+
|
|
256
|
+
function extractCompany(url, title) {
|
|
257
|
+
// Strategy 1: domain-based
|
|
258
|
+
if (typeof url === 'string' && url.length > 0) {
|
|
259
|
+
let host = '';
|
|
260
|
+
try {
|
|
261
|
+
const parsed = new URL(url);
|
|
262
|
+
host = parsed.hostname || '';
|
|
263
|
+
} catch (_e) { /* fall through */ }
|
|
264
|
+
if (host) {
|
|
265
|
+
// Strip leading www. then take the SLD (e.g. 'foo.example.com' -> 'example').
|
|
266
|
+
const stripped = host.replace(/^www\./, '');
|
|
267
|
+
const parts = stripped.split('.');
|
|
268
|
+
// 'example.com' -> ['example','com']; 'sub.example.com' -> ['sub','example','com']
|
|
269
|
+
let sld = '';
|
|
270
|
+
if (parts.length >= 2) {
|
|
271
|
+
sld = parts[parts.length - 2];
|
|
272
|
+
} else if (parts.length === 1) {
|
|
273
|
+
sld = parts[0];
|
|
274
|
+
}
|
|
275
|
+
if (sld && sld.length > 0 && sld !== 'example') {
|
|
276
|
+
// Title-case the SLD; preserve dashes.
|
|
277
|
+
const cased = sld.split('-').map(function (seg) {
|
|
278
|
+
return seg.length > 0 ? (seg.charAt(0).toUpperCase() + seg.slice(1).toLowerCase()) : seg;
|
|
279
|
+
}).join('-');
|
|
280
|
+
if (cased.length > 0) return cased;
|
|
281
|
+
}
|
|
282
|
+
if (sld === 'example') {
|
|
283
|
+
// Distinguish example-foo subdomains from generic example.com.
|
|
284
|
+
// For test fixtures: example-abc1234.com -> Example-Abc1234.
|
|
285
|
+
const cased = sld.charAt(0).toUpperCase() + sld.slice(1);
|
|
286
|
+
return cased;
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
}
|
|
290
|
+
// Strategy 2: first capitalized token from title
|
|
291
|
+
if (typeof title === 'string' && title.length > 0) {
|
|
292
|
+
const tokens = title.split(/\s+/);
|
|
293
|
+
for (const tok of tokens) {
|
|
294
|
+
if (tok.length === 0) continue;
|
|
295
|
+
const c0 = tok.charAt(0);
|
|
296
|
+
if (c0 >= 'A' && c0 <= 'Z') return tok;
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
return 'Unknown';
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
// ---------- parseTavilyResponse ----------
|
|
303
|
+
//
|
|
304
|
+
// Tavily JSON shape: { results: [{ title, url, content, score }, ... ], ... }.
|
|
305
|
+
// Throws on completely empty body (treated as api_error upstream); returns
|
|
306
|
+
// [] when results is absent or empty (graceful: API responded but had no
|
|
307
|
+
// matches).
|
|
308
|
+
|
|
309
|
+
function parseTavilyResponse(json) {
|
|
310
|
+
const out = [];
|
|
311
|
+
if (!json || typeof json !== 'object') {
|
|
312
|
+
throw new Error('tavily: empty or non-object body');
|
|
313
|
+
}
|
|
314
|
+
if (!Array.isArray(json.results)) return out;
|
|
315
|
+
for (const r of json.results) {
|
|
316
|
+
if (!r || typeof r !== 'object') continue;
|
|
317
|
+
const url = (typeof r.url === 'string' && r.url.length > 0) ? r.url : '';
|
|
318
|
+
const title = typeof r.title === 'string' ? r.title : '';
|
|
319
|
+
const content = typeof r.content === 'string' ? r.content : '';
|
|
320
|
+
if (!url) continue;
|
|
321
|
+
const company = extractCompany(url, title);
|
|
322
|
+
let signalText = (content || title || '').slice(0, SIGNAL_MAX_CHARS);
|
|
323
|
+
if (signalText.length === 0) signalText = title.slice(0, SIGNAL_MAX_CHARS);
|
|
324
|
+
if (signalText.length === 0) continue;
|
|
325
|
+
out.push({
|
|
326
|
+
company: company,
|
|
327
|
+
signal: signalText,
|
|
328
|
+
source: 'tavily',
|
|
329
|
+
url: url,
|
|
330
|
+
fetched_at: new Date().toISOString(),
|
|
331
|
+
});
|
|
332
|
+
}
|
|
333
|
+
return out;
|
|
334
|
+
}
|
|
335
|
+
|
|
336
|
+
// ---------- normalizeSignal ----------
|
|
337
|
+
//
|
|
338
|
+
// Final shape guarantee. parseTavilyResponse already produces this shape
|
|
339
|
+
// but normalizeSignal is exposed so future call sites can canonicalize
|
|
340
|
+
// hand-constructed records.
|
|
341
|
+
|
|
342
|
+
function normalizeSignal(raw) {
|
|
343
|
+
return {
|
|
344
|
+
company: raw && raw.company ? String(raw.company) : 'Unknown',
|
|
345
|
+
signal: raw && raw.signal ? String(raw.signal).slice(0, SIGNAL_MAX_CHARS) : '',
|
|
346
|
+
source: raw && raw.source ? String(raw.source) : '',
|
|
347
|
+
url: raw && raw.url ? String(raw.url) : '',
|
|
348
|
+
fetched_at: raw && raw.fetched_at ? String(raw.fetched_at) : new Date().toISOString(),
|
|
349
|
+
};
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
// ---------- Dedup key + dedupe ----------
|
|
353
|
+
//
|
|
354
|
+
// dedup key per CONTEXT.md: hash of (company_normalized + signal_text_first_50_chars_hash).
|
|
355
|
+
// First-seen wins so refined-template iteration order determines tie-breaking.
|
|
356
|
+
|
|
357
|
+
function dedupKey(s) {
|
|
358
|
+
if (!s || typeof s !== 'object') return 'unknown:' + JSON.stringify(s || {});
|
|
359
|
+
const company = (typeof s.company === 'string') ? s.company.toLowerCase().trim() : '';
|
|
360
|
+
const signalHead = (typeof s.signal === 'string') ? s.signal.slice(0, 50) : '';
|
|
361
|
+
const sha = crypto.createHash('sha256').update(company + '|' + signalHead).digest('hex').slice(0, 16);
|
|
362
|
+
return 'industry:' + sha;
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
function dedupe(signals) {
|
|
366
|
+
const seen = new Set();
|
|
367
|
+
const out = [];
|
|
368
|
+
for (const s of signals) {
|
|
369
|
+
const key = dedupKey(s);
|
|
370
|
+
if (seen.has(key)) continue;
|
|
371
|
+
seen.add(key);
|
|
372
|
+
out.push(s);
|
|
373
|
+
}
|
|
374
|
+
return out;
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
// ---------- normalizeIncoming (Phase 94 Plan 05 envelope helper) ----------
|
|
378
|
+
//
|
|
379
|
+
// Accepts either Tavily-shape {url, title, content/snippet} or canonical
|
|
380
|
+
// signal shape {company, signal, url} and returns the canonical signal
|
|
381
|
+
// shape. Used by the opts.tavily / opts.webSearch / opts.cacheReader
|
|
382
|
+
// injection seams so the envelope's results[] is always the same shape.
|
|
383
|
+
|
|
384
|
+
function normalizeIncoming(r, sourceTag) {
|
|
385
|
+
if (!r || typeof r !== 'object') {
|
|
386
|
+
return {
|
|
387
|
+
company: 'Unknown',
|
|
388
|
+
signal: '',
|
|
389
|
+
source: sourceTag || '',
|
|
390
|
+
url: '',
|
|
391
|
+
fetched_at: new Date().toISOString(),
|
|
392
|
+
};
|
|
393
|
+
}
|
|
394
|
+
// Already canonical signal shape?
|
|
395
|
+
if (typeof r.company === 'string' && typeof r.signal === 'string') {
|
|
396
|
+
return {
|
|
397
|
+
company: r.company,
|
|
398
|
+
signal: r.signal.slice(0, SIGNAL_MAX_CHARS),
|
|
399
|
+
source: typeof r.source === 'string' && r.source.length > 0 ? r.source : (sourceTag || ''),
|
|
400
|
+
url: typeof r.url === 'string' ? r.url : '',
|
|
401
|
+
fetched_at: typeof r.fetched_at === 'string' ? r.fetched_at : new Date().toISOString(),
|
|
402
|
+
};
|
|
403
|
+
}
|
|
404
|
+
// Tavily / WebSearch shape: derive company + signal.
|
|
405
|
+
const url = typeof r.url === 'string' ? r.url : '';
|
|
406
|
+
const title = typeof r.title === 'string' ? r.title : '';
|
|
407
|
+
const body = typeof r.content === 'string' ? r.content
|
|
408
|
+
: (typeof r.snippet === 'string' ? r.snippet : '');
|
|
409
|
+
const company = extractCompany(url, title);
|
|
410
|
+
let signal = (body || title || '').slice(0, SIGNAL_MAX_CHARS);
|
|
411
|
+
if (signal.length === 0) signal = title.slice(0, SIGNAL_MAX_CHARS);
|
|
412
|
+
return {
|
|
413
|
+
company: company,
|
|
414
|
+
signal: signal,
|
|
415
|
+
source: sourceTag || 'websearch',
|
|
416
|
+
url: url,
|
|
417
|
+
fetched_at: new Date().toISOString(),
|
|
418
|
+
};
|
|
419
|
+
}
|
|
420
|
+
|
|
421
|
+
// ---------- fetchIndustry ----------
|
|
422
|
+
//
|
|
423
|
+
// Top-level orchestrator. Tavily is the only source. For each user query:
|
|
424
|
+
// 1. Pre-flight Canon Part 8 audit on the raw user query (Layer 1).
|
|
425
|
+
// Mirrors Plan 89.2-03 Pattern 6: scan ALL queries BEFORE the source
|
|
426
|
+
// loop runs so adversarial input throws BEFORE any fetch() call.
|
|
427
|
+
// 2. Build via chokepoint (throws if api-key absent OR adversarial).
|
|
428
|
+
// The chokepoint also runs Layer 2 audit on each refined sub-query.
|
|
429
|
+
// 3. If skip (api_key_missing) -> recordTelemetry + record in-memory + return.
|
|
430
|
+
// 4. Check budget. budget==0 -> skip Tavily for this run.
|
|
431
|
+
// 5. For each refined sub-query: fetchWithTimeout -> parse -> accumulate.
|
|
432
|
+
// After all queries, dedupe and return {signals, telemetry}.
|
|
433
|
+
//
|
|
434
|
+
// fetchIndustry NEVER throws on rate-limit, timeout, parse error, or
|
|
435
|
+
// budget exhaustion. It ONLY throws on Canon Part 8 violation
|
|
436
|
+
// (ExternalEgressViolation propagated from buildIndustryQuery).
|
|
437
|
+
|
|
438
|
+
async function fetchIndustry(queries, opts) {
|
|
439
|
+
if (!Array.isArray(queries)) {
|
|
440
|
+
throw new TypeError('fetchIndustry: queries must be an array of non-empty strings');
|
|
441
|
+
}
|
|
442
|
+
for (const q of queries) {
|
|
443
|
+
if (typeof q !== 'string' || q.length === 0) {
|
|
444
|
+
throw new TypeError('fetchIndustry: each query must be a non-empty string');
|
|
445
|
+
}
|
|
446
|
+
}
|
|
447
|
+
opts = opts || {};
|
|
448
|
+
const budgetOverrides = opts.budget || {};
|
|
449
|
+
|
|
450
|
+
// ---- Phase 94 Plan 05 amendment: Tier 1 PAID injection seam ----
|
|
451
|
+
// If the caller injected an opts.tavily callable (typically the
|
|
452
|
+
// /mos:research command wiring; never null in production agent
|
|
453
|
+
// context), short-circuit the network path entirely and use the
|
|
454
|
+
// injected adapter. The adapter must return {results: [...]} where
|
|
455
|
+
// each entry has at least {url, title, snippet|content} OR the
|
|
456
|
+
// canonical signal shape {company, signal, url}. We accept both
|
|
457
|
+
// and normalize.
|
|
458
|
+
if (opts.tavily && typeof opts.tavily === 'function') {
|
|
459
|
+
let injected;
|
|
460
|
+
try {
|
|
461
|
+
// Pre-flight Canon Part 8 audit on the user query so adversarial
|
|
462
|
+
// input still throws even when the network is bypassed.
|
|
463
|
+
for (const q of queries) {
|
|
464
|
+
auditQueryString(q, 'industry');
|
|
465
|
+
}
|
|
466
|
+
injected = await opts.tavily(queries, opts);
|
|
467
|
+
} catch (err) {
|
|
468
|
+
// Adapter failed -> fall through to native + cache.
|
|
469
|
+
injected = null;
|
|
470
|
+
}
|
|
471
|
+
if (injected && Array.isArray(injected.results) && injected.results.length > 0) {
|
|
472
|
+
const signals = injected.results.map(function (r) { return normalizeIncoming(r, 'tavily'); });
|
|
473
|
+
return {
|
|
474
|
+
tier: 'paid',
|
|
475
|
+
source: 'tavily',
|
|
476
|
+
results: signals,
|
|
477
|
+
signals: signals,
|
|
478
|
+
telemetry: [{ source: 'tavily', status: 'ok', tier: 'paid' }],
|
|
479
|
+
};
|
|
480
|
+
}
|
|
481
|
+
// Fall through.
|
|
482
|
+
}
|
|
483
|
+
|
|
484
|
+
// Pre-flight Canon Part 8 audit (Pattern 6 from Plan 89.2-03):
|
|
485
|
+
// walk every query through the chokepoint BEFORE iterating sources.
|
|
486
|
+
// Without this, an adversarial query in position N would run for queries
|
|
487
|
+
// 0..N-1 first (issuing real fetch() calls), then throw on N. The tests
|
|
488
|
+
// assert ZERO captured URLs on adversarial input, so the audit must
|
|
489
|
+
// happen before any fetch loop runs. The chokepoint also runs Layer 2
|
|
490
|
+
// on each refined sub-query, so adversarial template overrides throw
|
|
491
|
+
// here too (Test 10 + Test 11 fences).
|
|
492
|
+
for (const q of queries) {
|
|
493
|
+
// Surfacing the chokepoint here gives BOTH layers of audit pre-flight
|
|
494
|
+
// (so Test 10 + Test 11 throw before fetch).
|
|
495
|
+
buildIndustryQuery(q, opts);
|
|
496
|
+
}
|
|
497
|
+
|
|
498
|
+
const out = { signals: [], telemetry: [] };
|
|
499
|
+
|
|
500
|
+
// Single source: tavily. Loop preserved for parity with academic +
|
|
501
|
+
// patents fetchers; future expansion (Crunchbase direct, AngelList, etc.)
|
|
502
|
+
// would slot in here.
|
|
503
|
+
for (const source of SOURCES) {
|
|
504
|
+
const budgetCap = (typeof budgetOverrides[source] === 'number')
|
|
505
|
+
? budgetOverrides[source]
|
|
506
|
+
: DEFAULT_BUDGETS[source];
|
|
507
|
+
const remaining = computeRemainingBudget(source, budgetCap);
|
|
508
|
+
if (remaining <= 0) {
|
|
509
|
+
// Skip source for this run; budget itself is the trace. We do NOT
|
|
510
|
+
// call recordTelemetry here ('budget_exhausted' is not in
|
|
511
|
+
// ALLOWED_STATUSES for the v1 telemetry primitive).
|
|
512
|
+
out.telemetry.push({ source: source, status: 'budget_exhausted' });
|
|
513
|
+
continue;
|
|
514
|
+
}
|
|
515
|
+
|
|
516
|
+
for (const query of queries) {
|
|
517
|
+
// Build (chokepoint). Throws ExternalEgressViolation on adversarial
|
|
518
|
+
// input, but the pre-flight loop above has already cleared every
|
|
519
|
+
// query. Re-running here is defense-in-depth.
|
|
520
|
+
const built = buildIndustryQuery(query, opts);
|
|
521
|
+
|
|
522
|
+
if (built.skip) {
|
|
523
|
+
// TAVILY_API_KEY missing.
|
|
524
|
+
recordTelemetry({
|
|
525
|
+
source: source,
|
|
526
|
+
query_text: query,
|
|
527
|
+
status: built.reason,
|
|
528
|
+
});
|
|
529
|
+
out.telemetry.push({ source: source, status: built.reason });
|
|
530
|
+
// Every query will hit the same gate; break early.
|
|
531
|
+
break;
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
// Per-refined-sub-query fetch loop.
|
|
535
|
+
for (const refined of built.refined_subqueries) {
|
|
536
|
+
const body = buildTavilyBody(refined, built.max_results_per_subquery);
|
|
537
|
+
const req = {
|
|
538
|
+
url: built.url,
|
|
539
|
+
method: built.method,
|
|
540
|
+
headers: built.headers,
|
|
541
|
+
body: body,
|
|
542
|
+
};
|
|
543
|
+
|
|
544
|
+
let res = null;
|
|
545
|
+
try {
|
|
546
|
+
res = await fetchWithTimeout(req, opts);
|
|
547
|
+
} catch (err) {
|
|
548
|
+
if (err && err.name === 'AbortError') {
|
|
549
|
+
recordTelemetry({
|
|
550
|
+
source: source,
|
|
551
|
+
query_text: refined,
|
|
552
|
+
status: 'timeout',
|
|
553
|
+
});
|
|
554
|
+
out.telemetry.push({ source: source, status: 'timeout' });
|
|
555
|
+
continue;
|
|
556
|
+
}
|
|
557
|
+
recordTelemetry({
|
|
558
|
+
source: source,
|
|
559
|
+
query_text: refined,
|
|
560
|
+
status: 'network_error',
|
|
561
|
+
});
|
|
562
|
+
out.telemetry.push({ source: source, status: 'network_error' });
|
|
563
|
+
continue;
|
|
564
|
+
}
|
|
565
|
+
|
|
566
|
+
if (!res.ok) {
|
|
567
|
+
const httpStatus = res.status || 0;
|
|
568
|
+
if (httpStatus === 429 || httpStatus === 503) {
|
|
569
|
+
recordTelemetry({
|
|
570
|
+
source: source,
|
|
571
|
+
query_text: refined,
|
|
572
|
+
status: 'rate_limited',
|
|
573
|
+
http_status: httpStatus,
|
|
574
|
+
});
|
|
575
|
+
out.telemetry.push({ source: source, status: 'rate_limited', http_status: httpStatus });
|
|
576
|
+
continue;
|
|
577
|
+
}
|
|
578
|
+
recordTelemetry({
|
|
579
|
+
source: source,
|
|
580
|
+
query_text: refined,
|
|
581
|
+
status: 'api_error',
|
|
582
|
+
http_status: httpStatus,
|
|
583
|
+
});
|
|
584
|
+
out.telemetry.push({ source: source, status: 'api_error', http_status: httpStatus });
|
|
585
|
+
continue;
|
|
586
|
+
}
|
|
587
|
+
|
|
588
|
+
let parsed = null;
|
|
589
|
+
try {
|
|
590
|
+
const payload = await res.json();
|
|
591
|
+
parsed = parseTavilyResponse(payload);
|
|
592
|
+
} catch (_err) {
|
|
593
|
+
recordTelemetry({
|
|
594
|
+
source: source,
|
|
595
|
+
query_text: refined,
|
|
596
|
+
status: 'api_error',
|
|
597
|
+
http_status: res.status || 0,
|
|
598
|
+
});
|
|
599
|
+
out.telemetry.push({ source: source, status: 'api_error', http_status: res.status || 0 });
|
|
600
|
+
continue;
|
|
601
|
+
}
|
|
602
|
+
|
|
603
|
+
const rateRemaining = parseRateLimit(res.headers);
|
|
604
|
+
const recOpts = {
|
|
605
|
+
source: source,
|
|
606
|
+
query_text: refined,
|
|
607
|
+
status: 'ok',
|
|
608
|
+
http_status: res.status || 200,
|
|
609
|
+
};
|
|
610
|
+
if (typeof rateRemaining === 'number') {
|
|
611
|
+
recOpts.rate_limit_remaining = rateRemaining;
|
|
612
|
+
}
|
|
613
|
+
recordTelemetry(recOpts);
|
|
614
|
+
out.telemetry.push({ source: source, status: 'ok', http_status: res.status || 200 });
|
|
615
|
+
|
|
616
|
+
for (const sig of parsed) {
|
|
617
|
+
out.signals.push(sig);
|
|
618
|
+
}
|
|
619
|
+
}
|
|
620
|
+
}
|
|
621
|
+
}
|
|
622
|
+
|
|
623
|
+
// First-seen wins on dedupe.
|
|
624
|
+
out.signals = dedupe(out.signals);
|
|
625
|
+
|
|
626
|
+
// ---- Phase 94 Plan 05 amendment: envelope wrap + tier annotation ----
|
|
627
|
+
// If Tavily produced results, this is the paid tier. Otherwise probe
|
|
628
|
+
// Tier 0 NATIVE (opts.webSearch) then Tier -1 CACHE (opts.cacheReader)
|
|
629
|
+
// before returning the empty-cache floor.
|
|
630
|
+
if (out.signals.length > 0) {
|
|
631
|
+
return {
|
|
632
|
+
tier: 'paid',
|
|
633
|
+
source: 'tavily',
|
|
634
|
+
results: out.signals.slice(),
|
|
635
|
+
signals: out.signals,
|
|
636
|
+
telemetry: out.telemetry,
|
|
637
|
+
};
|
|
638
|
+
}
|
|
639
|
+
|
|
640
|
+
// Tier 0 NATIVE: Anthropic native WebSearch via injection seam.
|
|
641
|
+
if (opts.webSearch && typeof opts.webSearch === 'function') {
|
|
642
|
+
const adapted = queries.map(function (q) {
|
|
643
|
+
return q + ' industry analysis OR market report';
|
|
644
|
+
});
|
|
645
|
+
const merged = [];
|
|
646
|
+
for (const adq of adapted) {
|
|
647
|
+
let res;
|
|
648
|
+
try {
|
|
649
|
+
// Canon Part 8 audit on the adapted query before egress.
|
|
650
|
+
auditQueryString(adq, 'industry');
|
|
651
|
+
res = await opts.webSearch(adq, opts);
|
|
652
|
+
} catch (_e) {
|
|
653
|
+
res = null;
|
|
654
|
+
}
|
|
655
|
+
if (res && Array.isArray(res.results)) {
|
|
656
|
+
for (const r of res.results) {
|
|
657
|
+
merged.push(normalizeIncoming(r, 'websearch'));
|
|
658
|
+
}
|
|
659
|
+
}
|
|
660
|
+
}
|
|
661
|
+
if (merged.length > 0) {
|
|
662
|
+
const deduped = dedupe(merged);
|
|
663
|
+
return {
|
|
664
|
+
tier: 'native',
|
|
665
|
+
source: 'websearch',
|
|
666
|
+
results: deduped,
|
|
667
|
+
signals: deduped,
|
|
668
|
+
telemetry: out.telemetry.concat([{ source: 'websearch', status: 'ok', tier: 'native' }]),
|
|
669
|
+
};
|
|
670
|
+
}
|
|
671
|
+
}
|
|
672
|
+
|
|
673
|
+
// Tier -1 CACHE: read most-recent fetched_results.json from
|
|
674
|
+
// <room>/.mindrian/ via injected cacheReader. The injection seam
|
|
675
|
+
// keeps the fetcher stateless about room layout.
|
|
676
|
+
if (opts.cacheReader && typeof opts.cacheReader === 'function') {
|
|
677
|
+
let cached = null;
|
|
678
|
+
try {
|
|
679
|
+
cached = opts.cacheReader(opts.roomDir || null);
|
|
680
|
+
} catch (_e) {
|
|
681
|
+
cached = null;
|
|
682
|
+
}
|
|
683
|
+
if (cached && Array.isArray(cached.results) && cached.results.length > 0) {
|
|
684
|
+
const cachedSignals = cached.results.map(function (r) {
|
|
685
|
+
return normalizeIncoming(r, 'cache');
|
|
686
|
+
});
|
|
687
|
+
return {
|
|
688
|
+
tier: 'cache',
|
|
689
|
+
source: 'cache',
|
|
690
|
+
results: cachedSignals,
|
|
691
|
+
signals: cachedSignals,
|
|
692
|
+
telemetry: out.telemetry.concat([{ source: 'cache', status: 'ok', tier: 'cache' }]),
|
|
693
|
+
};
|
|
694
|
+
}
|
|
695
|
+
}
|
|
696
|
+
|
|
697
|
+
// Floor: empty-cache. NEVER throw; envelope shape preserved.
|
|
698
|
+
return {
|
|
699
|
+
tier: 'cache',
|
|
700
|
+
source: 'cache',
|
|
701
|
+
results: [],
|
|
702
|
+
signals: [],
|
|
703
|
+
telemetry: out.telemetry.concat([{ source: 'cache', status: 'empty', tier: 'cache' }]),
|
|
704
|
+
};
|
|
705
|
+
}
|
|
706
|
+
|
|
707
|
+
// ---------- Exports ----------
|
|
708
|
+
|
|
709
|
+
module.exports = {
|
|
710
|
+
fetchIndustry,
|
|
711
|
+
buildIndustryQuery,
|
|
712
|
+
// Test surface (private; do NOT consume in production).
|
|
713
|
+
_test: {
|
|
714
|
+
dedupe,
|
|
715
|
+
dedupKey,
|
|
716
|
+
normalizeSignal,
|
|
717
|
+
extractCompany,
|
|
718
|
+
parseTavilyResponse,
|
|
719
|
+
fetchWithTimeout,
|
|
720
|
+
buildTavilyBody,
|
|
721
|
+
parseRateLimit,
|
|
722
|
+
REFINEMENT_TEMPLATES,
|
|
723
|
+
SOURCES,
|
|
724
|
+
SOURCE_ENV_VARS,
|
|
725
|
+
ENDPOINT_TAVILY,
|
|
726
|
+
DEFAULT_TIMEOUT_MS,
|
|
727
|
+
USER_AGENT,
|
|
728
|
+
SIGNAL_MAX_CHARS,
|
|
729
|
+
MAX_RESULTS_PER_SUBQUERY,
|
|
730
|
+
},
|
|
731
|
+
};
|