@ouro.bot/cli 0.1.0-alpha.54 → 0.1.0-alpha.540

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 (384) hide show
  1. package/README.md +133 -19
  2. package/RepairGuide.ouro/agent.json +5 -0
  3. package/RepairGuide.ouro/psyche/IDENTITY.md +19 -0
  4. package/RepairGuide.ouro/psyche/SOUL.md +55 -0
  5. package/RepairGuide.ouro/skills/diagnose-bootstrap-drift.md +54 -0
  6. package/RepairGuide.ouro/skills/diagnose-broken-remote.md +63 -0
  7. package/RepairGuide.ouro/skills/diagnose-stacked-typed-issues.md +35 -0
  8. package/RepairGuide.ouro/skills/diagnose-sync-blocked.md +54 -0
  9. package/RepairGuide.ouro/skills/diagnose-vault-expired.md +60 -0
  10. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
  11. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
  12. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  13. package/changelog.json +3494 -0
  14. package/dist/arc/attention-types.js +8 -0
  15. package/dist/arc/cares.js +140 -0
  16. package/dist/arc/episodes.js +117 -0
  17. package/dist/arc/intentions.js +133 -0
  18. package/dist/arc/json-store.js +117 -0
  19. package/dist/arc/obligations.js +237 -0
  20. package/dist/arc/packets.js +193 -0
  21. package/dist/arc/presence.js +185 -0
  22. package/dist/arc/task-lifecycle.js +65 -0
  23. package/dist/heart/active-work.js +837 -26
  24. package/dist/heart/agent-entry.js +58 -3
  25. package/dist/heart/attachments/image-normalize.js +194 -0
  26. package/dist/heart/attachments/materialize.js +97 -0
  27. package/dist/heart/attachments/originals.js +88 -0
  28. package/dist/heart/attachments/render.js +29 -0
  29. package/dist/heart/attachments/sources/adapter.js +2 -0
  30. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  31. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  32. package/dist/heart/attachments/sources/index.js +16 -0
  33. package/dist/heart/attachments/store.js +103 -0
  34. package/dist/heart/attachments/types.js +93 -0
  35. package/dist/heart/auth/auth-flow.js +427 -0
  36. package/dist/heart/background-operations.js +281 -0
  37. package/dist/heart/bundle-state.js +168 -0
  38. package/dist/heart/commitments.js +111 -0
  39. package/dist/heart/config-registry.js +304 -0
  40. package/dist/heart/config.js +114 -118
  41. package/dist/heart/core.js +925 -246
  42. package/dist/heart/cross-chat-delivery.js +3 -18
  43. package/dist/heart/daemon/agent-config-check.js +512 -0
  44. package/dist/heart/daemon/agent-discovery.js +102 -3
  45. package/dist/heart/daemon/agent-service.js +522 -0
  46. package/dist/heart/daemon/agentic-repair.js +554 -0
  47. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  48. package/dist/heart/daemon/boot-sync-probe.js +197 -0
  49. package/dist/heart/daemon/cadence.js +70 -0
  50. package/dist/heart/daemon/cli-defaults.js +643 -0
  51. package/dist/heart/daemon/cli-exec.js +7567 -0
  52. package/dist/heart/daemon/cli-help.js +498 -0
  53. package/dist/heart/daemon/cli-parse.js +1590 -0
  54. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  55. package/dist/heart/daemon/cli-render.js +755 -0
  56. package/dist/heart/daemon/cli-types.js +8 -0
  57. package/dist/heart/daemon/connect-bay.js +323 -0
  58. package/dist/heart/daemon/daemon-cli.js +29 -1672
  59. package/dist/heart/daemon/daemon-entry.js +416 -2
  60. package/dist/heart/daemon/daemon-health.js +183 -0
  61. package/dist/heart/daemon/daemon-rollup.js +58 -0
  62. package/dist/heart/daemon/daemon-runtime-sync.js +69 -13
  63. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  64. package/dist/heart/daemon/daemon.js +758 -71
  65. package/dist/heart/daemon/dns-workflow.js +394 -0
  66. package/dist/heart/daemon/doctor-types.js +8 -0
  67. package/dist/heart/daemon/doctor.js +844 -0
  68. package/dist/heart/daemon/drift-detection.js +146 -0
  69. package/dist/heart/daemon/health-monitor.js +122 -1
  70. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  71. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  72. package/dist/heart/daemon/http-health-probe.js +80 -0
  73. package/dist/heart/daemon/human-command-screens.js +234 -0
  74. package/dist/heart/daemon/human-readiness.js +114 -0
  75. package/dist/heart/daemon/inner-status.js +102 -0
  76. package/dist/heart/daemon/interactive-repair.js +394 -0
  77. package/dist/heart/daemon/launchd.js +15 -3
  78. package/dist/heart/daemon/log-tailer.js +82 -12
  79. package/dist/heart/daemon/logs-prune.js +110 -0
  80. package/dist/heart/daemon/mcp-canary.js +288 -0
  81. package/dist/heart/daemon/message-router.js +2 -2
  82. package/dist/heart/daemon/os-cron-deps.js +134 -0
  83. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  84. package/dist/heart/daemon/ouro-entry.js +3 -1
  85. package/dist/heart/daemon/process-manager.js +375 -33
  86. package/dist/heart/daemon/provider-discovery.js +137 -0
  87. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  88. package/dist/heart/daemon/pulse.js +475 -0
  89. package/dist/heart/daemon/readiness-repair.js +365 -0
  90. package/dist/heart/daemon/run-hooks.js +2 -0
  91. package/dist/heart/daemon/runtime-logging.js +67 -16
  92. package/dist/heart/daemon/runtime-metadata.js +3 -31
  93. package/dist/heart/daemon/safe-mode.js +161 -0
  94. package/dist/heart/daemon/sense-manager.js +325 -38
  95. package/dist/heart/daemon/session-id-resolver.js +131 -0
  96. package/dist/heart/daemon/skill-management-installer.js +94 -0
  97. package/dist/heart/daemon/socket-client.js +158 -11
  98. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  99. package/dist/heart/daemon/startup-tui.js +330 -0
  100. package/dist/heart/daemon/task-scheduler.js +3 -25
  101. package/dist/heart/daemon/terminal-ui.js +499 -0
  102. package/dist/heart/daemon/thoughts.js +162 -17
  103. package/dist/heart/daemon/up-progress.js +366 -0
  104. package/dist/heart/daemon/vault-items.js +56 -0
  105. package/dist/heart/delegation.js +1 -1
  106. package/dist/heart/habits/habit-migration.js +189 -0
  107. package/dist/heart/habits/habit-parser.js +140 -0
  108. package/dist/heart/habits/habit-runtime-state.js +100 -0
  109. package/dist/heart/habits/habit-scheduler.js +372 -0
  110. package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
  111. package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
  112. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  113. package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
  114. package/dist/heart/identity.js +200 -51
  115. package/dist/heart/kept-notes.js +357 -0
  116. package/dist/heart/kicks.js +1 -1
  117. package/dist/heart/machine-identity.js +161 -0
  118. package/dist/heart/mail-import-discovery.js +353 -0
  119. package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
  120. package/dist/heart/mailbox/mailbox-http-response.js +7 -0
  121. package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
  122. package/dist/heart/mailbox/mailbox-http-static.js +103 -0
  123. package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
  124. package/dist/heart/mailbox/mailbox-http.js +99 -0
  125. package/dist/heart/mailbox/mailbox-read.js +31 -0
  126. package/dist/heart/mailbox/mailbox-types.js +27 -0
  127. package/dist/heart/mailbox/mailbox-view.js +195 -0
  128. package/dist/heart/mailbox/readers/agent-machine.js +382 -0
  129. package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
  130. package/dist/heart/mailbox/readers/mail.js +362 -0
  131. package/dist/heart/mailbox/readers/runtime-readers.js +651 -0
  132. package/dist/heart/mailbox/readers/sessions.js +232 -0
  133. package/dist/heart/mailbox/readers/shared.js +111 -0
  134. package/dist/heart/mcp/mcp-server.js +683 -0
  135. package/dist/heart/migrate-config.js +100 -0
  136. package/dist/heart/model-capabilities.js +19 -0
  137. package/dist/heart/platform.js +81 -0
  138. package/dist/heart/provider-attempt.js +134 -0
  139. package/dist/heart/provider-binding-resolver.js +255 -0
  140. package/dist/heart/provider-credentials.js +425 -0
  141. package/dist/heart/provider-failover.js +301 -0
  142. package/dist/heart/provider-models.js +81 -0
  143. package/dist/heart/provider-ping.js +262 -0
  144. package/dist/heart/provider-state.js +216 -0
  145. package/dist/heart/provider-visibility.js +188 -0
  146. package/dist/heart/providers/anthropic-token.js +131 -0
  147. package/dist/heart/providers/anthropic.js +139 -52
  148. package/dist/heart/providers/azure.js +97 -13
  149. package/dist/heart/providers/error-classification.js +127 -0
  150. package/dist/heart/providers/github-copilot.js +145 -0
  151. package/dist/heart/providers/minimax-vlm.js +189 -0
  152. package/dist/heart/providers/minimax.js +26 -8
  153. package/dist/heart/providers/openai-codex.js +55 -40
  154. package/dist/heart/runtime-capability-check.js +170 -0
  155. package/dist/heart/runtime-credentials.js +367 -0
  156. package/dist/heart/runtime-cwd.js +87 -0
  157. package/dist/heart/sense-truth.js +11 -4
  158. package/dist/heart/session-activity.js +43 -22
  159. package/dist/heart/session-events.js +1149 -0
  160. package/dist/heart/session-playback-cli-main.js +5 -0
  161. package/dist/heart/session-playback-cli.js +36 -0
  162. package/dist/heart/session-playback.js +231 -0
  163. package/dist/heart/session-stats-cli-main.js +5 -0
  164. package/dist/heart/session-stats.js +182 -0
  165. package/dist/heart/session-transcript.js +243 -0
  166. package/dist/heart/start-of-turn-packet.js +345 -0
  167. package/dist/heart/streaming.js +44 -27
  168. package/dist/heart/sync-classification.js +176 -0
  169. package/dist/heart/sync.js +449 -0
  170. package/dist/heart/target-resolution.js +9 -5
  171. package/dist/heart/tempo.js +93 -0
  172. package/dist/heart/temporal-view.js +41 -0
  173. package/dist/heart/timeouts.js +101 -0
  174. package/dist/heart/tool-activity-callbacks.js +59 -0
  175. package/dist/heart/tool-description.js +139 -0
  176. package/dist/heart/tool-friction.js +55 -0
  177. package/dist/heart/tool-loop.js +200 -0
  178. package/dist/heart/turn-context.js +381 -0
  179. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
  180. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  181. package/dist/heart/versioning/ouro-path-installer.js +426 -0
  182. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  183. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  184. package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
  185. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  186. package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
  187. package/dist/mailbox-ui/assets/index-Cm51CY9W.js +61 -0
  188. package/dist/mailbox-ui/index.html +15 -0
  189. package/dist/mailroom/attention.js +167 -0
  190. package/dist/mailroom/autonomy.js +209 -0
  191. package/dist/mailroom/blob-store.js +674 -0
  192. package/dist/mailroom/body-cache.js +61 -0
  193. package/dist/mailroom/core.js +720 -0
  194. package/dist/mailroom/entry.js +160 -0
  195. package/dist/mailroom/file-store.js +430 -0
  196. package/dist/mailroom/mbox-import.js +383 -0
  197. package/dist/mailroom/outbound.js +380 -0
  198. package/dist/mailroom/policy.js +263 -0
  199. package/dist/mailroom/reader.js +233 -0
  200. package/dist/mailroom/search-cache.js +256 -0
  201. package/dist/mailroom/search-relevance.js +319 -0
  202. package/dist/mailroom/smtp-ingress.js +176 -0
  203. package/dist/mailroom/source-state.js +176 -0
  204. package/dist/mailroom/thread.js +109 -0
  205. package/dist/mailroom/travel-extract.js +89 -0
  206. package/dist/mind/bundle-manifest.js +7 -1
  207. package/dist/mind/context.js +165 -101
  208. package/dist/mind/diary-integrity.js +60 -0
  209. package/dist/mind/{memory.js → diary.js} +74 -93
  210. package/dist/mind/embedding-provider.js +60 -0
  211. package/dist/mind/file-state.js +179 -0
  212. package/dist/mind/friends/channel.js +30 -0
  213. package/dist/mind/friends/resolver.js +54 -2
  214. package/dist/mind/friends/store-file.js +39 -3
  215. package/dist/mind/friends/types.js +2 -2
  216. package/dist/mind/journal-index.js +161 -0
  217. package/dist/mind/note-search.js +268 -0
  218. package/dist/mind/obligation-steering.js +221 -0
  219. package/dist/mind/pending.js +4 -0
  220. package/dist/mind/prompt-refresh.js +3 -2
  221. package/dist/mind/prompt.js +995 -123
  222. package/dist/mind/provenance-trust.js +26 -0
  223. package/dist/mind/scrutiny.js +173 -0
  224. package/dist/nerves/cli-logging.js +7 -1
  225. package/dist/nerves/coverage/audit-rules.js +15 -6
  226. package/dist/nerves/coverage/audit.js +28 -2
  227. package/dist/nerves/coverage/cli.js +1 -1
  228. package/dist/nerves/coverage/contract.js +5 -5
  229. package/dist/nerves/coverage/file-completeness.js +139 -5
  230. package/dist/nerves/coverage/run-artifacts.js +1 -1
  231. package/dist/nerves/event-buffer.js +111 -0
  232. package/dist/nerves/index.js +224 -4
  233. package/dist/nerves/observation.js +20 -0
  234. package/dist/nerves/redact.js +79 -0
  235. package/dist/nerves/review/cli-main.js +5 -0
  236. package/dist/nerves/review/cli.js +156 -0
  237. package/dist/nerves/review/core.js +152 -0
  238. package/dist/nerves/runtime.js +5 -1
  239. package/dist/repertoire/ado-client.js +15 -56
  240. package/dist/repertoire/ado-semantic.js +11 -10
  241. package/dist/repertoire/api-client.js +97 -0
  242. package/dist/repertoire/bitwarden-store.js +816 -0
  243. package/dist/repertoire/bundle-templates.js +72 -0
  244. package/dist/repertoire/bw-installer.js +180 -0
  245. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  246. package/dist/repertoire/coding/context-pack.js +330 -0
  247. package/dist/repertoire/coding/feedback.js +197 -30
  248. package/dist/repertoire/coding/manager.js +158 -9
  249. package/dist/repertoire/coding/spawner.js +55 -9
  250. package/dist/repertoire/coding/tools.js +170 -7
  251. package/dist/repertoire/commerce-errors.js +109 -0
  252. package/dist/repertoire/commerce-self-test.js +156 -0
  253. package/dist/repertoire/credential-access.js +111 -0
  254. package/dist/repertoire/duffel-client.js +185 -0
  255. package/dist/repertoire/github-client.js +14 -55
  256. package/dist/repertoire/graph-client.js +11 -52
  257. package/dist/repertoire/guardrails.js +396 -0
  258. package/dist/repertoire/mcp-client.js +295 -0
  259. package/dist/repertoire/mcp-manager.js +362 -0
  260. package/dist/repertoire/mcp-tools.js +63 -0
  261. package/dist/repertoire/shell-sessions.js +133 -0
  262. package/dist/repertoire/skills.js +15 -24
  263. package/dist/repertoire/stripe-client.js +131 -0
  264. package/dist/repertoire/tasks/board.js +31 -5
  265. package/dist/repertoire/tasks/fix.js +182 -0
  266. package/dist/repertoire/tasks/index.js +16 -4
  267. package/dist/repertoire/tasks/lifecycle.js +2 -2
  268. package/dist/repertoire/tasks/parser.js +3 -2
  269. package/dist/repertoire/tasks/scanner.js +194 -37
  270. package/dist/repertoire/tasks/transitions.js +16 -78
  271. package/dist/repertoire/tool-results.js +29 -0
  272. package/dist/repertoire/tools-attachments.js +317 -0
  273. package/dist/repertoire/tools-base.js +47 -1075
  274. package/dist/repertoire/tools-bluebubbles.js +1 -0
  275. package/dist/repertoire/tools-bridge.js +142 -0
  276. package/dist/repertoire/tools-bundle.js +984 -0
  277. package/dist/repertoire/tools-config.js +185 -0
  278. package/dist/repertoire/tools-continuity.js +248 -0
  279. package/dist/repertoire/tools-credential.js +381 -0
  280. package/dist/repertoire/tools-files.js +342 -0
  281. package/dist/repertoire/tools-flight.js +224 -0
  282. package/dist/repertoire/tools-flow.js +119 -0
  283. package/dist/repertoire/tools-github.js +1 -7
  284. package/dist/repertoire/tools-mail.js +1857 -0
  285. package/dist/repertoire/tools-notes.js +421 -0
  286. package/dist/repertoire/tools-session.js +750 -0
  287. package/dist/repertoire/tools-shell.js +120 -0
  288. package/dist/repertoire/tools-stripe.js +180 -0
  289. package/dist/repertoire/tools-surface.js +243 -0
  290. package/dist/repertoire/tools-teams.js +9 -39
  291. package/dist/repertoire/tools-travel.js +125 -0
  292. package/dist/repertoire/tools-trip.js +604 -0
  293. package/dist/repertoire/tools-user-profile.js +144 -0
  294. package/dist/repertoire/tools-vault.js +40 -0
  295. package/dist/repertoire/tools.js +108 -100
  296. package/dist/repertoire/travel-api-client.js +360 -0
  297. package/dist/repertoire/user-profile.js +131 -0
  298. package/dist/repertoire/vault-setup.js +246 -0
  299. package/dist/repertoire/vault-unlock.js +561 -0
  300. package/dist/scripts/claude-code-hook.js +41 -0
  301. package/dist/scripts/claude-code-stop-hook.js +47 -0
  302. package/dist/senses/attention-queue.js +116 -0
  303. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  304. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  305. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
  306. package/dist/senses/bluebubbles/entry.js +77 -0
  307. package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
  308. package/dist/senses/bluebubbles/index.js +2238 -0
  309. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  310. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  311. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
  312. package/dist/senses/bluebubbles/processed-log.js +111 -0
  313. package/dist/senses/bluebubbles/replay.js +129 -0
  314. package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +16 -2
  315. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  316. package/dist/senses/cli/bracketed-paste.js +82 -0
  317. package/dist/senses/cli/image-paste.js +287 -0
  318. package/dist/senses/cli/image-ref-navigation.js +75 -0
  319. package/dist/senses/cli/ink-app.js +156 -0
  320. package/dist/senses/cli/inline-diff.js +64 -0
  321. package/dist/senses/cli/input-keys.js +174 -0
  322. package/dist/senses/cli/kill-ring.js +86 -0
  323. package/dist/senses/cli/message-list.js +51 -0
  324. package/dist/senses/cli/ouro-tui.js +607 -0
  325. package/dist/senses/cli/spinner-imperative.js +135 -0
  326. package/dist/senses/cli/spinner.js +101 -0
  327. package/dist/senses/cli/status-line.js +60 -0
  328. package/dist/senses/cli/streaming-markdown.js +526 -0
  329. package/dist/senses/cli/tool-display.js +85 -0
  330. package/dist/senses/cli/tool-render.js +85 -0
  331. package/dist/senses/cli/tui-store.js +240 -0
  332. package/dist/senses/cli/virtual-list.js +35 -0
  333. package/dist/senses/cli-entry.js +60 -8
  334. package/dist/senses/cli-layout.js +187 -0
  335. package/dist/senses/cli.js +520 -209
  336. package/dist/senses/commands.js +66 -3
  337. package/dist/senses/habit-turn-message.js +108 -0
  338. package/dist/senses/inner-dialog-worker.js +175 -21
  339. package/dist/senses/inner-dialog.js +330 -27
  340. package/dist/senses/mail-entry.js +66 -0
  341. package/dist/senses/mail.js +379 -0
  342. package/dist/senses/pipeline.js +569 -182
  343. package/dist/senses/proactive-content-guard.js +51 -0
  344. package/dist/senses/shared-turn.js +248 -0
  345. package/dist/senses/surface-tool.js +68 -0
  346. package/dist/senses/teams-entry.js +60 -8
  347. package/dist/senses/teams.js +387 -98
  348. package/dist/senses/trust-gate.js +100 -5
  349. package/dist/trips/core.js +138 -0
  350. package/dist/trips/store.js +146 -0
  351. package/package.json +38 -7
  352. package/skills/agent-commerce.md +106 -0
  353. package/skills/browser-navigation.md +117 -0
  354. package/skills/commerce-setup-guide.md +116 -0
  355. package/skills/commerce-setup.md +84 -0
  356. package/skills/configure-dev-tools.md +101 -0
  357. package/skills/travel-planning.md +138 -0
  358. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  359. package/dist/heart/daemon/subagent-installer.js +0 -166
  360. package/dist/heart/session-recall.js +0 -116
  361. package/dist/mind/associative-recall.js +0 -209
  362. package/dist/senses/bluebubbles-entry.js +0 -13
  363. package/dist/senses/bluebubbles.js +0 -1177
  364. package/dist/senses/debug-activity.js +0 -148
  365. package/subagents/README.md +0 -86
  366. package/subagents/work-doer.md +0 -237
  367. package/subagents/work-merger.md +0 -618
  368. package/subagents/work-planner.md +0 -390
  369. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  370. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  371. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  372. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  373. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  374. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  375. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  376. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  377. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  378. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  379. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  380. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  381. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  382. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  383. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  384. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -0,0 +1,161 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.SAFE_MODE_OVERRIDE_FILENAME = void 0;
37
+ exports.detectSafeMode = detectSafeMode;
38
+ exports.pruneOldCrashes = pruneOldCrashes;
39
+ const fs = __importStar(require("fs"));
40
+ const path = __importStar(require("path"));
41
+ const runtime_1 = require("../../nerves/runtime");
42
+ exports.SAFE_MODE_OVERRIDE_FILENAME = "safe-mode-override.json";
43
+ /** 3+ crashes within this window triggers safe mode */
44
+ const CRASH_WINDOW_MS = 5 * 60 * 1000; // 5 minutes
45
+ const CRASH_THRESHOLD = 3;
46
+ /**
47
+ * Reads crash history from the tombstone file and determines if safe mode should be active.
48
+ * Returns active if 3+ crashes occurred within 5 minutes.
49
+ *
50
+ * Safe mode is bypassed when:
51
+ * - devMode is true
52
+ * - A safe-mode override file exists (written by `ouro up --force`)
53
+ */
54
+ function detectSafeMode(tombstonePath, options) {
55
+ const inactive = { active: false, reason: "", enteredAt: "" };
56
+ // Dev mode: never enter safe mode
57
+ if (options?.devMode) {
58
+ return inactive;
59
+ }
60
+ // Check for override file (--force)
61
+ const overridePath = path.join(path.dirname(tombstonePath), exports.SAFE_MODE_OVERRIDE_FILENAME);
62
+ try {
63
+ if (fs.existsSync(overridePath)) {
64
+ return inactive;
65
+ }
66
+ }
67
+ catch {
68
+ // Best-effort check
69
+ }
70
+ // Read tombstone
71
+ let parsed;
72
+ try {
73
+ const raw = fs.readFileSync(tombstonePath, "utf-8");
74
+ parsed = JSON.parse(raw);
75
+ }
76
+ catch {
77
+ return inactive;
78
+ }
79
+ // Extract recentCrashes
80
+ if (!Array.isArray(parsed.recentCrashes)) {
81
+ return inactive;
82
+ }
83
+ const nowMs = options?.now ? options.now() : Date.now();
84
+ const windowStart = nowMs - CRASH_WINDOW_MS;
85
+ // Filter to valid string timestamps within the crash window
86
+ const recentInWindow = parsed.recentCrashes.filter((entry) => {
87
+ if (typeof entry !== "string")
88
+ return false;
89
+ const ts = new Date(entry).getTime();
90
+ if (isNaN(ts))
91
+ return false;
92
+ return ts >= windowStart;
93
+ });
94
+ if (recentInWindow.length < CRASH_THRESHOLD) {
95
+ return inactive;
96
+ }
97
+ const result = {
98
+ active: true,
99
+ reason: `crash loop detected: ${recentInWindow.length} crashes in last 5 minutes`,
100
+ enteredAt: new Date(nowMs).toISOString(),
101
+ };
102
+ (0, runtime_1.emitNervesEvent)({
103
+ level: "error",
104
+ component: "daemon",
105
+ event: "daemon.safe_mode_entered",
106
+ message: result.reason,
107
+ meta: {
108
+ crashCount: recentInWindow.length,
109
+ windowMs: CRASH_WINDOW_MS,
110
+ tombstonePath,
111
+ },
112
+ });
113
+ return result;
114
+ }
115
+ /**
116
+ * Prunes crash entries older than 5 minutes from the tombstone's recentCrashes.
117
+ * Also removes the safe-mode override file if present.
118
+ * Called after successful startup (uptime > stability threshold).
119
+ */
120
+ function pruneOldCrashes(tombstonePath, options) {
121
+ // Remove override file
122
+ const overridePath = path.join(path.dirname(tombstonePath), exports.SAFE_MODE_OVERRIDE_FILENAME);
123
+ try {
124
+ if (fs.existsSync(overridePath)) {
125
+ fs.unlinkSync(overridePath);
126
+ }
127
+ }
128
+ catch {
129
+ // Best-effort
130
+ }
131
+ // Read existing tombstone
132
+ let parsed;
133
+ try {
134
+ const raw = fs.readFileSync(tombstonePath, "utf-8");
135
+ parsed = JSON.parse(raw);
136
+ }
137
+ catch {
138
+ return;
139
+ }
140
+ if (!Array.isArray(parsed.recentCrashes)) {
141
+ return;
142
+ }
143
+ const nowMs = options?.now ? options.now() : Date.now();
144
+ const windowStart = nowMs - CRASH_WINDOW_MS;
145
+ // Keep only entries within the window
146
+ const pruned = parsed.recentCrashes.filter((entry) => {
147
+ if (typeof entry !== "string")
148
+ return false;
149
+ const ts = new Date(entry).getTime();
150
+ if (isNaN(ts))
151
+ return false;
152
+ return ts >= windowStart;
153
+ });
154
+ parsed.recentCrashes = pruned;
155
+ try {
156
+ fs.writeFileSync(tombstonePath, JSON.stringify(parsed, null, 2) + "\n", "utf-8");
157
+ }
158
+ catch {
159
+ // Best-effort
160
+ }
161
+ }
@@ -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 http_health_probe_1 = require("./http-health-probe");
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,7 @@ 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 },
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"]) {
80
84
  const rawSense = rawSenses[sense];
81
85
  if (!rawSense || typeof rawSense !== "object" || Array.isArray(rawSense)) {
82
86
  continue;
@@ -88,22 +92,6 @@ 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() : "";
@@ -112,17 +100,36 @@ function numberField(record, key, fallback) {
112
100
  const value = record?.[key];
113
101
  return typeof value === "number" && Number.isFinite(value) ? value : fallback;
114
102
  }
115
- function senseFactsFromSecrets(agent, senses, secretsPath) {
103
+ function compactRuntimeConfigError(agent, error) {
104
+ const compact = error.replace(/\s+/g, " ").trim();
105
+ if (/credential vault is locked|vault locked|vault is locked/i.test(compact)) {
106
+ 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`;
107
+ }
108
+ return compact || "unavailable";
109
+ }
110
+ function runtimeConfigUnavailableDetail(agent, runtimeConfig) {
111
+ if (runtimeConfig.ok)
112
+ return "";
113
+ const itemName = /^vault:[^:]+:(.+)$/.exec(runtimeConfig.itemPath)?.[1] ?? "runtime/config";
114
+ if (runtimeConfig.reason === "missing")
115
+ return `missing vault ${itemName} (${agent})`;
116
+ return `vault ${itemName} unavailable (${compactRuntimeConfigError(agent, runtimeConfig.error)})`;
117
+ }
118
+ function senseFactsFromRuntimeConfig(agent, senses, runtimeConfig, machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent)) {
116
119
  const base = {
117
120
  cli: { configured: true, detail: "local interactive terminal" },
118
121
  teams: { configured: false, detail: "not enabled in agent.json" },
119
122
  bluebubbles: { configured: false, detail: "not enabled in agent.json" },
123
+ mail: { configured: false, detail: "not enabled in agent.json" },
120
124
  };
121
- const { payload, error } = readSecretsPayload(secretsPath);
125
+ const payload = runtimeConfig.ok ? runtimeConfig.config : {};
126
+ const unavailableDetail = runtimeConfigUnavailableDetail(agent, runtimeConfig);
122
127
  const teams = payload.teams;
123
128
  const teamsChannel = payload.teamsChannel;
124
- const bluebubbles = payload.bluebubbles;
125
- const bluebubblesChannel = payload.bluebubblesChannel;
129
+ const machinePayload = machineRuntimeConfig.ok ? machineRuntimeConfig.config : {};
130
+ const bluebubbles = machinePayload.bluebubbles;
131
+ const bluebubblesChannel = machinePayload.bluebubblesChannel;
132
+ const mailroom = payload.mailroom;
126
133
  if (senses.teams.enabled) {
127
134
  const missing = [];
128
135
  if (!textField(teams, "clientId"))
@@ -138,9 +145,9 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
138
145
  }
139
146
  : {
140
147
  configured: false,
141
- detail: error && !fs.existsSync(secretsPath)
142
- ? `missing secrets.json (${agent})`
143
- : `missing ${missing.join("/")}`,
148
+ detail: runtimeConfig.ok
149
+ ? `missing ${missing.join("/")}`
150
+ : unavailableDetail,
144
151
  };
145
152
  }
146
153
  if (senses.bluebubbles.enabled) {
@@ -156,19 +163,52 @@ function senseFactsFromSecrets(agent, senses, secretsPath) {
156
163
  }
157
164
  : {
158
165
  configured: false,
159
- detail: error && !fs.existsSync(secretsPath)
160
- ? `missing secrets.json (${agent})`
161
- : `missing ${missing.join("/")}`,
166
+ optional: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing",
167
+ detail: !machineRuntimeConfig.ok && machineRuntimeConfig.reason === "missing"
168
+ ? "not attached on this machine"
169
+ : machineRuntimeConfig.ok
170
+ ? `missing ${missing.join("/")}`
171
+ : runtimeConfigUnavailableDetail(agent, machineRuntimeConfig),
172
+ };
173
+ }
174
+ if (senses.mail.enabled) {
175
+ const privateKeys = mailroom?.privateKeys;
176
+ const hasPrivateKeys = !!privateKeys && typeof privateKeys === "object" && !Array.isArray(privateKeys) && Object.values(privateKeys).some((value) => typeof value === "string" && value.trim().length > 0);
177
+ const mailboxAddress = textField(mailroom, "mailboxAddress");
178
+ const missing = [];
179
+ if (!mailboxAddress)
180
+ missing.push("mailroom.mailboxAddress");
181
+ if (!hasPrivateKeys)
182
+ missing.push("mailroom.privateKeys");
183
+ base.mail = missing.length === 0
184
+ ? { configured: true, detail: mailboxAddress }
185
+ : {
186
+ configured: false,
187
+ detail: runtimeConfig.ok
188
+ ? `missing ${missing.join("/")}`
189
+ : unavailableDetail,
162
190
  };
163
191
  }
164
192
  return base;
165
193
  }
194
+ function senseRepairHint(agent, sense) {
195
+ if (sense === "teams") {
196
+ return `Run 'ouro vault config set --agent ${agent} --key teams.clientId', teams.clientSecret, and teams.tenantId; then run 'ouro up' again.`;
197
+ }
198
+ if (sense === "mail") {
199
+ return `Agent-runnable: provision Mailroom access with 'ouro connect mail --agent ${agent}', then restart with 'ouro up'.`;
200
+ }
201
+ return `Run 'ouro connect bluebubbles --agent ${agent}' to attach BlueBubbles on this machine; then run 'ouro up' again.`;
202
+ }
203
+ function currentMachineId() {
204
+ return (0, machine_identity_1.loadOrCreateMachineIdentity)({ homeDir: os.homedir() }).machineId;
205
+ }
166
206
  function parseSenseSnapshotName(name) {
167
207
  const parts = name.split(":");
168
208
  if (parts.length !== 2)
169
209
  return null;
170
210
  const [agent, sense] = parts;
171
- if (sense !== "teams" && sense !== "bluebubbles")
211
+ if (sense !== "teams" && sense !== "bluebubbles" && sense !== "mail")
172
212
  return null;
173
213
  return { agent, sense };
174
214
  }
@@ -177,6 +217,32 @@ function runtimeInfoFor(status) {
177
217
  return { runtime: "running" };
178
218
  return { runtime: "error" };
179
219
  }
220
+ function managedSenseEntry(sense) {
221
+ if (sense === "teams")
222
+ return "senses/teams-entry.js";
223
+ if (sense === "bluebubbles")
224
+ return "senses/bluebubbles/entry.js";
225
+ return "senses/mail-entry.js";
226
+ }
227
+ function runtimeCredentialBootstrapFor(agent, sense) {
228
+ const runtime = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
229
+ const machineId = sense === "bluebubbles" ? currentMachineId() : undefined;
230
+ const machine = sense === "bluebubbles" ? (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent) : null;
231
+ const providerPool = (0, provider_credentials_1.readProviderCredentialPool)(agent);
232
+ const providerCredentialRecords = providerPool.ok
233
+ ? Object.values(providerPool.pool.providers).filter((record) => !!record)
234
+ : [];
235
+ const bootstrap = {
236
+ agentName: agent,
237
+ runtimeConfig: runtime.ok ? runtime.config : undefined,
238
+ machineRuntimeConfig: machine?.ok ? machine.config : undefined,
239
+ machineId,
240
+ providerCredentialRecords: providerCredentialRecords.length > 0 ? providerCredentialRecords : undefined,
241
+ };
242
+ if (!bootstrap.runtimeConfig && !bootstrap.machineRuntimeConfig && !bootstrap.providerCredentialRecords)
243
+ return null;
244
+ return bootstrap;
245
+ }
180
246
  function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
181
247
  if (!lastCheckedAt) {
182
248
  return false;
@@ -187,53 +253,162 @@ function blueBubblesRuntimeStateIsFresh(lastCheckedAt, now = Date.now()) {
187
253
  }
188
254
  return checkedAt >= now - BLUEBUBBLES_RUNTIME_FRESHNESS_WINDOW_MS;
189
255
  }
256
+ function readBlueBubblesRuntimeJson(runtimePath) {
257
+ try {
258
+ const raw = fs.readFileSync(runtimePath, "utf-8");
259
+ const parsed = JSON.parse(raw);
260
+ /* v8 ignore start -- branches: ternary fallbacks for missing/malformed BB runtime fields @preserve */
261
+ return {
262
+ upstreamStatus: parsed.upstreamStatus === "ok" || parsed.upstreamStatus === "error"
263
+ ? parsed.upstreamStatus
264
+ : "unknown",
265
+ detail: typeof parsed.detail === "string" && parsed.detail.trim()
266
+ ? parsed.detail
267
+ : "startup health probe pending",
268
+ lastCheckedAt: typeof parsed.lastCheckedAt === "string" ? parsed.lastCheckedAt : undefined,
269
+ proofMethod: typeof parsed.proofMethod === "string" && parsed.proofMethod.trim()
270
+ ? parsed.proofMethod
271
+ : undefined,
272
+ pendingRecoveryCount: typeof parsed.pendingRecoveryCount === "number" && Number.isFinite(parsed.pendingRecoveryCount)
273
+ ? parsed.pendingRecoveryCount
274
+ : 0,
275
+ failedRecoveryCount: typeof parsed.failedRecoveryCount === "number" && Number.isFinite(parsed.failedRecoveryCount)
276
+ ? parsed.failedRecoveryCount
277
+ : 0,
278
+ oldestPendingRecoveryAt: typeof parsed.oldestPendingRecoveryAt === "string" ? parsed.oldestPendingRecoveryAt : undefined,
279
+ oldestPendingRecoveryAgeMs: typeof parsed.oldestPendingRecoveryAgeMs === "number" && Number.isFinite(parsed.oldestPendingRecoveryAgeMs)
280
+ ? parsed.oldestPendingRecoveryAgeMs
281
+ : undefined,
282
+ };
283
+ /* v8 ignore stop */
284
+ /* v8 ignore start -- defensive: catch for missing/corrupt BB runtime state file @preserve */
285
+ }
286
+ catch {
287
+ return { upstreamStatus: "unknown", detail: "startup health probe pending", pendingRecoveryCount: 0, failedRecoveryCount: 0 };
288
+ }
289
+ /* v8 ignore stop */
290
+ }
190
291
  function readBlueBubblesRuntimeFacts(agent, bundlesRoot, snapshot) {
191
292
  const agentRoot = path.join(bundlesRoot, `${agent}.ouro`);
192
293
  const runtimePath = path.join(agentRoot, "state", "senses", "bluebubbles", "runtime.json");
193
294
  if (!fs.existsSync(runtimePath)) {
194
295
  return { runtime: snapshot?.runtime };
195
296
  }
196
- const state = (0, bluebubbles_runtime_state_1.readBlueBubblesRuntimeState)(agent, agentRoot);
297
+ const state = readBlueBubblesRuntimeJson(runtimePath);
298
+ const checkedAtMs = state.lastCheckedAt ? Date.parse(state.lastCheckedAt) : Number.NaN;
299
+ const proofFacts = {
300
+ proofMethod: state.proofMethod ?? "bluebubbles.checkHealth",
301
+ lastProofAt: state.lastCheckedAt,
302
+ proofAgeMs: Number.isFinite(checkedAtMs) ? Math.max(0, Date.now() - checkedAtMs) : undefined,
303
+ pendingRecoveryCount: state.pendingRecoveryCount,
304
+ failedRecoveryCount: state.failedRecoveryCount,
305
+ oldestPendingRecoveryAt: state.oldestPendingRecoveryAt,
306
+ oldestPendingRecoveryAgeMs: state.oldestPendingRecoveryAgeMs,
307
+ };
197
308
  if (!blueBubblesRuntimeStateIsFresh(state.lastCheckedAt)) {
198
- return { runtime: snapshot?.runtime };
309
+ return {
310
+ runtime: snapshot?.runtime,
311
+ lastFailure: state.lastCheckedAt ? "BlueBubbles proof is stale" : undefined,
312
+ failureLayer: state.lastCheckedAt ? "proof_freshness" : undefined,
313
+ };
314
+ }
315
+ if (snapshot?.runtime !== "running") {
316
+ return {
317
+ runtime: "error",
318
+ detail: "BlueBubbles listener is not running",
319
+ ...proofFacts,
320
+ lastFailure: "listener process is not running",
321
+ failureLayer: "listener",
322
+ recoveryAction: "daemon health monitor will restart the BlueBubbles listener when its probe fails",
323
+ };
199
324
  }
200
325
  if (state.upstreamStatus === "error") {
201
326
  return {
202
327
  runtime: "error",
203
328
  detail: state.detail,
329
+ ...proofFacts,
330
+ lastFailure: state.detail,
331
+ failureLayer: "upstream",
332
+ recoveryAction: "verify BlueBubbles server/app auth and local machine attachment, then retry the listener",
333
+ };
334
+ }
335
+ if (state.pendingRecoveryCount > 0) {
336
+ return {
337
+ runtime: "error",
338
+ detail: state.detail,
339
+ ...proofFacts,
340
+ lastFailure: state.detail,
341
+ failureLayer: "recovery_queue",
342
+ recoveryAction: "queued recovery will retry; inspect BlueBubbles inbound/recovery sidecar logs if age keeps growing",
204
343
  };
205
344
  }
206
345
  if (state.upstreamStatus === "ok") {
207
- return { runtime: "running" };
346
+ return {
347
+ runtime: "running",
348
+ ...proofFacts,
349
+ ...(state.failedRecoveryCount > 0 ? { detail: state.detail } : {}),
350
+ ...(state.failedRecoveryCount > 0 ? {
351
+ lastFailure: state.detail,
352
+ failureLayer: "recovery_quarantine",
353
+ recoveryAction: "inspect quarantined BlueBubbles recovery failures; live transport remains reachable",
354
+ } : {}),
355
+ };
208
356
  }
209
- return { runtime: snapshot?.runtime };
357
+ return { runtime: snapshot?.runtime, ...proofFacts };
210
358
  }
211
359
  class DaemonSenseManager {
212
360
  processManager;
213
361
  contexts;
362
+ pendingConfigRefreshes = new Set();
214
363
  bundlesRoot;
215
364
  constructor(options) {
216
365
  const bundlesRoot = options.bundlesRoot ?? path.join(os.homedir(), "AgentBundles");
217
- const secretsRoot = options.secretsRoot ?? path.join(os.homedir(), ".agentsecrets");
218
366
  this.bundlesRoot = bundlesRoot;
219
367
  this.contexts = new Map(options.agents.map((agent) => {
220
368
  const senses = readAgentSenses(path.join(bundlesRoot, `${agent}.ouro`, "agent.json"));
221
- const facts = senseFactsFromSecrets(agent, senses, path.join(secretsRoot, agent, "secrets.json"));
369
+ const facts = senseFactsFromRuntimeConfig(agent, senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
222
370
  return [agent, { senses, facts }];
223
371
  }));
224
372
  const managedSenseAgents = [...this.contexts.entries()].flatMap(([agent, context]) => {
225
- return ["teams", "bluebubbles"]
226
- .filter((sense) => context.senses[sense].enabled && context.facts[sense].configured)
373
+ return ["teams", "bluebubbles", "mail"]
374
+ .filter((sense) => context.senses[sense].enabled)
227
375
  .map((sense) => ({
228
376
  name: `${agent}:${sense}`,
229
377
  agentArg: agent,
230
- entry: sense === "teams" ? "senses/teams-entry.js" : "senses/bluebubbles-entry.js",
378
+ entry: managedSenseEntry(sense),
231
379
  channel: sense,
232
380
  autoStart: true,
381
+ getRuntimeCredentialBootstrap: () => runtimeCredentialBootstrapFor(agent, sense),
233
382
  }));
234
383
  });
235
384
  this.processManager = options.processManager ?? new process_manager_1.DaemonProcessManager({
236
385
  agents: managedSenseAgents,
386
+ configCheck: async (name) => {
387
+ const parsed = parseSenseSnapshotName(name);
388
+ if (!parsed)
389
+ return { ok: true };
390
+ const context = this.contexts.get(parsed.agent);
391
+ if (!context)
392
+ return { ok: true };
393
+ context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(parsed.agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent));
394
+ const fact = context.facts[parsed.sense];
395
+ if (fact.configured)
396
+ return { ok: true };
397
+ this.scheduleSenseConfigRefresh(name, parsed);
398
+ if (fact.optional) {
399
+ return {
400
+ ok: false,
401
+ skip: true,
402
+ error: `${parsed.sense} is enabled for ${parsed.agent} but not attached on this machine`,
403
+ };
404
+ }
405
+ return {
406
+ ok: false,
407
+ skip: true,
408
+ error: `${parsed.sense} is enabled for ${parsed.agent} but runtime credentials are not ready: ${fact.detail}`,
409
+ fix: senseRepairHint(parsed.agent, parsed.sense),
410
+ };
411
+ },
237
412
  });
238
413
  (0, runtime_1.emitNervesEvent)({
239
414
  component: "channels",
@@ -245,12 +420,106 @@ class DaemonSenseManager {
245
420
  },
246
421
  });
247
422
  }
423
+ scheduleSenseConfigRefresh(name, parsed) {
424
+ if (this.pendingConfigRefreshes.has(name))
425
+ return;
426
+ this.pendingConfigRefreshes.add(name);
427
+ void this.refreshSenseConfigAndRetry(name, parsed);
428
+ }
429
+ async refreshSenseConfigAndRetry(name, parsed) {
430
+ try {
431
+ const refreshed = await (0, runtime_credentials_1.refreshRuntimeCredentialConfig)(parsed.agent, { preserveCachedOnFailure: true });
432
+ const machineRefreshed = parsed.sense === "bluebubbles"
433
+ ? await (0, runtime_credentials_1.refreshMachineRuntimeCredentialConfig)(parsed.agent, currentMachineId(), { preserveCachedOnFailure: true })
434
+ : (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(parsed.agent);
435
+ const context = this.contexts.get(parsed.agent);
436
+ /* v8 ignore next -- defensive: config refreshes are only scheduled for known agent contexts @preserve */
437
+ if (!context)
438
+ return;
439
+ context.facts = senseFactsFromRuntimeConfig(parsed.agent, context.senses, refreshed, machineRefreshed);
440
+ if (!context.facts[parsed.sense].configured)
441
+ return;
442
+ setTimeout(() => {
443
+ void this.processManager.startAgent?.(name).catch((error) => {
444
+ (0, runtime_1.emitNervesEvent)({
445
+ level: "error",
446
+ component: "channels",
447
+ event: "channel.daemon_sense_autostart_error",
448
+ message: "sense autostart failed",
449
+ /* v8 ignore next -- defensive: process manager rejects with Error instances in normal use @preserve */
450
+ meta: { error: error instanceof Error ? error.message : String(error) },
451
+ });
452
+ });
453
+ }, 0);
454
+ }
455
+ catch (error) {
456
+ (0, runtime_1.emitNervesEvent)({
457
+ level: "error",
458
+ component: "channels",
459
+ event: "channel.daemon_sense_autostart_error",
460
+ message: "sense config refresh failed",
461
+ /* v8 ignore next -- defensive: runtime credential refresh rejects with Error instances in normal use @preserve */
462
+ meta: { error: error instanceof Error ? error.message : String(error) },
463
+ });
464
+ }
465
+ finally {
466
+ this.pendingConfigRefreshes.delete(name);
467
+ }
468
+ }
248
469
  async startAutoStartSenses() {
249
470
  await this.processManager.startAutoStartAgents();
250
471
  }
472
+ triggerAutoStartSenses() {
473
+ if (this.processManager.triggerAutoStartAgents) {
474
+ this.processManager.triggerAutoStartAgents();
475
+ return;
476
+ }
477
+ void this.processManager.startAutoStartAgents().catch((error) => {
478
+ (0, runtime_1.emitNervesEvent)({
479
+ level: "error",
480
+ component: "channels",
481
+ event: "channel.daemon_sense_autostart_error",
482
+ message: "sense autostart failed",
483
+ meta: { error: error instanceof Error ? error.message : String(error) },
484
+ });
485
+ });
486
+ }
251
487
  async stopAll() {
252
488
  await this.processManager.stopAll();
253
489
  }
490
+ /* v8 ignore start -- pid collection for orphan cleanup pidfile @preserve */
491
+ listManagedPids() {
492
+ return this.processManager.listAgentSnapshots()
493
+ .map((s) => s.pid)
494
+ .filter((pid) => pid !== null && pid !== undefined);
495
+ }
496
+ /* v8 ignore stop */
497
+ listHealthProbes() {
498
+ const probes = [];
499
+ for (const [agent, context] of this.contexts.entries()) {
500
+ const runtimeConfig = (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent);
501
+ const machineRuntimeConfig = (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent);
502
+ context.facts = senseFactsFromRuntimeConfig(agent, context.senses, runtimeConfig, machineRuntimeConfig);
503
+ if (!context.senses.bluebubbles.enabled || !context.facts.bluebubbles.configured || !machineRuntimeConfig.ok) {
504
+ continue;
505
+ }
506
+ const machinePayload = machineRuntimeConfig.config;
507
+ const bluebubblesChannel = machinePayload.bluebubblesChannel;
508
+ const port = numberField(bluebubblesChannel, "port", DEFAULT_BLUEBUBBLES_PORT);
509
+ probes.push({
510
+ ...(0, http_health_probe_1.createHttpHealthProbe)(`bluebubbles:${agent}`, port),
511
+ managedName: `${agent}:bluebubbles`,
512
+ });
513
+ }
514
+ return probes;
515
+ }
516
+ async restartSense(managedName) {
517
+ if (this.processManager.restartAgent) {
518
+ await this.processManager.restartAgent(managedName);
519
+ return;
520
+ }
521
+ await this.processManager.startAgent?.(managedName);
522
+ }
254
523
  listSenseRows() {
255
524
  const runtime = new Map();
256
525
  for (const snapshot of this.processManager.listAgentSnapshots()) {
@@ -262,6 +531,7 @@ class DaemonSenseManager {
262
531
  runtime.set(parsed.agent, current);
263
532
  }
264
533
  const rows = [...this.contexts.entries()].flatMap(([agent, context]) => {
534
+ context.facts = senseFactsFromRuntimeConfig(agent, context.senses, (0, runtime_credentials_1.readRuntimeCredentialConfig)(agent), (0, runtime_credentials_1.readMachineRuntimeCredentialConfig)(agent));
265
535
  const blueBubblesRuntimeFacts = readBlueBubblesRuntimeFacts(agent, this.bundlesRoot, runtime.get(agent)?.bluebubbles);
266
536
  const runtimeInfo = {
267
537
  cli: { configured: true },
@@ -271,8 +541,13 @@ class DaemonSenseManager {
271
541
  },
272
542
  bluebubbles: {
273
543
  configured: context.facts.bluebubbles.configured,
544
+ optional: context.facts.bluebubbles.optional,
274
545
  ...blueBubblesRuntimeFacts,
275
546
  },
547
+ mail: {
548
+ configured: context.facts.mail.configured,
549
+ ...(runtime.get(agent)?.mail ?? {}),
550
+ },
276
551
  };
277
552
  const inventory = (0, sense_truth_1.getSenseInventory)({ senses: context.senses }, runtimeInfo);
278
553
  return inventory.map((entry) => ({
@@ -287,6 +562,18 @@ class DaemonSenseManager {
287
562
  ?? context.facts[entry.sense].detail
288
563
  : context.facts[entry.sense].detail
289
564
  : "not enabled in agent.json",
565
+ ...(entry.sense === "bluebubbles" ? {
566
+ proofMethod: blueBubblesRuntimeFacts.proofMethod,
567
+ lastProofAt: blueBubblesRuntimeFacts.lastProofAt,
568
+ proofAgeMs: blueBubblesRuntimeFacts.proofAgeMs,
569
+ lastFailure: blueBubblesRuntimeFacts.lastFailure,
570
+ failureLayer: blueBubblesRuntimeFacts.failureLayer,
571
+ recoveryAction: blueBubblesRuntimeFacts.recoveryAction,
572
+ pendingRecoveryCount: blueBubblesRuntimeFacts.pendingRecoveryCount,
573
+ failedRecoveryCount: blueBubblesRuntimeFacts.failedRecoveryCount,
574
+ oldestPendingRecoveryAt: blueBubblesRuntimeFacts.oldestPendingRecoveryAt,
575
+ oldestPendingRecoveryAgeMs: blueBubblesRuntimeFacts.oldestPendingRecoveryAgeMs,
576
+ } : {}),
290
577
  }));
291
578
  });
292
579
  (0, runtime_1.emitNervesEvent)({