@ouro.bot/cli 0.1.0-alpha.61 → 0.1.0-alpha.612

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 (418) hide show
  1. package/README.md +127 -23
  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-broken-remote.md +63 -0
  6. package/RepairGuide.ouro/skills/diagnose-stacked-typed-issues.md +35 -0
  7. package/RepairGuide.ouro/skills/diagnose-sync-blocked.md +54 -0
  8. package/RepairGuide.ouro/skills/diagnose-vault-expired.md +60 -0
  9. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
  10. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
  11. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  12. package/changelog.json +3912 -0
  13. package/dist/arc/attention-types.js +8 -0
  14. package/dist/arc/cares.js +144 -0
  15. package/dist/arc/episodes.js +118 -0
  16. package/dist/arc/intentions.js +134 -0
  17. package/dist/arc/json-store.js +117 -0
  18. package/dist/arc/obligations.js +266 -0
  19. package/dist/arc/packets.js +194 -0
  20. package/dist/arc/presence.js +185 -0
  21. package/dist/arc/task-lifecycle.js +65 -0
  22. package/dist/heart/active-work.js +837 -26
  23. package/dist/heart/agent-entry.js +69 -3
  24. package/dist/heart/attachments/image-normalize.js +194 -0
  25. package/dist/heart/attachments/materialize.js +97 -0
  26. package/dist/heart/attachments/originals.js +88 -0
  27. package/dist/heart/attachments/render.js +29 -0
  28. package/dist/heart/attachments/sources/adapter.js +2 -0
  29. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  30. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  31. package/dist/heart/attachments/sources/index.js +16 -0
  32. package/dist/heart/attachments/store.js +103 -0
  33. package/dist/heart/attachments/types.js +93 -0
  34. package/dist/heart/auth/auth-flow.js +479 -0
  35. package/dist/heart/awaiting/await-alert.js +146 -0
  36. package/dist/heart/awaiting/await-expiry.js +108 -0
  37. package/dist/heart/awaiting/await-loader.js +91 -0
  38. package/dist/heart/awaiting/await-parser.js +141 -0
  39. package/dist/heart/awaiting/await-runtime-state.js +100 -0
  40. package/dist/heart/awaiting/await-scheduler.js +377 -0
  41. package/dist/heart/background-operations.js +281 -0
  42. package/dist/heart/bridges/store.js +14 -2
  43. package/dist/heart/bundle-state.js +168 -0
  44. package/dist/heart/commitments.js +142 -0
  45. package/dist/heart/config-registry.js +322 -0
  46. package/dist/heart/config.js +114 -119
  47. package/dist/heart/core.js +909 -246
  48. package/dist/heart/cross-chat-delivery.js +3 -18
  49. package/dist/heart/daemon/agent-config-check.js +419 -0
  50. package/dist/heart/daemon/agent-discovery.js +102 -3
  51. package/dist/heart/daemon/agent-service.js +522 -0
  52. package/dist/heart/daemon/agentic-repair.js +547 -0
  53. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  54. package/dist/heart/daemon/boot-sync-probe.js +197 -0
  55. package/dist/heart/daemon/cadence.js +70 -0
  56. package/dist/heart/daemon/cli-defaults.js +776 -0
  57. package/dist/heart/daemon/cli-exec.js +7579 -0
  58. package/dist/heart/daemon/cli-help.js +498 -0
  59. package/dist/heart/daemon/cli-parse.js +1599 -0
  60. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  61. package/dist/heart/daemon/cli-render.js +763 -0
  62. package/dist/heart/daemon/cli-types.js +8 -0
  63. package/dist/heart/daemon/connect-bay.js +323 -0
  64. package/dist/heart/daemon/daemon-cli.js +29 -1700
  65. package/dist/heart/daemon/daemon-entry.js +485 -2
  66. package/dist/heart/daemon/daemon-health.js +176 -0
  67. package/dist/heart/daemon/daemon-rollup.js +57 -0
  68. package/dist/heart/daemon/daemon-runtime-sync.js +88 -13
  69. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  70. package/dist/heart/daemon/daemon.js +904 -70
  71. package/dist/heart/daemon/dns-workflow.js +394 -0
  72. package/dist/heart/daemon/doctor-types.js +8 -0
  73. package/dist/heart/daemon/doctor.js +873 -0
  74. package/dist/heart/daemon/health-monitor.js +122 -1
  75. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  76. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  77. package/dist/heart/daemon/http-health-probe.js +80 -0
  78. package/dist/heart/daemon/human-command-screens.js +234 -0
  79. package/dist/heart/daemon/human-readiness.js +114 -0
  80. package/dist/heart/daemon/inner-status.js +89 -0
  81. package/dist/heart/daemon/interactive-repair.js +394 -0
  82. package/dist/heart/daemon/launchd.js +37 -8
  83. package/dist/heart/daemon/log-tailer.js +78 -9
  84. package/dist/heart/daemon/logs-prune.js +110 -0
  85. package/dist/heart/daemon/mcp-canary.js +297 -0
  86. package/dist/heart/daemon/os-cron-deps.js +135 -0
  87. package/dist/heart/daemon/os-cron.js +14 -12
  88. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  89. package/dist/heart/daemon/ouro-entry.js +3 -1
  90. package/dist/heart/daemon/process-manager.js +463 -34
  91. package/dist/heart/daemon/provider-discovery.js +137 -0
  92. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  93. package/dist/heart/daemon/pulse.js +475 -0
  94. package/dist/heart/daemon/readiness-repair.js +365 -0
  95. package/dist/heart/daemon/run-hooks.js +2 -0
  96. package/dist/heart/daemon/runtime-logging.js +10 -2
  97. package/dist/heart/daemon/runtime-metadata.js +2 -30
  98. package/dist/heart/daemon/safe-mode.js +161 -0
  99. package/dist/heart/daemon/sense-manager.js +493 -38
  100. package/dist/heart/daemon/session-id-resolver.js +131 -0
  101. package/dist/heart/daemon/skill-management-installer.js +28 -7
  102. package/dist/heart/daemon/socket-client.js +158 -11
  103. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  104. package/dist/heart/daemon/startup-tui.js +330 -0
  105. package/dist/heart/daemon/task-scheduler.js +3 -25
  106. package/dist/heart/daemon/terminal-ui.js +499 -0
  107. package/dist/heart/daemon/thoughts.js +229 -17
  108. package/dist/heart/daemon/up-progress.js +366 -0
  109. package/dist/heart/daemon/vault-items.js +56 -0
  110. package/dist/heart/delegation.js +1 -1
  111. package/dist/heart/habits/habit-migration.js +189 -0
  112. package/dist/heart/habits/habit-parser.js +140 -0
  113. package/dist/heart/habits/habit-runtime-state.js +100 -0
  114. package/dist/heart/habits/habit-scheduler.js +372 -0
  115. package/dist/heart/{daemon → hatch}/hatch-flow.js +32 -56
  116. package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
  117. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  118. package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
  119. package/dist/heart/identity.js +166 -55
  120. package/dist/heart/kept-notes.js +357 -0
  121. package/dist/heart/kicks.js +1 -1
  122. package/dist/heart/machine-identity.js +161 -0
  123. package/dist/heart/mail-import-discovery.js +353 -0
  124. package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
  125. package/dist/heart/mailbox/mailbox-http-response.js +7 -0
  126. package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
  127. package/dist/heart/mailbox/mailbox-http-static.js +103 -0
  128. package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
  129. package/dist/heart/mailbox/mailbox-http.js +99 -0
  130. package/dist/heart/mailbox/mailbox-read.js +31 -0
  131. package/dist/heart/mailbox/mailbox-types.js +27 -0
  132. package/dist/heart/mailbox/mailbox-view.js +197 -0
  133. package/dist/heart/mailbox/readers/agent-machine.js +425 -0
  134. package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
  135. package/dist/heart/mailbox/readers/mail.js +375 -0
  136. package/dist/heart/mailbox/readers/runtime-readers.js +756 -0
  137. package/dist/heart/mailbox/readers/sessions.js +232 -0
  138. package/dist/heart/mailbox/readers/shared.js +111 -0
  139. package/dist/heart/mcp/mcp-server.js +656 -0
  140. package/dist/heart/migrate-config.js +100 -0
  141. package/dist/heart/model-capabilities.js +19 -0
  142. package/dist/heart/platform.js +81 -0
  143. package/dist/heart/provider-attempt.js +134 -0
  144. package/dist/heart/provider-binding-resolver.js +267 -0
  145. package/dist/heart/provider-credentials.js +425 -0
  146. package/dist/heart/provider-failover.js +301 -0
  147. package/dist/heart/provider-models.js +81 -0
  148. package/dist/heart/provider-ping.js +262 -0
  149. package/dist/heart/provider-readiness-cache.js +40 -0
  150. package/dist/heart/provider-visibility.js +188 -0
  151. package/dist/heart/providers/anthropic-token.js +131 -0
  152. package/dist/heart/providers/anthropic.js +139 -52
  153. package/dist/heart/providers/azure.js +23 -11
  154. package/dist/heart/providers/error-classification.js +127 -0
  155. package/dist/heart/providers/github-copilot.js +145 -0
  156. package/dist/heart/providers/minimax-vlm.js +189 -0
  157. package/dist/heart/providers/minimax.js +26 -8
  158. package/dist/heart/providers/openai-codex.js +55 -40
  159. package/dist/heart/runtime-capability-check.js +170 -0
  160. package/dist/heart/runtime-credentials.js +367 -0
  161. package/dist/heart/runtime-cwd.js +87 -0
  162. package/dist/heart/sense-truth.js +13 -4
  163. package/dist/heart/session-activity.js +48 -24
  164. package/dist/heart/session-events.js +1156 -0
  165. package/dist/heart/session-playback-cli-main.js +5 -0
  166. package/dist/heart/session-playback-cli.js +36 -0
  167. package/dist/heart/session-playback.js +231 -0
  168. package/dist/heart/session-stats-cli-main.js +5 -0
  169. package/dist/heart/session-stats.js +182 -0
  170. package/dist/heart/session-transcript.js +133 -0
  171. package/dist/heart/start-of-turn-packet.js +345 -0
  172. package/dist/heart/streaming.js +44 -27
  173. package/dist/heart/sync-classification.js +176 -0
  174. package/dist/heart/sync.js +449 -0
  175. package/dist/heart/target-resolution.js +9 -5
  176. package/dist/heart/tempo.js +93 -0
  177. package/dist/heart/temporal-view.js +41 -0
  178. package/dist/heart/timeouts.js +101 -0
  179. package/dist/heart/tool-activity-callbacks.js +59 -0
  180. package/dist/heart/tool-description.js +143 -0
  181. package/dist/heart/tool-friction.js +55 -0
  182. package/dist/heart/tool-loop.js +200 -0
  183. package/dist/heart/turn-context.js +421 -0
  184. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
  185. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  186. package/dist/heart/versioning/ouro-path-installer.js +426 -0
  187. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  188. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  189. package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
  190. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  191. package/dist/mailbox-ui/assets/index-9-AxCxuB.js +61 -0
  192. package/dist/mailbox-ui/assets/index-CWzt267f.css +1 -0
  193. package/dist/mailbox-ui/index.html +15 -0
  194. package/dist/mailroom/attention.js +167 -0
  195. package/dist/mailroom/autonomy.js +209 -0
  196. package/dist/mailroom/blob-store.js +712 -0
  197. package/dist/mailroom/body-cache.js +61 -0
  198. package/dist/mailroom/core.js +788 -0
  199. package/dist/mailroom/entry.js +160 -0
  200. package/dist/mailroom/file-store.js +460 -0
  201. package/dist/mailroom/mbox-import.js +393 -0
  202. package/dist/mailroom/migration.js +164 -0
  203. package/dist/mailroom/outbound.js +380 -0
  204. package/dist/mailroom/policy.js +263 -0
  205. package/dist/mailroom/reader.js +233 -0
  206. package/dist/mailroom/search-cache.js +268 -0
  207. package/dist/mailroom/search-relevance.js +319 -0
  208. package/dist/mailroom/smtp-ingress.js +176 -0
  209. package/dist/mailroom/source-state.js +176 -0
  210. package/dist/mailroom/thread.js +109 -0
  211. package/dist/mailroom/travel-extract.js +89 -0
  212. package/dist/mind/bundle-manifest.js +7 -1
  213. package/dist/mind/context.js +250 -101
  214. package/dist/mind/diary-integrity.js +60 -0
  215. package/dist/mind/{memory.js → diary.js} +68 -76
  216. package/dist/mind/embedding-provider.js +60 -0
  217. package/dist/mind/file-state.js +179 -0
  218. package/dist/mind/friends/channel.js +39 -0
  219. package/dist/mind/friends/resolver.js +54 -2
  220. package/dist/mind/friends/store-file.js +48 -4
  221. package/dist/mind/friends/types.js +2 -2
  222. package/dist/mind/journal-index.js +162 -0
  223. package/dist/mind/note-search.js +268 -0
  224. package/dist/mind/obligation-steering.js +221 -0
  225. package/dist/mind/pending.js +6 -1
  226. package/dist/mind/prompt-refresh.js +3 -2
  227. package/dist/mind/prompt.js +1051 -135
  228. package/dist/mind/provenance-trust.js +26 -0
  229. package/dist/mind/scrutiny.js +173 -0
  230. package/dist/nerves/cli-logging.js +7 -1
  231. package/dist/nerves/coverage/audit-rules.js +15 -6
  232. package/dist/nerves/coverage/audit.js +28 -2
  233. package/dist/nerves/coverage/cli.js +1 -1
  234. package/dist/nerves/coverage/contract.js +5 -5
  235. package/dist/nerves/coverage/file-completeness.js +129 -5
  236. package/dist/nerves/event-buffer.js +111 -0
  237. package/dist/nerves/index.js +224 -4
  238. package/dist/nerves/observation.js +20 -0
  239. package/dist/nerves/redact.js +79 -0
  240. package/dist/nerves/review/cli-main.js +5 -0
  241. package/dist/nerves/review/cli.js +156 -0
  242. package/dist/nerves/review/core.js +152 -0
  243. package/dist/nerves/runtime.js +5 -1
  244. package/dist/repertoire/ado-client.js +15 -56
  245. package/dist/repertoire/ado-semantic.js +11 -10
  246. package/dist/repertoire/api-client.js +97 -0
  247. package/dist/repertoire/bitwarden-store.js +997 -0
  248. package/dist/repertoire/bundle-templates.js +72 -0
  249. package/dist/repertoire/bw-installer.js +180 -0
  250. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  251. package/dist/repertoire/coding/context-pack.js +331 -0
  252. package/dist/repertoire/coding/feedback.js +197 -30
  253. package/dist/repertoire/coding/manager.js +163 -10
  254. package/dist/repertoire/coding/spawner.js +55 -9
  255. package/dist/repertoire/coding/tools.js +170 -7
  256. package/dist/repertoire/commerce-errors.js +109 -0
  257. package/dist/repertoire/commerce-self-test.js +156 -0
  258. package/dist/repertoire/credential-access.js +178 -0
  259. package/dist/repertoire/duffel-client.js +185 -0
  260. package/dist/repertoire/github-client.js +14 -55
  261. package/dist/repertoire/graph-client.js +11 -52
  262. package/dist/repertoire/guardrails.js +396 -0
  263. package/dist/repertoire/mcp-client.js +295 -0
  264. package/dist/repertoire/mcp-manager.js +362 -0
  265. package/dist/repertoire/mcp-tools.js +63 -0
  266. package/dist/repertoire/shell-sessions.js +133 -0
  267. package/dist/repertoire/skills.js +18 -4
  268. package/dist/repertoire/stripe-client.js +131 -0
  269. package/dist/repertoire/tasks/board.js +31 -5
  270. package/dist/repertoire/tasks/fix.js +182 -0
  271. package/dist/repertoire/tasks/index.js +16 -4
  272. package/dist/repertoire/tasks/lifecycle.js +2 -2
  273. package/dist/repertoire/tasks/parser.js +3 -2
  274. package/dist/repertoire/tasks/scanner.js +194 -37
  275. package/dist/repertoire/tasks/transitions.js +16 -78
  276. package/dist/repertoire/tool-results.js +29 -0
  277. package/dist/repertoire/tools-attachments.js +317 -0
  278. package/dist/repertoire/tools-awaiting.js +365 -0
  279. package/dist/repertoire/tools-base.js +55 -1082
  280. package/dist/repertoire/tools-bluebubbles.js +1 -0
  281. package/dist/repertoire/tools-bridge.js +141 -0
  282. package/dist/repertoire/tools-bundle.js +984 -0
  283. package/dist/repertoire/tools-config.js +185 -0
  284. package/dist/repertoire/tools-continuity.js +248 -0
  285. package/dist/repertoire/tools-credential.js +381 -0
  286. package/dist/repertoire/tools-files.js +342 -0
  287. package/dist/repertoire/tools-flight.js +224 -0
  288. package/dist/repertoire/tools-flow.js +119 -0
  289. package/dist/repertoire/tools-github.js +1 -7
  290. package/dist/repertoire/tools-mail.js +1916 -0
  291. package/dist/repertoire/tools-notes.js +421 -0
  292. package/dist/repertoire/tools-obligations.js +142 -0
  293. package/dist/repertoire/tools-record.js +463 -0
  294. package/dist/repertoire/tools-runtime.js +148 -0
  295. package/dist/repertoire/tools-session.js +781 -0
  296. package/dist/repertoire/tools-shell.js +120 -0
  297. package/dist/repertoire/tools-stripe.js +180 -0
  298. package/dist/repertoire/tools-surface.js +345 -0
  299. package/dist/repertoire/tools-teams.js +9 -39
  300. package/dist/repertoire/tools-travel.js +125 -0
  301. package/dist/repertoire/tools-trip.js +604 -0
  302. package/dist/repertoire/tools-user-profile.js +144 -0
  303. package/dist/repertoire/tools-vault.js +40 -0
  304. package/dist/repertoire/tools-voice.js +144 -0
  305. package/dist/repertoire/tools.js +115 -103
  306. package/dist/repertoire/travel-api-client.js +360 -0
  307. package/dist/repertoire/user-profile.js +131 -0
  308. package/dist/repertoire/vault-setup.js +246 -0
  309. package/dist/repertoire/vault-unlock.js +594 -0
  310. package/dist/scripts/claude-code-hook.js +41 -0
  311. package/dist/scripts/claude-code-stop-hook.js +47 -0
  312. package/dist/senses/attention-queue.js +116 -0
  313. package/dist/senses/await-turn-message.js +58 -0
  314. package/dist/senses/bluebubbles/active-turns.js +216 -0
  315. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  316. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  317. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
  318. package/dist/senses/bluebubbles/entry.js +77 -0
  319. package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
  320. package/dist/senses/bluebubbles/index.js +2613 -0
  321. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -71
  322. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  323. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
  324. package/dist/senses/bluebubbles/processed-log.js +133 -0
  325. package/dist/senses/bluebubbles/replay.js +137 -0
  326. package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
  327. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  328. package/dist/senses/bluebubbles-meta-guard.js +40 -0
  329. package/dist/senses/cli/bracketed-paste.js +82 -0
  330. package/dist/senses/cli/image-paste.js +287 -0
  331. package/dist/senses/cli/image-ref-navigation.js +75 -0
  332. package/dist/senses/cli/ink-app.js +156 -0
  333. package/dist/senses/cli/inline-diff.js +64 -0
  334. package/dist/senses/cli/input-keys.js +174 -0
  335. package/dist/senses/cli/kill-ring.js +86 -0
  336. package/dist/senses/cli/message-list.js +51 -0
  337. package/dist/senses/cli/ouro-tui.js +607 -0
  338. package/dist/senses/cli/spinner-imperative.js +135 -0
  339. package/dist/senses/cli/spinner.js +101 -0
  340. package/dist/senses/cli/status-line.js +60 -0
  341. package/dist/senses/cli/streaming-markdown.js +526 -0
  342. package/dist/senses/cli/tool-display.js +85 -0
  343. package/dist/senses/cli/tool-render.js +85 -0
  344. package/dist/senses/cli/tui-store.js +240 -0
  345. package/dist/senses/cli/virtual-list.js +35 -0
  346. package/dist/senses/cli-entry.js +60 -8
  347. package/dist/senses/cli-layout.js +100 -0
  348. package/dist/senses/cli.js +516 -204
  349. package/dist/senses/commands.js +66 -3
  350. package/dist/senses/habit-turn-message.js +108 -0
  351. package/dist/senses/inner-dialog-worker.js +185 -21
  352. package/dist/senses/inner-dialog.js +469 -39
  353. package/dist/senses/mail-entry.js +66 -0
  354. package/dist/senses/mail.js +379 -0
  355. package/dist/senses/pipeline.js +654 -181
  356. package/dist/senses/proactive-content-guard.js +51 -0
  357. package/dist/senses/shared-turn.js +392 -0
  358. package/dist/senses/surface-tool.js +70 -0
  359. package/dist/senses/teams-entry.js +60 -8
  360. package/dist/senses/teams.js +387 -98
  361. package/dist/senses/trust-gate.js +100 -5
  362. package/dist/senses/voice/audio-playback.js +237 -0
  363. package/dist/senses/voice/audio-routing.js +119 -0
  364. package/dist/senses/voice/elevenlabs.js +202 -0
  365. package/dist/senses/voice/floor-control.js +431 -0
  366. package/dist/senses/voice/floor-controller.js +115 -0
  367. package/dist/senses/voice/golden-path.js +116 -0
  368. package/dist/senses/voice/index.js +29 -0
  369. package/dist/senses/voice/meeting.js +113 -0
  370. package/dist/senses/voice/outbound.js +190 -0
  371. package/dist/senses/voice/phone.js +33 -0
  372. package/dist/senses/voice/playback.js +139 -0
  373. package/dist/senses/voice/realtime-eval.js +496 -0
  374. package/dist/senses/voice/realtime-trace.js +531 -0
  375. package/dist/senses/voice/transcript.js +70 -0
  376. package/dist/senses/voice/turn.js +191 -0
  377. package/dist/senses/voice/twilio-phone-runtime.js +807 -0
  378. package/dist/senses/voice/twilio-phone.js +5079 -0
  379. package/dist/senses/voice/types.js +2 -0
  380. package/dist/senses/voice/whisper.js +161 -0
  381. package/dist/senses/voice-entry.js +81 -0
  382. package/dist/senses/voice-realtime-eval-command.js +99 -0
  383. package/dist/senses/voice-realtime-eval-entry.js +21 -0
  384. package/dist/senses/voice-twilio-entry.js +87 -0
  385. package/dist/trips/core.js +138 -0
  386. package/dist/trips/store.js +265 -0
  387. package/package.json +41 -7
  388. package/skills/agent-commerce.md +106 -0
  389. package/skills/browser-navigation.md +117 -0
  390. package/skills/commerce-setup-guide.md +116 -0
  391. package/skills/commerce-setup.md +84 -0
  392. package/skills/configure-dev-tools.md +99 -0
  393. package/skills/travel-planning.md +138 -0
  394. package/dist/heart/daemon/auth-flow.js +0 -351
  395. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  396. package/dist/heart/safe-workspace.js +0 -228
  397. package/dist/heart/session-recall.js +0 -116
  398. package/dist/mind/associative-recall.js +0 -209
  399. package/dist/senses/bluebubbles-entry.js +0 -13
  400. package/dist/senses/bluebubbles.js +0 -1177
  401. package/dist/senses/debug-activity.js +0 -148
  402. package/subagents/README.md +0 -7
  403. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  404. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  405. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  406. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  407. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  408. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  409. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  410. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  411. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  412. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  413. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  414. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  415. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  416. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  417. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  418. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -38,9 +38,11 @@ const fs = __importStar(require("fs"));
38
38
  const os = __importStar(require("os"));
39
39
  const path = __importStar(require("path"));
40
40
  const runtime_1 = require("../../nerves/runtime");
41
- const bluebubbles_runtime_state_1 = require("../../senses/bluebubbles-runtime-state");
42
41
  const identity_1 = require("../identity");
42
+ const runtime_credentials_1 = require("../runtime-credentials");
43
+ const provider_credentials_1 = require("../provider-credentials");
43
44
  const sense_truth_1 = require("../sense-truth");
45
+ const machine_identity_1 = require("../machine-identity");
44
46
  const process_manager_1 = require("./process-manager");
45
47
  const DEFAULT_TEAMS_PORT = 3978;
46
48
  const DEFAULT_BLUEBUBBLES_PORT = 18790;
@@ -51,6 +53,8 @@ function defaultSenses() {
51
53
  cli: { ...identity_1.DEFAULT_AGENT_SENSES.cli },
52
54
  teams: { ...identity_1.DEFAULT_AGENT_SENSES.teams },
53
55
  bluebubbles: { ...identity_1.DEFAULT_AGENT_SENSES.bluebubbles },
56
+ mail: { ...identity_1.DEFAULT_AGENT_SENSES.mail },
57
+ voice: { ...identity_1.DEFAULT_AGENT_SENSES.voice },
54
58
  };
55
59
  }
56
60
  function readAgentSenses(agentJsonPath) {
@@ -76,7 +80,7 @@ function readAgentSenses(agentJsonPath) {
76
80
  if (!rawSenses || typeof rawSenses !== "object" || Array.isArray(rawSenses)) {
77
81
  return defaults;
78
82
  }
79
- for (const sense of ["cli", "teams", "bluebubbles"]) {
83
+ for (const sense of ["cli", "teams", "bluebubbles", "mail", "voice"]) {
80
84
  const rawSense = rawSenses[sense];
81
85
  if (!rawSense || typeof rawSense !== "object" || Array.isArray(rawSense)) {
82
86
  continue;
@@ -88,41 +92,50 @@ function readAgentSenses(agentJsonPath) {
88
92
  }
89
93
  return defaults;
90
94
  }
91
- function readSecretsPayload(secretsPath) {
92
- try {
93
- const raw = fs.readFileSync(secretsPath, "utf-8");
94
- const parsed = JSON.parse(raw);
95
- if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
96
- return { payload: {}, error: "invalid secrets.json object" };
97
- }
98
- return { payload: parsed, error: null };
99
- }
100
- catch (error) {
101
- return {
102
- payload: {},
103
- error: error instanceof Error ? error.message : String(error),
104
- };
105
- }
106
- }
107
95
  function textField(record, key) {
108
96
  const value = record?.[key];
109
97
  return typeof value === "string" ? value.trim() : "";
110
98
  }
99
+ function booleanField(record, key) {
100
+ return record?.[key] === true;
101
+ }
111
102
  function numberField(record, key, fallback) {
112
103
  const value = record?.[key];
113
104
  return typeof value === "number" && Number.isFinite(value) ? value : fallback;
114
105
  }
115
- function senseFactsFromSecrets(agent, senses, secretsPath) {
106
+ function compactRuntimeConfigError(agent, error) {
107
+ const compact = error.replace(/\s+/g, " ").trim();
108
+ if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
109
+ return `vault locked; run 'ouro vault unlock --agent ${agent}' if you have the saved secret, or 'ouro vault replace --agent ${agent}' if none was saved`;
110
+ }
111
+ return compact || "unavailable";
112
+ }
113
+ function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
114
+ if (runtimeConfig.ok)
115
+ return "";
116
+ const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
117
+ if (runtimeConfig.reason === "missing")
118
+ return `missing vault ${itemName} (${agent})`;
119
+ return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
120
+ }
121
+ function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
116
122
  const base = {
117
123
  cli: { configured: true, detail: "local interactive terminal" },
118
124
  teams: { configured: false, detail: "not enabled in agent.json" },
119
125
  bluebubbles: { configured: false, detail: "not enabled in agent.json" },
126
+ mail: { configured: false, detail: "not enabled in agent.json" },
127
+ voice: { configured: false, detail: "not enabled in agent.json" },
120
128
  };
121
- const { payload, error } = readSecretsPayload(secretsPath);
129
+ const payload = runtimeConfig.ok ? runtimeConfig.config : {};
130
+ const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
122
131
  const teams = payload.teams;
123
132
  const teamsChannel = payload.teamsChannel;
124
- const bluebubbles = payload.bluebubbles;
125
- const bluebubblesChannel = payload.bluebubblesChannel;
133
+ const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
134
+ const bluebubbles = machinePayload.bluebubbles;
135
+ const bluebubblesChannel = machinePayload.bluebubblesChannel;
136
+ const mailroom = payload.mailroom;
137
+ const integrations = payload.integrations;
138
+ const voice = machinePayload.voice;
126
139
  if (senses.teams.enabled) {
127
140
  const missing = [];
128
141
  if (!textField(teams, "clientId"))
@@ -138,9 +151,9 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
138
151
  }
139
152
  : {
140
153
  configured: false,
141
- detail: error && !fs.existsSync(secretsPath)
142
- ? `missing secrets.json (${agent})`
143
- : `missing ${missing.join("/")}`,
154
+ detail: runtimeConfig.ok
155
+ ? `missing ${missing.join("/")}`
156
+ : unavailableDetail,
144
157
  };
145
158
  }
146
159
  if (senses.bluebubbles.enabled) {
@@ -156,19 +169,128 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
156
169
  }
157
170
  : {
158
171
  configured: false,
159
- detail: error && !fs.existsSync(secretsPath)
160
- ? `missing secrets.json (${agent})`
161
- : `missing ${missing.join("/")}`,
172
+ optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
173
+ detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
174
+ ? "not attached on this machine"
175
+ : machineRuntimeConfig.ok
176
+ ? `missing ${missing.join("/")}`
177
+ : runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
178
+ };
179
+ }
180
+ if (senses.mail.enabled) {
181
+ const privateKeys = mailroom?.privateKeys;
182
+ const hasPrivateKeys = !!privateKeys && typeof privateKeys === "object" && !Array.isArray(privateKeys) && Object.values(privateKeys).some((value) => typeof value === "string" && value.trim().length > 0);
183
+ const mailboxAddress = textField(mailroom, "mailboxAddress");
184
+ const missing = [];
185
+ if (!mailboxAddress)
186
+ missing.push("mailroom.mailboxAddress");
187
+ if (!hasPrivateKeys)
188
+ missing.push("mailroom.privateKeys");
189
+ base.mail = missing.length === 0
190
+ ? { configured: true, detail: mailboxAddress }
191
+ : {
192
+ configured: false,
193
+ detail: runtimeConfig.ok
194
+ ? `missing ${missing.join("/")}`
195
+ : unavailableDetail,
196
+ };
197
+ }
198
+ if (senses.voice.enabled) {
199
+ const portableVoice = payload.voice;
200
+ const conversationEngine = (textField(voice, "twilioConversationEngine")
201
+ || textField(voice, "conversationEngine")
202
+ || textField(portableVoice, "twilioConversationEngine")
203
+ || textField(portableVoice, "conversationEngine")
204
+ || "cascade").toLowerCase();
205
+ const twilioTransportMode = (textField(voice, "twilioTransportMode") || "record-play").toLowerCase();
206
+ const twilioPublicUrl = textField(voice, "twilioPublicUrl");
207
+ const hasOpenAIRealtimeKey = !!(textField(portableVoice, "openaiRealtimeApiKey")
208
+ || textField(integrations, "openaiApiKey")
209
+ || textField(integrations, "openaiEmbeddingsApiKey"));
210
+ const missing = [];
211
+ /* v8 ignore start -- voice setup missing-field matrix is enforced by the voice runtime resolver; sense-manager only renders human-readable readiness facts @preserve */
212
+ if (conversationEngine === "openai-realtime" || conversationEngine === "openai-sip") {
213
+ if (conversationEngine === "openai-realtime" && twilioTransportMode !== "media-stream") {
214
+ missing.push("voice.twilioTransportMode=media-stream");
215
+ }
216
+ if (!hasOpenAIRealtimeKey) {
217
+ missing.push("voice.openaiRealtimeApiKey");
218
+ }
219
+ if (conversationEngine === "openai-sip") {
220
+ if (!textField(portableVoice, "openaiSipProjectId") && !textField(voice, "openaiSipProjectId")) {
221
+ missing.push("voice.openaiSipProjectId");
222
+ }
223
+ const allowUnsignedWebhooks = booleanField(portableVoice, "openaiSipAllowUnsignedWebhooks")
224
+ || booleanField(voice, "openaiSipAllowUnsignedWebhooks");
225
+ if (!allowUnsignedWebhooks && !textField(portableVoice, "openaiSipWebhookSecret") && !textField(voice, "openaiSipWebhookSecret")) {
226
+ missing.push("voice.openaiSipWebhookSecret");
227
+ }
228
+ }
229
+ }
230
+ else {
231
+ if (!textField(integrations, "elevenLabsApiKey"))
232
+ missing.push("integrations.elevenLabsApiKey");
233
+ if (!textField(integrations, "elevenLabsVoiceId") && !textField(portableVoice, "elevenLabsVoiceId")) {
234
+ missing.push("integrations.elevenLabsVoiceId");
235
+ }
236
+ if (!textField(voice, "whisperCliPath"))
237
+ missing.push("voice.whisperCliPath");
238
+ if (!textField(voice, "whisperModelPath"))
239
+ missing.push("voice.whisperModelPath");
240
+ }
241
+ /* v8 ignore stop */
242
+ base.voice = missing.length === 0
243
+ ? {
244
+ configured: true,
245
+ /* v8 ignore start -- voice detail copy mirrors runtime resolver modes; resolver tests own the matrix @preserve */
246
+ detail: conversationEngine === "openai-sip"
247
+ ? twilioPublicUrl
248
+ ? "OpenAI Realtime SIP speech-to-speech; Twilio phone transport attached"
249
+ : "OpenAI Realtime SIP speech-to-speech"
250
+ : conversationEngine === "openai-realtime"
251
+ ? twilioPublicUrl
252
+ ? "OpenAI Realtime speech-to-speech; Twilio phone transport attached"
253
+ : "OpenAI Realtime speech-to-speech"
254
+ : twilioPublicUrl
255
+ ? "local Whisper.cpp STT + ElevenLabs TTS; Twilio phone transport attached"
256
+ : "local Whisper.cpp STT + ElevenLabs TTS",
257
+ /* v8 ignore stop */
258
+ }
259
+ : {
260
+ configured: false,
261
+ optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
262
+ detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
263
+ ? "not attached on this machine"
264
+ : runtimeConfig.ok && machineRuntimeConfig.ok
265
+ ? `missing ${missing.join("/")}`
266
+ : !runtimeConfig.ok
267
+ ? unavailableDetail
268
+ : runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
162
269
  };
163
270
  }
164
271
  return base;
165
272
  }
273
+ function senseRepairHint(agent, sense) {
274
+ if (sense === "teams") {
275
+ return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
276
+ }
277
+ if (sense === "mail") {
278
+ return `Agent-runnable: provision Mailroom access with 'ouro connect mail --agent ${agent}', then restart with 'ouro up'.`;
279
+ }
280
+ if (sense === "voice") {
281
+ return `Agent-runnable: run 'ouro connect voice --agent ${agent}' for config guidance; use voice.twilioConversationEngine=openai-sip with voice.openaiRealtimeApiKey, voice.openaiSipProjectId, and voice.openaiSipWebhookSecret for preferred SIP phone voice; use openai-realtime for Media Streams fallback, or save ElevenLabs and local Whisper.cpp settings for cascade fallback; then run 'ouro up' again.`;
282
+ }
283
+ return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
284
+ }
285
+ function currentMachineId() {
286
+ return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
287
+ }
166
288
  function parseSenseSnapshotName(name) {
167
289
  const parts = name.split(":");
168
290
  if (parts.length !== 2)
169
291
  return null;
170
292
  const [agent, sense] = parts;
171
- if (sense !== "teams" && sense !== "bluebubbles")
293
+ if (sense !== "teams" && sense !== "bluebubbles" && sense !== "mail" && sense !== "voice")
172
294
  return null;
173
295
  return { agent, sense };
174
296
  }
@@ -177,6 +299,34 @@ function runtimeInfoFor(status) {
177
299
  return { runtime: "running" };
178
300
  return { runtime: "error" };
179
301
  }
302
+ function managedSenseEntry(sense) {
303
+ if (sense === "teams")
304
+ return "senses/teams-entry.js";
305
+ if (sense === "bluebubbles")
306
+ return "senses/bluebubbles/entry.js";
307
+ if (sense === "voice")
308
+ return "senses/voice-entry.js";
309
+ return "senses/mail-entry.js";
310
+ }
311
+ function runtimeCredentialBootstrapFor(agent, sense) {
312
+ const runtime = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
313
+ const machineId = sense === "bluebubbles" || sense === "voice" ? currentMachineId() : undefined;
314
+ const machine = sense === "bluebubbles" || sense === "voice" ? (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent) : null;
315
+ const providerPool = (0, provider_credentials_1.readProviderCredentialPool)(agent);
316
+ const providerCredentialRecords = providerPool.ok
317
+ ? Object.values(providerPool.pool.providers).filter((record) => !!record)
318
+ : [];
319
+ const bootstrap = {
320
+ agentName: agent,
321
+ runtimeConfig: runtime.ok ? runtime.config : undefined,
322
+ machineRuntimeConfig: machine?.ok ? machine.config : undefined,
323
+ machineId,
324
+ providerCredentialRecords: providerCredentialRecords.length > 0 ? providerCredentialRecords : undefined,
325
+ };
326
+ if (!bootstrap.runtimeConfig && !bootstrap.machineRuntimeConfig && !bootstrap.providerCredentialRecords)
327
+ return null;
328
+ return bootstrap;
329
+ }
180
330
  function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
181
331
  if (!lastCheckedAt) {
182
332
  return false;
@@ -187,53 +337,186 @@ function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
187
337
  }
188
338
  return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
189
339
  }
340
+ function readBlueBubblesRuntimeJson(runtimePath) {
341
+ try {
342
+ const raw = fs.readFileSync(runtimePath, "utf-8");
343
+ const parsed = JSON.parse(raw);
344
+ /* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
345
+ return {
346
+ upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
347
+ ? parsed.upstreamStatus
348
+ : "unknown",
349
+ detail: typeof parsed.detail === "string" && parsed.detail.trim()
350
+ ? parsed.detail
351
+ : "startup health probe pending",
352
+ lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
353
+ proofMethod: typeof parsed.proofMethod === "string" && parsed.proofMethod.trim()
354
+ ? parsed.proofMethod
355
+ : undefined,
356
+ pendingRecoveryCount: typeof parsed.pendingRecoveryCount === "number" && Number.isFinite(parsed.pendingRecoveryCount)
357
+ ? parsed.pendingRecoveryCount
358
+ : 0,
359
+ failedRecoveryCount: typeof parsed.failedRecoveryCount === "number" && Number.isFinite(parsed.failedRecoveryCount)
360
+ ? parsed.failedRecoveryCount
361
+ : 0,
362
+ oldestPendingRecoveryAt: typeof parsed.oldestPendingRecoveryAt === "string" ? parsed.oldestPendingRecoveryAt : undefined,
363
+ oldestPendingRecoveryAgeMs: typeof parsed.oldestPendingRecoveryAgeMs === "number" && Number.isFinite(parsed.oldestPendingRecoveryAgeMs)
364
+ ? parsed.oldestPendingRecoveryAgeMs
365
+ : undefined,
366
+ activeTurnCount: typeof parsed.activeTurnCount === "number" && Number.isFinite(parsed.activeTurnCount)
367
+ ? parsed.activeTurnCount
368
+ : undefined,
369
+ stalledTurnCount: typeof parsed.stalledTurnCount === "number" && Number.isFinite(parsed.stalledTurnCount)
370
+ ? parsed.stalledTurnCount
371
+ : undefined,
372
+ oldestActiveTurnStartedAt: typeof parsed.oldestActiveTurnStartedAt === "string" ? parsed.oldestActiveTurnStartedAt : undefined,
373
+ oldestActiveTurnAgeMs: typeof parsed.oldestActiveTurnAgeMs === "number" && Number.isFinite(parsed.oldestActiveTurnAgeMs)
374
+ ? parsed.oldestActiveTurnAgeMs
375
+ : undefined,
376
+ };
377
+ /* v8 ignore stop */
378
+ /* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
379
+ }
380
+ catch {
381
+ return { upstreamStatus: "unknown", detail: "startup health probe pending", pendingRecoveryCount: 0, failedRecoveryCount: 0 };
382
+ }
383
+ /* v8 ignore stop */
384
+ }
190
385
  function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
191
386
  const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
192
387
  const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
193
388
  if (!fs.existsSync(runtimePath)) {
194
389
  return { runtime: snapshot?.runtime };
195
390
  }
196
- const state = (0, bluebubbles_runtime_state_1.readBlueBubblesRuntimeState)(agent, agentRoot);
391
+ const state = readBlueBubblesRuntimeJson(runtimePath);
392
+ const checkedAtMs = state.lastCheckedAt ? Date.parse(state.lastCheckedAt) : Number.NaN;
393
+ const proofFacts = {
394
+ proofMethod: state.proofMethod ?? "bluebubbles.checkHealth",
395
+ lastProofAt: state.lastCheckedAt,
396
+ proofAgeMs: Number.isFinite(checkedAtMs) ? Math.max(0, Date.now() - checkedAtMs) : undefined,
397
+ pendingRecoveryCount: state.pendingRecoveryCount,
398
+ failedRecoveryCount: state.failedRecoveryCount,
399
+ oldestPendingRecoveryAt: state.oldestPendingRecoveryAt,
400
+ oldestPendingRecoveryAgeMs: state.oldestPendingRecoveryAgeMs,
401
+ activeTurnCount: state.activeTurnCount,
402
+ stalledTurnCount: state.stalledTurnCount,
403
+ oldestActiveTurnStartedAt: state.oldestActiveTurnStartedAt,
404
+ oldestActiveTurnAgeMs: state.oldestActiveTurnAgeMs,
405
+ };
197
406
  if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
198
- return { runtime: snapshot?.runtime };
407
+ return {
408
+ runtime: snapshot?.runtime,
409
+ lastFailure: state.lastCheckedAt ? "BlueBubbles proof is stale" : undefined,
410
+ failureLayer: state.lastCheckedAt ? "proof_freshness" : undefined,
411
+ };
412
+ }
413
+ if (snapshot?.runtime !== "running") {
414
+ return {
415
+ runtime: "error",
416
+ detail: "BlueBubbles listener is not running",
417
+ ...proofFacts,
418
+ lastFailure: "listener process is not running",
419
+ failureLayer: "listener",
420
+ recoveryAction: "daemon health monitor will restart the BlueBubbles listener when its probe fails",
421
+ };
199
422
  }
200
423
  if (state.upstreamStatus === "error") {
201
424
  return {
202
425
  runtime: "error",
203
426
  detail: state.detail,
427
+ ...proofFacts,
428
+ lastFailure: state.detail,
429
+ failureLayer: "upstream",
430
+ recoveryAction: "verify BlueBubbles server/app auth and local machine attachment, then retry the listener",
431
+ };
432
+ }
433
+ if (state.pendingRecoveryCount > 0) {
434
+ return {
435
+ runtime: "error",
436
+ detail: state.detail,
437
+ ...proofFacts,
438
+ lastFailure: state.detail,
439
+ failureLayer: "recovery_queue",
440
+ recoveryAction: "queued recovery will retry; inspect BlueBubbles inbound/recovery sidecar logs if age keeps growing",
441
+ };
442
+ }
443
+ if ((state.stalledTurnCount ?? 0) > 0) {
444
+ return {
445
+ runtime: "error",
446
+ detail: state.detail,
447
+ ...proofFacts,
448
+ lastFailure: state.detail,
449
+ failureLayer: "live_turn_stall",
450
+ recoveryAction: "live iMessage turn timeout/watchdog will release the lane and recovery will retry captured messages",
204
451
  };
205
452
  }
206
453
  if (state.upstreamStatus === "ok") {
207
- return { runtime: "running" };
454
+ return {
455
+ runtime: "running",
456
+ ...proofFacts,
457
+ ...(state.failedRecoveryCount > 0 ? { detail: state.detail } : {}),
458
+ ...(state.failedRecoveryCount > 0 ? {
459
+ lastFailure: state.detail,
460
+ failureLayer: "recovery_quarantine",
461
+ recoveryAction: "inspect quarantined BlueBubbles recovery failures; live transport remains reachable",
462
+ } : {}),
463
+ };
208
464
  }
209
- return { runtime: snapshot?.runtime };
465
+ return { runtime: snapshot?.runtime, ...proofFacts };
210
466
  }
211
467
  class DaemonSenseManager {
212
468
  processManager;
213
469
  contexts;
470
+ pendingConfigRefreshes = new Set();
214
471
  bundlesRoot;
215
472
  constructor(options) {
216
473
  const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
217
- const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
218
474
  this.bundlesRoot = bundlesRoot;
219
475
  this.contexts = new Map(options.agents.map((agent) => {
220
476
  const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
221
- const facts = senseFactsFromSecrets(agent, senses, path.join(secretsRoot, agent, "secrets.json"));
477
+ const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
222
478
  return [agent, { senses, facts }];
223
479
  }));
224
480
  const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
225
- return ["teams", "bluebubbles"]
226
- .filter((sense) => context.senses[sense].enabled && context.facts[sense].configured)
481
+ return ["teams", "bluebubbles", "mail", "voice"]
482
+ .filter((sense) => context.senses[sense].enabled)
227
483
  .map((sense) => ({
228
484
  name: `${agent}:${sense}`,
229
485
  agentArg: agent,
230
- entry: sense === "teams" ? "senses/teams-entry.js" : "senses/bluebubbles-entry.js",
486
+ entry: managedSenseEntry(sense),
231
487
  channel: sense,
232
488
  autoStart: true,
489
+ getRuntimeCredentialBootstrap: () => runtimeCredentialBootstrapFor(agent, sense),
233
490
  }));
234
491
  });
235
492
  this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
236
493
  agents: managedSenseAgents,
494
+ configCheck: async (name) => {
495
+ const parsed = parseSenseSnapshotName(name);
496
+ if (!parsed)
497
+ return { ok: true };
498
+ const context = this.contexts.get(parsed.agent);
499
+ if (!context)
500
+ return { ok: true };
501
+ context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(parsed.agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent));
502
+ const fact = context.facts[parsed.sense];
503
+ if (fact.configured)
504
+ return { ok: true };
505
+ this.scheduleSenseConfigRefresh(name, parsed);
506
+ if (fact.optional) {
507
+ return {
508
+ ok: false,
509
+ skip: true,
510
+ error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
511
+ };
512
+ }
513
+ return {
514
+ ok: false,
515
+ skip: true,
516
+ error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
517
+ fix: senseRepairHint(parsed.agent, parsed.sense),
518
+ };
519
+ },
237
520
  });
238
521
  (0, runtime_1.emitNervesEvent)({
239
522
  component: "channels",
@@ -245,12 +528,157 @@ class DaemonSenseManager {
245
528
  },
246
529
  });
247
530
  }
531
+ scheduleSenseConfigRefresh(name, parsed) {
532
+ if (this.pendingConfigRefreshes.has(name))
533
+ return;
534
+ this.pendingConfigRefreshes.add(name);
535
+ void this.refreshSenseConfigAndRetry(name, parsed);
536
+ }
537
+ async refreshSenseConfigAndRetry(name, parsed) {
538
+ try {
539
+ const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
540
+ const machineRefreshed = parsed.sense === "bluebubbles" || parsed.sense === "voice"
541
+ ? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
542
+ : (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
543
+ const context = this.contexts.get(parsed.agent);
544
+ /* v8 ignore next -- defensive: config refreshes are only scheduled for known agent contexts @preserve */
545
+ if (!context)
546
+ return;
547
+ context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
548
+ if (!context.facts[parsed.sense].configured)
549
+ return;
550
+ setTimeout(() => {
551
+ void this.processManager.startAgent?.(name).catch((error) => {
552
+ (0, runtime_1.emitNervesEvent)({
553
+ level: "error",
554
+ component: "channels",
555
+ event: "channel.daemon_sense_autostart_error",
556
+ message: "sense autostart failed",
557
+ /* v8 ignore next -- defensive: process manager rejects with Error instances in normal use @preserve */
558
+ meta: { error: error instanceof Error ? error.message : String(error) },
559
+ });
560
+ });
561
+ }, 0);
562
+ }
563
+ catch (error) {
564
+ (0, runtime_1.emitNervesEvent)({
565
+ level: "error",
566
+ component: "channels",
567
+ event: "channel.daemon_sense_autostart_error",
568
+ message: "sense config refresh failed",
569
+ /* v8 ignore next -- defensive: runtime credential refresh rejects with Error instances in normal use @preserve */
570
+ meta: { error: error instanceof Error ? error.message : String(error) },
571
+ });
572
+ }
573
+ finally {
574
+ this.pendingConfigRefreshes.delete(name);
575
+ }
576
+ }
577
+ async refreshEnabledSenseConfigs() {
578
+ const refreshes = [...this.contexts.entries()].map(async ([agent, context]) => {
579
+ const enabledManagedSenses = ["teams", "bluebubbles", "mail", "voice"]
580
+ .filter((sense) => context.senses[sense].enabled);
581
+ /* v8 ignore next -- periodic refresh work only exists when a managed background sense is enabled @preserve */
582
+ if (enabledManagedSenses.length === 0)
583
+ return;
584
+ /* v8 ignore start -- periodic freshness refresh uses the same runtime readers covered by startup integration tests @preserve */
585
+ const runtimeConfig = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(agent, { preserveCachedOnFailure: true });
586
+ const needsMachineConfig = enabledManagedSenses.some((sense) => sense === "bluebubbles" || sense === "voice");
587
+ const machineRuntimeConfig = needsMachineConfig
588
+ ? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(agent, currentMachineId(), { preserveCachedOnFailure: true })
589
+ : (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
590
+ /* v8 ignore stop */
591
+ context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
592
+ });
593
+ await Promise.all(refreshes);
594
+ }
248
595
  async startAutoStartSenses() {
596
+ await this.refreshEnabledSenseConfigs();
249
597
  await this.processManager.startAutoStartAgents();
250
598
  }
599
+ triggerAutoStartSenses() {
600
+ if (this.processManager.triggerAutoStartAgents) {
601
+ this.processManager.triggerAutoStartAgents();
602
+ return;
603
+ }
604
+ void this.processManager.startAutoStartAgents().catch((error) => {
605
+ (0, runtime_1.emitNervesEvent)({
606
+ level: "error",
607
+ component: "channels",
608
+ event: "channel.daemon_sense_autostart_error",
609
+ message: "sense autostart failed",
610
+ meta: { error: error instanceof Error ? error.message : String(error) },
611
+ });
612
+ });
613
+ }
251
614
  async stopAll() {
252
615
  await this.processManager.stopAll();
253
616
  }
617
+ /* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
618
+ listManagedPids() {
619
+ return this.processManager.listAgentSnapshots()
620
+ .map((s) => s.pid)
621
+ .filter((pid) => pid !== null && pid !== undefined);
622
+ }
623
+ /* v8 ignore stop */
624
+ listHealthProbes() {
625
+ const probes = [];
626
+ for (const [agent, context] of this.contexts.entries()) {
627
+ const runtimeConfig = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
628
+ const machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
629
+ context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
630
+ if (!context.senses.bluebubbles.enabled || !context.facts.bluebubbles.configured || !machineRuntimeConfig.ok) {
631
+ continue;
632
+ }
633
+ // DELIBERATELY no HTTP health probe for BlueBubbles.
634
+ //
635
+ // We used to register `createHttpHealthProbe(...)` here, which GETs the
636
+ // sense's /health endpoint every ~60s with a 5s timeout. On 2026-05-11
637
+ // that caused a death spiral:
638
+ // 1. Sense gets busy with real work (e.g. VLM image describe → 20+s)
639
+ // 2. /health probe times out at 5s
640
+ // 3. Daemon declares the sense "critical" → SIGTERMs it mid-work
641
+ // 4. Sense respawns, recovery loop replays the same inbound message
642
+ // into the agent's BB session (visible side-effect — slugger saw
643
+ // the same user text injected 76 times)
644
+ // 5. New sense hits the same VLM call, gets killed at 5s, repeat
645
+ //
646
+ // The probe was redundant supervision: dead processes are already
647
+ // recaptured by `processManager`'s child-process exit handler. The
648
+ // probe specifically caught "alive but hung" cases — but the cost
649
+ // (killing genuinely-busy processes and replaying messages) far
650
+ // outweighed the benefit. For "alive but hung" detection we now
651
+ // rely on the agent's own awareness: BB sense's runtime.json carries
652
+ // pendingRecoveryCount + lastRecoveredAt, surfaced in the agent
653
+ // prompt. If recovery has been wedged for too long, the agent can
654
+ // call `restart_runtime` itself (see alpha.598 / PR #723).
655
+ //
656
+ // The respawn-loop guard in processManager is the backstop: even if
657
+ // something else triggers a tight respawn cycle for any reason, the
658
+ // guard fires and refuses further restarts after N attempts in M
659
+ // minutes, so we can never re-enter the 2026-05-11 spiral.
660
+ }
661
+ return probes;
662
+ }
663
+ async restartSense(managedName) {
664
+ if (this.processManager.restartAgent) {
665
+ await this.processManager.restartAgent(managedName);
666
+ return;
667
+ }
668
+ await this.processManager.startAgent?.(managedName);
669
+ }
670
+ async reviveSense(agent, sense) {
671
+ const parsed = parseSenseSnapshotName(`${agent}:${sense}`);
672
+ if (!parsed)
673
+ return null;
674
+ const context = this.contexts.get(parsed.agent);
675
+ if (!context || !context.senses[parsed.sense].enabled)
676
+ return null;
677
+ const managedName = `${parsed.agent}:${parsed.sense}`;
678
+ this.processManager.resetAgentFailureState?.(managedName);
679
+ await this.processManager.startAgent?.(managedName);
680
+ return this.listSenseRows().find((row) => row.agent === parsed.agent && row.sense === parsed.sense) ?? null;
681
+ }
254
682
  listSenseRows() {
255
683
  const runtime = new Map();
256
684
  for (const snapshot of this.processManager.listAgentSnapshots()) {
@@ -262,6 +690,7 @@ class DaemonSenseManager {
262
690
  runtime.set(parsed.agent, current);
263
691
  }
264
692
  const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
693
+ context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
265
694
  const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
266
695
  const runtimeInfo = {
267
696
  cli: { configured: true },
@@ -271,8 +700,18 @@ class DaemonSenseManager {
271
700
  },
272
701
  bluebubbles: {
273
702
  configured: context.facts.bluebubbles.configured,
703
+ optional: context.facts.bluebubbles.optional,
274
704
  ...blueBubblesRuntimeFacts,
275
705
  },
706
+ mail: {
707
+ configured: context.facts.mail.configured,
708
+ ...(runtime.get(agent)?.mail ?? {}),
709
+ },
710
+ voice: {
711
+ configured: context.facts.voice.configured,
712
+ optional: context.facts.voice.optional,
713
+ ...(runtime.get(agent)?.voice ?? {}),
714
+ },
276
715
  };
277
716
  const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
278
717
  return inventory.map((entry) => ({
@@ -287,6 +726,22 @@ class DaemonSenseManager {
287
726
  ?? context.facts[entry.sense].detail
288
727
  : context.facts[entry.sense].detail
289
728
  : "not enabled in agent.json",
729
+ ...(entry.sense === "bluebubbles" ? {
730
+ proofMethod: blueBubblesRuntimeFacts.proofMethod,
731
+ lastProofAt: blueBubblesRuntimeFacts.lastProofAt,
732
+ proofAgeMs: blueBubblesRuntimeFacts.proofAgeMs,
733
+ lastFailure: blueBubblesRuntimeFacts.lastFailure,
734
+ failureLayer: blueBubblesRuntimeFacts.failureLayer,
735
+ recoveryAction: blueBubblesRuntimeFacts.recoveryAction,
736
+ pendingRecoveryCount: blueBubblesRuntimeFacts.pendingRecoveryCount,
737
+ failedRecoveryCount: blueBubblesRuntimeFacts.failedRecoveryCount,
738
+ oldestPendingRecoveryAt: blueBubblesRuntimeFacts.oldestPendingRecoveryAt,
739
+ oldestPendingRecoveryAgeMs: blueBubblesRuntimeFacts.oldestPendingRecoveryAgeMs,
740
+ activeTurnCount: blueBubblesRuntimeFacts.activeTurnCount,
741
+ stalledTurnCount: blueBubblesRuntimeFacts.stalledTurnCount,
742
+ oldestActiveTurnStartedAt: blueBubblesRuntimeFacts.oldestActiveTurnStartedAt,
743
+ oldestActiveTurnAgeMs: blueBubblesRuntimeFacts.oldestActiveTurnAgeMs,
744
+ } : {}),
290
745
  }));
291
746
  });
292
747
  (0, runtime_1.emitNervesEvent)({