@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.
Files changed (597) hide show
  1. package/.claude-plugin/plugin.json +21 -0
  2. package/.mcp.json +9 -0
  3. package/CHANGELOG.md +3333 -0
  4. package/LICENSE +123 -0
  5. package/README.md +673 -0
  6. package/agents/brain-query.md +80 -0
  7. package/agents/framework-runner.md +237 -0
  8. package/agents/grading.md +188 -0
  9. package/agents/investor.md +128 -0
  10. package/agents/larry-extended.md +135 -0
  11. package/agents/opportunity-scanner.md +91 -0
  12. package/agents/persona-analyst.md +132 -0
  13. package/agents/research.md +89 -0
  14. package/agents/reverse-salient-agent.md +27 -0
  15. package/bin/cli.js +142 -0
  16. package/bin/mindrian-mcp-server.cjs +182 -0
  17. package/bin/mindrian-tools.cjs +765 -0
  18. package/commands/act.md +439 -0
  19. package/commands/admin.md +404 -0
  20. package/commands/analyze-needs.md +42 -0
  21. package/commands/analyze-systems.md +39 -0
  22. package/commands/analyze-timing.md +42 -0
  23. package/commands/auto-explore.md +64 -0
  24. package/commands/beautiful-question.md +40 -0
  25. package/commands/brain-derive.md +78 -0
  26. package/commands/build-knowledge.md +42 -0
  27. package/commands/build-thesis.md +46 -0
  28. package/commands/causal.md +234 -0
  29. package/commands/challenge-assumptions.md +33 -0
  30. package/commands/compare-ventures.md +83 -0
  31. package/commands/dashboard.md +110 -0
  32. package/commands/deep-grade.md +82 -0
  33. package/commands/diagnose.md +58 -0
  34. package/commands/diagnostics.md +151 -0
  35. package/commands/doctor.md +151 -0
  36. package/commands/dominant-designs.md +40 -0
  37. package/commands/explain-decision.md +87 -0
  38. package/commands/explore-domains.md +42 -0
  39. package/commands/explore-futures.md +40 -0
  40. package/commands/explore-trends.md +42 -0
  41. package/commands/export.md +103 -0
  42. package/commands/file-meeting.md +724 -0
  43. package/commands/find-analogies.md +188 -0
  44. package/commands/find-bottlenecks.md +62 -0
  45. package/commands/find-connections.md +76 -0
  46. package/commands/funding.md +81 -0
  47. package/commands/grade.md +203 -0
  48. package/commands/graph.md +128 -0
  49. package/commands/hat-briefing.md +125 -0
  50. package/commands/heal.md +196 -0
  51. package/commands/help.md +399 -0
  52. package/commands/hmi-status.md +172 -0
  53. package/commands/jtbd.md +241 -0
  54. package/commands/leadership.md +73 -0
  55. package/commands/lean-canvas.md +40 -0
  56. package/commands/macro-trends.md +40 -0
  57. package/commands/map-unknowns.md +40 -0
  58. package/commands/memory.md +173 -0
  59. package/commands/models.md +175 -0
  60. package/commands/mos-reason.md +285 -0
  61. package/commands/mullins.md +120 -0
  62. package/commands/new-project.md +481 -0
  63. package/commands/onboard.md +434 -0
  64. package/commands/operator.md +149 -0
  65. package/commands/opportunities.md +144 -0
  66. package/commands/organize.md +497 -0
  67. package/commands/persona.md +198 -0
  68. package/commands/pipeline.md +112 -0
  69. package/commands/present.md +91 -0
  70. package/commands/publish.md +201 -0
  71. package/commands/query.md +124 -0
  72. package/commands/radar.md +72 -0
  73. package/commands/reanalyze.md +91 -0
  74. package/commands/research.md +196 -0
  75. package/commands/room.md +352 -0
  76. package/commands/rooms.md +598 -0
  77. package/commands/root-cause.md +40 -0
  78. package/commands/rs-experts.md +85 -0
  79. package/commands/rs-explain.md +100 -0
  80. package/commands/rs-fetch.md +94 -0
  81. package/commands/rs-thesis.md +85 -0
  82. package/commands/scenario-plan.md +40 -0
  83. package/commands/scheduled-tasks.md +285 -0
  84. package/commands/score-innovation.md +43 -0
  85. package/commands/scout.md +239 -0
  86. package/commands/setup.md +618 -0
  87. package/commands/snapshot.md +147 -0
  88. package/commands/speakers.md +84 -0
  89. package/commands/splash.md +28 -0
  90. package/commands/status.md +75 -0
  91. package/commands/structure-argument.md +42 -0
  92. package/commands/suggest-next.md +80 -0
  93. package/commands/systems-thinking.md +40 -0
  94. package/commands/think-hats.md +42 -0
  95. package/commands/update.md +181 -0
  96. package/commands/user-needs.md +40 -0
  97. package/commands/validate.md +40 -0
  98. package/commands/value-proposition.md +61 -0
  99. package/commands/vault.md +180 -0
  100. package/commands/visualize.md +52 -0
  101. package/commands/whitespace.md +507 -0
  102. package/commands/wiki.md +69 -0
  103. package/hooks/hooks.json +381 -0
  104. package/hooks/run-hook.cmd +64 -0
  105. package/lib/__init__.py +0 -0
  106. package/lib/__pycache__/__init__.cpython-312.pyc +0 -0
  107. package/lib/agents/auto-explore-agent.cjs +1043 -0
  108. package/lib/agents/reverse-salient-agent.cjs +679 -0
  109. package/lib/agents/tension-hook-agent.cjs +544 -0
  110. package/lib/brain/ROOM.md +44 -0
  111. package/lib/brain/chain-recommender.cjs +301 -0
  112. package/lib/chat/chat-context.js +185 -0
  113. package/lib/chat/chat-panel.js +721 -0
  114. package/lib/chat/fabric-chat.cjs +288 -0
  115. package/lib/chat/generative-tools.js +219 -0
  116. package/lib/conversation/ROOM.md +39 -0
  117. package/lib/conversation/classifier-rules.json +38 -0
  118. package/lib/conversation/classifier.cjs +264 -0
  119. package/lib/conversation/operator.cjs +287 -0
  120. package/lib/copy/115-spec-strings.cjs +55 -0
  121. package/lib/core/__init__.py +0 -0
  122. package/lib/core/__nav-stub.cjs +14 -0
  123. package/lib/core/__pycache__/__init__.cpython-312.pyc +0 -0
  124. package/lib/core/__pycache__/rs-math.cpython-312.pyc +0 -0
  125. package/lib/core/__pycache__/rs_cache.cpython-312.pyc +0 -0
  126. package/lib/core/__pycache__/rs_corpus.cpython-312.pyc +0 -0
  127. package/lib/core/__pycache__/rs_hybrid.cpython-312.pyc +0 -0
  128. package/lib/core/__pycache__/rs_math.cpython-312.pyc +0 -0
  129. package/lib/core/__pycache__/rs_rooms.cpython-312.pyc +0 -0
  130. package/lib/core/artifact-id.cjs +148 -0
  131. package/lib/core/asset-ops.cjs +151 -0
  132. package/lib/core/auto-commit-throttle.cjs +129 -0
  133. package/lib/core/bearer-token.cjs +199 -0
  134. package/lib/core/brain-client.cjs +865 -0
  135. package/lib/core/brain-derivation-prompts.cjs +326 -0
  136. package/lib/core/brain-derivation-queue.cjs +431 -0
  137. package/lib/core/brain-derivation.cjs +580 -0
  138. package/lib/core/brain-md-schema.cjs +528 -0
  139. package/lib/core/brain-md-staleness.cjs +357 -0
  140. package/lib/core/brain-response-sanitize.cjs +188 -0
  141. package/lib/core/bridge-writer.cjs +477 -0
  142. package/lib/core/chat-context-builder.cjs +253 -0
  143. package/lib/core/cross-room-aggregator.cjs +762 -0
  144. package/lib/core/daily-briefing.cjs +438 -0
  145. package/lib/core/decision-capture.cjs +618 -0
  146. package/lib/core/deep-links.cjs +82 -0
  147. package/lib/core/dispatch-optimizer.cjs +354 -0
  148. package/lib/core/dual-path-detector.cjs +84 -0
  149. package/lib/core/dual-path-detector.test.cjs +334 -0
  150. package/lib/core/exports-log.cjs +79 -0
  151. package/lib/core/feynman-minto-invariants.cjs +605 -0
  152. package/lib/core/folder-memory-async.cjs +338 -0
  153. package/lib/core/folder-memory-shared.cjs +890 -0
  154. package/lib/core/folder-memory.cjs +416 -0
  155. package/lib/core/framework-chain-composer.cjs +411 -0
  156. package/lib/core/frontmatter-schemas.cjs +330 -0
  157. package/lib/core/git-ops.cjs +141 -0
  158. package/lib/core/graph-ops.cjs +258 -0
  159. package/lib/core/hat-persistence.cjs +362 -0
  160. package/lib/core/index.cjs +60 -0
  161. package/lib/core/integration-registry.cjs +232 -0
  162. package/lib/core/intelligence-cascade.cjs +661 -0
  163. package/lib/core/lazygraph-ops.cjs +1057 -0
  164. package/lib/core/lru-cache.cjs +139 -0
  165. package/lib/core/mcp-profiles.cjs +182 -0
  166. package/lib/core/meeting-ops.cjs +54 -0
  167. package/lib/core/memory-ops.cjs +600 -0
  168. package/lib/core/migrations/ROOM.md +33 -0
  169. package/lib/core/migrations/phase-109-nodes-provenance.cjs +339 -0
  170. package/lib/core/migrations/phase-109-session-focus.cjs +99 -0
  171. package/lib/core/model-profiles.cjs +246 -0
  172. package/lib/core/mullins-scaffold.cjs +160 -0
  173. package/lib/core/nav-dial.cjs +316 -0
  174. package/lib/core/navigation/ROOM.md +15 -0
  175. package/lib/core/navigation/explanation.cjs +43 -0
  176. package/lib/core/navigation/focus.cjs +135 -0
  177. package/lib/core/navigation/ingestion.cjs +82 -0
  178. package/lib/core/navigation/insights.cjs +350 -0
  179. package/lib/core/navigation/memory-events.cjs +118 -0
  180. package/lib/core/navigation/neighborhood.cjs +78 -0
  181. package/lib/core/navigation/packet.cjs +182 -0
  182. package/lib/core/navigation/room-home.cjs +127 -0
  183. package/lib/core/navigation/transitions.cjs +82 -0
  184. package/lib/core/navigation-engine-shared.cjs +242 -0
  185. package/lib/core/navigation-engine.cjs +664 -0
  186. package/lib/core/navigation.cjs +60 -0
  187. package/lib/core/nl-graph-queries.cjs +164 -0
  188. package/lib/core/offer-presenter.cjs +406 -0
  189. package/lib/core/opportunity-extractor.cjs +183 -0
  190. package/lib/core/opportunity-ops.cjs +1371 -0
  191. package/lib/core/persona-ops.cjs +537 -0
  192. package/lib/core/persona-taxonomy.cjs +190 -0
  193. package/lib/core/platform-gates.cjs +120 -0
  194. package/lib/core/platform.cjs +257 -0
  195. package/lib/core/proactive-intelligence.cjs +528 -0
  196. package/lib/core/problem-type-router.cjs +315 -0
  197. package/lib/core/reasoning-ops.cjs +639 -0
  198. package/lib/core/reverse-salient-persona-suffix.cjs +115 -0
  199. package/lib/core/room-classifier-strict-mode.cjs +229 -0
  200. package/lib/core/room-db.cjs +127 -0
  201. package/lib/core/room-ops-async.cjs +92 -0
  202. package/lib/core/room-ops-shared.cjs +64 -0
  203. package/lib/core/room-ops-sync.cjs +70 -0
  204. package/lib/core/room-ops.cjs +32 -0
  205. package/lib/core/room-type-detector.cjs +386 -0
  206. package/lib/core/rs-brain-substrate-prompts.cjs +129 -0
  207. package/lib/core/rs-brain-substrate.cjs +570 -0
  208. package/lib/core/rs-breakthrough-scorer.cjs +255 -0
  209. package/lib/core/rs-canon-violations.cjs +82 -0
  210. package/lib/core/rs-chain-feeder.cjs +343 -0
  211. package/lib/core/rs-commercial-assessor.cjs +280 -0
  212. package/lib/core/rs-differential-scorer.cjs +376 -0
  213. package/lib/core/rs-domain-analyzer.cjs +385 -0
  214. package/lib/core/rs-egress-prompts.cjs +113 -0
  215. package/lib/core/rs-egress-telemetry.cjs +225 -0
  216. package/lib/core/rs-egress-violations.cjs +53 -0
  217. package/lib/core/rs-expert-mapper.cjs +467 -0
  218. package/lib/core/rs-fetcher-academic.cjs +697 -0
  219. package/lib/core/rs-fetcher-experts.cjs +314 -0
  220. package/lib/core/rs-fetcher-industry.cjs +731 -0
  221. package/lib/core/rs-fetcher-patents.cjs +564 -0
  222. package/lib/core/rs-innovation-classifier.cjs +194 -0
  223. package/lib/core/rs-mind-map.cjs +656 -0
  224. package/lib/core/rs-neo4j-writer.cjs +388 -0
  225. package/lib/core/rs-nl-to-query.cjs +425 -0
  226. package/lib/core/rs-pinecone-bridge.cjs +303 -0
  227. package/lib/core/rs-preprocessor.cjs +350 -0
  228. package/lib/core/rs-query-matrix.cjs +316 -0
  229. package/lib/core/rs-query-to-text.cjs +438 -0
  230. package/lib/core/rs-sqlite-mirror.cjs +443 -0
  231. package/lib/core/rs-thesis-generator.cjs +188 -0
  232. package/lib/core/rs_cache.py +479 -0
  233. package/lib/core/rs_corpus.py +468 -0
  234. package/lib/core/rs_hybrid.py +586 -0
  235. package/lib/core/rs_math.py +287 -0
  236. package/lib/core/rs_rooms.py +193 -0
  237. package/lib/core/scheduled-scanner.cjs +463 -0
  238. package/lib/core/scratchpad-ops.cjs +201 -0
  239. package/lib/core/section-8-trace-schema.cjs +138 -0
  240. package/lib/core/section-registry.cjs +111 -0
  241. package/lib/core/session-state.cjs +144 -0
  242. package/lib/core/shallow-doc-parser.cjs +174 -0
  243. package/lib/core/shallow-doc-parser.test.cjs +226 -0
  244. package/lib/core/skill-activation-router.cjs +284 -0
  245. package/lib/core/state-ops.cjs +46 -0
  246. package/lib/core/statusline-cache.cjs +266 -0
  247. package/lib/core/token-estimator.cjs +348 -0
  248. package/lib/core/user-archetype.cjs +239 -0
  249. package/lib/core/user-md-ops.cjs +524 -0
  250. package/lib/core/visual-ops.cjs +624 -0
  251. package/lib/core/write-lock.cjs +149 -0
  252. package/lib/graph/canvas-graph.js +467 -0
  253. package/lib/graph/constellation-config.cjs +299 -0
  254. package/lib/graph/graph-detail-panel.js +165 -0
  255. package/lib/hmi/ROOM.md +47 -0
  256. package/lib/hmi/across-session-memory.cjs +604 -0
  257. package/lib/hmi/cross-room-memory.cjs +575 -0
  258. package/lib/hmi/decoy-tier.cjs +395 -0
  259. package/lib/hmi/jtbd-classifier.cjs +219 -0
  260. package/lib/hmi/jtbd-state.cjs +199 -0
  261. package/lib/hmi/jtbd-taxonomy.json +392 -0
  262. package/lib/hmi/selector-dispatcher.cjs +546 -0
  263. package/lib/hmi/selector-telemetry.cjs +263 -0
  264. package/lib/hmi/shape-f0-renderer.cjs +139 -0
  265. package/lib/hmi/shape-f1-fallback.cjs +80 -0
  266. package/lib/hmi/shape-f1-renderer.cjs +138 -0
  267. package/lib/hmi/shape-f2-renderer.cjs +132 -0
  268. package/lib/hmi/shape-f3-renderer.cjs +66 -0
  269. package/lib/hmi/shape-f4-renderer.cjs +72 -0
  270. package/lib/hmi/shape-f5-renderer.cjs +155 -0
  271. package/lib/hmi/shape-f6-plan-review-renderer.cjs +312 -0
  272. package/lib/hmi/shape-f6-renderer.cjs +144 -0
  273. package/lib/hmi/shape-g-renderer.cjs +219 -0
  274. package/lib/hmi/shape-h-renderer.cjs +222 -0
  275. package/lib/hmi/tier-check.cjs +63 -0
  276. package/lib/import/PRECONDITIONS.md +41 -0
  277. package/lib/import/branding.cjs +210 -0
  278. package/lib/import/branding.test.cjs +235 -0
  279. package/lib/import/classifications-sync.cjs +104 -0
  280. package/lib/import/classifications-sync.test.cjs +129 -0
  281. package/lib/import/enricher.cjs +296 -0
  282. package/lib/import/enricher.test.cjs +273 -0
  283. package/lib/import/integration.test.cjs +376 -0
  284. package/lib/import/manifest.cjs +129 -0
  285. package/lib/import/manifest.schema.json +185 -0
  286. package/lib/import/manifest.test.cjs +123 -0
  287. package/lib/import/meeting-detector.cjs +92 -0
  288. package/lib/import/meeting-detector.test.cjs +100 -0
  289. package/lib/import/person-detector.cjs +229 -0
  290. package/lib/import/person-detector.test.cjs +149 -0
  291. package/lib/import/report.cjs +186 -0
  292. package/lib/import/report.test.cjs +186 -0
  293. package/lib/import/room-md-scaffolder.cjs +49 -0
  294. package/lib/import/router.cjs +224 -0
  295. package/lib/import/router.test.cjs +356 -0
  296. package/lib/import/run-all-tests.cjs +36 -0
  297. package/lib/import/smoke-test.cjs +213 -0
  298. package/lib/import/smoke-test.test.cjs +148 -0
  299. package/lib/import/test-fixtures/collision-vault/preexisting-room/STATE.md +8 -0
  300. package/lib/import/test-fixtures/collision-vault/preexisting-room/problem-definition/onboarding/onboarding.md +7 -0
  301. package/lib/import/test-fixtures/collision-vault/source/onboarding.md +5 -0
  302. package/lib/import/test-fixtures/obsidian-vault/.obsidian/workspace.json +1 -0
  303. package/lib/import/test-fixtures/obsidian-vault/notes/with-wikilinks.md +4 -0
  304. package/lib/import/test-fixtures/tiny-vault/notes/2026-01-15-team-sync.md +9 -0
  305. package/lib/import/test-fixtures/tiny-vault/notes/empty.md +3 -0
  306. package/lib/import/test-fixtures/tiny-vault/notes/onboarding.md +5 -0
  307. package/lib/import/test-fixtures/tiny-vault/notes/pricing.md +5 -0
  308. package/lib/import/test-fixtures/tiny-vault/notes/random.md +4 -0
  309. package/lib/import/undo.test.cjs +199 -0
  310. package/lib/import/vault-scanner.cjs +105 -0
  311. package/lib/import/vault-scanner.test.cjs +67 -0
  312. package/lib/mcp/app-html/dashboard.html +316 -0
  313. package/lib/mcp/app-html/graph.html +428 -0
  314. package/lib/mcp/app-html/mindrian-platform.html +1841 -0
  315. package/lib/mcp/app-html/wiki.html +383 -0
  316. package/lib/mcp/app-views.cjs +322 -0
  317. package/lib/mcp/brain-router.cjs +418 -0
  318. package/lib/mcp/capability-registry.cjs +62 -0
  319. package/lib/mcp/larry-context.cjs +46 -0
  320. package/lib/mcp/larry-server-instructions.md +114 -0
  321. package/lib/mcp/pipeline-state.cjs +275 -0
  322. package/lib/mcp/prompts.cjs +302 -0
  323. package/lib/mcp/resources.cjs +227 -0
  324. package/lib/mcp/session-catchup.cjs +327 -0
  325. package/lib/mcp/surface-detect.cjs +75 -0
  326. package/lib/mcp/tool-router.cjs +1034 -0
  327. package/lib/memory/aaak-compress.cjs +403 -0
  328. package/lib/memory/aaak-compress.test.cjs +288 -0
  329. package/lib/memory/async-artifact-auto-commit.test.cjs +223 -0
  330. package/lib/memory/bearer-token.test.cjs +315 -0
  331. package/lib/memory/brain-cache-lru.test.cjs +259 -0
  332. package/lib/memory/brain-client-query-shape.test.cjs +160 -0
  333. package/lib/memory/brain-derivation-graceful-degradation.test.cjs +1019 -0
  334. package/lib/memory/brain-derivation-queue.test.cjs +539 -0
  335. package/lib/memory/brain-derivation.test.cjs +634 -0
  336. package/lib/memory/brain-derive-command.test.cjs +534 -0
  337. package/lib/memory/brain-md-invariants-validator.test.cjs +704 -0
  338. package/lib/memory/brain-md-schema.test.cjs +467 -0
  339. package/lib/memory/brain-md-staleness.test.cjs +525 -0
  340. package/lib/memory/brain-server-resolution.test.cjs +314 -0
  341. package/lib/memory/chain-recommender.test.cjs +233 -0
  342. package/lib/memory/chat-context.test.cjs +128 -0
  343. package/lib/memory/command-registry.test.cjs +220 -0
  344. package/lib/memory/cross-room-aggregator.test.cjs +909 -0
  345. package/lib/memory/dashboard-server.test.cjs +256 -0
  346. package/lib/memory/debouncer-drain-at-prompt.test.cjs +389 -0
  347. package/lib/memory/decision-capture.test.cjs +632 -0
  348. package/lib/memory/decision-capture.worker.cjs +70 -0
  349. package/lib/memory/explain-decision-command.test.cjs +521 -0
  350. package/lib/memory/explain-decision-footer.test.cjs +316 -0
  351. package/lib/memory/explored-materials-store.cjs +392 -0
  352. package/lib/memory/feynman-minto-guardian.test.cjs +736 -0
  353. package/lib/memory/feynman-minto-invariants.test.cjs +511 -0
  354. package/lib/memory/feynman-prompts-drift.test.cjs +144 -0
  355. package/lib/memory/feynman-prompts.cjs +151 -0
  356. package/lib/memory/feynman-prompts.test.cjs +96 -0
  357. package/lib/memory/folder-memory-quadruple.test.cjs +548 -0
  358. package/lib/memory/folder-memory.test.cjs +503 -0
  359. package/lib/memory/framework-chain-composer.test.cjs +515 -0
  360. package/lib/memory/frontmatter-schema-validator.test.cjs +290 -0
  361. package/lib/memory/heal-command.test.cjs +604 -0
  362. package/lib/memory/index-artifact-transaction.test.cjs +333 -0
  363. package/lib/memory/lazygraph-rs-discoveries-view.test.cjs +122 -0
  364. package/lib/memory/mcp-input-validation.test.cjs +240 -0
  365. package/lib/memory/mcp-server-brain-deps.test.cjs +270 -0
  366. package/lib/memory/mcp-stack-fallback.test.cjs +433 -0
  367. package/lib/memory/minto-debouncer.test.cjs +407 -0
  368. package/lib/memory/minto-debouncer.worker.cjs +46 -0
  369. package/lib/memory/minto-migration-v88.test.cjs +265 -0
  370. package/lib/memory/minto-schema-v88.test.cjs +390 -0
  371. package/lib/memory/mos-status-renderer.test.cjs +631 -0
  372. package/lib/memory/narrative-schema.cjs +376 -0
  373. package/lib/memory/narrative-schema.test.cjs +209 -0
  374. package/lib/memory/nav-dial.test.cjs +414 -0
  375. package/lib/memory/navigation-engine-core.test.cjs +722 -0
  376. package/lib/memory/navigation-invariants.test.cjs +483 -0
  377. package/lib/memory/offer-presenter.test.cjs +554 -0
  378. package/lib/memory/on-stop-snapshot.test.cjs +404 -0
  379. package/lib/memory/pending-tension-store.cjs +373 -0
  380. package/lib/memory/post-compact-reinjection.test.cjs +854 -0
  381. package/lib/memory/post-write-triple.test.cjs +317 -0
  382. package/lib/memory/pre-compact-snapshot.test.cjs +495 -0
  383. package/lib/memory/problem-type-router.test.cjs +656 -0
  384. package/lib/memory/query-efficiency-telemetry.test.cjs +370 -0
  385. package/lib/memory/recompile-room-references.test.cjs +392 -0
  386. package/lib/memory/recompile-room-references.worker.cjs +42 -0
  387. package/lib/memory/record-decision-dual-write.test.cjs +454 -0
  388. package/lib/memory/room-classifier-strict-mode.test.cjs +417 -0
  389. package/lib/memory/room-minto-hook.test.cjs +398 -0
  390. package/lib/memory/rs-discovery-engine.test.cjs +323 -0
  391. package/lib/memory/run-feynman-tests.cjs +1247 -0
  392. package/lib/memory/security-trifecta.test.cjs +312 -0
  393. package/lib/memory/session-start-brain-staleness.test.cjs +363 -0
  394. package/lib/memory/session-start-triple-injection.test.cjs +514 -0
  395. package/lib/memory/sessionstart-banner-formatter.cjs +318 -0
  396. package/lib/memory/sessionstart-minto-banner.test.cjs +373 -0
  397. package/lib/memory/skill-activation-router.test.cjs +419 -0
  398. package/lib/memory/stamp-artifact-write.test.cjs +304 -0
  399. package/lib/memory/statusline-active-room.test.cjs +315 -0
  400. package/lib/memory/statusline-minto-segment.test.cjs +292 -0
  401. package/lib/memory/sync-async-entry-points.test.cjs +204 -0
  402. package/lib/memory/test-bridge-writer-enhanced.cjs +452 -0
  403. package/lib/memory/test-rs-brain-substrate-shape.cjs +529 -0
  404. package/lib/memory/test-rs-brain-substrate.cjs +636 -0
  405. package/lib/memory/test-rs-breakthrough-scorer.cjs +375 -0
  406. package/lib/memory/test-rs-canon-violations.cjs +218 -0
  407. package/lib/memory/test-rs-chain-feeder-core.cjs +344 -0
  408. package/lib/memory/test-rs-chain-feeder-skill-spawn.cjs +297 -0
  409. package/lib/memory/test-rs-commercial-assessor.cjs +385 -0
  410. package/lib/memory/test-rs-differential-scorer.cjs +480 -0
  411. package/lib/memory/test-rs-discovery-engine.cjs +603 -0
  412. package/lib/memory/test-rs-domain-analyzer.cjs +492 -0
  413. package/lib/memory/test-rs-egress-primitives.cjs +420 -0
  414. package/lib/memory/test-rs-expert-mapper.cjs +547 -0
  415. package/lib/memory/test-rs-explain-command.cjs +443 -0
  416. package/lib/memory/test-rs-fetcher-academic.cjs +848 -0
  417. package/lib/memory/test-rs-fetcher-experts.cjs +496 -0
  418. package/lib/memory/test-rs-fetcher-industry.cjs +702 -0
  419. package/lib/memory/test-rs-fetcher-patents.cjs +674 -0
  420. package/lib/memory/test-rs-innovation-classifier.cjs +301 -0
  421. package/lib/memory/test-rs-mind-map.cjs +646 -0
  422. package/lib/memory/test-rs-neo4j-writer.cjs +518 -0
  423. package/lib/memory/test-rs-nl-to-query.cjs +449 -0
  424. package/lib/memory/test-rs-pinecone-bridge.cjs +277 -0
  425. package/lib/memory/test-rs-preprocessor.cjs +433 -0
  426. package/lib/memory/test-rs-query-matrix.cjs +391 -0
  427. package/lib/memory/test-rs-query-to-text.cjs +551 -0
  428. package/lib/memory/test-rs-sqlite-mirror.cjs +649 -0
  429. package/lib/memory/test-rs-thesis-generator.cjs +360 -0
  430. package/lib/memory/triple-context-formatter.cjs +473 -0
  431. package/lib/memory/triple-context-formatter.test.cjs +442 -0
  432. package/lib/memory/user-md-persona.test.cjs +565 -0
  433. package/lib/memory/userpromptsubmit-integration.test.cjs +690 -0
  434. package/lib/memory/validators/README.md +157 -0
  435. package/lib/memory/validators/brain-md-invariants.cjs +475 -0
  436. package/lib/memory/validators/brain-substrate-invariants.cjs +285 -0
  437. package/lib/memory/validators/external-academic-invariants.cjs +249 -0
  438. package/lib/memory/validators/external-industry-invariants.cjs +271 -0
  439. package/lib/memory/validators/external-patents-invariants.cjs +266 -0
  440. package/lib/memory/validators/minto-invariants.cjs +62 -0
  441. package/lib/memory/validators/navigation-invariants.cjs +340 -0
  442. package/lib/memory/validators/queue-health.cjs +95 -0
  443. package/lib/memory/validators/snapshot-integrity.cjs +129 -0
  444. package/lib/memory/validators/stale-lifecycle.cjs +116 -0
  445. package/lib/memory/vault-section-minto-generator-atomic.test.cjs +556 -0
  446. package/lib/memory/vault-section-minto-generator-atomic.worker.cjs +73 -0
  447. package/lib/memory/write-lock-atomic.test.cjs +137 -0
  448. package/lib/memory/write-lock-atomic.worker.cjs +55 -0
  449. package/lib/parity/check-parity.cjs +83 -0
  450. package/lib/presentation/presentation-server.cjs +101 -0
  451. package/lib/presentation/presentation-watcher.cjs +123 -0
  452. package/lib/quickview/hub-server.cjs +719 -0
  453. package/lib/quickview/server.cjs +533 -0
  454. package/lib/render/JTBD-PALETTES.md +145 -0
  455. package/lib/render/ROOM.md +59 -0
  456. package/lib/render/render-v2.cjs +486 -0
  457. package/lib/render/render-v2.test.cjs +267 -0
  458. package/lib/render/render.cjs +65 -0
  459. package/lib/state/ROOM.md +46 -0
  460. package/lib/state/state-md-parser.cjs +215 -0
  461. package/lib/statusline/ROOM.md +38 -0
  462. package/lib/statusline/banner-suppression.cjs +50 -0
  463. package/lib/statusline/surface-detect.cjs +85 -0
  464. package/lib/update-bootstrap.sh.template +145 -0
  465. package/lib/vault/frontmatter-schema.cjs +297 -0
  466. package/lib/vault/room-scanner.cjs +352 -0
  467. package/lib/vault/wikilink-builder.cjs +231 -0
  468. package/lib/vault/wikilink-builder.test.cjs +182 -0
  469. package/lib/wiki/graph-links.cjs +281 -0
  470. package/lib/wiki/page-renderer.cjs +229 -0
  471. package/lib/wiki/wiki-chat.cjs +81 -0
  472. package/lib/wiki/wiki-layout.cjs +1459 -0
  473. package/lib/wiki/wiki-search.cjs +142 -0
  474. package/lib/wiki/wiki-server.cjs +678 -0
  475. package/lib/wiki/wiki-watcher.cjs +105 -0
  476. package/lib/workflow/ROOM.md +47 -0
  477. package/lib/workflow/command-resolver.cjs +155 -0
  478. package/lib/workflow/command-resolver.test.cjs +235 -0
  479. package/package.json +44 -0
  480. package/pipelines/analogy/01-decompose.md +80 -0
  481. package/pipelines/analogy/02-abstract.md +87 -0
  482. package/pipelines/analogy/03-search.md +135 -0
  483. package/pipelines/analogy/04-transfer.md +101 -0
  484. package/pipelines/analogy/05-validate.md +106 -0
  485. package/pipelines/analogy/CHAIN.md +56 -0
  486. package/pipelines/discovery/01-explore-domains.md +44 -0
  487. package/pipelines/discovery/02-think-hats.md +50 -0
  488. package/pipelines/discovery/03-analyze-needs.md +54 -0
  489. package/pipelines/discovery/CHAIN.md +37 -0
  490. package/pipelines/thesis/01-structure-argument.md +45 -0
  491. package/pipelines/thesis/02-challenge-assumptions.md +48 -0
  492. package/pipelines/thesis/03-build-thesis.md +54 -0
  493. package/pipelines/thesis/CHAIN.md +37 -0
  494. package/references/brain/causal-directives.md +91 -0
  495. package/references/brain/causal-enrichment.cypher +165 -0
  496. package/references/brain/command-triggers-schema.md +226 -0
  497. package/references/brain/graph-architecture.md +317 -0
  498. package/references/brain/query-patterns.md +460 -0
  499. package/references/brain/room-hierarchy-schema.md +218 -0
  500. package/references/brain/schema.md +76 -0
  501. package/references/capability-radar/capabilities-index.md +241 -0
  502. package/references/capability-radar/changelog-cache.md +81 -0
  503. package/references/causal/causal-schema.md +103 -0
  504. package/references/design/email-template-standard.md +155 -0
  505. package/references/design/graph-visualization-standard.md +178 -0
  506. package/references/document-generation.md +179 -0
  507. package/references/hsi/HSI-TOOLS-REFERENCE.md +222 -0
  508. package/references/import-config.md +141 -0
  509. package/references/integrations/detection-patterns.md +101 -0
  510. package/references/meeting/artifact-template.md +377 -0
  511. package/references/meeting/cross-meeting-intelligence.md +216 -0
  512. package/references/meeting/cross-relationship-patterns.md +202 -0
  513. package/references/meeting/live-join-interface.md +244 -0
  514. package/references/meeting/section-mapping.md +192 -0
  515. package/references/meeting/segment-classification.md +258 -0
  516. package/references/meeting/speaker-profile-template.md +219 -0
  517. package/references/meeting/summary-template.md +348 -0
  518. package/references/meeting/transcript-patterns.md +226 -0
  519. package/references/methodology/analyze-needs.md +135 -0
  520. package/references/methodology/analyze-systems.md +121 -0
  521. package/references/methodology/analyze-timing.md +149 -0
  522. package/references/methodology/beautiful-question.md +109 -0
  523. package/references/methodology/build-knowledge.md +161 -0
  524. package/references/methodology/build-thesis.md +237 -0
  525. package/references/methodology/challenge-assumptions.md +127 -0
  526. package/references/methodology/diagnose.md +169 -0
  527. package/references/methodology/dominant-designs.md +212 -0
  528. package/references/methodology/explore-domains.md +147 -0
  529. package/references/methodology/explore-futures.md +163 -0
  530. package/references/methodology/explore-trends.md +129 -0
  531. package/references/methodology/find-bottlenecks.md +131 -0
  532. package/references/methodology/grade.md +211 -0
  533. package/references/methodology/index.md +97 -0
  534. package/references/methodology/leadership.md +200 -0
  535. package/references/methodology/lean-canvas.md +116 -0
  536. package/references/methodology/macro-trends.md +192 -0
  537. package/references/methodology/map-unknowns.md +137 -0
  538. package/references/methodology/mullins-7-domains.md +104 -0
  539. package/references/methodology/problem-types.md +65 -0
  540. package/references/methodology/root-cause.md +178 -0
  541. package/references/methodology/sapphire-encoding.md +355 -0
  542. package/references/methodology/scenario-plan.md +178 -0
  543. package/references/methodology/score-innovation.md +154 -0
  544. package/references/methodology/structure-argument.md +158 -0
  545. package/references/methodology/systems-thinking.md +159 -0
  546. package/references/methodology/think-hats.md +147 -0
  547. package/references/methodology/triz-matrix.json +751 -0
  548. package/references/methodology/triz-principles.md +501 -0
  549. package/references/methodology/user-needs.md +199 -0
  550. package/references/methodology/validate.md +163 -0
  551. package/references/methodology/value-proposition.md +244 -0
  552. package/references/opportunities/funding-lifecycle.md +103 -0
  553. package/references/opportunities/grant-api-patterns.md +99 -0
  554. package/references/opportunities/opportunity-template.md +84 -0
  555. package/references/personality/assessment-philosophy.md +72 -0
  556. package/references/personality/lexicon.md +100 -0
  557. package/references/personality/persona-chains.md +56 -0
  558. package/references/personality/pws-lexicon-full.md +499 -0
  559. package/references/personality/voice-dna.md +156 -0
  560. package/references/personas/hat-perspectives.md +76 -0
  561. package/references/personas/persona-template.md +63 -0
  562. package/references/pipeline/act-output-contract.md +88 -0
  563. package/references/pipeline/chains-index.md +39 -0
  564. package/references/pws-profile-generation.md +79 -0
  565. package/references/reasoning/reasoning-schema.md +143 -0
  566. package/references/reasoning/reasoning-template.md +68 -0
  567. package/references/reasoning/run-template.md +38 -0
  568. package/references/research/RESEARCH_14_CLAUDE_CODE_SOURCE_ARCHITECTURE.md +209 -0
  569. package/references/research/RESEARCH_15_V1.8_OPTIMIZATION_JTBD.md +375 -0
  570. package/references/research/RESEARCH_16_NATIVE_FIRST_PLUGIN_ARCHITECTURE.md +575 -0
  571. package/references/research/RESEARCH_17_MCP_UI_FRAMEWORKS.md +272 -0
  572. package/references/taxonomy/TAXONOMY.md +192 -0
  573. package/references/templates/MINTO.md +36 -0
  574. package/references/user-research/2026-04-05-leah-lawrence-session.md +202 -0
  575. package/references/vault-kit/README.md +35 -0
  576. package/references/vault-kit/app.json +12 -0
  577. package/references/vault-kit/appearance.json +12 -0
  578. package/references/vault-kit/graph.json +35 -0
  579. package/references/vault-kit/snippets/mindrian-destijl.css +297 -0
  580. package/references/vault-kit/templates/new-artifact.md +37 -0
  581. package/references/vault-kit/templates/new-meeting-note.md +35 -0
  582. package/references/vault-kit/templates/new-team-profile.md +29 -0
  583. package/references/vault-kit/templates/new-xref.md +35 -0
  584. package/references/visual/symbol-system.md +151 -0
  585. package/skills/MOSDeckEngine/SKILL.md +325 -0
  586. package/skills/brain-connector/SKILL.md +114 -0
  587. package/skills/context-engine/SKILL.md +147 -0
  588. package/skills/conversation-mode/SKILL.md +102 -0
  589. package/skills/larry-personality/SKILL.md +219 -0
  590. package/skills/larry-personality/framework-chains.md +92 -0
  591. package/skills/larry-personality/mode-engine.md +185 -0
  592. package/skills/mullins-scaffold/SKILL.md +61 -0
  593. package/skills/mullins-scaffold/scaffold.json +146 -0
  594. package/skills/pws-methodology/SKILL.md +49 -0
  595. package/skills/room-passive/SKILL.md +165 -0
  596. package/skills/room-proactive/SKILL.md +250 -0
  597. 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
+ };