@ouro.bot/cli 0.1.0-alpha.66 → 0.1.0-alpha.660

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