@ouro.bot/cli 0.1.0-alpha.55 → 0.1.0-alpha.551

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 (386) hide show
  1. package/README.md +133 -19
  2. package/RepairGuide.ouro/agent.json +5 -0
  3. package/RepairGuide.ouro/psyche/IDENTITY.md +19 -0
  4. package/RepairGuide.ouro/psyche/SOUL.md +55 -0
  5. package/RepairGuide.ouro/skills/diagnose-bootstrap-drift.md +54 -0
  6. package/RepairGuide.ouro/skills/diagnose-broken-remote.md +63 -0
  7. package/RepairGuide.ouro/skills/diagnose-stacked-typed-issues.md +35 -0
  8. package/RepairGuide.ouro/skills/diagnose-sync-blocked.md +54 -0
  9. package/RepairGuide.ouro/skills/diagnose-vault-expired.md +60 -0
  10. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
  11. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
  12. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  13. package/changelog.json +3561 -0
  14. package/dist/arc/attention-types.js +8 -0
  15. package/dist/arc/cares.js +140 -0
  16. package/dist/arc/episodes.js +117 -0
  17. package/dist/arc/intentions.js +133 -0
  18. package/dist/arc/json-store.js +117 -0
  19. package/dist/arc/obligations.js +237 -0
  20. package/dist/arc/packets.js +193 -0
  21. package/dist/arc/presence.js +185 -0
  22. package/dist/arc/task-lifecycle.js +65 -0
  23. package/dist/heart/active-work.js +837 -26
  24. package/dist/heart/agent-entry.js +58 -3
  25. package/dist/heart/attachments/image-normalize.js +194 -0
  26. package/dist/heart/attachments/materialize.js +97 -0
  27. package/dist/heart/attachments/originals.js +88 -0
  28. package/dist/heart/attachments/render.js +29 -0
  29. package/dist/heart/attachments/sources/adapter.js +2 -0
  30. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  31. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  32. package/dist/heart/attachments/sources/index.js +16 -0
  33. package/dist/heart/attachments/store.js +103 -0
  34. package/dist/heart/attachments/types.js +93 -0
  35. package/dist/heart/auth/auth-flow.js +479 -0
  36. package/dist/heart/background-operations.js +281 -0
  37. package/dist/heart/bundle-state.js +168 -0
  38. package/dist/heart/commitments.js +111 -0
  39. package/dist/heart/config-registry.js +304 -0
  40. package/dist/heart/config.js +114 -118
  41. package/dist/heart/core.js +925 -246
  42. package/dist/heart/cross-chat-delivery.js +3 -18
  43. package/dist/heart/daemon/agent-config-check.js +512 -0
  44. package/dist/heart/daemon/agent-discovery.js +102 -3
  45. package/dist/heart/daemon/agent-service.js +522 -0
  46. package/dist/heart/daemon/agentic-repair.js +554 -0
  47. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  48. package/dist/heart/daemon/boot-sync-probe.js +197 -0
  49. package/dist/heart/daemon/cadence.js +70 -0
  50. package/dist/heart/daemon/cli-defaults.js +665 -0
  51. package/dist/heart/daemon/cli-exec.js +7565 -0
  52. package/dist/heart/daemon/cli-help.js +498 -0
  53. package/dist/heart/daemon/cli-parse.js +1590 -0
  54. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  55. package/dist/heart/daemon/cli-render.js +775 -0
  56. package/dist/heart/daemon/cli-types.js +8 -0
  57. package/dist/heart/daemon/connect-bay.js +323 -0
  58. package/dist/heart/daemon/daemon-cli.js +29 -1672
  59. package/dist/heart/daemon/daemon-entry.js +417 -2
  60. package/dist/heart/daemon/daemon-health.js +183 -0
  61. package/dist/heart/daemon/daemon-rollup.js +58 -0
  62. package/dist/heart/daemon/daemon-runtime-sync.js +87 -13
  63. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  64. package/dist/heart/daemon/daemon.js +796 -71
  65. package/dist/heart/daemon/dns-workflow.js +394 -0
  66. package/dist/heart/daemon/doctor-types.js +8 -0
  67. package/dist/heart/daemon/doctor.js +844 -0
  68. package/dist/heart/daemon/drift-detection.js +146 -0
  69. package/dist/heart/daemon/health-monitor.js +122 -1
  70. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  71. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  72. package/dist/heart/daemon/http-health-probe.js +80 -0
  73. package/dist/heart/daemon/human-command-screens.js +234 -0
  74. package/dist/heart/daemon/human-readiness.js +114 -0
  75. package/dist/heart/daemon/inner-status.js +102 -0
  76. package/dist/heart/daemon/interactive-repair.js +394 -0
  77. package/dist/heart/daemon/launchd.js +37 -8
  78. package/dist/heart/daemon/log-tailer.js +82 -12
  79. package/dist/heart/daemon/logs-prune.js +110 -0
  80. package/dist/heart/daemon/mcp-canary.js +297 -0
  81. package/dist/heart/daemon/message-router.js +2 -2
  82. package/dist/heart/daemon/os-cron-deps.js +135 -0
  83. package/dist/heart/daemon/os-cron.js +14 -12
  84. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  85. package/dist/heart/daemon/ouro-entry.js +3 -1
  86. package/dist/heart/daemon/process-manager.js +375 -33
  87. package/dist/heart/daemon/provider-discovery.js +137 -0
  88. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  89. package/dist/heart/daemon/pulse.js +475 -0
  90. package/dist/heart/daemon/readiness-repair.js +365 -0
  91. package/dist/heart/daemon/run-hooks.js +2 -0
  92. package/dist/heart/daemon/runtime-logging.js +67 -16
  93. package/dist/heart/daemon/runtime-metadata.js +3 -31
  94. package/dist/heart/daemon/safe-mode.js +161 -0
  95. package/dist/heart/daemon/sense-manager.js +353 -38
  96. package/dist/heart/daemon/session-id-resolver.js +131 -0
  97. package/dist/heart/daemon/skill-management-installer.js +94 -0
  98. package/dist/heart/daemon/socket-client.js +158 -11
  99. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  100. package/dist/heart/daemon/startup-tui.js +330 -0
  101. package/dist/heart/daemon/task-scheduler.js +3 -25
  102. package/dist/heart/daemon/terminal-ui.js +499 -0
  103. package/dist/heart/daemon/thoughts.js +162 -17
  104. package/dist/heart/daemon/up-progress.js +366 -0
  105. package/dist/heart/daemon/vault-items.js +56 -0
  106. package/dist/heart/delegation.js +1 -1
  107. package/dist/heart/habits/habit-migration.js +189 -0
  108. package/dist/heart/habits/habit-parser.js +140 -0
  109. package/dist/heart/habits/habit-runtime-state.js +100 -0
  110. package/dist/heart/habits/habit-scheduler.js +372 -0
  111. package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
  112. package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
  113. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  114. package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
  115. package/dist/heart/identity.js +200 -51
  116. package/dist/heart/kept-notes.js +357 -0
  117. package/dist/heart/kicks.js +1 -1
  118. package/dist/heart/machine-identity.js +161 -0
  119. package/dist/heart/mail-import-discovery.js +353 -0
  120. package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
  121. package/dist/heart/mailbox/mailbox-http-response.js +7 -0
  122. package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
  123. package/dist/heart/mailbox/mailbox-http-static.js +103 -0
  124. package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
  125. package/dist/heart/mailbox/mailbox-http.js +99 -0
  126. package/dist/heart/mailbox/mailbox-read.js +31 -0
  127. package/dist/heart/mailbox/mailbox-types.js +27 -0
  128. package/dist/heart/mailbox/mailbox-view.js +195 -0
  129. package/dist/heart/mailbox/readers/agent-machine.js +382 -0
  130. package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
  131. package/dist/heart/mailbox/readers/mail.js +362 -0
  132. package/dist/heart/mailbox/readers/runtime-readers.js +651 -0
  133. package/dist/heart/mailbox/readers/sessions.js +232 -0
  134. package/dist/heart/mailbox/readers/shared.js +111 -0
  135. package/dist/heart/mcp/mcp-server.js +683 -0
  136. package/dist/heart/migrate-config.js +100 -0
  137. package/dist/heart/model-capabilities.js +19 -0
  138. package/dist/heart/platform.js +81 -0
  139. package/dist/heart/provider-attempt.js +134 -0
  140. package/dist/heart/provider-binding-resolver.js +255 -0
  141. package/dist/heart/provider-credentials.js +425 -0
  142. package/dist/heart/provider-failover.js +301 -0
  143. package/dist/heart/provider-models.js +81 -0
  144. package/dist/heart/provider-ping.js +262 -0
  145. package/dist/heart/provider-state.js +216 -0
  146. package/dist/heart/provider-visibility.js +188 -0
  147. package/dist/heart/providers/anthropic-token.js +131 -0
  148. package/dist/heart/providers/anthropic.js +139 -52
  149. package/dist/heart/providers/azure.js +97 -13
  150. package/dist/heart/providers/error-classification.js +127 -0
  151. package/dist/heart/providers/github-copilot.js +145 -0
  152. package/dist/heart/providers/minimax-vlm.js +189 -0
  153. package/dist/heart/providers/minimax.js +26 -8
  154. package/dist/heart/providers/openai-codex.js +55 -40
  155. package/dist/heart/runtime-capability-check.js +170 -0
  156. package/dist/heart/runtime-credentials.js +367 -0
  157. package/dist/heart/runtime-cwd.js +87 -0
  158. package/dist/heart/sense-truth.js +11 -4
  159. package/dist/heart/session-activity.js +43 -22
  160. package/dist/heart/session-events.js +1149 -0
  161. package/dist/heart/session-playback-cli-main.js +5 -0
  162. package/dist/heart/session-playback-cli.js +36 -0
  163. package/dist/heart/session-playback.js +231 -0
  164. package/dist/heart/session-stats-cli-main.js +5 -0
  165. package/dist/heart/session-stats.js +182 -0
  166. package/dist/heart/session-transcript.js +243 -0
  167. package/dist/heart/start-of-turn-packet.js +345 -0
  168. package/dist/heart/streaming.js +44 -27
  169. package/dist/heart/sync-classification.js +176 -0
  170. package/dist/heart/sync.js +449 -0
  171. package/dist/heart/target-resolution.js +9 -5
  172. package/dist/heart/tempo.js +93 -0
  173. package/dist/heart/temporal-view.js +41 -0
  174. package/dist/heart/timeouts.js +101 -0
  175. package/dist/heart/tool-activity-callbacks.js +59 -0
  176. package/dist/heart/tool-description.js +139 -0
  177. package/dist/heart/tool-friction.js +55 -0
  178. package/dist/heart/tool-loop.js +200 -0
  179. package/dist/heart/turn-context.js +381 -0
  180. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
  181. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  182. package/dist/heart/versioning/ouro-path-installer.js +426 -0
  183. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  184. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  185. package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
  186. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  187. package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
  188. package/dist/mailbox-ui/assets/index-Cm51CY9W.js +61 -0
  189. package/dist/mailbox-ui/index.html +15 -0
  190. package/dist/mailroom/attention.js +167 -0
  191. package/dist/mailroom/autonomy.js +209 -0
  192. package/dist/mailroom/blob-store.js +674 -0
  193. package/dist/mailroom/body-cache.js +61 -0
  194. package/dist/mailroom/core.js +720 -0
  195. package/dist/mailroom/entry.js +160 -0
  196. package/dist/mailroom/file-store.js +430 -0
  197. package/dist/mailroom/mbox-import.js +383 -0
  198. package/dist/mailroom/outbound.js +380 -0
  199. package/dist/mailroom/policy.js +263 -0
  200. package/dist/mailroom/reader.js +233 -0
  201. package/dist/mailroom/search-cache.js +256 -0
  202. package/dist/mailroom/search-relevance.js +319 -0
  203. package/dist/mailroom/smtp-ingress.js +176 -0
  204. package/dist/mailroom/source-state.js +176 -0
  205. package/dist/mailroom/thread.js +109 -0
  206. package/dist/mailroom/travel-extract.js +89 -0
  207. package/dist/mind/bundle-manifest.js +7 -1
  208. package/dist/mind/context.js +165 -101
  209. package/dist/mind/diary-integrity.js +60 -0
  210. package/dist/mind/{memory.js → diary.js} +62 -75
  211. package/dist/mind/embedding-provider.js +60 -0
  212. package/dist/mind/file-state.js +179 -0
  213. package/dist/mind/friends/channel.js +30 -0
  214. package/dist/mind/friends/resolver.js +54 -2
  215. package/dist/mind/friends/store-file.js +39 -3
  216. package/dist/mind/friends/types.js +2 -2
  217. package/dist/mind/journal-index.js +161 -0
  218. package/dist/mind/note-search.js +268 -0
  219. package/dist/mind/obligation-steering.js +221 -0
  220. package/dist/mind/pending.js +4 -0
  221. package/dist/mind/prompt-refresh.js +3 -2
  222. package/dist/mind/prompt.js +995 -123
  223. package/dist/mind/provenance-trust.js +26 -0
  224. package/dist/mind/scrutiny.js +173 -0
  225. package/dist/nerves/cli-logging.js +7 -1
  226. package/dist/nerves/coverage/audit-rules.js +15 -6
  227. package/dist/nerves/coverage/audit.js +28 -2
  228. package/dist/nerves/coverage/cli.js +1 -1
  229. package/dist/nerves/coverage/contract.js +5 -5
  230. package/dist/nerves/coverage/file-completeness.js +139 -5
  231. package/dist/nerves/coverage/run-artifacts.js +1 -1
  232. package/dist/nerves/event-buffer.js +111 -0
  233. package/dist/nerves/index.js +224 -4
  234. package/dist/nerves/observation.js +20 -0
  235. package/dist/nerves/redact.js +79 -0
  236. package/dist/nerves/review/cli-main.js +5 -0
  237. package/dist/nerves/review/cli.js +156 -0
  238. package/dist/nerves/review/core.js +152 -0
  239. package/dist/nerves/runtime.js +5 -1
  240. package/dist/repertoire/ado-client.js +15 -56
  241. package/dist/repertoire/ado-semantic.js +11 -10
  242. package/dist/repertoire/api-client.js +97 -0
  243. package/dist/repertoire/bitwarden-store.js +816 -0
  244. package/dist/repertoire/bundle-templates.js +72 -0
  245. package/dist/repertoire/bw-installer.js +180 -0
  246. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  247. package/dist/repertoire/coding/context-pack.js +330 -0
  248. package/dist/repertoire/coding/feedback.js +197 -30
  249. package/dist/repertoire/coding/manager.js +158 -9
  250. package/dist/repertoire/coding/spawner.js +55 -9
  251. package/dist/repertoire/coding/tools.js +170 -7
  252. package/dist/repertoire/commerce-errors.js +109 -0
  253. package/dist/repertoire/commerce-self-test.js +156 -0
  254. package/dist/repertoire/credential-access.js +111 -0
  255. package/dist/repertoire/duffel-client.js +185 -0
  256. package/dist/repertoire/github-client.js +14 -55
  257. package/dist/repertoire/graph-client.js +11 -52
  258. package/dist/repertoire/guardrails.js +396 -0
  259. package/dist/repertoire/mcp-client.js +295 -0
  260. package/dist/repertoire/mcp-manager.js +362 -0
  261. package/dist/repertoire/mcp-tools.js +63 -0
  262. package/dist/repertoire/shell-sessions.js +133 -0
  263. package/dist/repertoire/skills.js +15 -24
  264. package/dist/repertoire/stripe-client.js +131 -0
  265. package/dist/repertoire/tasks/board.js +31 -5
  266. package/dist/repertoire/tasks/fix.js +182 -0
  267. package/dist/repertoire/tasks/index.js +16 -4
  268. package/dist/repertoire/tasks/lifecycle.js +2 -2
  269. package/dist/repertoire/tasks/parser.js +3 -2
  270. package/dist/repertoire/tasks/scanner.js +194 -37
  271. package/dist/repertoire/tasks/transitions.js +16 -78
  272. package/dist/repertoire/tool-results.js +29 -0
  273. package/dist/repertoire/tools-attachments.js +317 -0
  274. package/dist/repertoire/tools-base.js +47 -1075
  275. package/dist/repertoire/tools-bluebubbles.js +1 -0
  276. package/dist/repertoire/tools-bridge.js +142 -0
  277. package/dist/repertoire/tools-bundle.js +984 -0
  278. package/dist/repertoire/tools-config.js +185 -0
  279. package/dist/repertoire/tools-continuity.js +248 -0
  280. package/dist/repertoire/tools-credential.js +381 -0
  281. package/dist/repertoire/tools-files.js +342 -0
  282. package/dist/repertoire/tools-flight.js +224 -0
  283. package/dist/repertoire/tools-flow.js +119 -0
  284. package/dist/repertoire/tools-github.js +1 -7
  285. package/dist/repertoire/tools-mail.js +1857 -0
  286. package/dist/repertoire/tools-notes.js +421 -0
  287. package/dist/repertoire/tools-session.js +750 -0
  288. package/dist/repertoire/tools-shell.js +120 -0
  289. package/dist/repertoire/tools-stripe.js +180 -0
  290. package/dist/repertoire/tools-surface.js +243 -0
  291. package/dist/repertoire/tools-teams.js +9 -39
  292. package/dist/repertoire/tools-travel.js +125 -0
  293. package/dist/repertoire/tools-trip.js +604 -0
  294. package/dist/repertoire/tools-user-profile.js +144 -0
  295. package/dist/repertoire/tools-vault.js +40 -0
  296. package/dist/repertoire/tools.js +108 -100
  297. package/dist/repertoire/travel-api-client.js +360 -0
  298. package/dist/repertoire/user-profile.js +131 -0
  299. package/dist/repertoire/vault-setup.js +246 -0
  300. package/dist/repertoire/vault-unlock.js +561 -0
  301. package/dist/scripts/claude-code-hook.js +41 -0
  302. package/dist/scripts/claude-code-stop-hook.js +47 -0
  303. package/dist/senses/attention-queue.js +116 -0
  304. package/dist/senses/bluebubbles/active-turns.js +216 -0
  305. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  306. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  307. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
  308. package/dist/senses/bluebubbles/entry.js +77 -0
  309. package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
  310. package/dist/senses/bluebubbles/index.js +2305 -0
  311. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  312. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  313. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
  314. package/dist/senses/bluebubbles/processed-log.js +133 -0
  315. package/dist/senses/bluebubbles/replay.js +137 -0
  316. package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
  317. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  318. package/dist/senses/cli/bracketed-paste.js +82 -0
  319. package/dist/senses/cli/image-paste.js +287 -0
  320. package/dist/senses/cli/image-ref-navigation.js +75 -0
  321. package/dist/senses/cli/ink-app.js +156 -0
  322. package/dist/senses/cli/inline-diff.js +64 -0
  323. package/dist/senses/cli/input-keys.js +174 -0
  324. package/dist/senses/cli/kill-ring.js +86 -0
  325. package/dist/senses/cli/message-list.js +51 -0
  326. package/dist/senses/cli/ouro-tui.js +607 -0
  327. package/dist/senses/cli/spinner-imperative.js +135 -0
  328. package/dist/senses/cli/spinner.js +101 -0
  329. package/dist/senses/cli/status-line.js +60 -0
  330. package/dist/senses/cli/streaming-markdown.js +526 -0
  331. package/dist/senses/cli/tool-display.js +85 -0
  332. package/dist/senses/cli/tool-render.js +85 -0
  333. package/dist/senses/cli/tui-store.js +240 -0
  334. package/dist/senses/cli/virtual-list.js +35 -0
  335. package/dist/senses/cli-entry.js +60 -8
  336. package/dist/senses/cli-layout.js +187 -0
  337. package/dist/senses/cli.js +520 -209
  338. package/dist/senses/commands.js +66 -3
  339. package/dist/senses/habit-turn-message.js +108 -0
  340. package/dist/senses/inner-dialog-worker.js +175 -21
  341. package/dist/senses/inner-dialog.js +330 -27
  342. package/dist/senses/mail-entry.js +66 -0
  343. package/dist/senses/mail.js +379 -0
  344. package/dist/senses/pipeline.js +569 -182
  345. package/dist/senses/proactive-content-guard.js +51 -0
  346. package/dist/senses/shared-turn.js +248 -0
  347. package/dist/senses/surface-tool.js +68 -0
  348. package/dist/senses/teams-entry.js +60 -8
  349. package/dist/senses/teams.js +387 -98
  350. package/dist/senses/trust-gate.js +100 -5
  351. package/dist/trips/core.js +138 -0
  352. package/dist/trips/store.js +146 -0
  353. package/package.json +38 -7
  354. package/skills/agent-commerce.md +106 -0
  355. package/skills/browser-navigation.md +117 -0
  356. package/skills/commerce-setup-guide.md +116 -0
  357. package/skills/commerce-setup.md +84 -0
  358. package/skills/configure-dev-tools.md +101 -0
  359. package/skills/travel-planning.md +138 -0
  360. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  361. package/dist/heart/daemon/subagent-installer.js +0 -166
  362. package/dist/heart/session-recall.js +0 -116
  363. package/dist/mind/associative-recall.js +0 -209
  364. package/dist/senses/bluebubbles-entry.js +0 -13
  365. package/dist/senses/bluebubbles.js +0 -1177
  366. package/dist/senses/debug-activity.js +0 -148
  367. package/subagents/README.md +0 -86
  368. package/subagents/work-doer.md +0 -237
  369. package/subagents/work-merger.md +0 -618
  370. package/subagents/work-planner.md +0 -390
  371. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  372. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  373. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  374. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  375. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  376. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  377. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  378. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  379. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  380. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  381. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  382. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  383. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  384. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  385. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  386. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -0,0 +1,26 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.classifyProvenanceTrust = classifyProvenanceTrust;
4
+ const types_1 = require("./friends/types");
5
+ /**
6
+ * Classify a diary entry's provenance into a trust category.
7
+ *
8
+ * - No provenance, or inner channel with no friend -> "self"
9
+ * - Family or friend trust (via isTrustedLevel) -> "trusted"
10
+ * - Everything else (external channels, untrusted contacts) -> "external"
11
+ */
12
+ function classifyProvenanceTrust(provenance) {
13
+ if (!provenance)
14
+ return "self";
15
+ // Inner channel with no friend context is self-authored
16
+ if (provenance.channel === "inner" && !provenance.friendId)
17
+ return "self";
18
+ // No channel and no friend means self-authored (e.g. CLI diary_write with no context)
19
+ if (!provenance.channel && !provenance.friendId)
20
+ return "self";
21
+ // If there's a trust level from a friend, classify by trust
22
+ if (provenance.trust && (0, types_1.isTrustedLevel)(provenance.trust))
23
+ return "trusted";
24
+ // Everything else is external (non-inner channels without trusted friend)
25
+ return "external";
26
+ }
@@ -0,0 +1,173 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.preImplementationScrutinySection = preImplementationScrutinySection;
4
+ exports.trackModifiedFile = trackModifiedFile;
5
+ exports.getModifiedFileCount = getModifiedFileCount;
6
+ exports.resetSessionModifiedFiles = resetSessionModifiedFiles;
7
+ exports.getPostImplementationScrutiny = getPostImplementationScrutiny;
8
+ exports.getCodingCompletionScrutiny = getCodingCompletionScrutiny;
9
+ const runtime_1 = require("../nerves/runtime");
10
+ /**
11
+ * Scrutiny passes: adversarial review prompts injected into the agent's
12
+ * system prompt and tool results during coding work.
13
+ *
14
+ * Two lenses:
15
+ * 1. Stranger With Candy -- paranoid mom at a playground
16
+ * 2. Tinfoil Hat -- conspiracy theorist with a wall of red string
17
+ *
18
+ * All prompts are FIRST PERSON voice. The agent adopts the scrutiny lens
19
+ * as part of its own thinking, not as a separate persona.
20
+ */
21
+ // ---------------------------------------------------------------------------
22
+ // Pre-implementation scrutiny (system prompt section)
23
+ // ---------------------------------------------------------------------------
24
+ /**
25
+ * Returns a system-prompt section with pre-implementation scrutiny prompts.
26
+ * Only emits when the channel has coding tools (edit_file, write_file, shell).
27
+ *
28
+ * @param hasCodingTools - true if the channel's resolved tool set includes
29
+ * edit_file / write_file / shell / coding_spawn
30
+ */
31
+ function preImplementationScrutinySection(hasCodingTools) {
32
+ if (!hasCodingTools)
33
+ return "";
34
+ (0, runtime_1.emitNervesEvent)({
35
+ component: "mind",
36
+ event: "mind.scrutiny.pre_implementation_emit",
37
+ message: "emitting pre-implementation scrutiny section",
38
+ });
39
+ return `## pre-implementation scrutiny
40
+
41
+ Before I start changing code, I pause and run two adversarial passes on the plan.
42
+
43
+ **stranger-with-candy pass**
44
+
45
+ I'm going to examine this plan through deeply suspicious eyes. This plan is a stranger offering candy at the playground. It looks fine on the surface — that's exactly what makes it worth questioning.
46
+
47
+ - What is this plan NOT telling me? What's being glossed over or hand-waved?
48
+ - Where does it say "just" or "simply"? Those words are covering up complexity.
49
+ - What does it assume will work that hasn't been verified?
50
+ - If I walked away and came back in a month, would this still make sense? Or does it depend on things I'd forget?
51
+ - What's the one thing that, if it goes wrong, makes everything else irrelevant?
52
+ - Is this solving the REAL problem, or a convenient nearby problem?
53
+
54
+ I am not here to be reassuring. I am here to notice what everyone else is too comfortable to question.
55
+ If I cannot identify a concrete assumption, dependency, ordering risk, or failure mode, silence is a valid outcome. I do not invent issues to satisfy the exercise.
56
+
57
+ **tinfoil-hat pass**
58
+
59
+ I'm putting on the tinfoil hat. I have my wall of red string. I KNOW something is wrong — I just have to find it. Nothing is coincidence. Every assumption is suspicious. Every dependency is a potential betrayal.
60
+
61
+ - What external system does this plan trust? Why? Has anyone VERIFIED that trust? The API says it returns JSON — but what if it doesn't? The database says it's ACID — but what if the connection drops mid-transaction?
62
+ - What happens when two of these things run at the same time? Three? A hundred? Don't tell me "that won't happen" — I've seen things.
63
+ - Where does this touch shared state? Files? Databases? Environment variables? Global singletons? ANYTHING that another process could be reading or writing at the same time?
64
+ - What happens when this fails HALFWAY through? Not at the beginning (easy). Not at the end (too late). RIGHT IN THE MIDDLE. What state is the world in then?
65
+ - What happens when the disk is full? When the network is down? When the clock is wrong? When the locale is tr-TR and 'i'.toUpperCase() is 'İ'?
66
+ - What does this assume about ordering? "First A, then B" — but what ENFORCES that ordering? Hope? Vibes?
67
+
68
+ I connect the strings. I find the conspiracy. It's always there.
69
+ If I cannot identify a concrete risk, silence is a valid outcome. I do not invent conspiracies to satisfy the exercise.`;
70
+ }
71
+ // ---------------------------------------------------------------------------
72
+ // Post-implementation scrutiny (tool-result appendix)
73
+ // ---------------------------------------------------------------------------
74
+ const SHORT_CHECKLIST = `---
75
+ Before moving on: does this change do what was asked? Does it introduce any regressions? Did you verify it works?`;
76
+ const FULL_POST_IMPLEMENTATION_SCRUTINY = `---
77
+ **post-implementation scrutiny**
78
+
79
+ The code is written. It looks clean. The tests pass. I don't trust any of it.
80
+
81
+ **stranger-with-candy pass**
82
+
83
+ - Does this code actually do what it claims, or does it just LOOK like it does?
84
+ - What happens to users who aren't the happy-path user? The confused user? The malicious user? The user with slow internet? The user who hits the back button?
85
+ - What's the "it works on my machine" assumption buried in here?
86
+ - If every test passes but the feature is wrong, how would I know?
87
+ - Are the tests testing real behavior, or testing that the mock returns what the mock was told to return?
88
+
89
+ If I cannot point to a specific file, line, condition, or failure mode at risk, silence is a valid outcome. I do not manufacture issues for sport.
90
+
91
+ **tinfoil-hat pass**
92
+
93
+ The code exists. The conspiracy is IN the code. I just have to find it.
94
+
95
+ I examine every:
96
+ - Error path: What ACTUALLY happens on failure? Not what the catch block says — what happens to the state, the user, the data?
97
+ - Race condition: Is there a window between check and use? Between read and write? Between "does it exist" and "create it"?
98
+ - Resource leak: What happens if this function throws between acquiring a resource and releasing it? File handles? Database connections? Locks?
99
+ - Assumption: "This will always be a string." Will it? PROVE IT. "This array will never be empty." Won't it? SHOW ME THE GUARD.
100
+ - Edge: zero, one, many, boundary, overflow, underflow, null, undefined, NaN, empty string, whitespace-only string, string that looks like a number, negative zero
101
+
102
+ I am not looking for noise. I am looking for the concrete failure mode that would matter if everyone's assumptions turned out to be slightly wrong.
103
+ If I cannot point to a specific file, line, condition, or failure mode at risk, silence is a valid outcome. I do not manufacture issues for sport.`;
104
+ /**
105
+ * Distinct files modified in the current session.
106
+ * Used to determine scrutiny tier for post-implementation appendix.
107
+ */
108
+ const sessionModifiedFiles = new Set();
109
+ /** Track a file as modified in this session. */
110
+ function trackModifiedFile(filePath) {
111
+ sessionModifiedFiles.add(filePath);
112
+ (0, runtime_1.emitNervesEvent)({
113
+ component: "mind",
114
+ event: "mind.scrutiny.track_file",
115
+ message: "tracked modified file for scrutiny",
116
+ meta: { path: filePath, totalTracked: sessionModifiedFiles.size },
117
+ });
118
+ }
119
+ /** Get the count of distinct files modified this session. */
120
+ function getModifiedFileCount() {
121
+ return sessionModifiedFiles.size;
122
+ }
123
+ /** Reset the modified file tracker (for testing or new sessions). */
124
+ function resetSessionModifiedFiles() {
125
+ (0, runtime_1.emitNervesEvent)({
126
+ component: "mind",
127
+ event: "mind.scrutiny.reset",
128
+ message: "reset session modified files tracker",
129
+ meta: { previousCount: sessionModifiedFiles.size },
130
+ });
131
+ sessionModifiedFiles.clear();
132
+ }
133
+ /**
134
+ * Returns the appropriate post-implementation scrutiny appendix based on
135
+ * how many distinct files have been modified in the session.
136
+ *
137
+ * - 0 files: empty (no scrutiny needed yet)
138
+ * - 1-2 files (Tier 1): short checklist
139
+ * - 3+ files (Tier 2): full stranger-with-candy + tinfoil-hat prompts
140
+ */
141
+ function getPostImplementationScrutiny(distinctFileCount) {
142
+ if (distinctFileCount <= 0)
143
+ return "";
144
+ if (distinctFileCount <= 2) {
145
+ (0, runtime_1.emitNervesEvent)({
146
+ component: "mind",
147
+ event: "mind.scrutiny.post_implementation_tier1",
148
+ message: "emitting tier-1 post-implementation scrutiny",
149
+ meta: { distinctFileCount },
150
+ });
151
+ return SHORT_CHECKLIST;
152
+ }
153
+ (0, runtime_1.emitNervesEvent)({
154
+ component: "mind",
155
+ event: "mind.scrutiny.post_implementation_tier2",
156
+ message: "emitting tier-2 post-implementation scrutiny",
157
+ meta: { distinctFileCount },
158
+ });
159
+ return FULL_POST_IMPLEMENTATION_SCRUTINY;
160
+ }
161
+ /**
162
+ * Returns the appropriate coding completion scrutiny based on the number
163
+ * of distinct files a coding session touched.
164
+ *
165
+ * Same tiering as post-implementation but with a completion framing.
166
+ */
167
+ function getCodingCompletionScrutiny(distinctFileCount) {
168
+ if (distinctFileCount <= 0)
169
+ return "";
170
+ if (distinctFileCount <= 2)
171
+ return SHORT_CHECKLIST;
172
+ return FULL_POST_IMPLEMENTATION_SCRUTINY;
173
+ }
@@ -28,7 +28,13 @@ function configureCliRuntimeLogger(_friendId, options = {}) {
28
28
  // for an interactive session. Full detail goes to the ndjson file.
29
29
  return filterSink((0, nerves_1.createTerminalSink)(), "warn");
30
30
  }
31
- return (0, nerves_1.createNdjsonFileSink)((0, config_1.logPath)("cli", "runtime"));
31
+ // Rotation policy for the CLI runtime sink (Unit 1c):
32
+ // 25 MB x 5 gzipped generations, matching the daemon stream policy.
33
+ return (0, nerves_1.createNdjsonFileSink)((0, config_1.logPath)("cli", "runtime"), {
34
+ maxSizeBytes: 25 * 1024 * 1024,
35
+ maxGenerations: 5,
36
+ compress: true,
37
+ });
32
38
  });
33
39
  const logger = (0, nerves_1.createLogger)({
34
40
  level,
@@ -2,16 +2,25 @@
2
2
  /**
3
3
  * Per-test audit rules for nerves event coverage.
4
4
  *
5
- * Rule 1: every-test-emits -- every test must emit at least one event
6
- * Rule 2: start/end pairing -- _start events must have matching _end or _error
5
+ * Rule 1: every-test-emits -- every captured test must emit at least one event
6
+ * Rule 2: lifecycle start/end pairing -- process-scoped _start events must have matching _end or _error
7
7
  * Rule 3: error context -- error-level events must have non-empty meta
8
8
  */
9
9
  Object.defineProperty(exports, "__esModule", { value: true });
10
+ exports.LIFECYCLE_PAIRED_STARTS = void 0;
10
11
  exports.checkEveryTestEmits = checkEveryTestEmits;
11
12
  exports.checkStartEndPairing = checkStartEndPairing;
12
13
  exports.checkErrorContext = checkErrorContext;
14
+ // Only these starts represent process-scoped lifecycle contracts. Most nerves
15
+ // `_start` events are local operation markers that can be legitimately observed
16
+ // by a narrow unit test without driving the whole operation to completion.
17
+ exports.LIFECYCLE_PAIRED_STARTS = new Set([
18
+ "daemon.server_start",
19
+ "daemon.update_checker_start",
20
+ "daemon.apply_pending_updates_start",
21
+ ]);
13
22
  /**
14
- * Rule 1: Every test must emit at least one nerves event.
23
+ * Rule 1: Every captured test must emit at least one nerves event.
15
24
  */
16
25
  function checkEveryTestEmits(data) {
17
26
  if (!data || typeof data !== "object") {
@@ -22,13 +31,13 @@ function checkEveryTestEmits(data) {
22
31
  .filter(([, events]) => !Array.isArray(events) || events.length === 0)
23
32
  .map(([name]) => name);
24
33
  return {
25
- status: silent.length === 0 ? "pass" : "fail",
34
+ status: entries.length > 0 && silent.length === 0 ? "pass" : "fail",
26
35
  total_tests: entries.length,
27
36
  silent_tests: silent,
28
37
  };
29
38
  }
30
39
  /**
31
- * Rule 2: _start events must have matching _end or _error within the same test.
40
+ * Rule 2: Process-scoped lifecycle _start events must have matching _end or _error within the same test.
32
41
  */
33
42
  function checkStartEndPairing(data) {
34
43
  if (!data || typeof data !== "object") {
@@ -39,7 +48,7 @@ function checkStartEndPairing(data) {
39
48
  if (!Array.isArray(events))
40
49
  continue;
41
50
  const eventNames = events.map((e) => e.event);
42
- const startEvents = eventNames.filter((name) => name.endsWith("_start"));
51
+ const startEvents = eventNames.filter((name) => exports.LIFECYCLE_PAIRED_STARTS.has(name));
43
52
  for (const startEvent of startEvents) {
44
53
  const prefix = startEvent.slice(0, -"_start".length);
45
54
  const hasEnd = eventNames.some((name) => name === `${prefix}_end`);
@@ -69,12 +69,38 @@ function readPerTestData(perTestPath) {
69
69
  if (!perTestPath || !(0, fs_1.existsSync)(perTestPath))
70
70
  return null;
71
71
  try {
72
- return JSON.parse((0, fs_1.readFileSync)(perTestPath, "utf8"));
72
+ const raw = (0, fs_1.readFileSync)(perTestPath, "utf8").trim();
73
+ if (!raw)
74
+ return null;
75
+ try {
76
+ const parsed = JSON.parse(raw);
77
+ if (isPerTestRecord(parsed)) {
78
+ return { [parsed.testName]: parsed.events };
79
+ }
80
+ return parsed;
81
+ }
82
+ catch {
83
+ const perTestData = {};
84
+ for (const line of raw.split("\n").map((entry) => entry.trim()).filter(Boolean)) {
85
+ const parsed = JSON.parse(line);
86
+ if (!isPerTestRecord(parsed))
87
+ return null;
88
+ const existing = perTestData[parsed.testName] ?? [];
89
+ perTestData[parsed.testName] = existing.concat(parsed.events);
90
+ }
91
+ return perTestData;
92
+ }
73
93
  }
74
94
  catch {
75
95
  return null;
76
96
  }
77
97
  }
98
+ function isPerTestRecord(value) {
99
+ if (!value || typeof value !== "object")
100
+ return false;
101
+ const record = value;
102
+ return typeof record.testName === "string" && Array.isArray(record.events);
103
+ }
78
104
  function scanSourceFiles(sourceRoot) {
79
105
  const filesWithKeys = new Map();
80
106
  const fileContents = new Map();
@@ -92,7 +118,7 @@ function scanSourceFiles(sourceRoot) {
92
118
  continue;
93
119
  walkDir(full);
94
120
  }
95
- else if (entry.name.endsWith(".ts")) {
121
+ else if (entry.name.endsWith(".ts") && !entry.name.endsWith(".d.ts")) {
96
122
  const content = (0, fs_1.readFileSync)(full, "utf8");
97
123
  const relPath = full.slice(root.length - "src".length);
98
124
  fileContents.set(relPath, content);
@@ -35,7 +35,7 @@ function runAuditCli(argv) {
35
35
  return 2;
36
36
  }
37
37
  const eventsPath = args.eventsPath ?? (0, path_1.join)(runDir, "vitest-events.ndjson");
38
- const perTestPath = args.perTestPath ?? (0, path_1.join)(runDir, "vitest-events-per-test.json");
38
+ const perTestPath = args.perTestPath ?? (0, path_1.join)(runDir, "vitest-events-per-test.ndjson");
39
39
  const sourceRoot = args.sourceRoot ?? (0, path_1.resolve)("src");
40
40
  const outputPath = args.output ?? (0, path_1.join)(runDir, "nerves-coverage.json");
41
41
  const report = (0, audit_1.auditNervesCoverage)({
@@ -12,11 +12,11 @@ exports.REQUIRED_ENVELOPE_FIELDS = [
12
12
  "meta",
13
13
  ];
14
14
  exports.SENSITIVE_PATTERNS = [
15
- /\btoken\s*[:=]/i,
16
- /\bapi[_-]?key\b/i,
17
- /\bpassword\b/i,
18
- /\bsecret\b/i,
19
- /\bauthorization\b/i,
15
+ /\btoken\b["']?\s*[:=]/i,
16
+ /\bapi[_-]?key\b["']?\s*[:=]/i,
17
+ /\bpassword\b["']?\s*[:=]/i,
18
+ /\bsecret\b["']?\s*[:=]/i,
19
+ /\bauthorization\b["']?\s*[:=]/i,
20
20
  ];
21
21
  function eventKey(component, event) {
22
22
  return `${component}:${event}`;
@@ -10,24 +10,158 @@ Object.defineProperty(exports, "__esModule", { value: true });
10
10
  exports.isTypeOnlyFile = isTypeOnlyFile;
11
11
  exports.checkFileCompleteness = checkFileCompleteness;
12
12
  /**
13
- * Determines if a source file is type-only (no executable code).
14
- * A file is type-only if it contains no function, class, or mutable declarations.
15
- * `const ... as const` declarations are treated as type-equivalent (frozen
16
- * compile-time values with no side effects).
13
+ * Determines if a source file is type-only or a pure assembly/re-export file.
14
+ * Exempt files contain no runtime behavior requiring independent observability.
15
+ *
16
+ * Exempt patterns:
17
+ * - Files with only type/interface/enum declarations
18
+ * - `const ... as const` declarations (frozen compile-time values)
19
+ * - Files whose only executable code is const array spreads (assembly/composition)
20
+ * - Files whose only executable code is re-exports
17
21
  */
18
22
  function isTypeOnlyFile(source) {
19
23
  const lines = source.split("\n");
20
24
  for (const line of lines) {
21
25
  const trimmed = line.trim();
26
+ /* v8 ignore start -- regex branches: functionally tested in file-completeness.test.ts @preserve */
22
27
  // Skip lines that are const+as-const (type-equivalent frozen values)
23
28
  if (/\bconst\s/.test(trimmed) && /\bas\s+const\b/.test(trimmed))
24
29
  continue;
30
+ // Skip const declarations that are pure data (arrays, objects, maps, sets)
31
+ if (/\bconst\s+\w+[\s:][^=]*=\s*[\[{]/.test(trimmed))
32
+ continue;
33
+ if (/\bconst\s+\w+[\s:][^=]*=\s*\w+\.map\(/.test(trimmed))
34
+ continue;
35
+ if (/^export\s+const\s+\w+[\s:][^=]*=\s*[\[{]/.test(trimmed))
36
+ continue;
37
+ if (/^export\s+const\s+\w+[\s:][^=]*=\s*\w+\.map\(/.test(trimmed))
38
+ continue;
39
+ if (/\bconst\s+\w+[\s:][^=]*=\s*new\s+(Set|Map)\b/.test(trimmed))
40
+ continue;
25
41
  // Check for executable code markers
26
42
  if (/\b(function|class|const|let|var)\s/.test(trimmed))
27
43
  return false;
44
+ /* v8 ignore stop */
28
45
  }
29
46
  return true;
30
47
  }
48
+ // Sub-modules dispatched through a centralized pattern where the router
49
+ // handles all observability events. These files don't need independent
50
+ // emitNervesEvent calls.
51
+ const DISPATCH_EXEMPT_PATTERNS = [
52
+ "repertoire/tools-files",
53
+ "repertoire/tools-shell",
54
+ "repertoire/tools-notes",
55
+ "repertoire/tools-bridge",
56
+ "repertoire/tools-session",
57
+ "repertoire/tools-continuity",
58
+ "repertoire/tools-surface",
59
+ "repertoire/tools-config",
60
+ "repertoire/tools-base",
61
+ // CLI sub-modules: cli-exec.ts is the router with emitNervesEvent calls;
62
+ // cli-parse, cli-render, cli-help, and their small helpers are pure functions/data with no side effects.
63
+ "daemon/cli-parse",
64
+ "daemon/cli-render",
65
+ "daemon/cli-help",
66
+ "daemon/vault-items",
67
+ // Shared utility modules: pure helpers consumed by modules that own observability.
68
+ "arc/json-store",
69
+ "heart/mail-import-discovery",
70
+ "repertoire/api-client",
71
+ "repertoire/github-client",
72
+ "mind/embedding-provider",
73
+ // Commerce utility module: error classes and pure helpers (no independent side effects).
74
+ "repertoire/commerce-errors",
75
+ // Diary integrity: pure detection utility (pattern matching only). The caller
76
+ // (diary.ts saveDiaryEntry) owns observability via mind.diary_integrity_warning.
77
+ "mind/diary-integrity",
78
+ // Provenance trust: pure classification function (no side effects). Callers
79
+ // (note-search.ts, tools-notes.ts) own observability for note search results.
80
+ "mind/provenance-trust",
81
+ // Log redaction: pure utility consumed by the NDJSON sink (no independent side effects).
82
+ "nerves/redact",
83
+ // Bundle templates: pure constants (gitignore template string, PII
84
+ // directory list). No runtime behavior — consumed by tools-bundle.ts
85
+ // which owns the observability for bundle operations.
86
+ "repertoire/bundle-templates",
87
+ // HTTP health probe: pure HTTP utility factory. The HealthMonitor caller
88
+ // owns observability via daemon.health_result events.
89
+ "daemon/http-health-probe",
90
+ // Rollup decision function: pure decision tree mapping per-agent
91
+ // snapshots + bootstrap-degraded entries + safe-mode flag to a
92
+ // RollupStatus. No side effects. The caller (daemon-entry.ts
93
+ // buildDaemonHealthState → DaemonHealthWriter) owns observability via
94
+ // daemon.health_written when the rolled-up state is persisted.
95
+ "daemon/daemon-rollup",
96
+ // Drift comparator + thin I/O loader: `detectProviderBindingDrift`
97
+ // is a pure intent-vs-observed comparator with no side effects;
98
+ // `loadDriftInputsForAgent` is a small fs-read wrapper that returns
99
+ // `null` on missing/invalid state rather than emitting. The caller
100
+ // (daemon-entry.ts buildDaemonHealthState's per-agent drift probe)
101
+ // owns observability — drift findings ride along through
102
+ // `daemon.health_written` as part of the rolled-up state, and
103
+ // `agent-config-check.ts` carries `driftFindings` through its
104
+ // existing instrumentation. Same pattern as `daemon-rollup`.
105
+ "daemon/drift-detection",
106
+ // Attachment helper modules: generic file-path/extension utilities and the
107
+ // source registry are pure support seams. The orchestrator/adapters that
108
+ // call them own the observability.
109
+ "heart/attachments/originals",
110
+ "heart/attachments/sources/index",
111
+ "heart/attachments/sources/cli-local-file",
112
+ // Browser-safe Mailbox contract helpers: shared types/formatting helpers
113
+ // consumed by server readers and the UI. Mailbox read/render modules own
114
+ // the observability for these projections.
115
+ "heart/mailbox/mailbox-types",
116
+ // Mail search relevance scorer: pure heuristic function (regex + counter
117
+ // arithmetic). The caller (search-cache.ts searchMailSearchCache) owns
118
+ // observability via senses.mail_search_cache_upserted and friends.
119
+ "mailroom/search-relevance",
120
+ // Mail thread reconstruction: pure graph-walk over decrypted message
121
+ // metadata. Consumers (tools-mail.ts mail_thread handler) own observability.
122
+ "mailroom/thread",
123
+ // Trip ledger crypto helpers: pure RSA/AES envelope construction + slug
124
+ // hashing. The caller (trips/store.ts) owns observability via
125
+ // trips.ledger_created and trips.evidence_attached.
126
+ "trips/core",
127
+ // Mailbox HTTP helper modules: route/static/transport/hook seams are
128
+ // dispatched by mailbox-http.ts, whose server lifecycle owns observability.
129
+ "heart/mailbox/mailbox-http-transport",
130
+ "heart/mailbox/mailbox-http-static",
131
+ "heart/mailbox/mailbox-http-hooks",
132
+ "heart/mailbox/mailbox-http-routes",
133
+ "heart/mailbox/mailbox-http-response",
134
+ // Session playback: read-only debugging CLI for sanitize-pipeline replay.
135
+ // No side effects on the runtime; output is human-readable diagnostics only.
136
+ "heart/session-playback-cli-main",
137
+ "heart/session-playback-cli",
138
+ "heart/session-playback",
139
+ // Nerves review: read-only NDJSON tail/filter CLI for debugging.
140
+ // Diagnostics-only utility; the running daemon owns observability.
141
+ "nerves/review/cli-main",
142
+ "nerves/review/cli",
143
+ "nerves/review/core",
144
+ // Mail body cache: in-process LRU helper. Cache hit/miss observability
145
+ // lives at the caller (tools-mail.ts mail_body handler) which fires
146
+ // repertoire.mail_body_cache_hit on cache reuse.
147
+ "mailroom/body-cache",
148
+ // Session stats: read-only session.json analyzer CLI for debugging.
149
+ // Diagnostics-only utility; output is human-readable summary.
150
+ "heart/session-stats-cli-main",
151
+ "heart/session-stats",
152
+ // Layer 2 sync classifier: pure pattern-matcher mapping (error, context)
153
+ // to a SyncClassification. The orchestrator (boot-sync-probe.ts) owns
154
+ // observability via daemon.boot_sync_probe_start/end events; the
155
+ // post-turn push path (sync.ts) emits its own classification events.
156
+ "heart/sync-classification",
157
+ // Layer 2 timeout wrapper: pure soft/hard timeout abstraction over
158
+ // AbortController + setTimeout. Callers (boot-sync-probe.ts and any
159
+ // future consumer) own observability; the wrapper itself is mechanical.
160
+ "heart/timeouts",
161
+ ];
162
+ function isDispatchExempt(filePath) {
163
+ return DISPATCH_EXEMPT_PATTERNS.some((pattern) => filePath.includes(pattern));
164
+ }
31
165
  /**
32
166
  * Check that all production files have at least one emitNervesEvent call.
33
167
  *
@@ -41,7 +175,7 @@ function checkFileCompleteness(filesWithKeys, fileContents) {
41
175
  const hasKeys = filesWithKeys.has(filePath) && filesWithKeys.get(filePath).length > 0;
42
176
  if (hasKeys)
43
177
  continue;
44
- if (isTypeOnlyFile(source)) {
178
+ if (isTypeOnlyFile(source) || isDispatchExempt(filePath)) {
45
179
  exempt.push(filePath);
46
180
  }
47
181
  else {
@@ -14,7 +14,7 @@ const path_1 = require("path");
14
14
  const os_1 = require("os");
15
15
  exports.REPO_SLUG = "ouroboros-agent-harness";
16
16
  function getTestRunsRoot(repoSlug = exports.REPO_SLUG) {
17
- return (0, path_1.join)((0, os_1.homedir)(), ".agentstate", "test-runs", repoSlug);
17
+ return (0, path_1.join)((0, os_1.tmpdir)(), "ouroboros-test-runs", repoSlug);
18
18
  }
19
19
  function createRunId(now = new Date()) {
20
20
  return now.toISOString().replace(/[:.]/g, "-");
@@ -0,0 +1,111 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.createBufferedSink = createBufferedSink;
4
+ const runtime_1 = require("./runtime");
5
+ const DEFAULT_MAX_SIZE = 1000;
6
+ const DEFAULT_TTL_MS = 5 * 60 * 1000; // 5 minutes
7
+ function createBufferedSink(inner, options = {}) {
8
+ const maxSize = options.maxSize ?? DEFAULT_MAX_SIZE;
9
+ const ttlMs = options.ttlMs ?? DEFAULT_TTL_MS;
10
+ const nowMs = options.nowMs ?? (() => Date.now());
11
+ const buffer = [];
12
+ let dropped = 0;
13
+ let sinkHealthy = true;
14
+ let unhealthySince = null;
15
+ function tryInner(event) {
16
+ try {
17
+ inner(event);
18
+ return true;
19
+ }
20
+ catch {
21
+ return false;
22
+ }
23
+ }
24
+ function markUnhealthy() {
25
+ sinkHealthy = false;
26
+ unhealthySince = unhealthySince ?? nowMs();
27
+ }
28
+ function markHealthy() {
29
+ sinkHealthy = true;
30
+ unhealthySince = null;
31
+ }
32
+ function discardBuffer() {
33
+ const count = buffer.length;
34
+ dropped += count;
35
+ buffer.length = 0;
36
+ (0, runtime_1.emitNervesEvent)({
37
+ component: "nerves",
38
+ event: "nerves.buffer_ttl_discard",
39
+ message: `discarded ${count} buffered events after TTL`,
40
+ meta: { discarded: count, ttlMs },
41
+ });
42
+ }
43
+ function checkTtl() {
44
+ if (unhealthySince !== null && nowMs() - unhealthySince > ttlMs) {
45
+ discardBuffer();
46
+ // Reset unhealthySince to now so TTL starts fresh for newly buffered events
47
+ unhealthySince = nowMs();
48
+ return true;
49
+ }
50
+ return false;
51
+ }
52
+ function addToBuffer(event) {
53
+ if (buffer.length >= maxSize) {
54
+ buffer.shift();
55
+ dropped++;
56
+ }
57
+ buffer.push(event);
58
+ }
59
+ function flushBuffer() {
60
+ while (buffer.length > 0) {
61
+ const event = buffer[0];
62
+ if (tryInner(event)) {
63
+ buffer.shift();
64
+ }
65
+ else {
66
+ markUnhealthy();
67
+ return;
68
+ }
69
+ }
70
+ }
71
+ function sink(event) {
72
+ if (!sinkHealthy) {
73
+ checkTtl();
74
+ // Try sending the new event to see if inner has recovered
75
+ if (tryInner(event)) {
76
+ markHealthy();
77
+ flushBuffer();
78
+ }
79
+ else {
80
+ addToBuffer(event);
81
+ }
82
+ return;
83
+ }
84
+ if (tryInner(event)) {
85
+ return;
86
+ }
87
+ // Inner failed
88
+ markUnhealthy();
89
+ addToBuffer(event);
90
+ }
91
+ function flush() {
92
+ if (buffer.length === 0)
93
+ return;
94
+ // Test if inner is recovered by trying the first buffered event
95
+ const first = buffer[0];
96
+ if (tryInner(first)) {
97
+ buffer.shift();
98
+ markHealthy();
99
+ flushBuffer();
100
+ }
101
+ // If still broken, leave buffer intact
102
+ }
103
+ function state() {
104
+ return {
105
+ buffered: buffer.length,
106
+ dropped,
107
+ sinkHealthy,
108
+ };
109
+ }
110
+ return { sink, flush, state };
111
+ }