@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,586 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ rs_hybrid.py -- Room x External Unified Corpus (Phase 89 Plan 89-05)
4
+ =====================================================================
5
+
6
+ Mode C builder: unifies a user room's artifacts with an external research
7
+ corpus into a single indexable corpus, plus a cross-corpus post-filter
8
+ that keeps only pairs where one side is a room artifact and the other is
9
+ an external document.
10
+
11
+ Architecture (Plan 89-05):
12
+ - Room side: walk the room filesystem (skip lazygraph, obsidian, etc.),
13
+ reuse the discover_artifacts pattern from scripts/rs-engine.py so Mode
14
+ A and Mode C see the same artifact shape.
15
+ - External side: reuse the Pinecone rs-external cache from Plan 89-03.
16
+ Cold path fetches via lib.core.rs_corpus and upserts; warm path reads
17
+ cached multilingual-e5-large vectors directly.
18
+ - Unified corpus: flat list of dicts with `origin` in {room, external}
19
+ and `global_id` that is globally unique across both sides.
20
+ - origin_mask: numpy bool array, True=room, False=external. Enables O(1)
21
+ cross-corpus pair filtering in rs-engine.
22
+
23
+ Public API:
24
+ build_unified_corpus(room_path, topic, external_target=2000)
25
+ -> (corpus, origin_mask, metadata)
26
+ filter_cross_corpus_pairs(pairs, origin_mask, topk)
27
+ -> list[dict] pairs where origin_mask[i] != origin_mask[j]
28
+
29
+ Part 8 Graph Boundary (Canon, mandatory):
30
+ Zero user-content egress. Room artifacts are read locally and used as
31
+ LSA / semantic inputs in-process only. The external corpus metadata
32
+ stored in Pinecone (via rs_cache.upsert_corpus) is strictly public
33
+ OpenAlex/arXiv/Tavily data. Room content never touches Pinecone.
34
+
35
+ Three-surface coverage (CLAUDE.md tri-polar rule):
36
+ CLI Consumed by scripts/rs-engine.py --mode hybrid.
37
+ Desktop ReverseSalientAgent invokes rs-engine.py as a child process
38
+ (Plan 89-07 wires the conversational trigger).
39
+ Cowork When COWORK=1, the hybrid results symlink into 00_Context/
40
+ at the rs-engine CLI layer (unchanged from Mode A/B).
41
+
42
+ License: BSL-1.1 (see LICENSE at repo root).
43
+ Licensed under the Business Source License 1.1. Change Date 2029-04-24.
44
+ Change License Apache 2.0.
45
+ """
46
+
47
+ from __future__ import annotations
48
+
49
+ import os
50
+ import re
51
+ import sys
52
+ from datetime import datetime, timezone
53
+ from pathlib import Path
54
+ from typing import Any, Dict, List, Optional, Sequence, Tuple
55
+
56
+ try:
57
+ import numpy as np
58
+ except ImportError:
59
+ print(
60
+ "rs_hybrid requires numpy. Run: pip install -r requirements-hsi.txt",
61
+ file=sys.stderr,
62
+ )
63
+ raise
64
+
65
+
66
+ # --- Shared constants mirrored from scripts/rs-engine.py --------------------
67
+
68
+ # Files we never treat as artifacts (room identity / state, not content).
69
+ SKIP_FILES = {"STATE.md", "ROOM.md", "MINTO.md"}
70
+ # Directories we never descend into (metadata / tooling, not room artifacts).
71
+ SKIP_DIRS = {".lazygraph", ".git", ".mindrian", "node_modules", ".obsidian"}
72
+ # Minimum body length to treat a .md file as an artifact. Matches
73
+ # scripts/rs-engine.py:discover_artifacts and scripts/compute-hsi.py so the
74
+ # Mode A and Mode C corpora see identical artifact inclusion rules.
75
+ MIN_BODY_CHARS = 50
76
+
77
+ # Maximum external docs we will fold into the unified corpus. Plan 89-05
78
+ # canonical value is 2000 (aligns with Pinecone rs-external namespace cap
79
+ # MAX_NAMESPACE_VECTORS = 10_000 / 5, leaving room for multiple topics).
80
+ DEFAULT_EXTERNAL_TARGET = 2000
81
+
82
+ # External target is bounded so a misconfigured caller cannot balloon the
83
+ # corpus into a 50k-vector LSA fit that blows memory.
84
+ MAX_EXTERNAL_TARGET = 5000
85
+
86
+
87
+ # --- Frontmatter + body helpers (mirror scripts/rs-engine.py) ---------------
88
+
89
+ def _parse_frontmatter_skip(content: str) -> str:
90
+ """Strip a YAML frontmatter block if present; return the rest."""
91
+ fm_match = re.match(r"^---\n[\s\S]*?\n---\n?", content)
92
+ if fm_match:
93
+ return content[fm_match.end():]
94
+ return content
95
+
96
+
97
+ def _extract_title(content: str, filepath: Path) -> str:
98
+ match = re.search(r"^# (.+)$", content, re.MULTILINE)
99
+ if match:
100
+ return match.group(1).strip()
101
+ return filepath.stem.replace("-", " ").title()
102
+
103
+
104
+ # --- Room artifact loader ---------------------------------------------------
105
+
106
+ def _load_room_artifacts(room_path: Path) -> List[Dict[str, Any]]:
107
+ """Walk room_path for .md artifacts. Returns room-origin dicts.
108
+
109
+ Each artifact carries:
110
+ - global_id: f"room::{room_id}::{artifact_id}" (unique across corpus)
111
+ - origin: "room"
112
+ - room_id: basename of the room directory
113
+ - artifact_id: relative path stem (section/name)
114
+ - section: first path segment under the room
115
+ - title: first H1 or humanized filename
116
+ - text: post-frontmatter body (the LSA / semantic input)
117
+
118
+ Pattern reuse: mirrors scripts/rs-engine.py:discover_artifacts byte-for-
119
+ byte so single-room Mode A and hybrid Mode C file the same artifact the
120
+ same way. Plan 89-01 Deviation #2 (no SQLite artifacts table) carries
121
+ through -- we walk the filesystem.
122
+ """
123
+ artifacts: List[Dict[str, Any]] = []
124
+ room_id = room_path.name
125
+
126
+ for root, dirs, files in os.walk(room_path):
127
+ dirs[:] = [d for d in dirs if d not in SKIP_DIRS]
128
+ rel_root = Path(root).relative_to(room_path)
129
+ if str(rel_root) == ".":
130
+ # Skip room-root files -- not part of any section.
131
+ continue
132
+ section = str(rel_root).split(os.sep)[0]
133
+ for fname in sorted(files):
134
+ if not fname.endswith(".md"):
135
+ continue
136
+ if fname in SKIP_FILES:
137
+ continue
138
+ fpath = Path(root) / fname
139
+ try:
140
+ content = fpath.read_text(encoding="utf-8")
141
+ except (OSError, UnicodeDecodeError):
142
+ continue
143
+ body = _parse_frontmatter_skip(content).strip()
144
+ if len(body) < MIN_BODY_CHARS:
145
+ continue
146
+ artifact_id = str(rel_root / Path(fname).stem).replace(os.sep, "/")
147
+ title = _extract_title(content, fpath)
148
+ artifacts.append({
149
+ "global_id": f"room::{room_id}::{artifact_id}",
150
+ "origin": "room",
151
+ "room_id": room_id,
152
+ "artifact_id": artifact_id,
153
+ "section": section,
154
+ "title": title,
155
+ "path": str(fpath.relative_to(room_path)),
156
+ "text": body,
157
+ })
158
+ return artifacts
159
+
160
+
161
+ # --- External loader (via rs_cache) -----------------------------------------
162
+
163
+ def _load_external_records(
164
+ topic: str,
165
+ external_target: int,
166
+ ) -> Tuple[List[Dict[str, Any]], Dict[str, Any]]:
167
+ """Fetch (cold) or read (warm) the external corpus via Plan 89-03 cache.
168
+
169
+ Returns (records, cache_meta).
170
+ records: list of {id, values, metadata} as returned by rs_cache.
171
+ cache_meta: {cache_mode, cache_age_days, cache_namespace, cache_ttl_days,
172
+ corpus_size}.
173
+
174
+ Routes:
175
+ - If PINECONE_API_KEY is missing or RS_EMBEDDING_MODEL=minilm:
176
+ returns ([], {"cache_mode": "bypass", ...}). Caller falls back to
177
+ local MiniLM + rs_corpus fetch (Plan 89-02 path).
178
+ - If cache is warm (age <= TTL): skip fetch, return cached records.
179
+ - If cold: fetch via rs_corpus, upsert to Pinecone, re-read so the
180
+ semantic vectors are the server-side multilingual-e5-large ones.
181
+
182
+ Consumes Plan 89-03's public API verbatim. No new Pinecone calls live
183
+ here; this module is a composition layer.
184
+ """
185
+ try:
186
+ from lib.core.rs_cache import (
187
+ namespace_slug,
188
+ get_namespace_freshness,
189
+ upsert_corpus,
190
+ fetch_all_from_namespace,
191
+ is_fresh,
192
+ TTL_DAYS,
193
+ )
194
+ from lib.core.rs_corpus import fetch_corpus
195
+ except Exception as e:
196
+ return [], {
197
+ "cache_mode": "bypass",
198
+ "cache_age_days": None,
199
+ "cache_namespace": None,
200
+ "cache_ttl_days": None,
201
+ "corpus_size": 0,
202
+ "skip_reason": f"rs_cache import failed: {e}",
203
+ }
204
+
205
+ # Bypass check: the Plan 89-03 bypass conditions.
206
+ api_key = os.environ.get("PINECONE_API_KEY")
207
+ model_env = os.environ.get("RS_EMBEDDING_MODEL", "").strip().lower()
208
+ if not api_key:
209
+ return [], {
210
+ "cache_mode": "bypass",
211
+ "cache_age_days": None,
212
+ "cache_namespace": None,
213
+ "cache_ttl_days": None,
214
+ "corpus_size": 0,
215
+ "skip_reason": "PINECONE_API_KEY not set",
216
+ }
217
+ if model_env == "minilm":
218
+ return [], {
219
+ "cache_mode": "bypass",
220
+ "cache_age_days": None,
221
+ "cache_namespace": None,
222
+ "cache_ttl_days": None,
223
+ "corpus_size": 0,
224
+ "skip_reason": "RS_EMBEDDING_MODEL=minilm (user opted out)",
225
+ }
226
+
227
+ ns = namespace_slug(topic)
228
+ try:
229
+ age_days = get_namespace_freshness(ns)
230
+ except Exception as e:
231
+ print(
232
+ f"[rs-hybrid] freshness check failed ({e}); treating as cold.",
233
+ file=sys.stderr,
234
+ )
235
+ age_days = None
236
+
237
+ if is_fresh(age_days, ttl_days=TTL_DAYS):
238
+ # WARM path.
239
+ print(
240
+ f"[rs-hybrid] Cache hit (age={age_days:.1f} days, ttl={TTL_DAYS}). "
241
+ f"Using Pinecone namespace {ns}.",
242
+ file=sys.stderr,
243
+ )
244
+ try:
245
+ records = fetch_all_from_namespace(ns, limit=external_target)
246
+ return records, {
247
+ "cache_mode": "warm",
248
+ "cache_age_days": round(age_days, 2) if age_days is not None else None,
249
+ "cache_namespace": ns,
250
+ "cache_ttl_days": TTL_DAYS,
251
+ "corpus_size": len(records),
252
+ }
253
+ except Exception as e:
254
+ print(
255
+ f"[rs-hybrid] warm-path fetch failed ({e}); dropping to cold.",
256
+ file=sys.stderr,
257
+ )
258
+
259
+ # COLD path.
260
+ age_label = "unset" if age_days is None else f"{age_days:.1f} days"
261
+ print(
262
+ f"[rs-hybrid] Cold path: fetching external corpus for {topic!r} "
263
+ f"(cache age={age_label}).",
264
+ file=sys.stderr,
265
+ )
266
+ try:
267
+ # Overshoot fetch target so dedup + abstract-present filter still
268
+ # yields near-external_target usable docs (same pattern as Mode B).
269
+ fetch_target = max(external_target * 2, external_target + 200)
270
+ docs = fetch_corpus(topic, target_n=fetch_target)
271
+ except Exception as e:
272
+ print(f"[rs-hybrid] corpus fetch failed ({e}).", file=sys.stderr)
273
+ return [], {
274
+ "cache_mode": "bypass",
275
+ "cache_age_days": None,
276
+ "cache_namespace": ns,
277
+ "cache_ttl_days": TTL_DAYS,
278
+ "corpus_size": 0,
279
+ "skip_reason": f"fetch_corpus failed: {e}",
280
+ }
281
+
282
+ if len(docs) < 2:
283
+ return [], {
284
+ "cache_mode": "bypass",
285
+ "cache_age_days": None,
286
+ "cache_namespace": ns,
287
+ "cache_ttl_days": TTL_DAYS,
288
+ "corpus_size": len(docs),
289
+ "skip_reason": "cold fetch returned < 2 usable docs",
290
+ }
291
+
292
+ try:
293
+ upsert_corpus(topic, docs)
294
+ records = fetch_all_from_namespace(ns, limit=external_target)
295
+ except Exception as e:
296
+ print(
297
+ f"[rs-hybrid] upsert/fetch failed ({e}); dropping to bypass.",
298
+ file=sys.stderr,
299
+ )
300
+ return [], {
301
+ "cache_mode": "bypass",
302
+ "cache_age_days": None,
303
+ "cache_namespace": ns,
304
+ "cache_ttl_days": TTL_DAYS,
305
+ "corpus_size": 0,
306
+ "skip_reason": f"upsert/fetch failed: {e}",
307
+ }
308
+
309
+ return records, {
310
+ "cache_mode": "cold",
311
+ "cache_age_days": 0.0,
312
+ "cache_namespace": ns,
313
+ "cache_ttl_days": TTL_DAYS,
314
+ "corpus_size": len(records),
315
+ }
316
+
317
+
318
+ def _external_records_to_dicts(records: Sequence[Dict[str, Any]]) -> List[Dict[str, Any]]:
319
+ """Shape rs_cache records into unified-corpus dicts (origin=external).
320
+
321
+ Fields mirrored from Plan 89-02 / 89-03 Mode B output so downstream
322
+ bridge-writer (Plan 89-06) can resolve both sides via the existing
323
+ resolvePairIdentity() without per-mode branches.
324
+
325
+ Attached _vector field carries the cached multilingual-e5-large
326
+ embedding when present; rs-engine will prefer it over local MiniLM to
327
+ keep warm/cold semantic consistency with Mode B.
328
+ """
329
+ out: List[Dict[str, Any]] = []
330
+ for rec in records:
331
+ rid = rec.get("id") or ""
332
+ meta = rec.get("metadata", {}) or {}
333
+ abstract = (meta.get("abstract") or "").strip()
334
+ if not abstract:
335
+ continue
336
+ source = (meta.get("source") or "external").strip() or "external"
337
+ title = (meta.get("title") or "").strip() or rid
338
+ doi = (meta.get("doi") or "").strip() or None
339
+ year = meta.get("year") or None
340
+ # Build a stable artifact-style id that matches Mode B cold-path shape.
341
+ eid_slug = re.sub(r"[^a-zA-Z0-9]+", "-", rid).strip("-")
342
+ if not eid_slug:
343
+ eid_slug = re.sub(r"[^a-zA-Z0-9]+", "-", title or "untitled").strip("-")
344
+ external_id = f"{source}/{eid_slug}"
345
+ values = rec.get("values") or []
346
+ out.append({
347
+ "global_id": f"external::{external_id}",
348
+ "origin": "external",
349
+ "external_id": external_id,
350
+ "pinecone_id": rid,
351
+ "source": source,
352
+ "title": title,
353
+ "doi": doi,
354
+ "year": year,
355
+ "path": f"https://doi.org/{doi}" if doi else rid,
356
+ "text": abstract,
357
+ "_vector": list(values) if values else [],
358
+ })
359
+ return out
360
+
361
+
362
+ def _external_docs_to_dicts(docs: Sequence[Dict[str, Any]]) -> List[Dict[str, Any]]:
363
+ """Bypass-path shape: raw rs_corpus docs -> unified-corpus dicts.
364
+
365
+ Used when Pinecone is unavailable and the caller cold-fetched via
366
+ rs_corpus.fetch_corpus directly. Vectors are not attached (caller embeds
367
+ locally via MiniLM).
368
+ """
369
+ out: List[Dict[str, Any]] = []
370
+ for doc in docs:
371
+ abstract = (doc.get("abstract") or "").strip()
372
+ if not abstract:
373
+ continue
374
+ rid = doc.get("external_id") or ""
375
+ source = (doc.get("source") or "external").strip() or "external"
376
+ title = (doc.get("title") or "").strip() or rid
377
+ doi = (doc.get("doi") or "").strip() or None
378
+ year = doc.get("year")
379
+ eid_slug = re.sub(r"[^a-zA-Z0-9]+", "-", rid).strip("-")
380
+ if not eid_slug:
381
+ eid_slug = re.sub(r"[^a-zA-Z0-9]+", "-", title or "untitled").strip("-")
382
+ external_id = f"{source}/{eid_slug}"
383
+ out.append({
384
+ "global_id": f"external::{external_id}",
385
+ "origin": "external",
386
+ "external_id": external_id,
387
+ "pinecone_id": rid,
388
+ "source": source,
389
+ "title": title,
390
+ "doi": doi,
391
+ "year": year,
392
+ "path": doc.get("url") or rid,
393
+ "text": abstract,
394
+ "_vector": [], # bypass path -- no cached vector.
395
+ })
396
+ return out
397
+
398
+
399
+ # --- Public builder ---------------------------------------------------------
400
+
401
+ def build_unified_corpus(
402
+ room_path: str,
403
+ topic: str,
404
+ external_target: int = DEFAULT_EXTERNAL_TARGET,
405
+ ) -> Tuple[List[Dict[str, Any]], "np.ndarray", Dict[str, Any]]:
406
+ """Build a unified (room + external) corpus for Mode C.
407
+
408
+ Returns:
409
+ corpus: flat list of dicts. Room artifacts first (preserved
410
+ discover order), then external docs (Pinecone namespace
411
+ order or fetch order on bypass path).
412
+ origin_mask: np.ndarray[bool], True=room, False=external. Same length
413
+ as corpus. Used by filter_cross_corpus_pairs.
414
+ metadata: {room_count, external_count, room_id, topic, topic_slug,
415
+ cache_mode, cache_age_days, cache_namespace,
416
+ cache_ttl_days, external_target, fetched_at}.
417
+
418
+ Edge cases:
419
+ - Empty room: returns ([], array([]), {...room_count=0...}). Caller
420
+ should short-circuit (no cross-corpus pair is constructible).
421
+ - Empty external: same pattern. Callers should emit a clear "no
422
+ external corpus available" message rather than running the LSA.
423
+ - Bypass path with rs_corpus importable: falls through to cold fetch
424
+ via rs_corpus.fetch_corpus and shapes docs directly.
425
+ - Bypass path with rs_corpus NOT importable: returns empty external
426
+ side (caller decides what to do; typically short-circuits).
427
+
428
+ external_target is clamped to MAX_EXTERNAL_TARGET so misconfigured
429
+ callers cannot request 50k docs and blow memory.
430
+ """
431
+ # Clamp external_target defensively.
432
+ external_target = max(1, min(int(external_target), MAX_EXTERNAL_TARGET))
433
+
434
+ room_path_obj = Path(room_path).resolve()
435
+ if not room_path_obj.is_dir():
436
+ raise ValueError(f"rs_hybrid: room path is not a directory: {room_path_obj}")
437
+
438
+ room_artifacts = _load_room_artifacts(room_path_obj)
439
+ room_id = room_path_obj.name
440
+
441
+ # External side.
442
+ records, cache_meta = _load_external_records(topic, external_target)
443
+ if records:
444
+ external_dicts = _external_records_to_dicts(records)
445
+ else:
446
+ # Attempt the Plan 89-02 bypass path: fetch via rs_corpus directly
447
+ # if it is available. If rs_corpus import failed upstream, the
448
+ # cache_meta carries skip_reason and we return an empty external
449
+ # side -- caller decides whether to abort or continue with only
450
+ # room artifacts (which cross-corpus filter will discard anyway).
451
+ external_dicts = []
452
+ try:
453
+ from lib.core.rs_corpus import fetch_corpus
454
+ print(
455
+ "[rs-hybrid] Bypass path: fetching via rs_corpus (no Pinecone).",
456
+ file=sys.stderr,
457
+ )
458
+ fetch_target = max(external_target * 2, external_target + 200)
459
+ docs = fetch_corpus(topic, target_n=fetch_target)
460
+ external_dicts = _external_docs_to_dicts(docs[:external_target])
461
+ # Update cache_meta corpus_size to reflect actual bypass-path
462
+ # usable count.
463
+ cache_meta["corpus_size"] = len(external_dicts)
464
+ except Exception as e:
465
+ print(
466
+ f"[rs-hybrid] bypass-path fetch also failed ({e}); "
467
+ f"proceeding with empty external corpus.",
468
+ file=sys.stderr,
469
+ )
470
+
471
+ # Cap external side at external_target explicitly (defense-in-depth in
472
+ # case cache returned more than requested).
473
+ external_dicts = external_dicts[:external_target]
474
+
475
+ corpus: List[Dict[str, Any]] = list(room_artifacts) + list(external_dicts)
476
+
477
+ # Build origin_mask. True=room, False=external. Matches plan contract.
478
+ origin_mask = np.array(
479
+ [c["origin"] == "room" for c in corpus],
480
+ dtype=bool,
481
+ )
482
+
483
+ # Resolve topic slug the same way Mode B does so the hybrid output
484
+ # path aligns with research/{topic-slug}/ conventions.
485
+ try:
486
+ from lib.core.rs_corpus import topic_slug
487
+ slug = topic_slug(topic)
488
+ except Exception:
489
+ slug = re.sub(r"[^a-z0-9]+", "-", (topic or "").lower()).strip("-")
490
+
491
+ metadata: Dict[str, Any] = {
492
+ "room_count": int(origin_mask.sum()),
493
+ "external_count": int((~origin_mask).sum()),
494
+ "room_id": room_id,
495
+ "room_path": str(room_path_obj),
496
+ "topic": topic,
497
+ "topic_slug": slug,
498
+ "external_target": external_target,
499
+ "fetched_at": datetime.now(timezone.utc).isoformat(),
500
+ }
501
+ # Surface the 89-03 cache provenance so rs-engine can write it to
502
+ # .rs-engine-results.json and downstream consumers see warm/cold mode.
503
+ metadata.update({k: v for k, v in cache_meta.items() if k != "corpus_size"})
504
+
505
+ return corpus, origin_mask, metadata
506
+
507
+
508
+ # --- Cross-corpus pair filter -----------------------------------------------
509
+
510
+ def filter_cross_corpus_pairs(
511
+ pairs: Sequence[Any],
512
+ origin_mask: "np.ndarray",
513
+ topk: int,
514
+ ) -> List[Dict[str, Any]]:
515
+ """Keep only pairs where one side is room and other is external.
516
+
517
+ Accepts either:
518
+ - abs_diff_topk tuple output: list of (i, j, signed, abs_diff) tuples.
519
+ - Dict-shaped pairs already carrying i/j (future-proofing; not used by
520
+ the current rs-engine call site but allows unit testing with mocks).
521
+
522
+ Returns a list of dicts augmented with room_side and external_side
523
+ (integer indices into the unified corpus). The bridge-writer (Plan
524
+ 89-06) consumes these through resolvePairIdentity.
525
+
526
+ Stops as soon as `topk` cross-corpus pairs have been collected. The
527
+ caller should request abs_diff_topk with an overshoot multiplier so
528
+ intra-corpus pairs (room-room or external-external) can be discarded
529
+ without starving the final result below topk.
530
+ """
531
+ out: List[Dict[str, Any]] = []
532
+ topk = max(1, int(topk))
533
+
534
+ for p in pairs:
535
+ # Normalize input shape.
536
+ if isinstance(p, dict):
537
+ i = int(p.get("i", -1))
538
+ j = int(p.get("j", -1))
539
+ signed = p.get("signed_diff", p.get("signed"))
540
+ absv = p.get("abs_diff", p.get("abs"))
541
+ else:
542
+ try:
543
+ i, j, signed, absv = p
544
+ except (ValueError, TypeError):
545
+ continue
546
+ i, j = int(i), int(j)
547
+
548
+ if i < 0 or j < 0:
549
+ continue
550
+ if i >= len(origin_mask) or j >= len(origin_mask):
551
+ continue
552
+ if bool(origin_mask[i]) == bool(origin_mask[j]):
553
+ # Same origin -- intra-corpus pair. Discard.
554
+ continue
555
+
556
+ # Canonical orientation: room_side is the room index, external_side
557
+ # is the external index. Matches plan Detailed Steps snippet.
558
+ if bool(origin_mask[i]):
559
+ room_side, external_side = i, j
560
+ else:
561
+ room_side, external_side = j, i
562
+
563
+ out.append({
564
+ "i": i,
565
+ "j": j,
566
+ "signed_diff": float(signed) if signed is not None else 0.0,
567
+ "abs_diff": float(absv) if absv is not None else 0.0,
568
+ "room_side": room_side,
569
+ "external_side": external_side,
570
+ })
571
+
572
+ if len(out) >= topk:
573
+ break
574
+
575
+ return out
576
+
577
+
578
+ __all__ = [
579
+ "build_unified_corpus",
580
+ "filter_cross_corpus_pairs",
581
+ "DEFAULT_EXTERNAL_TARGET",
582
+ "MAX_EXTERNAL_TARGET",
583
+ "SKIP_FILES",
584
+ "SKIP_DIRS",
585
+ "MIN_BODY_CHARS",
586
+ ]