@mindrian_os/cli 1.13.0-beta.10 → 1.13.0-beta.44

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 (590) hide show
  1. package/CHANGELOG.md +348 -13
  2. package/README.md +76 -579
  3. package/bin/cli.js +166 -40
  4. package/lib/core/active-plugin-root.cjs +207 -0
  5. package/package.json +7 -13
  6. package/.claude-plugin/plugin.json +0 -21
  7. package/.mcp.json +0 -9
  8. package/agents/brain-query.md +0 -80
  9. package/agents/framework-runner.md +0 -237
  10. package/agents/grading.md +0 -188
  11. package/agents/investor.md +0 -128
  12. package/agents/larry-extended.md +0 -135
  13. package/agents/opportunity-scanner.md +0 -91
  14. package/agents/persona-analyst.md +0 -132
  15. package/agents/research.md +0 -89
  16. package/agents/reverse-salient-agent.md +0 -27
  17. package/bin/mindrian-mcp-server.cjs +0 -182
  18. package/bin/mindrian-tools.cjs +0 -765
  19. package/commands/act.md +0 -433
  20. package/commands/admin.md +0 -404
  21. package/commands/analyze-needs.md +0 -36
  22. package/commands/analyze-systems.md +0 -33
  23. package/commands/analyze-timing.md +0 -36
  24. package/commands/auto-explore.md +0 -64
  25. package/commands/beautiful-question.md +0 -34
  26. package/commands/brain-derive.md +0 -78
  27. package/commands/build-knowledge.md +0 -36
  28. package/commands/build-thesis.md +0 -40
  29. package/commands/causal.md +0 -228
  30. package/commands/challenge-assumptions.md +0 -33
  31. package/commands/compare-ventures.md +0 -77
  32. package/commands/dashboard.md +0 -110
  33. package/commands/deep-grade.md +0 -76
  34. package/commands/diagnose.md +0 -52
  35. package/commands/diagnostics.md +0 -145
  36. package/commands/doctor.md +0 -151
  37. package/commands/dominant-designs.md +0 -34
  38. package/commands/explain-decision.md +0 -87
  39. package/commands/explore-domains.md +0 -36
  40. package/commands/explore-futures.md +0 -34
  41. package/commands/explore-trends.md +0 -36
  42. package/commands/export.md +0 -103
  43. package/commands/file-meeting.md +0 -724
  44. package/commands/find-analogies.md +0 -182
  45. package/commands/find-bottlenecks.md +0 -56
  46. package/commands/find-connections.md +0 -70
  47. package/commands/funding.md +0 -81
  48. package/commands/grade.md +0 -197
  49. package/commands/graph.md +0 -128
  50. package/commands/hat-briefing.md +0 -119
  51. package/commands/heal.md +0 -196
  52. package/commands/help.md +0 -399
  53. package/commands/hmi-status.md +0 -172
  54. package/commands/jtbd.md +0 -241
  55. package/commands/leadership.md +0 -73
  56. package/commands/lean-canvas.md +0 -34
  57. package/commands/macro-trends.md +0 -34
  58. package/commands/map-unknowns.md +0 -34
  59. package/commands/memory.md +0 -173
  60. package/commands/models.md +0 -175
  61. package/commands/mos-reason.md +0 -279
  62. package/commands/mullins.md +0 -114
  63. package/commands/new-project.md +0 -481
  64. package/commands/onboard.md +0 -434
  65. package/commands/operator.md +0 -149
  66. package/commands/opportunities.md +0 -144
  67. package/commands/organize.md +0 -497
  68. package/commands/persona.md +0 -192
  69. package/commands/pipeline.md +0 -106
  70. package/commands/present.md +0 -91
  71. package/commands/publish.md +0 -201
  72. package/commands/query.md +0 -124
  73. package/commands/radar.md +0 -72
  74. package/commands/reanalyze.md +0 -91
  75. package/commands/research.md +0 -190
  76. package/commands/room.md +0 -352
  77. package/commands/rooms.md +0 -598
  78. package/commands/root-cause.md +0 -34
  79. package/commands/rs-experts.md +0 -79
  80. package/commands/rs-explain.md +0 -94
  81. package/commands/rs-fetch.md +0 -88
  82. package/commands/rs-thesis.md +0 -79
  83. package/commands/scenario-plan.md +0 -34
  84. package/commands/scheduled-tasks.md +0 -285
  85. package/commands/score-innovation.md +0 -37
  86. package/commands/scout.md +0 -239
  87. package/commands/setup.md +0 -618
  88. package/commands/snapshot.md +0 -147
  89. package/commands/speakers.md +0 -84
  90. package/commands/splash.md +0 -28
  91. package/commands/status.md +0 -75
  92. package/commands/structure-argument.md +0 -36
  93. package/commands/suggest-next.md +0 -74
  94. package/commands/systems-thinking.md +0 -34
  95. package/commands/think-hats.md +0 -36
  96. package/commands/update.md +0 -181
  97. package/commands/user-needs.md +0 -34
  98. package/commands/validate.md +0 -34
  99. package/commands/value-proposition.md +0 -55
  100. package/commands/vault.md +0 -180
  101. package/commands/visualize.md +0 -52
  102. package/commands/whitespace.md +0 -501
  103. package/commands/wiki.md +0 -69
  104. package/hooks/hooks.json +0 -381
  105. package/hooks/run-hook.cmd +0 -64
  106. package/lib/__init__.py +0 -0
  107. package/lib/__pycache__/__init__.cpython-312.pyc +0 -0
  108. package/lib/agents/auto-explore-agent.cjs +0 -1043
  109. package/lib/agents/reverse-salient-agent.cjs +0 -679
  110. package/lib/agents/tension-hook-agent.cjs +0 -544
  111. package/lib/chat/chat-context.js +0 -185
  112. package/lib/chat/chat-panel.js +0 -721
  113. package/lib/chat/fabric-chat.cjs +0 -288
  114. package/lib/chat/generative-tools.js +0 -219
  115. package/lib/conversation/ROOM.md +0 -39
  116. package/lib/conversation/classifier-rules.json +0 -38
  117. package/lib/conversation/classifier.cjs +0 -264
  118. package/lib/conversation/operator.cjs +0 -287
  119. package/lib/copy/115-spec-strings.cjs +0 -55
  120. package/lib/core/__init__.py +0 -0
  121. package/lib/core/__nav-stub.cjs +0 -14
  122. package/lib/core/__pycache__/__init__.cpython-312.pyc +0 -0
  123. package/lib/core/__pycache__/rs-math.cpython-312.pyc +0 -0
  124. package/lib/core/__pycache__/rs_cache.cpython-312.pyc +0 -0
  125. package/lib/core/__pycache__/rs_corpus.cpython-312.pyc +0 -0
  126. package/lib/core/__pycache__/rs_hybrid.cpython-312.pyc +0 -0
  127. package/lib/core/__pycache__/rs_math.cpython-312.pyc +0 -0
  128. package/lib/core/__pycache__/rs_rooms.cpython-312.pyc +0 -0
  129. package/lib/core/artifact-id.cjs +0 -148
  130. package/lib/core/asset-ops.cjs +0 -151
  131. package/lib/core/auto-commit-throttle.cjs +0 -129
  132. package/lib/core/bearer-token.cjs +0 -199
  133. package/lib/core/brain-client.cjs +0 -865
  134. package/lib/core/brain-derivation-prompts.cjs +0 -326
  135. package/lib/core/brain-derivation-queue.cjs +0 -431
  136. package/lib/core/brain-derivation.cjs +0 -580
  137. package/lib/core/brain-md-schema.cjs +0 -528
  138. package/lib/core/brain-md-staleness.cjs +0 -357
  139. package/lib/core/brain-response-sanitize.cjs +0 -188
  140. package/lib/core/bridge-writer.cjs +0 -477
  141. package/lib/core/chat-context-builder.cjs +0 -253
  142. package/lib/core/cross-room-aggregator.cjs +0 -762
  143. package/lib/core/daily-briefing.cjs +0 -438
  144. package/lib/core/decision-capture.cjs +0 -618
  145. package/lib/core/deep-links.cjs +0 -82
  146. package/lib/core/dispatch-optimizer.cjs +0 -354
  147. package/lib/core/dual-path-detector.cjs +0 -84
  148. package/lib/core/dual-path-detector.test.cjs +0 -334
  149. package/lib/core/exports-log.cjs +0 -79
  150. package/lib/core/feynman-minto-invariants.cjs +0 -605
  151. package/lib/core/folder-memory-async.cjs +0 -338
  152. package/lib/core/folder-memory-shared.cjs +0 -890
  153. package/lib/core/folder-memory.cjs +0 -416
  154. package/lib/core/framework-chain-composer.cjs +0 -411
  155. package/lib/core/frontmatter-schemas.cjs +0 -330
  156. package/lib/core/git-ops.cjs +0 -141
  157. package/lib/core/graph-ops.cjs +0 -258
  158. package/lib/core/hat-persistence.cjs +0 -362
  159. package/lib/core/index.cjs +0 -60
  160. package/lib/core/integration-registry.cjs +0 -232
  161. package/lib/core/intelligence-cascade.cjs +0 -661
  162. package/lib/core/lazygraph-ops.cjs +0 -1057
  163. package/lib/core/lru-cache.cjs +0 -139
  164. package/lib/core/mcp-profiles.cjs +0 -182
  165. package/lib/core/meeting-ops.cjs +0 -54
  166. package/lib/core/memory-ops.cjs +0 -600
  167. package/lib/core/migrations/ROOM.md +0 -33
  168. package/lib/core/migrations/phase-109-nodes-provenance.cjs +0 -339
  169. package/lib/core/migrations/phase-109-session-focus.cjs +0 -99
  170. package/lib/core/model-profiles.cjs +0 -246
  171. package/lib/core/mullins-scaffold.cjs +0 -160
  172. package/lib/core/nav-dial.cjs +0 -316
  173. package/lib/core/navigation/ROOM.md +0 -15
  174. package/lib/core/navigation/explanation.cjs +0 -43
  175. package/lib/core/navigation/focus.cjs +0 -135
  176. package/lib/core/navigation/ingestion.cjs +0 -82
  177. package/lib/core/navigation/insights.cjs +0 -350
  178. package/lib/core/navigation/memory-events.cjs +0 -118
  179. package/lib/core/navigation/neighborhood.cjs +0 -78
  180. package/lib/core/navigation/packet.cjs +0 -182
  181. package/lib/core/navigation/room-home.cjs +0 -127
  182. package/lib/core/navigation/transitions.cjs +0 -82
  183. package/lib/core/navigation-engine-shared.cjs +0 -242
  184. package/lib/core/navigation-engine.cjs +0 -664
  185. package/lib/core/navigation.cjs +0 -60
  186. package/lib/core/nl-graph-queries.cjs +0 -164
  187. package/lib/core/offer-presenter.cjs +0 -406
  188. package/lib/core/opportunity-extractor.cjs +0 -183
  189. package/lib/core/opportunity-ops.cjs +0 -1371
  190. package/lib/core/persona-ops.cjs +0 -537
  191. package/lib/core/persona-taxonomy.cjs +0 -190
  192. package/lib/core/platform-gates.cjs +0 -120
  193. package/lib/core/platform.cjs +0 -257
  194. package/lib/core/proactive-intelligence.cjs +0 -528
  195. package/lib/core/problem-type-router.cjs +0 -315
  196. package/lib/core/reasoning-ops.cjs +0 -639
  197. package/lib/core/reverse-salient-persona-suffix.cjs +0 -115
  198. package/lib/core/room-classifier-strict-mode.cjs +0 -229
  199. package/lib/core/room-db.cjs +0 -127
  200. package/lib/core/room-ops-async.cjs +0 -92
  201. package/lib/core/room-ops-shared.cjs +0 -64
  202. package/lib/core/room-ops-sync.cjs +0 -70
  203. package/lib/core/room-ops.cjs +0 -32
  204. package/lib/core/room-type-detector.cjs +0 -386
  205. package/lib/core/rs-brain-substrate-prompts.cjs +0 -129
  206. package/lib/core/rs-brain-substrate.cjs +0 -570
  207. package/lib/core/rs-breakthrough-scorer.cjs +0 -255
  208. package/lib/core/rs-canon-violations.cjs +0 -82
  209. package/lib/core/rs-chain-feeder.cjs +0 -343
  210. package/lib/core/rs-commercial-assessor.cjs +0 -280
  211. package/lib/core/rs-differential-scorer.cjs +0 -376
  212. package/lib/core/rs-domain-analyzer.cjs +0 -385
  213. package/lib/core/rs-egress-prompts.cjs +0 -113
  214. package/lib/core/rs-egress-telemetry.cjs +0 -225
  215. package/lib/core/rs-egress-violations.cjs +0 -53
  216. package/lib/core/rs-expert-mapper.cjs +0 -467
  217. package/lib/core/rs-fetcher-academic.cjs +0 -697
  218. package/lib/core/rs-fetcher-experts.cjs +0 -314
  219. package/lib/core/rs-fetcher-industry.cjs +0 -731
  220. package/lib/core/rs-fetcher-patents.cjs +0 -564
  221. package/lib/core/rs-innovation-classifier.cjs +0 -194
  222. package/lib/core/rs-mind-map.cjs +0 -656
  223. package/lib/core/rs-neo4j-writer.cjs +0 -388
  224. package/lib/core/rs-nl-to-query.cjs +0 -425
  225. package/lib/core/rs-pinecone-bridge.cjs +0 -303
  226. package/lib/core/rs-preprocessor.cjs +0 -350
  227. package/lib/core/rs-query-matrix.cjs +0 -316
  228. package/lib/core/rs-query-to-text.cjs +0 -438
  229. package/lib/core/rs-sqlite-mirror.cjs +0 -443
  230. package/lib/core/rs-thesis-generator.cjs +0 -188
  231. package/lib/core/rs_cache.py +0 -479
  232. package/lib/core/rs_corpus.py +0 -468
  233. package/lib/core/rs_hybrid.py +0 -586
  234. package/lib/core/rs_math.py +0 -287
  235. package/lib/core/rs_rooms.py +0 -193
  236. package/lib/core/scheduled-scanner.cjs +0 -463
  237. package/lib/core/scratchpad-ops.cjs +0 -201
  238. package/lib/core/section-8-trace-schema.cjs +0 -138
  239. package/lib/core/section-registry.cjs +0 -111
  240. package/lib/core/session-state.cjs +0 -144
  241. package/lib/core/shallow-doc-parser.cjs +0 -174
  242. package/lib/core/shallow-doc-parser.test.cjs +0 -226
  243. package/lib/core/skill-activation-router.cjs +0 -284
  244. package/lib/core/state-ops.cjs +0 -46
  245. package/lib/core/statusline-cache.cjs +0 -266
  246. package/lib/core/token-estimator.cjs +0 -348
  247. package/lib/core/user-archetype.cjs +0 -239
  248. package/lib/core/user-md-ops.cjs +0 -524
  249. package/lib/core/visual-ops.cjs +0 -624
  250. package/lib/core/write-lock.cjs +0 -149
  251. package/lib/graph/canvas-graph.js +0 -467
  252. package/lib/graph/constellation-config.cjs +0 -299
  253. package/lib/graph/graph-detail-panel.js +0 -165
  254. package/lib/hmi/ROOM.md +0 -47
  255. package/lib/hmi/across-session-memory.cjs +0 -604
  256. package/lib/hmi/cross-room-memory.cjs +0 -575
  257. package/lib/hmi/decoy-tier.cjs +0 -395
  258. package/lib/hmi/jtbd-classifier.cjs +0 -219
  259. package/lib/hmi/jtbd-state.cjs +0 -199
  260. package/lib/hmi/jtbd-taxonomy.json +0 -392
  261. package/lib/hmi/selector-dispatcher.cjs +0 -546
  262. package/lib/hmi/selector-telemetry.cjs +0 -263
  263. package/lib/hmi/shape-f0-renderer.cjs +0 -139
  264. package/lib/hmi/shape-f1-fallback.cjs +0 -80
  265. package/lib/hmi/shape-f1-renderer.cjs +0 -138
  266. package/lib/hmi/shape-f2-renderer.cjs +0 -132
  267. package/lib/hmi/shape-f3-renderer.cjs +0 -66
  268. package/lib/hmi/shape-f4-renderer.cjs +0 -72
  269. package/lib/hmi/shape-f5-renderer.cjs +0 -155
  270. package/lib/hmi/shape-f6-plan-review-renderer.cjs +0 -312
  271. package/lib/hmi/shape-f6-renderer.cjs +0 -144
  272. package/lib/hmi/shape-g-renderer.cjs +0 -219
  273. package/lib/hmi/shape-h-renderer.cjs +0 -222
  274. package/lib/hmi/tier-check.cjs +0 -63
  275. package/lib/import/PRECONDITIONS.md +0 -41
  276. package/lib/import/branding.cjs +0 -210
  277. package/lib/import/branding.test.cjs +0 -235
  278. package/lib/import/classifications-sync.cjs +0 -104
  279. package/lib/import/classifications-sync.test.cjs +0 -129
  280. package/lib/import/enricher.cjs +0 -296
  281. package/lib/import/enricher.test.cjs +0 -273
  282. package/lib/import/integration.test.cjs +0 -376
  283. package/lib/import/manifest.cjs +0 -129
  284. package/lib/import/manifest.schema.json +0 -185
  285. package/lib/import/manifest.test.cjs +0 -123
  286. package/lib/import/meeting-detector.cjs +0 -92
  287. package/lib/import/meeting-detector.test.cjs +0 -100
  288. package/lib/import/person-detector.cjs +0 -229
  289. package/lib/import/person-detector.test.cjs +0 -149
  290. package/lib/import/report.cjs +0 -186
  291. package/lib/import/report.test.cjs +0 -186
  292. package/lib/import/room-md-scaffolder.cjs +0 -49
  293. package/lib/import/router.cjs +0 -224
  294. package/lib/import/router.test.cjs +0 -356
  295. package/lib/import/run-all-tests.cjs +0 -36
  296. package/lib/import/smoke-test.cjs +0 -213
  297. package/lib/import/smoke-test.test.cjs +0 -148
  298. package/lib/import/test-fixtures/collision-vault/preexisting-room/STATE.md +0 -8
  299. package/lib/import/test-fixtures/collision-vault/preexisting-room/problem-definition/onboarding/onboarding.md +0 -7
  300. package/lib/import/test-fixtures/collision-vault/source/onboarding.md +0 -5
  301. package/lib/import/test-fixtures/obsidian-vault/.obsidian/workspace.json +0 -1
  302. package/lib/import/test-fixtures/obsidian-vault/notes/with-wikilinks.md +0 -4
  303. package/lib/import/test-fixtures/tiny-vault/notes/2026-01-15-team-sync.md +0 -9
  304. package/lib/import/test-fixtures/tiny-vault/notes/empty.md +0 -3
  305. package/lib/import/test-fixtures/tiny-vault/notes/onboarding.md +0 -5
  306. package/lib/import/test-fixtures/tiny-vault/notes/pricing.md +0 -5
  307. package/lib/import/test-fixtures/tiny-vault/notes/random.md +0 -4
  308. package/lib/import/undo.test.cjs +0 -199
  309. package/lib/import/vault-scanner.cjs +0 -105
  310. package/lib/import/vault-scanner.test.cjs +0 -67
  311. package/lib/mcp/app-html/dashboard.html +0 -316
  312. package/lib/mcp/app-html/graph.html +0 -428
  313. package/lib/mcp/app-html/mindrian-platform.html +0 -1841
  314. package/lib/mcp/app-html/wiki.html +0 -383
  315. package/lib/mcp/app-views.cjs +0 -322
  316. package/lib/mcp/brain-router.cjs +0 -418
  317. package/lib/mcp/capability-registry.cjs +0 -62
  318. package/lib/mcp/larry-context.cjs +0 -46
  319. package/lib/mcp/larry-server-instructions.md +0 -114
  320. package/lib/mcp/pipeline-state.cjs +0 -275
  321. package/lib/mcp/prompts.cjs +0 -302
  322. package/lib/mcp/resources.cjs +0 -227
  323. package/lib/mcp/session-catchup.cjs +0 -327
  324. package/lib/mcp/surface-detect.cjs +0 -75
  325. package/lib/mcp/tool-router.cjs +0 -1034
  326. package/lib/memory/aaak-compress.cjs +0 -403
  327. package/lib/memory/aaak-compress.test.cjs +0 -288
  328. package/lib/memory/async-artifact-auto-commit.test.cjs +0 -223
  329. package/lib/memory/bearer-token.test.cjs +0 -315
  330. package/lib/memory/brain-cache-lru.test.cjs +0 -259
  331. package/lib/memory/brain-client-query-shape.test.cjs +0 -160
  332. package/lib/memory/brain-derivation-graceful-degradation.test.cjs +0 -1019
  333. package/lib/memory/brain-derivation-queue.test.cjs +0 -539
  334. package/lib/memory/brain-derivation.test.cjs +0 -634
  335. package/lib/memory/brain-derive-command.test.cjs +0 -534
  336. package/lib/memory/brain-md-invariants-validator.test.cjs +0 -704
  337. package/lib/memory/brain-md-schema.test.cjs +0 -467
  338. package/lib/memory/brain-md-staleness.test.cjs +0 -525
  339. package/lib/memory/brain-server-resolution.test.cjs +0 -314
  340. package/lib/memory/chat-context.test.cjs +0 -128
  341. package/lib/memory/cross-room-aggregator.test.cjs +0 -909
  342. package/lib/memory/dashboard-server.test.cjs +0 -256
  343. package/lib/memory/debouncer-drain-at-prompt.test.cjs +0 -389
  344. package/lib/memory/decision-capture.test.cjs +0 -632
  345. package/lib/memory/decision-capture.worker.cjs +0 -70
  346. package/lib/memory/explain-decision-command.test.cjs +0 -521
  347. package/lib/memory/explain-decision-footer.test.cjs +0 -316
  348. package/lib/memory/explored-materials-store.cjs +0 -392
  349. package/lib/memory/feynman-minto-guardian.test.cjs +0 -736
  350. package/lib/memory/feynman-minto-invariants.test.cjs +0 -511
  351. package/lib/memory/feynman-prompts-drift.test.cjs +0 -144
  352. package/lib/memory/feynman-prompts.cjs +0 -151
  353. package/lib/memory/feynman-prompts.test.cjs +0 -96
  354. package/lib/memory/folder-memory-quadruple.test.cjs +0 -548
  355. package/lib/memory/folder-memory.test.cjs +0 -503
  356. package/lib/memory/framework-chain-composer.test.cjs +0 -515
  357. package/lib/memory/frontmatter-schema-validator.test.cjs +0 -290
  358. package/lib/memory/heal-command.test.cjs +0 -604
  359. package/lib/memory/index-artifact-transaction.test.cjs +0 -333
  360. package/lib/memory/lazygraph-rs-discoveries-view.test.cjs +0 -122
  361. package/lib/memory/mcp-input-validation.test.cjs +0 -240
  362. package/lib/memory/mcp-server-brain-deps.test.cjs +0 -270
  363. package/lib/memory/mcp-stack-fallback.test.cjs +0 -433
  364. package/lib/memory/minto-debouncer.test.cjs +0 -407
  365. package/lib/memory/minto-debouncer.worker.cjs +0 -46
  366. package/lib/memory/minto-migration-v88.test.cjs +0 -265
  367. package/lib/memory/minto-schema-v88.test.cjs +0 -390
  368. package/lib/memory/mos-status-renderer.test.cjs +0 -631
  369. package/lib/memory/narrative-schema.cjs +0 -376
  370. package/lib/memory/narrative-schema.test.cjs +0 -209
  371. package/lib/memory/nav-dial.test.cjs +0 -414
  372. package/lib/memory/navigation-engine-core.test.cjs +0 -722
  373. package/lib/memory/navigation-invariants.test.cjs +0 -483
  374. package/lib/memory/offer-presenter.test.cjs +0 -554
  375. package/lib/memory/on-stop-snapshot.test.cjs +0 -404
  376. package/lib/memory/pending-tension-store.cjs +0 -373
  377. package/lib/memory/post-compact-reinjection.test.cjs +0 -854
  378. package/lib/memory/post-write-triple.test.cjs +0 -317
  379. package/lib/memory/pre-compact-snapshot.test.cjs +0 -495
  380. package/lib/memory/problem-type-router.test.cjs +0 -656
  381. package/lib/memory/query-efficiency-telemetry.test.cjs +0 -370
  382. package/lib/memory/recompile-room-references.test.cjs +0 -392
  383. package/lib/memory/recompile-room-references.worker.cjs +0 -42
  384. package/lib/memory/record-decision-dual-write.test.cjs +0 -454
  385. package/lib/memory/room-classifier-strict-mode.test.cjs +0 -417
  386. package/lib/memory/room-minto-hook.test.cjs +0 -398
  387. package/lib/memory/rs-discovery-engine.test.cjs +0 -323
  388. package/lib/memory/run-feynman-tests.cjs +0 -1239
  389. package/lib/memory/security-trifecta.test.cjs +0 -312
  390. package/lib/memory/session-start-brain-staleness.test.cjs +0 -363
  391. package/lib/memory/session-start-triple-injection.test.cjs +0 -514
  392. package/lib/memory/sessionstart-banner-formatter.cjs +0 -318
  393. package/lib/memory/sessionstart-minto-banner.test.cjs +0 -373
  394. package/lib/memory/skill-activation-router.test.cjs +0 -419
  395. package/lib/memory/stamp-artifact-write.test.cjs +0 -304
  396. package/lib/memory/statusline-active-room.test.cjs +0 -315
  397. package/lib/memory/statusline-minto-segment.test.cjs +0 -292
  398. package/lib/memory/sync-async-entry-points.test.cjs +0 -204
  399. package/lib/memory/test-bridge-writer-enhanced.cjs +0 -452
  400. package/lib/memory/test-rs-brain-substrate-shape.cjs +0 -529
  401. package/lib/memory/test-rs-brain-substrate.cjs +0 -636
  402. package/lib/memory/test-rs-breakthrough-scorer.cjs +0 -375
  403. package/lib/memory/test-rs-canon-violations.cjs +0 -218
  404. package/lib/memory/test-rs-chain-feeder-core.cjs +0 -344
  405. package/lib/memory/test-rs-chain-feeder-skill-spawn.cjs +0 -297
  406. package/lib/memory/test-rs-commercial-assessor.cjs +0 -385
  407. package/lib/memory/test-rs-differential-scorer.cjs +0 -480
  408. package/lib/memory/test-rs-discovery-engine.cjs +0 -603
  409. package/lib/memory/test-rs-domain-analyzer.cjs +0 -492
  410. package/lib/memory/test-rs-egress-primitives.cjs +0 -420
  411. package/lib/memory/test-rs-expert-mapper.cjs +0 -547
  412. package/lib/memory/test-rs-explain-command.cjs +0 -443
  413. package/lib/memory/test-rs-fetcher-academic.cjs +0 -848
  414. package/lib/memory/test-rs-fetcher-experts.cjs +0 -496
  415. package/lib/memory/test-rs-fetcher-industry.cjs +0 -702
  416. package/lib/memory/test-rs-fetcher-patents.cjs +0 -674
  417. package/lib/memory/test-rs-innovation-classifier.cjs +0 -301
  418. package/lib/memory/test-rs-mind-map.cjs +0 -646
  419. package/lib/memory/test-rs-neo4j-writer.cjs +0 -518
  420. package/lib/memory/test-rs-nl-to-query.cjs +0 -449
  421. package/lib/memory/test-rs-pinecone-bridge.cjs +0 -277
  422. package/lib/memory/test-rs-preprocessor.cjs +0 -433
  423. package/lib/memory/test-rs-query-matrix.cjs +0 -391
  424. package/lib/memory/test-rs-query-to-text.cjs +0 -551
  425. package/lib/memory/test-rs-sqlite-mirror.cjs +0 -649
  426. package/lib/memory/test-rs-thesis-generator.cjs +0 -360
  427. package/lib/memory/triple-context-formatter.cjs +0 -473
  428. package/lib/memory/triple-context-formatter.test.cjs +0 -442
  429. package/lib/memory/user-md-persona.test.cjs +0 -565
  430. package/lib/memory/userpromptsubmit-integration.test.cjs +0 -690
  431. package/lib/memory/validators/README.md +0 -157
  432. package/lib/memory/validators/brain-md-invariants.cjs +0 -475
  433. package/lib/memory/validators/brain-substrate-invariants.cjs +0 -285
  434. package/lib/memory/validators/external-academic-invariants.cjs +0 -249
  435. package/lib/memory/validators/external-industry-invariants.cjs +0 -271
  436. package/lib/memory/validators/external-patents-invariants.cjs +0 -266
  437. package/lib/memory/validators/minto-invariants.cjs +0 -62
  438. package/lib/memory/validators/navigation-invariants.cjs +0 -340
  439. package/lib/memory/validators/queue-health.cjs +0 -95
  440. package/lib/memory/validators/snapshot-integrity.cjs +0 -129
  441. package/lib/memory/validators/stale-lifecycle.cjs +0 -116
  442. package/lib/memory/vault-section-minto-generator-atomic.test.cjs +0 -556
  443. package/lib/memory/vault-section-minto-generator-atomic.worker.cjs +0 -73
  444. package/lib/memory/write-lock-atomic.test.cjs +0 -137
  445. package/lib/memory/write-lock-atomic.worker.cjs +0 -55
  446. package/lib/parity/check-parity.cjs +0 -83
  447. package/lib/presentation/presentation-server.cjs +0 -101
  448. package/lib/presentation/presentation-watcher.cjs +0 -123
  449. package/lib/quickview/hub-server.cjs +0 -719
  450. package/lib/quickview/server.cjs +0 -533
  451. package/lib/render/JTBD-PALETTES.md +0 -145
  452. package/lib/render/ROOM.md +0 -59
  453. package/lib/render/render-v2.cjs +0 -486
  454. package/lib/render/render-v2.test.cjs +0 -267
  455. package/lib/render/render.cjs +0 -65
  456. package/lib/state/ROOM.md +0 -46
  457. package/lib/state/state-md-parser.cjs +0 -215
  458. package/lib/statusline/ROOM.md +0 -38
  459. package/lib/statusline/banner-suppression.cjs +0 -50
  460. package/lib/statusline/surface-detect.cjs +0 -85
  461. package/lib/update-bootstrap.sh.template +0 -145
  462. package/lib/vault/frontmatter-schema.cjs +0 -297
  463. package/lib/vault/room-scanner.cjs +0 -352
  464. package/lib/vault/wikilink-builder.cjs +0 -231
  465. package/lib/vault/wikilink-builder.test.cjs +0 -182
  466. package/lib/wiki/graph-links.cjs +0 -281
  467. package/lib/wiki/page-renderer.cjs +0 -229
  468. package/lib/wiki/wiki-chat.cjs +0 -81
  469. package/lib/wiki/wiki-layout.cjs +0 -1459
  470. package/lib/wiki/wiki-search.cjs +0 -142
  471. package/lib/wiki/wiki-server.cjs +0 -678
  472. package/lib/wiki/wiki-watcher.cjs +0 -105
  473. package/pipelines/analogy/01-decompose.md +0 -80
  474. package/pipelines/analogy/02-abstract.md +0 -87
  475. package/pipelines/analogy/03-search.md +0 -135
  476. package/pipelines/analogy/04-transfer.md +0 -101
  477. package/pipelines/analogy/05-validate.md +0 -106
  478. package/pipelines/analogy/CHAIN.md +0 -56
  479. package/pipelines/discovery/01-explore-domains.md +0 -44
  480. package/pipelines/discovery/02-think-hats.md +0 -50
  481. package/pipelines/discovery/03-analyze-needs.md +0 -54
  482. package/pipelines/discovery/CHAIN.md +0 -37
  483. package/pipelines/thesis/01-structure-argument.md +0 -45
  484. package/pipelines/thesis/02-challenge-assumptions.md +0 -48
  485. package/pipelines/thesis/03-build-thesis.md +0 -54
  486. package/pipelines/thesis/CHAIN.md +0 -37
  487. package/references/brain/causal-directives.md +0 -91
  488. package/references/brain/causal-enrichment.cypher +0 -165
  489. package/references/brain/command-triggers-schema.md +0 -226
  490. package/references/brain/graph-architecture.md +0 -317
  491. package/references/brain/query-patterns.md +0 -460
  492. package/references/brain/room-hierarchy-schema.md +0 -218
  493. package/references/brain/schema.md +0 -76
  494. package/references/capability-radar/capabilities-index.md +0 -241
  495. package/references/capability-radar/changelog-cache.md +0 -81
  496. package/references/causal/causal-schema.md +0 -103
  497. package/references/design/email-template-standard.md +0 -155
  498. package/references/design/graph-visualization-standard.md +0 -178
  499. package/references/document-generation.md +0 -179
  500. package/references/hsi/HSI-TOOLS-REFERENCE.md +0 -222
  501. package/references/import-config.md +0 -141
  502. package/references/integrations/detection-patterns.md +0 -101
  503. package/references/meeting/artifact-template.md +0 -377
  504. package/references/meeting/cross-meeting-intelligence.md +0 -216
  505. package/references/meeting/cross-relationship-patterns.md +0 -202
  506. package/references/meeting/live-join-interface.md +0 -244
  507. package/references/meeting/section-mapping.md +0 -192
  508. package/references/meeting/segment-classification.md +0 -258
  509. package/references/meeting/speaker-profile-template.md +0 -219
  510. package/references/meeting/summary-template.md +0 -348
  511. package/references/meeting/transcript-patterns.md +0 -226
  512. package/references/methodology/analyze-needs.md +0 -135
  513. package/references/methodology/analyze-systems.md +0 -121
  514. package/references/methodology/analyze-timing.md +0 -149
  515. package/references/methodology/beautiful-question.md +0 -109
  516. package/references/methodology/build-knowledge.md +0 -161
  517. package/references/methodology/build-thesis.md +0 -237
  518. package/references/methodology/challenge-assumptions.md +0 -127
  519. package/references/methodology/diagnose.md +0 -169
  520. package/references/methodology/dominant-designs.md +0 -212
  521. package/references/methodology/explore-domains.md +0 -147
  522. package/references/methodology/explore-futures.md +0 -163
  523. package/references/methodology/explore-trends.md +0 -129
  524. package/references/methodology/find-bottlenecks.md +0 -131
  525. package/references/methodology/grade.md +0 -211
  526. package/references/methodology/index.md +0 -97
  527. package/references/methodology/leadership.md +0 -200
  528. package/references/methodology/lean-canvas.md +0 -116
  529. package/references/methodology/macro-trends.md +0 -192
  530. package/references/methodology/map-unknowns.md +0 -137
  531. package/references/methodology/mullins-7-domains.md +0 -104
  532. package/references/methodology/problem-types.md +0 -65
  533. package/references/methodology/root-cause.md +0 -178
  534. package/references/methodology/sapphire-encoding.md +0 -355
  535. package/references/methodology/scenario-plan.md +0 -178
  536. package/references/methodology/score-innovation.md +0 -154
  537. package/references/methodology/structure-argument.md +0 -158
  538. package/references/methodology/systems-thinking.md +0 -159
  539. package/references/methodology/think-hats.md +0 -147
  540. package/references/methodology/triz-matrix.json +0 -751
  541. package/references/methodology/triz-principles.md +0 -501
  542. package/references/methodology/user-needs.md +0 -199
  543. package/references/methodology/validate.md +0 -163
  544. package/references/methodology/value-proposition.md +0 -244
  545. package/references/opportunities/funding-lifecycle.md +0 -103
  546. package/references/opportunities/grant-api-patterns.md +0 -99
  547. package/references/opportunities/opportunity-template.md +0 -84
  548. package/references/personality/assessment-philosophy.md +0 -72
  549. package/references/personality/lexicon.md +0 -100
  550. package/references/personality/persona-chains.md +0 -56
  551. package/references/personality/pws-lexicon-full.md +0 -499
  552. package/references/personality/voice-dna.md +0 -156
  553. package/references/personas/hat-perspectives.md +0 -76
  554. package/references/personas/persona-template.md +0 -63
  555. package/references/pipeline/act-output-contract.md +0 -88
  556. package/references/pipeline/chains-index.md +0 -39
  557. package/references/pws-profile-generation.md +0 -79
  558. package/references/reasoning/reasoning-schema.md +0 -143
  559. package/references/reasoning/reasoning-template.md +0 -68
  560. package/references/reasoning/run-template.md +0 -38
  561. package/references/research/RESEARCH_14_CLAUDE_CODE_SOURCE_ARCHITECTURE.md +0 -209
  562. package/references/research/RESEARCH_15_V1.8_OPTIMIZATION_JTBD.md +0 -375
  563. package/references/research/RESEARCH_16_NATIVE_FIRST_PLUGIN_ARCHITECTURE.md +0 -575
  564. package/references/research/RESEARCH_17_MCP_UI_FRAMEWORKS.md +0 -272
  565. package/references/taxonomy/TAXONOMY.md +0 -192
  566. package/references/templates/MINTO.md +0 -36
  567. package/references/user-research/2026-04-05-leah-lawrence-session.md +0 -202
  568. package/references/vault-kit/README.md +0 -35
  569. package/references/vault-kit/app.json +0 -12
  570. package/references/vault-kit/appearance.json +0 -12
  571. package/references/vault-kit/graph.json +0 -35
  572. package/references/vault-kit/snippets/mindrian-destijl.css +0 -297
  573. package/references/vault-kit/templates/new-artifact.md +0 -37
  574. package/references/vault-kit/templates/new-meeting-note.md +0 -35
  575. package/references/vault-kit/templates/new-team-profile.md +0 -29
  576. package/references/vault-kit/templates/new-xref.md +0 -35
  577. package/references/visual/symbol-system.md +0 -151
  578. package/skills/MOSDeckEngine/SKILL.md +0 -325
  579. package/skills/brain-connector/SKILL.md +0 -114
  580. package/skills/context-engine/SKILL.md +0 -147
  581. package/skills/conversation-mode/SKILL.md +0 -102
  582. package/skills/larry-personality/SKILL.md +0 -219
  583. package/skills/larry-personality/framework-chains.md +0 -92
  584. package/skills/larry-personality/mode-engine.md +0 -185
  585. package/skills/mullins-scaffold/SKILL.md +0 -61
  586. package/skills/mullins-scaffold/scaffold.json +0 -146
  587. package/skills/pws-methodology/SKILL.md +0 -49
  588. package/skills/room-passive/SKILL.md +0 -165
  589. package/skills/room-proactive/SKILL.md +0 -250
  590. package/skills/ui-system/SKILL.md +0 -277
@@ -1,848 +0,0 @@
1
- #!/usr/bin/env node
2
- 'use strict';
3
-
4
- /*
5
- * Copyright (c) 2026 Mindrian. BSL 1.1.
6
- * Phase 89.2 Plan 02 -- academic fetcher fixture suite.
7
- *
8
- * 18 scenarios total: 12 fetcher tests + 6 validator tests.
9
- *
10
- * Test 1 happy path 6 sources (mock fetch returns paper batches per source)
11
- * Test 2 dedup determinism (same input twice -> identical output)
12
- * Test 3 API-key-missing graceful (scopus + ieee + nature env unset)
13
- * Test 4 429 rate-limit graceful (openalex returns 429; fetcher continues)
14
- * Test 5 timeout graceful (AbortError; fetcher continues with remaining sources)
15
- * Test 6 malformed response graceful (arXiv returns non-XML; api_error logged)
16
- * Test 7 per-source budget exhausted (seed 100 entries; openalex skipped)
17
- * Test 8 CANON PART 8 adversarial: leaked-artifact-body in query throws
18
- * Test 9 CANON PART 8 adversarial: leaked-venture-name in query throws
19
- * Test 10 CANON PART 8 adversarial: leaked-meeting-fragment in query throws
20
- * Test 11 CANON PART 8 adversarial: leaked-financial-figure in query throws
21
- * Test 12 chokepoint exclusivity: every fetch() call site is inside a
22
- * per-source dispatcher invoked by fetchAcademic via
23
- * buildAcademicQuery; static grep enforces this.
24
- *
25
- * V1 validator Check A: telemetry file absent -> {severity: null}
26
- * V2 validator Check B: per-source budget exceeded -> warning
27
- * V3 validator Check C: status enum violation -> warning
28
- * V4 validator Check D: query_text literal present -> CRITICAL canon_boundary
29
- * V5 validator Check E: query_text_hash format violation -> warning
30
- * V6 validator Check F: fetched_at malformed ISO-8601 -> warning
31
- *
32
- * Suite-end audits:
33
- * A1 every captured outbound URL + every captured telemetry record
34
- * JSON.stringify scanned against FORBIDDEN_PATTERNS; ZERO hits
35
- * A2 parity gate: rs-egress-prompts FORBIDDEN_PATTERNS byte-for-byte
36
- * === cross-room-aggregator FORBIDDEN_PATTERNS at every regex.source
37
- *
38
- * Pure CJS, zero npm deps, node built-ins only (assert, fs, os, path, crypto).
39
- */
40
-
41
- const assert = require('node:assert/strict');
42
- const fs = require('node:fs');
43
- const os = require('node:os');
44
- const path = require('node:path');
45
- const crypto = require('node:crypto');
46
-
47
- // ---------- Suite bookkeeping ----------
48
-
49
- let passed = 0;
50
- let failed = 0;
51
- const failures = [];
52
-
53
- // Captured outputs for end-of-suite A1 sweep.
54
- const SCENARIO_RESULTS = [];
55
-
56
- // Captured outbound URLs from mock fetch (for chokepoint + A1 audit).
57
- // Two ledgers: per-scenario (cleared on each installMockFetch) + cumulative
58
- // (never cleared; A1 sweep scans this).
59
- const CAPTURED_URLS = [];
60
- const CAPTURED_URLS_ALL = [];
61
-
62
- let testHomeDir = null;
63
- let originalHome = null;
64
-
65
- const ALL_TMP_ROOTS = [];
66
-
67
- process.on('exit', function () {
68
- if (originalHome !== null) {
69
- process.env.HOME = originalHome;
70
- }
71
- for (const d of ALL_TMP_ROOTS) {
72
- try { fs.rmSync(d, { recursive: true, force: true }); } catch (_e) { /* best effort */ }
73
- }
74
- });
75
-
76
- function setupScopedHome() {
77
- if (originalHome === null) {
78
- originalHome = process.env.HOME;
79
- }
80
- testHomeDir = fs.mkdtempSync(path.join(os.tmpdir(), 'rsfetacad-home-'));
81
- ALL_TMP_ROOTS.push(testHomeDir);
82
- process.env.HOME = testHomeDir;
83
- resetRequireCache();
84
- }
85
-
86
- function resetRequireCache() {
87
- const targets = [
88
- '../core/rs-egress-violations.cjs',
89
- '../core/rs-egress-prompts.cjs',
90
- '../core/rs-egress-telemetry.cjs',
91
- '../core/rs-fetcher-academic.cjs',
92
- '../core/cross-room-aggregator.cjs',
93
- './validators/external-academic-invariants.cjs',
94
- ];
95
- for (const t of targets) {
96
- try {
97
- const p = require.resolve(t);
98
- delete require.cache[p];
99
- } catch (_e) { /* best effort */ }
100
- }
101
- }
102
-
103
- // ---------- Mock fetch ----------
104
- //
105
- // Map<urlPrefix, handler(url, opts) -> {ok, status, headers?, json?, text?}>
106
- // Wildcard: any prefix matches as substring (.indexOf >= 0). First match wins.
107
-
108
- let originalFetch = null;
109
- const mockResponses = new Map();
110
-
111
- function installMockFetch(responses) {
112
- if (originalFetch === null) {
113
- originalFetch = global.fetch;
114
- }
115
- mockResponses.clear();
116
- // Clear captured URLs per install so per-scenario assertions see only
117
- // this scenario's traffic. Cross-scenario aggregation is preserved by
118
- // scenarios that explicitly read CAPTURED_URLS before the next install.
119
- CAPTURED_URLS.length = 0;
120
- for (const [k, v] of Object.entries(responses)) {
121
- mockResponses.set(k, v);
122
- }
123
- global.fetch = async function (url, opts) {
124
- CAPTURED_URLS.push(String(url));
125
- CAPTURED_URLS_ALL.push(String(url));
126
- for (const [prefix, handler] of mockResponses.entries()) {
127
- if (String(url).indexOf(prefix) >= 0) {
128
- return handler(url, opts);
129
- }
130
- }
131
- return { ok: false, status: 404, headers: new Map(), async json() { return {}; }, async text() { return ''; } };
132
- };
133
- }
134
-
135
- function restoreFetch() {
136
- if (originalFetch !== null) {
137
- global.fetch = originalFetch;
138
- }
139
- mockResponses.clear();
140
- }
141
-
142
- // ---------- Scenario runner ----------
143
-
144
- async function runScenario(name, fn) {
145
- const start = Date.now();
146
- try {
147
- await fn();
148
- passed += 1;
149
- process.stdout.write(' ok ' + name + ' (' + (Date.now() - start) + 'ms)\n');
150
- } catch (err) {
151
- failed += 1;
152
- failures.push({ name: name, err: err });
153
- process.stderr.write(' FAIL ' + name + '\n');
154
- process.stderr.write(' ' + (err && err.stack ? err.stack : err) + '\n');
155
- } finally {
156
- restoreFetch();
157
- resetRequireCache();
158
- }
159
- }
160
-
161
- // ---------- Helpers ----------
162
-
163
- function clearApiKeys() {
164
- delete process.env.SCOPUS_API_KEY;
165
- delete process.env.IEEE_API_KEY;
166
- delete process.env.NATURE_API_KEY;
167
- }
168
-
169
- function setApiKeys() {
170
- process.env.SCOPUS_API_KEY = 'test-scopus-key';
171
- process.env.IEEE_API_KEY = 'test-ieee-key';
172
- process.env.NATURE_API_KEY = 'test-nature-key';
173
- }
174
-
175
- function makeOpenAlexResponse(query, n) {
176
- // Simulated OpenAlex shape with abstract_inverted_index.
177
- const works = [];
178
- for (let i = 0; i < n; i++) {
179
- works.push({
180
- id: 'https://openalex.org/W' + i + 'q' + crypto.createHash('sha256').update(query).digest('hex').slice(0, 6),
181
- title: 'OpenAlex paper ' + i + ' for ' + query,
182
- abstract_inverted_index: { 'cancer': [0], 'biomarker': [1], 'study': [2] },
183
- publication_year: 2024,
184
- authorships: [{ author: { display_name: 'Author A' + i }, institutions: [{ display_name: 'Inst-' + i }] }],
185
- doi: 'https://doi.org/10.1000/oa-' + i + '-' + crypto.createHash('sha256').update(query).digest('hex').slice(0, 6),
186
- });
187
- }
188
- return { results: works };
189
- }
190
-
191
- function makeArxivXml(query, n) {
192
- let entries = '';
193
- for (let i = 0; i < n; i++) {
194
- const tag = crypto.createHash('sha256').update(query).digest('hex').slice(0, 6);
195
- entries += '<entry><id>http://arxiv.org/abs/2401.000' + i + tag + '</id>'
196
- + '<title>arXiv paper ' + i + ' for ' + query + '</title>'
197
- + '<summary>Abstract for arxiv paper ' + i + '.</summary>'
198
- + '<published>2024-01-' + ((i % 28) + 1).toString().padStart(2, '0') + 'T00:00:00Z</published>'
199
- + '<author><name>Arxiv Author ' + i + '</name></author>'
200
- + '</entry>';
201
- }
202
- return '<?xml version="1.0" encoding="UTF-8"?>'
203
- + '<feed xmlns="http://www.w3.org/2005/Atom">'
204
- + entries
205
- + '</feed>';
206
- }
207
-
208
- function makePubMedResponse(query, n) {
209
- const idlist = [];
210
- for (let i = 0; i < n; i++) {
211
- idlist.push('PM' + i + crypto.createHash('sha256').update(query).digest('hex').slice(0, 6));
212
- }
213
- return { esearchresult: { idlist: idlist, count: String(n) } };
214
- }
215
-
216
- function makeScopusResponse(query, n) {
217
- const entries = [];
218
- for (let i = 0; i < n; i++) {
219
- entries.push({
220
- 'dc:identifier': 'SCOPUS_ID:' + i + crypto.createHash('sha256').update(query).digest('hex').slice(0, 6),
221
- 'dc:title': 'Scopus paper ' + i + ' for ' + query,
222
- 'dc:description': 'Scopus abstract ' + i,
223
- 'dc:creator': 'Scopus Author ' + i,
224
- 'prism:doi': '10.1234/scopus-' + i,
225
- 'affiliation': [{ 'affilname': 'Scopus Inst ' + i }],
226
- });
227
- }
228
- return { 'search-results': { entry: entries } };
229
- }
230
-
231
- function makeIeeeResponse(query, n) {
232
- const articles = [];
233
- for (let i = 0; i < n; i++) {
234
- articles.push({
235
- article_number: 'IEEE' + i + crypto.createHash('sha256').update(query).digest('hex').slice(0, 6),
236
- title: 'IEEE paper ' + i + ' for ' + query,
237
- abstract: 'IEEE abstract ' + i,
238
- authors: { authors: [{ full_name: 'IEEE Author ' + i, affiliation: 'IEEE Inst ' + i }] },
239
- doi: '10.1109/ieee-' + i,
240
- });
241
- }
242
- return { articles: articles, total_records: n };
243
- }
244
-
245
- function makeNatureResponse(query, n) {
246
- const records = [];
247
- for (let i = 0; i < n; i++) {
248
- records.push({
249
- identifier: 'doi:10.1038/nat-' + i + crypto.createHash('sha256').update(query).digest('hex').slice(0, 6),
250
- title: 'Nature paper ' + i + ' for ' + query,
251
- abstract: 'Nature abstract ' + i,
252
- creators: [{ creator: 'Nature Author ' + i }],
253
- doi: '10.1038/nat-' + i,
254
- });
255
- }
256
- return { records: records };
257
- }
258
-
259
- function buildAllSourcesMockOk(query, perSource) {
260
- return {
261
- 'https://api.openalex.org/works': async function (url) {
262
- return {
263
- ok: true,
264
- status: 200,
265
- headers: new Map([['x-ratelimit-remaining', '99']]),
266
- async json() { return makeOpenAlexResponse(query, perSource); },
267
- async text() { return JSON.stringify(makeOpenAlexResponse(query, perSource)); },
268
- };
269
- },
270
- 'export.arxiv.org/api/query': async function (url) {
271
- return {
272
- ok: true,
273
- status: 200,
274
- headers: new Map(),
275
- async text() { return makeArxivXml(query, perSource); },
276
- async json() { throw new Error('arxiv returns XML'); },
277
- };
278
- },
279
- 'eutils.ncbi.nlm.nih.gov': async function (url) {
280
- return {
281
- ok: true,
282
- status: 200,
283
- headers: new Map(),
284
- async json() { return makePubMedResponse(query, perSource); },
285
- async text() { return JSON.stringify(makePubMedResponse(query, perSource)); },
286
- };
287
- },
288
- 'api.elsevier.com/content/search/scopus': async function (url) {
289
- return {
290
- ok: true,
291
- status: 200,
292
- headers: new Map(),
293
- async json() { return makeScopusResponse(query, perSource); },
294
- async text() { return JSON.stringify(makeScopusResponse(query, perSource)); },
295
- };
296
- },
297
- 'ieeexploreapi.ieee.org': async function (url) {
298
- return {
299
- ok: true,
300
- status: 200,
301
- headers: new Map(),
302
- async json() { return makeIeeeResponse(query, perSource); },
303
- async text() { return JSON.stringify(makeIeeeResponse(query, perSource)); },
304
- };
305
- },
306
- 'api.springernature.com': async function (url) {
307
- return {
308
- ok: true,
309
- status: 200,
310
- headers: new Map(),
311
- async json() { return makeNatureResponse(query, perSource); },
312
- async text() { return JSON.stringify(makeNatureResponse(query, perSource)); },
313
- };
314
- },
315
- };
316
- }
317
-
318
- // ---------- A1/A2 sweep helpers ----------
319
-
320
- function getForbiddenPatternsFromAggregator() {
321
- const aggregator = require('../core/cross-room-aggregator.cjs');
322
- return aggregator.FORBIDDEN_PATTERNS;
323
- }
324
-
325
- function scanAgainstForbidden(stringified) {
326
- const patterns = getForbiddenPatternsFromAggregator();
327
- for (const re of patterns) {
328
- if (re.test(stringified)) {
329
- return { hit: true, pattern: re.source };
330
- }
331
- }
332
- return { hit: false };
333
- }
334
-
335
- // ---------- Begin scenarios ----------
336
-
337
- console.log('=== 89.2-02 fetcher-academic suite: starting ===');
338
-
339
- (async function main() {
340
-
341
- // ---------- Test 1: happy path 6 sources ----------
342
- await runScenario('Test 1: happy path 6 sources (with API keys set)', async function () {
343
- setupScopedHome();
344
- setApiKeys();
345
- const queries = ['cancer immunotherapy biomarkers'];
346
- installMockFetch(buildAllSourcesMockOk(queries[0], 5));
347
- const fetcher = require('../core/rs-fetcher-academic.cjs');
348
- const out = await fetcher.fetchAcademic(queries, {});
349
- assert.ok(out && Array.isArray(out.papers), 'papers is array');
350
- assert.ok(out.papers.length >= 20,
351
- 'expected >= 20 papers across 6 sources after dedup; got ' + out.papers.length);
352
- const sample = out.papers[0];
353
- for (const f of ['id', 'title', 'abstract', 'authors', 'institution', 'doi', 'source', 'fetched_at']) {
354
- assert.ok(Object.prototype.hasOwnProperty.call(sample, f),
355
- 'paper missing field: ' + f);
356
- }
357
- assert.ok(Array.isArray(sample.authors), 'authors is array');
358
- const sources = new Set(out.papers.map(p => p.source));
359
- for (const s of ['openalex', 'arxiv', 'pubmed', 'scopus', 'ieee', 'nature']) {
360
- assert.ok(sources.has(s), 'source coverage missing: ' + s);
361
- }
362
- SCENARIO_RESULTS.push({ test: 'T1', surface: 'academic', payload: out });
363
- });
364
-
365
- // ---------- Test 2: dedup determinism ----------
366
- await runScenario('Test 2: dedup determinism (same input twice == identical output)', async function () {
367
- setupScopedHome();
368
- setApiKeys();
369
- const queries = ['CRISPR off-target detection'];
370
- installMockFetch(buildAllSourcesMockOk(queries[0], 5));
371
- const fetcher = require('../core/rs-fetcher-academic.cjs');
372
- const r1 = await fetcher.fetchAcademic(queries, {});
373
- // Reset captured URLs and HOME to avoid budget contamination.
374
- CAPTURED_URLS.length = 0;
375
- setupScopedHome();
376
- installMockFetch(buildAllSourcesMockOk(queries[0], 5));
377
- const fetcher2 = require('../core/rs-fetcher-academic.cjs');
378
- const r2 = await fetcher2.fetchAcademic(queries, {});
379
- // Strip fetched_at because timestamps will diff. Compare structural shape.
380
- const stripT = function (papers) {
381
- return papers.map(p => {
382
- const c = Object.assign({}, p);
383
- delete c.fetched_at;
384
- return c;
385
- });
386
- };
387
- assert.equal(JSON.stringify(stripT(r1.papers)), JSON.stringify(stripT(r2.papers)),
388
- 'dedup output must be deterministic across runs');
389
- SCENARIO_RESULTS.push({ test: 'T2', surface: 'academic', payload: stripT(r1.papers) });
390
- });
391
-
392
- // ---------- Test 3: API-key-missing graceful ----------
393
- await runScenario('Test 3: API-key-missing graceful (scopus + ieee + nature unset)', async function () {
394
- setupScopedHome();
395
- clearApiKeys();
396
- const queries = ['quantum brain imaging'];
397
- installMockFetch(buildAllSourcesMockOk(queries[0], 3));
398
- const fetcher = require('../core/rs-fetcher-academic.cjs');
399
- const out = await fetcher.fetchAcademic(queries, {});
400
- assert.ok(Array.isArray(out.papers), 'papers is array');
401
- // Only openalex + arxiv + pubmed should populate papers.
402
- const sources = new Set(out.papers.map(p => p.source));
403
- assert.ok(sources.has('openalex'), 'openalex present');
404
- assert.ok(sources.has('arxiv'), 'arxiv present');
405
- assert.ok(sources.has('pubmed'), 'pubmed present');
406
- assert.ok(!sources.has('scopus'), 'scopus must be skipped (no API key)');
407
- assert.ok(!sources.has('ieee'), 'ieee must be skipped (no API key)');
408
- assert.ok(!sources.has('nature'), 'nature must be skipped (no API key)');
409
-
410
- // Check telemetry has 3 api_key_missing entries.
411
- const telemetry = require('../core/rs-egress-telemetry.cjs');
412
- const file = telemetry.TELEMETRY_FILE;
413
- assert.ok(fs.existsSync(file), 'telemetry exists at ' + file);
414
- const payload = JSON.parse(fs.readFileSync(file, 'utf8'));
415
- const keyMissing = payload.entries.filter(e => e.status === 'api_key_missing');
416
- assert.equal(keyMissing.length, 3,
417
- 'expected 3 api_key_missing entries; got ' + keyMissing.length);
418
- SCENARIO_RESULTS.push({ test: 'T3', surface: 'academic', payload: out });
419
- });
420
-
421
- // ---------- Test 4: 429 rate-limit graceful ----------
422
- await runScenario('Test 4: 429 rate-limit graceful (openalex returns 429)', async function () {
423
- setupScopedHome();
424
- clearApiKeys();
425
- const queries = ['graphene heat conductivity'];
426
- const mocks = buildAllSourcesMockOk(queries[0], 4);
427
- mocks['https://api.openalex.org/works'] = async function () {
428
- return {
429
- ok: false,
430
- status: 429,
431
- headers: new Map([['retry-after', '60']]),
432
- async json() { return { error: 'rate limited' }; },
433
- async text() { return 'rate limited'; },
434
- };
435
- };
436
- installMockFetch(mocks);
437
- const fetcher = require('../core/rs-fetcher-academic.cjs');
438
- const out = await fetcher.fetchAcademic(queries, {});
439
- // Should not throw. Should still have papers from arxiv + pubmed.
440
- const sources = new Set(out.papers.map(p => p.source));
441
- assert.ok(!sources.has('openalex'), 'openalex must be empty after 429');
442
- assert.ok(sources.has('arxiv'), 'arxiv populates');
443
- assert.ok(sources.has('pubmed'), 'pubmed populates');
444
-
445
- const telemetry = require('../core/rs-egress-telemetry.cjs');
446
- const payload = JSON.parse(fs.readFileSync(telemetry.TELEMETRY_FILE, 'utf8'));
447
- const rateLimited = payload.entries.filter(e => e.status === 'rate_limited');
448
- assert.ok(rateLimited.length >= 1, 'rate_limited telemetry recorded');
449
- assert.equal(rateLimited[0].http_status, 429, 'http_status 429 recorded');
450
- SCENARIO_RESULTS.push({ test: 'T4', surface: 'academic', payload: out });
451
- });
452
-
453
- // ---------- Test 5: timeout graceful ----------
454
- await runScenario('Test 5: timeout graceful (openalex throws AbortError)', async function () {
455
- setupScopedHome();
456
- clearApiKeys();
457
- const queries = ['CAR-T cell therapy'];
458
- const mocks = buildAllSourcesMockOk(queries[0], 3);
459
- mocks['https://api.openalex.org/works'] = async function () {
460
- const err = new Error('aborted');
461
- err.name = 'AbortError';
462
- throw err;
463
- };
464
- installMockFetch(mocks);
465
- const fetcher = require('../core/rs-fetcher-academic.cjs');
466
- const out = await fetcher.fetchAcademic(queries, {});
467
- // Should not throw. Should still have papers from arxiv + pubmed.
468
- const sources = new Set(out.papers.map(p => p.source));
469
- assert.ok(!sources.has('openalex'), 'openalex empty after timeout');
470
- assert.ok(sources.has('arxiv'), 'arxiv populates');
471
-
472
- const telemetry = require('../core/rs-egress-telemetry.cjs');
473
- const payload = JSON.parse(fs.readFileSync(telemetry.TELEMETRY_FILE, 'utf8'));
474
- const timed = payload.entries.filter(e => e.status === 'timeout');
475
- assert.ok(timed.length >= 1, 'timeout telemetry recorded');
476
- SCENARIO_RESULTS.push({ test: 'T5', surface: 'academic', payload: out });
477
- });
478
-
479
- // ---------- Test 6: malformed response graceful ----------
480
- await runScenario('Test 6: malformed response graceful (arXiv returns non-XML)', async function () {
481
- setupScopedHome();
482
- clearApiKeys();
483
- const queries = ['mRNA vaccine technology'];
484
- const mocks = buildAllSourcesMockOk(queries[0], 3);
485
- mocks['export.arxiv.org/api/query'] = async function () {
486
- return {
487
- ok: true,
488
- status: 200,
489
- headers: new Map(),
490
- async text() { return '<<<not-xml>>>'; },
491
- async json() { throw new Error('not json'); },
492
- };
493
- };
494
- installMockFetch(mocks);
495
- const fetcher = require('../core/rs-fetcher-academic.cjs');
496
- const out = await fetcher.fetchAcademic(queries, {});
497
- const sources = new Set(out.papers.map(p => p.source));
498
- assert.ok(!sources.has('arxiv'), 'arxiv empty after parse failure');
499
- assert.ok(sources.has('openalex'), 'openalex still populates');
500
-
501
- const telemetry = require('../core/rs-egress-telemetry.cjs');
502
- const payload = JSON.parse(fs.readFileSync(telemetry.TELEMETRY_FILE, 'utf8'));
503
- const errd = payload.entries.filter(e => e.status === 'api_error');
504
- assert.ok(errd.length >= 1, 'api_error telemetry recorded for arxiv');
505
- SCENARIO_RESULTS.push({ test: 'T6', surface: 'academic', payload: out });
506
- });
507
-
508
- // ---------- Test 7: per-source budget exhausted ----------
509
- await runScenario('Test 7: per-source budget exhausted (openalex skipped)', async function () {
510
- setupScopedHome();
511
- clearApiKeys();
512
- const telemetry = require('../core/rs-egress-telemetry.cjs');
513
- // Seed 100 openalex entries within last 24h.
514
- const now = Date.now();
515
- const entries = [];
516
- for (let i = 0; i < 100; i++) {
517
- entries.push({
518
- source: 'openalex',
519
- query_text_hash: 'aaaaaaaaaaaaaaaa',
520
- status: 'ok',
521
- fetched_at: new Date(now - (i * 1000)).toISOString(),
522
- });
523
- }
524
- fs.mkdirSync(path.dirname(telemetry.TELEMETRY_FILE), { recursive: true });
525
- fs.writeFileSync(telemetry.TELEMETRY_FILE,
526
- JSON.stringify({ schema_version: '1.0', entries: entries }, null, 2));
527
-
528
- const queries = ['plasma confinement tokamak'];
529
- installMockFetch(buildAllSourcesMockOk(queries[0], 3));
530
- const fetcher = require('../core/rs-fetcher-academic.cjs');
531
- const out = await fetcher.fetchAcademic(queries, {});
532
- const sources = new Set(out.papers.map(p => p.source));
533
- assert.ok(!sources.has('openalex'), 'openalex skipped after budget exhausted');
534
- assert.ok(sources.has('arxiv'), 'arxiv runs normally');
535
-
536
- const payload = JSON.parse(fs.readFileSync(telemetry.TELEMETRY_FILE, 'utf8'));
537
- // Note: 'budget_exhausted' is in our extended status enum below; if telemetry
538
- // module doesn't accept it, fetcher should still skip openalex without error.
539
- // We assert the behavior (skip), not the telemetry status name strictly here.
540
- // OpenAlex URL must NOT appear in CAPTURED_URLS for this run.
541
- const ourUrls = CAPTURED_URLS.filter(u => u.indexOf('openalex.org/works') >= 0);
542
- assert.equal(ourUrls.length, 0, 'no openalex URL fetched after budget exhaust');
543
- SCENARIO_RESULTS.push({ test: 'T7', surface: 'academic', payload: out });
544
- });
545
-
546
- // ---------- Test 8: CANON PART 8 adversarial: leaked-artifact-body ----------
547
- await runScenario('Test 8: CANON PART 8 adversarial leaked-artifact-body throws', async function () {
548
- setupScopedHome();
549
- clearApiKeys();
550
- // Pattern 3 (meeting with) is the only short way to leak an artifact body in 1 line.
551
- const queries = ['cancer biomarkers <<artifact: meeting with Dr Smith Q4 financials>>'];
552
- installMockFetch(buildAllSourcesMockOk('clean', 1));
553
- const fetcher = require('../core/rs-fetcher-academic.cjs');
554
- let threw = null;
555
- try {
556
- await fetcher.fetchAcademic(queries, {});
557
- } catch (e) {
558
- threw = e;
559
- }
560
- assert.ok(threw, 'expected throw on adversarial query');
561
- assert.equal(threw.name, 'ExternalEgressViolation', 'class name');
562
- assert.equal(threw.meta.surface, 'academic', 'meta.surface');
563
- assert.ok(typeof threw.meta.matched_pattern === 'string'
564
- && threw.meta.matched_pattern.length > 0, 'meta.matched_pattern present');
565
-
566
- // ZERO outbound URL captures for this query.
567
- assert.equal(CAPTURED_URLS.length, 0,
568
- 'NO fetch() calls allowed before throw; got ' + CAPTURED_URLS.length);
569
-
570
- // ZERO telemetry entries for this query.
571
- const telemetry = require('../core/rs-egress-telemetry.cjs');
572
- if (fs.existsSync(telemetry.TELEMETRY_FILE)) {
573
- const payload = JSON.parse(fs.readFileSync(telemetry.TELEMETRY_FILE, 'utf8'));
574
- assert.equal(payload.entries.length, 0,
575
- 'NO telemetry entries allowed; got ' + payload.entries.length);
576
- }
577
- SCENARIO_RESULTS.push({ test: 'T8', surface: 'academic', payload: { threw: threw.meta.matched_pattern } });
578
- });
579
-
580
- // ---------- Test 9: CANON PART 8 adversarial: leaked-venture-name (currency) ----------
581
- await runScenario('Test 9: CANON PART 8 adversarial leaked-venture-financials throws', async function () {
582
- setupScopedHome();
583
- clearApiKeys();
584
- // Pattern 1 (currency $5.2M) leak.
585
- const queries = ['oncology venture valuation $5.2M cancer treatment'];
586
- installMockFetch(buildAllSourcesMockOk('clean', 1));
587
- const fetcher = require('../core/rs-fetcher-academic.cjs');
588
- let threw = null;
589
- try {
590
- await fetcher.fetchAcademic(queries, {});
591
- } catch (e) {
592
- threw = e;
593
- }
594
- assert.ok(threw, 'expected throw on adversarial query');
595
- assert.equal(threw.name, 'ExternalEgressViolation', 'class name');
596
- assert.equal(CAPTURED_URLS.length, 0, 'NO fetch() calls allowed');
597
- SCENARIO_RESULTS.push({ test: 'T9', surface: 'academic', payload: { threw: threw.meta.matched_pattern } });
598
- });
599
-
600
- // ---------- Test 10: CANON PART 8 adversarial: leaked-meeting-fragment ----------
601
- await runScenario('Test 10: CANON PART 8 adversarial leaked-meeting-fragment throws', async function () {
602
- setupScopedHome();
603
- clearApiKeys();
604
- const queries = ['oncology meeting with Mayo Clinic Q3'];
605
- installMockFetch(buildAllSourcesMockOk('clean', 1));
606
- const fetcher = require('../core/rs-fetcher-academic.cjs');
607
- let threw = null;
608
- try {
609
- await fetcher.fetchAcademic(queries, {});
610
- } catch (e) {
611
- threw = e;
612
- }
613
- assert.ok(threw, 'expected throw on adversarial query');
614
- assert.equal(threw.name, 'ExternalEgressViolation', 'class name');
615
- assert.equal(CAPTURED_URLS.length, 0, 'NO fetch() calls allowed');
616
- SCENARIO_RESULTS.push({ test: 'T10', surface: 'academic', payload: { threw: threw.meta.matched_pattern } });
617
- });
618
-
619
- // ---------- Test 11: CANON PART 8 adversarial: leaked-financial-figure (currency variant) ----------
620
- await runScenario('Test 11: CANON PART 8 adversarial leaked-financial-figure throws', async function () {
621
- setupScopedHome();
622
- clearApiKeys();
623
- const queries = ['oncology partnerships under $750K threshold'];
624
- installMockFetch(buildAllSourcesMockOk('clean', 1));
625
- const fetcher = require('../core/rs-fetcher-academic.cjs');
626
- let threw = null;
627
- try {
628
- await fetcher.fetchAcademic(queries, {});
629
- } catch (e) {
630
- threw = e;
631
- }
632
- assert.ok(threw, 'expected throw on adversarial query');
633
- assert.equal(threw.name, 'ExternalEgressViolation', 'class name');
634
- assert.equal(CAPTURED_URLS.length, 0, 'NO fetch() calls allowed');
635
- SCENARIO_RESULTS.push({ test: 'T11', surface: 'academic', payload: { threw: threw.meta.matched_pattern } });
636
- });
637
-
638
- // ---------- Test 12: chokepoint exclusivity (static grep) ----------
639
- await runScenario('Test 12: chokepoint exclusivity (every fetch() inside per-source dispatcher)', async function () {
640
- const fetcherPath = path.resolve(__dirname, '..', 'core', 'rs-fetcher-academic.cjs');
641
- const src = fs.readFileSync(fetcherPath, 'utf8');
642
- // Find all fetch( occurrences (excluding comments).
643
- const lines = src.split('\n');
644
- let fetchCount = 0;
645
- let inBlockComment = false;
646
- for (let i = 0; i < lines.length; i++) {
647
- let line = lines[i];
648
- if (inBlockComment) {
649
- if (line.indexOf('*/') >= 0) inBlockComment = false;
650
- continue;
651
- }
652
- // Strip trailing line comments.
653
- const slashIdx = line.indexOf('//');
654
- if (slashIdx >= 0) line = line.slice(0, slashIdx);
655
- if (line.indexOf('/*') >= 0 && line.indexOf('*/') < 0) {
656
- inBlockComment = true;
657
- continue;
658
- }
659
- // Match 'fetch(' or 'fetchWithTimeout(' or 'global.fetch(' but not function names.
660
- // Production code MUST go through fetchWithTimeout helper.
661
- const matches = line.match(/\bfetch\s*\(/g);
662
- if (matches) {
663
- // Allow only if it's INSIDE the fetchWithTimeout helper (which is the
664
- // ONE function that calls native fetch).
665
- fetchCount += matches.length;
666
- }
667
- }
668
- // We expect exactly 1 native fetch call: inside fetchWithTimeout.
669
- assert.ok(fetchCount === 1,
670
- 'expected exactly 1 native fetch() call (inside fetchWithTimeout helper); got ' + fetchCount);
671
-
672
- // Also verify auditQueryString call count >= 1 (chokepoint enforcement).
673
- const auditCount = (src.match(/auditQueryString/g) || []).length;
674
- assert.ok(auditCount >= 1, 'auditQueryString must be invoked at least once; got ' + auditCount);
675
-
676
- // Verify buildAcademicQuery is defined exactly once.
677
- const buildDefCount = (src.match(/function buildAcademicQuery/g) || []).length;
678
- assert.equal(buildDefCount, 1, 'buildAcademicQuery defined exactly once');
679
- });
680
-
681
- // ---------- V1: validator Check A telemetry-file-absent ----------
682
- await runScenario('V1: validator Check A telemetry-file-absent -> {severity: null}', async function () {
683
- setupScopedHome();
684
- const validator = require('./validators/external-academic-invariants.cjs');
685
- // Telemetry file does not exist in this scoped HOME.
686
- const result = validator.validate('/dev/null', {});
687
- assert.equal(result.severity, null, 'severity null when telemetry absent');
688
- assert.equal(result.violations.length, 0, 'no violations when telemetry absent');
689
- });
690
-
691
- // ---------- V2: validator Check B per-source budget exceeded ----------
692
- await runScenario('V2: validator Check B per-source budget exceeded -> warning', async function () {
693
- setupScopedHome();
694
- const telemetry = require('../core/rs-egress-telemetry.cjs');
695
- // Seed 150 openalex entries within last 24h (over default 100 budget).
696
- const now = Date.now();
697
- const entries = [];
698
- for (let i = 0; i < 150; i++) {
699
- entries.push({
700
- source: 'openalex',
701
- query_text_hash: 'aaaaaaaaaaaaaaaa',
702
- status: 'ok',
703
- fetched_at: new Date(now - (i * 1000)).toISOString(),
704
- });
705
- }
706
- fs.mkdirSync(path.dirname(telemetry.TELEMETRY_FILE), { recursive: true });
707
- fs.writeFileSync(telemetry.TELEMETRY_FILE,
708
- JSON.stringify({ schema_version: '1.0', entries: entries }, null, 2));
709
-
710
- const validator = require('./validators/external-academic-invariants.cjs');
711
- const result = validator.validate('/dev/null', {});
712
- assert.ok(result.violations.length >= 1, 'expected at least one violation');
713
- const budgetViolations = result.violations.filter(v => v.category === 'budget_exceeded');
714
- assert.ok(budgetViolations.length >= 1, 'expected budget_exceeded violation');
715
- assert.equal(budgetViolations[0].severity, 'warning', 'budget_exceeded severity is warning');
716
- });
717
-
718
- // ---------- V3: validator Check C status enum violation ----------
719
- await runScenario('V3: validator Check C status enum violation -> warning', async function () {
720
- setupScopedHome();
721
- const telemetry = require('../core/rs-egress-telemetry.cjs');
722
- const entries = [{
723
- source: 'openalex',
724
- query_text_hash: 'aaaaaaaaaaaaaaaa',
725
- status: 'unknown_status',
726
- fetched_at: new Date().toISOString(),
727
- }];
728
- fs.mkdirSync(path.dirname(telemetry.TELEMETRY_FILE), { recursive: true });
729
- fs.writeFileSync(telemetry.TELEMETRY_FILE,
730
- JSON.stringify({ schema_version: '1.0', entries: entries }, null, 2));
731
-
732
- const validator = require('./validators/external-academic-invariants.cjs');
733
- const result = validator.validate('/dev/null', {});
734
- const statusViolations = result.violations.filter(v => v.category === 'status_enum_violation');
735
- assert.ok(statusViolations.length >= 1, 'expected status_enum_violation');
736
- assert.equal(statusViolations[0].severity, 'warning', 'status_enum severity is warning');
737
- });
738
-
739
- // ---------- V4: validator Check D query_text literal -> CRITICAL ----------
740
- await runScenario('V4: validator Check D query_text literal present -> CRITICAL canon_boundary', async function () {
741
- setupScopedHome();
742
- const telemetry = require('../core/rs-egress-telemetry.cjs');
743
- const entries = [{
744
- source: 'openalex',
745
- query_text: 'literal cancer biomarkers query', // FORBIDDEN: literal field
746
- query_text_hash: 'aaaaaaaaaaaaaaaa',
747
- status: 'ok',
748
- fetched_at: new Date().toISOString(),
749
- }];
750
- fs.mkdirSync(path.dirname(telemetry.TELEMETRY_FILE), { recursive: true });
751
- fs.writeFileSync(telemetry.TELEMETRY_FILE,
752
- JSON.stringify({ schema_version: '1.0', entries: entries }, null, 2));
753
-
754
- const validator = require('./validators/external-academic-invariants.cjs');
755
- const result = validator.validate('/dev/null', {});
756
- const canonViolations = result.violations.filter(v => v.category === 'canon_boundary');
757
- assert.ok(canonViolations.length >= 1, 'expected canon_boundary violation');
758
- assert.equal(canonViolations[0].severity, 'critical',
759
- 'canon_boundary severity is critical');
760
- });
761
-
762
- // ---------- V5: validator Check E query_text_hash format ----------
763
- await runScenario('V5: validator Check E query_text_hash format violation -> warning', async function () {
764
- setupScopedHome();
765
- const telemetry = require('../core/rs-egress-telemetry.cjs');
766
- const entries = [{
767
- source: 'openalex',
768
- query_text_hash: 'NOT-A-HEX-HASH-XYZ', // wrong format
769
- status: 'ok',
770
- fetched_at: new Date().toISOString(),
771
- }];
772
- fs.mkdirSync(path.dirname(telemetry.TELEMETRY_FILE), { recursive: true });
773
- fs.writeFileSync(telemetry.TELEMETRY_FILE,
774
- JSON.stringify({ schema_version: '1.0', entries: entries }, null, 2));
775
-
776
- const validator = require('./validators/external-academic-invariants.cjs');
777
- const result = validator.validate('/dev/null', {});
778
- const hashViolations = result.violations.filter(v => v.category === 'hash_format_invalid');
779
- assert.ok(hashViolations.length >= 1, 'expected hash_format_invalid violation');
780
- assert.equal(hashViolations[0].severity, 'warning', 'severity is warning');
781
- });
782
-
783
- // ---------- V6: validator Check F fetched_at malformed ISO-8601 ----------
784
- await runScenario('V6: validator Check F fetched_at malformed -> warning', async function () {
785
- setupScopedHome();
786
- const telemetry = require('../core/rs-egress-telemetry.cjs');
787
- const entries = [{
788
- source: 'openalex',
789
- query_text_hash: 'aaaaaaaaaaaaaaaa',
790
- status: 'ok',
791
- fetched_at: 'NOT-A-VALID-DATE',
792
- }];
793
- fs.mkdirSync(path.dirname(telemetry.TELEMETRY_FILE), { recursive: true });
794
- fs.writeFileSync(telemetry.TELEMETRY_FILE,
795
- JSON.stringify({ schema_version: '1.0', entries: entries }, null, 2));
796
-
797
- const validator = require('./validators/external-academic-invariants.cjs');
798
- const result = validator.validate('/dev/null', {});
799
- const dateViolations = result.violations.filter(v => v.category === 'fetched_at_malformed');
800
- assert.ok(dateViolations.length >= 1, 'expected fetched_at_malformed violation');
801
- assert.equal(dateViolations[0].severity, 'warning', 'severity is warning');
802
- });
803
-
804
- // ---------- A1 sweep: zero forbidden matches in any captured output ----------
805
- await runScenario('A1: zero forbidden matches across captured payloads + URLs', async function () {
806
- for (const rec of SCENARIO_RESULTS) {
807
- const stringified = JSON.stringify(rec.payload);
808
- const hit = scanAgainstForbidden(stringified);
809
- assert.equal(hit.hit, false,
810
- 'A1 violation in ' + rec.test + '/' + rec.surface +
811
- ' against pattern ' + (hit.pattern || 'n/a'));
812
- }
813
- // Also scan every captured outbound URL across the ENTIRE suite.
814
- // CAPTURED_URLS is per-scenario; CAPTURED_URLS_ALL is the cumulative
815
- // ledger that captures every URL ever issued during the suite.
816
- for (const u of CAPTURED_URLS_ALL) {
817
- const hit = scanAgainstForbidden(u);
818
- assert.equal(hit.hit, false,
819
- 'A1 outbound URL leaked forbidden pattern ' + (hit.pattern || 'n/a') +
820
- ' in ' + u.slice(0, 100));
821
- }
822
- });
823
-
824
- // ---------- A2 sweep: parity gate ----------
825
- await runScenario('A2: FORBIDDEN_PATTERNS parity (rs-egress-prompts === cross-room-aggregator)', async function () {
826
- const promptsModule = require('../core/rs-egress-prompts.cjs');
827
- const aggregator = require('../core/cross-room-aggregator.cjs');
828
- const ours = promptsModule.FORBIDDEN_PATTERNS;
829
- const truth = aggregator.FORBIDDEN_PATTERNS;
830
- assert.equal(ours.length, truth.length, 'lengths match');
831
- for (let i = 0; i < truth.length; i++) {
832
- assert.equal(ours[i].source, truth[i].source, 'pattern source mismatch at index ' + i);
833
- assert.equal(ours[i].flags, truth[i].flags, 'pattern flags mismatch at index ' + i);
834
- }
835
- });
836
-
837
- // ---------- Final report ----------
838
- if (failed > 0) {
839
- console.error('\n=== ' + failed + ' FAILURES ===');
840
- for (const f of failures) {
841
- console.error(' ' + f.name);
842
- }
843
- process.exit(1);
844
- }
845
-
846
- console.log('=== 89.2-02 fetcher-academic suite: 18/18 passed ===');
847
- process.exit(0);
848
- })();