@ouro.bot/cli 0.1.0-alpha.53 → 0.1.0-alpha.530

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 (382) 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 +3427 -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 +119 -129
  41. package/dist/heart/core.js +948 -243
  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 +7544 -0
  52. package/dist/heart/daemon/cli-help.js +498 -0
  53. package/dist/heart/daemon/cli-parse.js +1561 -0
  54. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  55. package/dist/heart/daemon/cli-render.js +728 -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 -1650
  59. package/dist/heart/daemon/daemon-entry.js +384 -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 +190 -12
  63. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  64. package/dist/heart/daemon/daemon.js +798 -69
  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 +837 -0
  68. package/dist/heart/daemon/drift-detection.js +146 -0
  69. package/dist/heart/daemon/health-monitor.js +117 -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 +22 -5
  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/message-router.js +2 -2
  81. package/dist/heart/daemon/os-cron-deps.js +134 -0
  82. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  83. package/dist/heart/daemon/ouro-entry.js +3 -1
  84. package/dist/heart/daemon/process-manager.js +375 -33
  85. package/dist/heart/daemon/provider-discovery.js +137 -0
  86. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  87. package/dist/heart/daemon/pulse.js +475 -0
  88. package/dist/heart/daemon/readiness-repair.js +365 -0
  89. package/dist/heart/daemon/run-hooks.js +2 -0
  90. package/dist/heart/daemon/runtime-logging.js +67 -16
  91. package/dist/heart/daemon/runtime-metadata.js +73 -0
  92. package/dist/heart/daemon/safe-mode.js +161 -0
  93. package/dist/heart/daemon/sense-manager.js +341 -37
  94. package/dist/heart/daemon/session-id-resolver.js +131 -0
  95. package/dist/heart/daemon/skill-management-installer.js +94 -0
  96. package/dist/heart/daemon/socket-client.js +109 -4
  97. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  98. package/dist/heart/daemon/startup-tui.js +330 -0
  99. package/dist/heart/daemon/task-scheduler.js +3 -25
  100. package/dist/heart/daemon/terminal-ui.js +499 -0
  101. package/dist/heart/daemon/thoughts.js +162 -17
  102. package/dist/heart/daemon/up-progress.js +366 -0
  103. package/dist/heart/daemon/vault-items.js +56 -0
  104. package/dist/heart/delegation.js +1 -1
  105. package/dist/heart/habits/habit-migration.js +189 -0
  106. package/dist/heart/habits/habit-parser.js +140 -0
  107. package/dist/heart/habits/habit-runtime-state.js +100 -0
  108. package/dist/heart/habits/habit-scheduler.js +372 -0
  109. package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
  110. package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
  111. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  112. package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
  113. package/dist/heart/identity.js +205 -66
  114. package/dist/heart/kept-notes.js +357 -0
  115. package/dist/heart/kicks.js +1 -1
  116. package/dist/heart/machine-identity.js +161 -0
  117. package/dist/heart/mail-import-discovery.js +353 -0
  118. package/dist/heart/mailbox/mailbox-http-hooks.js +66 -0
  119. package/dist/heart/mailbox/mailbox-http-response.js +7 -0
  120. package/dist/heart/mailbox/mailbox-http-routes.js +246 -0
  121. package/dist/heart/mailbox/mailbox-http-static.js +103 -0
  122. package/dist/heart/mailbox/mailbox-http-transport.js +116 -0
  123. package/dist/heart/mailbox/mailbox-http.js +99 -0
  124. package/dist/heart/mailbox/mailbox-read.js +31 -0
  125. package/dist/heart/mailbox/mailbox-types.js +27 -0
  126. package/dist/heart/mailbox/mailbox-view.js +195 -0
  127. package/dist/heart/mailbox/readers/agent-machine.js +382 -0
  128. package/dist/heart/mailbox/readers/continuity-readers.js +338 -0
  129. package/dist/heart/mailbox/readers/mail.js +362 -0
  130. package/dist/heart/mailbox/readers/runtime-readers.js +651 -0
  131. package/dist/heart/mailbox/readers/sessions.js +232 -0
  132. package/dist/heart/mailbox/readers/shared.js +111 -0
  133. package/dist/heart/mcp/mcp-server.js +653 -0
  134. package/dist/heart/migrate-config.js +100 -0
  135. package/dist/heart/model-capabilities.js +19 -0
  136. package/dist/heart/platform.js +81 -0
  137. package/dist/heart/provider-attempt.js +134 -0
  138. package/dist/heart/provider-binding-resolver.js +255 -0
  139. package/dist/heart/provider-credentials.js +425 -0
  140. package/dist/heart/provider-failover.js +301 -0
  141. package/dist/heart/provider-models.js +81 -0
  142. package/dist/heart/provider-ping.js +262 -0
  143. package/dist/heart/provider-state.js +216 -0
  144. package/dist/heart/provider-visibility.js +188 -0
  145. package/dist/heart/providers/anthropic-token.js +131 -0
  146. package/dist/heart/providers/anthropic.js +139 -52
  147. package/dist/heart/providers/azure.js +97 -13
  148. package/dist/heart/providers/error-classification.js +127 -0
  149. package/dist/heart/providers/github-copilot.js +145 -0
  150. package/dist/heart/providers/minimax-vlm.js +189 -0
  151. package/dist/heart/providers/minimax.js +26 -8
  152. package/dist/heart/providers/openai-codex.js +55 -40
  153. package/dist/heart/runtime-capability-check.js +170 -0
  154. package/dist/heart/runtime-credentials.js +367 -0
  155. package/dist/heart/sense-truth.js +11 -4
  156. package/dist/heart/session-activity.js +43 -22
  157. package/dist/heart/session-events.js +1149 -0
  158. package/dist/heart/session-playback-cli-main.js +5 -0
  159. package/dist/heart/session-playback-cli.js +36 -0
  160. package/dist/heart/session-playback.js +231 -0
  161. package/dist/heart/session-stats-cli-main.js +5 -0
  162. package/dist/heart/session-stats.js +182 -0
  163. package/dist/heart/session-transcript.js +243 -0
  164. package/dist/heart/start-of-turn-packet.js +345 -0
  165. package/dist/heart/streaming.js +44 -27
  166. package/dist/heart/sync-classification.js +176 -0
  167. package/dist/heart/sync.js +449 -0
  168. package/dist/heart/target-resolution.js +9 -5
  169. package/dist/heart/tempo.js +93 -0
  170. package/dist/heart/temporal-view.js +41 -0
  171. package/dist/heart/timeouts.js +101 -0
  172. package/dist/heart/tool-activity-callbacks.js +59 -0
  173. package/dist/heart/tool-description.js +139 -0
  174. package/dist/heart/tool-friction.js +55 -0
  175. package/dist/heart/tool-loop.js +200 -0
  176. package/dist/heart/turn-context.js +381 -0
  177. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  178. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  179. package/dist/heart/versioning/ouro-path-installer.js +425 -0
  180. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  181. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  182. package/dist/heart/{daemon → versioning}/update-checker.js +5 -1
  183. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  184. package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
  185. package/dist/mailbox-ui/assets/index-Cm51CY9W.js +61 -0
  186. package/dist/mailbox-ui/index.html +15 -0
  187. package/dist/mailroom/attention.js +167 -0
  188. package/dist/mailroom/autonomy.js +209 -0
  189. package/dist/mailroom/blob-store.js +606 -0
  190. package/dist/mailroom/body-cache.js +61 -0
  191. package/dist/mailroom/core.js +672 -0
  192. package/dist/mailroom/entry.js +160 -0
  193. package/dist/mailroom/file-store.js +426 -0
  194. package/dist/mailroom/mbox-import.js +382 -0
  195. package/dist/mailroom/outbound.js +380 -0
  196. package/dist/mailroom/policy.js +263 -0
  197. package/dist/mailroom/reader.js +228 -0
  198. package/dist/mailroom/search-cache.js +182 -0
  199. package/dist/mailroom/search-relevance.js +319 -0
  200. package/dist/mailroom/smtp-ingress.js +176 -0
  201. package/dist/mailroom/source-state.js +176 -0
  202. package/dist/mailroom/thread.js +109 -0
  203. package/dist/mailroom/travel-extract.js +89 -0
  204. package/dist/mind/bundle-manifest.js +7 -1
  205. package/dist/mind/context.js +165 -101
  206. package/dist/mind/diary-integrity.js +60 -0
  207. package/dist/mind/{memory.js → diary.js} +74 -93
  208. package/dist/mind/embedding-provider.js +60 -0
  209. package/dist/mind/file-state.js +179 -0
  210. package/dist/mind/friends/channel.js +30 -0
  211. package/dist/mind/friends/resolver.js +54 -2
  212. package/dist/mind/friends/store-file.js +39 -3
  213. package/dist/mind/friends/types.js +2 -2
  214. package/dist/mind/journal-index.js +161 -0
  215. package/dist/mind/note-search.js +268 -0
  216. package/dist/mind/obligation-steering.js +221 -0
  217. package/dist/mind/pending.js +4 -0
  218. package/dist/mind/prompt-refresh.js +3 -2
  219. package/dist/mind/prompt.js +942 -122
  220. package/dist/mind/provenance-trust.js +26 -0
  221. package/dist/mind/scrutiny.js +173 -0
  222. package/dist/nerves/cli-logging.js +7 -1
  223. package/dist/nerves/coverage/audit-rules.js +15 -6
  224. package/dist/nerves/coverage/audit.js +28 -2
  225. package/dist/nerves/coverage/cli.js +1 -1
  226. package/dist/nerves/coverage/contract.js +5 -5
  227. package/dist/nerves/coverage/file-completeness.js +139 -5
  228. package/dist/nerves/coverage/run-artifacts.js +1 -1
  229. package/dist/nerves/event-buffer.js +111 -0
  230. package/dist/nerves/index.js +224 -4
  231. package/dist/nerves/observation.js +20 -0
  232. package/dist/nerves/redact.js +79 -0
  233. package/dist/nerves/review/cli-main.js +5 -0
  234. package/dist/nerves/review/cli.js +156 -0
  235. package/dist/nerves/review/core.js +152 -0
  236. package/dist/nerves/runtime.js +5 -1
  237. package/dist/repertoire/ado-client.js +15 -56
  238. package/dist/repertoire/ado-semantic.js +11 -10
  239. package/dist/repertoire/api-client.js +97 -0
  240. package/dist/repertoire/bitwarden-store.js +816 -0
  241. package/dist/repertoire/bundle-templates.js +72 -0
  242. package/dist/repertoire/bw-installer.js +180 -0
  243. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  244. package/dist/repertoire/coding/context-pack.js +330 -0
  245. package/dist/repertoire/coding/feedback.js +197 -30
  246. package/dist/repertoire/coding/manager.js +158 -9
  247. package/dist/repertoire/coding/spawner.js +55 -9
  248. package/dist/repertoire/coding/tools.js +170 -7
  249. package/dist/repertoire/commerce-errors.js +109 -0
  250. package/dist/repertoire/commerce-self-test.js +156 -0
  251. package/dist/repertoire/credential-access.js +111 -0
  252. package/dist/repertoire/duffel-client.js +185 -0
  253. package/dist/repertoire/github-client.js +14 -55
  254. package/dist/repertoire/graph-client.js +11 -52
  255. package/dist/repertoire/guardrails.js +396 -0
  256. package/dist/repertoire/mcp-client.js +255 -0
  257. package/dist/repertoire/mcp-manager.js +305 -0
  258. package/dist/repertoire/mcp-tools.js +63 -0
  259. package/dist/repertoire/shell-sessions.js +133 -0
  260. package/dist/repertoire/skills.js +15 -24
  261. package/dist/repertoire/stripe-client.js +131 -0
  262. package/dist/repertoire/tasks/board.js +31 -5
  263. package/dist/repertoire/tasks/fix.js +182 -0
  264. package/dist/repertoire/tasks/index.js +16 -4
  265. package/dist/repertoire/tasks/lifecycle.js +2 -2
  266. package/dist/repertoire/tasks/parser.js +3 -2
  267. package/dist/repertoire/tasks/scanner.js +194 -37
  268. package/dist/repertoire/tasks/transitions.js +16 -78
  269. package/dist/repertoire/tool-results.js +29 -0
  270. package/dist/repertoire/tools-attachments.js +317 -0
  271. package/dist/repertoire/tools-base.js +47 -1075
  272. package/dist/repertoire/tools-bluebubbles.js +1 -0
  273. package/dist/repertoire/tools-bridge.js +142 -0
  274. package/dist/repertoire/tools-bundle.js +984 -0
  275. package/dist/repertoire/tools-config.js +185 -0
  276. package/dist/repertoire/tools-continuity.js +248 -0
  277. package/dist/repertoire/tools-credential.js +381 -0
  278. package/dist/repertoire/tools-files.js +342 -0
  279. package/dist/repertoire/tools-flight.js +224 -0
  280. package/dist/repertoire/tools-flow.js +119 -0
  281. package/dist/repertoire/tools-github.js +1 -7
  282. package/dist/repertoire/tools-mail.js +1477 -0
  283. package/dist/repertoire/tools-notes.js +421 -0
  284. package/dist/repertoire/tools-session.js +750 -0
  285. package/dist/repertoire/tools-shell.js +120 -0
  286. package/dist/repertoire/tools-stripe.js +180 -0
  287. package/dist/repertoire/tools-surface.js +243 -0
  288. package/dist/repertoire/tools-teams.js +9 -39
  289. package/dist/repertoire/tools-travel.js +125 -0
  290. package/dist/repertoire/tools-trip.js +422 -0
  291. package/dist/repertoire/tools-user-profile.js +144 -0
  292. package/dist/repertoire/tools-vault.js +40 -0
  293. package/dist/repertoire/tools.js +108 -100
  294. package/dist/repertoire/travel-api-client.js +360 -0
  295. package/dist/repertoire/user-profile.js +131 -0
  296. package/dist/repertoire/vault-setup.js +246 -0
  297. package/dist/repertoire/vault-unlock.js +561 -0
  298. package/dist/scripts/claude-code-hook.js +41 -0
  299. package/dist/scripts/claude-code-stop-hook.js +47 -0
  300. package/dist/senses/attention-queue.js +116 -0
  301. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  302. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  303. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
  304. package/dist/senses/bluebubbles/entry.js +77 -0
  305. package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
  306. package/dist/senses/bluebubbles/index.js +2236 -0
  307. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  308. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  309. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
  310. package/dist/senses/bluebubbles/processed-log.js +111 -0
  311. package/dist/senses/bluebubbles/replay.js +129 -0
  312. package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +16 -2
  313. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  314. package/dist/senses/cli/bracketed-paste.js +82 -0
  315. package/dist/senses/cli/image-paste.js +287 -0
  316. package/dist/senses/cli/image-ref-navigation.js +75 -0
  317. package/dist/senses/cli/ink-app.js +156 -0
  318. package/dist/senses/cli/inline-diff.js +64 -0
  319. package/dist/senses/cli/input-keys.js +174 -0
  320. package/dist/senses/cli/kill-ring.js +86 -0
  321. package/dist/senses/cli/message-list.js +51 -0
  322. package/dist/senses/cli/ouro-tui.js +607 -0
  323. package/dist/senses/cli/spinner-imperative.js +135 -0
  324. package/dist/senses/cli/spinner.js +101 -0
  325. package/dist/senses/cli/status-line.js +60 -0
  326. package/dist/senses/cli/streaming-markdown.js +526 -0
  327. package/dist/senses/cli/tool-display.js +85 -0
  328. package/dist/senses/cli/tool-render.js +85 -0
  329. package/dist/senses/cli/tui-store.js +240 -0
  330. package/dist/senses/cli/virtual-list.js +35 -0
  331. package/dist/senses/cli-entry.js +60 -8
  332. package/dist/senses/cli-layout.js +187 -0
  333. package/dist/senses/cli.js +520 -209
  334. package/dist/senses/commands.js +66 -3
  335. package/dist/senses/habit-turn-message.js +108 -0
  336. package/dist/senses/inner-dialog-worker.js +175 -21
  337. package/dist/senses/inner-dialog.js +330 -27
  338. package/dist/senses/mail-entry.js +66 -0
  339. package/dist/senses/mail.js +379 -0
  340. package/dist/senses/pipeline.js +569 -182
  341. package/dist/senses/proactive-content-guard.js +51 -0
  342. package/dist/senses/shared-turn.js +248 -0
  343. package/dist/senses/surface-tool.js +68 -0
  344. package/dist/senses/teams-entry.js +60 -8
  345. package/dist/senses/teams.js +387 -98
  346. package/dist/senses/trust-gate.js +100 -5
  347. package/dist/trips/core.js +138 -0
  348. package/dist/trips/store.js +146 -0
  349. package/package.json +38 -7
  350. package/skills/agent-commerce.md +106 -0
  351. package/skills/browser-navigation.md +117 -0
  352. package/skills/commerce-setup-guide.md +116 -0
  353. package/skills/commerce-setup.md +84 -0
  354. package/skills/configure-dev-tools.md +101 -0
  355. package/skills/travel-planning.md +138 -0
  356. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  357. package/dist/heart/daemon/subagent-installer.js +0 -166
  358. package/dist/heart/session-recall.js +0 -116
  359. package/dist/mind/associative-recall.js +0 -209
  360. package/dist/senses/bluebubbles-entry.js +0 -13
  361. package/dist/senses/bluebubbles.js +0 -1177
  362. package/dist/senses/debug-activity.js +0 -148
  363. package/subagents/README.md +0 -86
  364. package/subagents/work-doer.md +0 -237
  365. package/subagents/work-merger.md +0 -618
  366. package/subagents/work-planner.md +0 -390
  367. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  368. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  369. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  370. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  371. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  372. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  373. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  374. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  375. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  376. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  377. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  378. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  379. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  380. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  381. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  382. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -39,14 +39,27 @@ const path = __importStar(require("path"));
39
39
  const process_manager_1 = require("./process-manager");
40
40
  const daemon_1 = require("./daemon");
41
41
  const runtime_1 = require("../../nerves/runtime");
42
+ const index_1 = require("../../nerves/index");
42
43
  const message_router_1 = require("./message-router");
43
44
  const health_monitor_1 = require("./health-monitor");
45
+ const daemon_health_1 = require("./daemon-health");
46
+ const daemon_rollup_1 = require("./daemon-rollup");
44
47
  const task_scheduler_1 = require("./task-scheduler");
45
48
  const runtime_logging_1 = require("./runtime-logging");
46
49
  const sense_manager_1 = require("./sense-manager");
47
50
  const agent_discovery_1 = require("./agent-discovery");
48
51
  const identity_1 = require("../identity");
49
52
  const runtime_mode_1 = require("./runtime-mode");
53
+ const habit_scheduler_1 = require("../habits/habit-scheduler");
54
+ const habit_migration_1 = require("../habits/habit-migration");
55
+ const os_cron_deps_1 = require("./os-cron-deps");
56
+ const os_cron_1 = require("./os-cron");
57
+ const daemon_tombstone_1 = require("./daemon-tombstone");
58
+ const agent_config_check_1 = require("./agent-config-check");
59
+ const drift_detection_1 = require("./drift-detection");
60
+ const pulse_1 = require("./pulse");
61
+ const socket_client_1 = require("./socket-client");
62
+ const bundle_manifest_1 = require("../../mind/bundle-manifest");
50
63
  function parseSocketPath(argv) {
51
64
  const socketIndex = argv.indexOf("--socket");
52
65
  if (socketIndex >= 0) {
@@ -66,6 +79,16 @@ const mode = (0, runtime_mode_1.detectRuntimeMode)((0, identity_1.getRepoRoot)()
66
79
  message: "starting daemon entrypoint",
67
80
  meta: { socketPath, entryPath, mode },
68
81
  });
82
+ /* v8 ignore next -- dev-mode indicator: false branch (production) tested in daemon-boot-updates.test.ts @preserve */
83
+ if (mode === "dev") {
84
+ const repoRoot = (0, identity_1.getRepoRoot)();
85
+ (0, runtime_1.emitNervesEvent)({
86
+ component: "daemon",
87
+ event: "daemon.dev_mode_indicator",
88
+ message: `[dev] running from ${repoRoot}`,
89
+ meta: { repoRoot },
90
+ });
91
+ }
69
92
  const managedAgents = (0, agent_discovery_1.listEnabledBundleAgents)();
70
93
  const processManager = new process_manager_1.DaemonProcessManager({
71
94
  agents: managedAgents.map((agent) => ({
@@ -75,6 +98,28 @@ const processManager = new process_manager_1.DaemonProcessManager({
75
98
  autoStart: true,
76
99
  })),
77
100
  existsSync: fs.existsSync,
101
+ /* v8 ignore next 4 -- wiring: delegates to checkAgentConfigWithProviderHealth which has full unit tests @preserve */
102
+ configCheck: async (agent) => {
103
+ const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
104
+ return (0, agent_config_check_1.checkAgentConfigWithProviderHealth)(agent, bundlesRoot);
105
+ },
106
+ /* v8 ignore start -- pulse flush wiring: integration code; flushPulse itself has full unit tests @preserve */
107
+ onSnapshotChange: () => {
108
+ (0, pulse_1.flushPulse)({
109
+ snapshots: processManager.listAgentSnapshots(),
110
+ bundlesRoot: (0, identity_1.getAgentBundlesRoot)(),
111
+ daemonVersion: (0, bundle_manifest_1.getPackageVersion)(),
112
+ now: new Date(),
113
+ // Default I/O wired into pulse.ts (writePulse, readPulse, etc.)
114
+ // Wake recipient: send inner.wake over the daemon's own socket so
115
+ // the recipient agent runs an inner-dialog turn that picks up the
116
+ // pulse alert. Catch errors silently — pulse is best-effort.
117
+ fireInnerWake: (agent) => {
118
+ (0, socket_client_1.sendDaemonCommand)(socketPath, { kind: "inner.wake", agent }).catch(() => { });
119
+ },
120
+ });
121
+ },
122
+ /* v8 ignore stop */
78
123
  });
79
124
  const scheduler = new task_scheduler_1.TaskDrivenScheduler({
80
125
  agents: [...managedAgents],
@@ -86,6 +131,7 @@ const senseManager = new sense_manager_1.DaemonSenseManager({
86
131
  const healthMonitor = new health_monitor_1.HealthMonitor({
87
132
  processManager,
88
133
  scheduler,
134
+ senseProbeProvider: () => senseManager.listHealthProbes(),
89
135
  alertSink: (message) => {
90
136
  (0, runtime_1.emitNervesEvent)({
91
137
  level: "error",
@@ -95,7 +141,40 @@ const healthMonitor = new health_monitor_1.HealthMonitor({
95
141
  meta: { message },
96
142
  });
97
143
  },
144
+ /* v8 ignore next 3 -- wiring: delegates to processManager.restartAgent which has full unit tests @preserve */
145
+ onCriticalAgent: (agentName) => {
146
+ try {
147
+ processManager.restartAgent(agentName);
148
+ }
149
+ catch { /* recovery is best-effort */ }
150
+ },
151
+ /* v8 ignore next 3 -- wiring: delegates to senseManager.restartSense which has focused tests @preserve */
152
+ onCriticalSense: (managedName) => {
153
+ try {
154
+ void senseManager.restartSense(managedName);
155
+ }
156
+ catch { /* recovery is best-effort */ }
157
+ },
98
158
  });
159
+ const habitSchedulers = [];
160
+ let entryRuntimeStopping = false;
161
+ let stopCommandExitScheduled = false;
162
+ function stopEntryRuntime() {
163
+ if (entryRuntimeStopping)
164
+ return;
165
+ entryRuntimeStopping = true;
166
+ for (const s of habitSchedulers) {
167
+ s.stopWatching();
168
+ s.stop();
169
+ }
170
+ healthMonitor.stopPeriodicChecks();
171
+ }
172
+ function scheduleCleanProcessExitAfterStopCommand() {
173
+ if (stopCommandExitScheduled)
174
+ return;
175
+ stopCommandExitScheduled = true;
176
+ setTimeout(() => process.exit(0), 100);
177
+ }
99
178
  const daemon = new daemon_1.OuroDaemon({
100
179
  socketPath,
101
180
  processManager,
@@ -103,21 +182,324 @@ const daemon = new daemon_1.OuroDaemon({
103
182
  scheduler,
104
183
  healthMonitor,
105
184
  router,
185
+ mode,
186
+ onStopCommandComplete: () => {
187
+ stopEntryRuntime();
188
+ scheduleCleanProcessExitAfterStopCommand();
189
+ },
106
190
  });
107
- void daemon.start().catch(async () => {
191
+ const daemonStartedAt = new Date().toISOString();
192
+ const degradedComponents = [];
193
+ function buildDaemonHealthState() {
194
+ const snapshots = processManager.listAgentSnapshots();
195
+ const agentDegradedComponents = snapshots
196
+ .filter((snapshot) => snapshot.status !== "running")
197
+ .map((snapshot) => {
198
+ const reasonParts = [
199
+ snapshot.errorReason ?? `${snapshot.channel} is ${snapshot.status}`,
200
+ snapshot.fixHint ? `Fix: ${snapshot.fixHint}` : null,
201
+ ].filter((part) => part !== null);
202
+ return {
203
+ component: `agent:${snapshot.name}`,
204
+ reason: reasonParts.join(" "),
205
+ since: snapshot.lastCrashAt ?? daemonStartedAt,
206
+ };
207
+ });
208
+ // Preserved for backwards-compatible inspection: callers (status
209
+ // command, mailbox surface, etc.) may still read this combined list
210
+ // for per-component reasons. The rollup status field above is what
211
+ // changed meaning — the array is still the union of bootstrap +
212
+ // agent-derived degradation entries.
213
+ const degraded = [
214
+ ...degradedComponents.map((entry) => ({ ...entry })),
215
+ ...agentDegradedComponents,
216
+ ];
217
+ // Layer 4: probe each enabled agent for drift between agent.json
218
+ // (intent) and state/providers.json (observed binding). The result is
219
+ // a single boolean — driftDetected — that downgrades the rollup from
220
+ // healthy to partial when true. Per-finding detail is consumed by
221
+ // render-side surfaces (inner-status, --no-repair summary) and by
222
+ // Layer 3 RepairGuide; the rollup itself only cares about presence.
223
+ // Best-effort: a single agent's read failure is not blocking — the
224
+ // function returns "no drift detected for that agent" and continues.
225
+ const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
226
+ const drift = [];
227
+ for (const snapshot of snapshots) {
228
+ try {
229
+ const inputs = (0, drift_detection_1.loadDriftInputsForAgent)(bundlesRoot, snapshot.name);
230
+ const findings = (0, drift_detection_1.detectProviderBindingDrift)({
231
+ agentName: snapshot.name,
232
+ agentJson: inputs.agentJson,
233
+ providerState: inputs.providerState,
234
+ });
235
+ if (findings.length > 0) {
236
+ drift.push(...findings);
237
+ }
238
+ }
239
+ catch {
240
+ // best-effort: continue scanning
241
+ }
242
+ }
243
+ const driftDetected = drift.length > 0;
244
+ // Layer 1 rollup: project per-agent snapshots into the minimal
245
+ // AgentRollupInput shape and let computeDaemonRollup decide. The
246
+ // input is "every enabled agent" — managedAgents was filtered via
247
+ // listEnabledBundleAgents at module init, and snapshots only covers
248
+ // agents the process manager was told to manage, so by construction
249
+ // these entries are all enabled. The rollup function is a pure
250
+ // declarative function on the data we hand it.
251
+ //
252
+ // Note: safe-mode is wired as `false` here. Existing crash-loop
253
+ // detection (safe-mode.ts) already runs at the daemon-up boot path
254
+ // (cli-exec.ts), not from inside the daemon process itself. Once
255
+ // the daemon is up and reaching this rollup, safe mode no longer
256
+ // applies — the daemon is by definition past the crash-loop gate.
257
+ // If a future PR moves safe-mode signal into the running daemon,
258
+ // wire it through this third argument.
259
+ const rollupStatus = (0, daemon_rollup_1.computeDaemonRollup)({
260
+ enabledAgents: snapshots.map((snapshot) => ({
261
+ name: snapshot.name,
262
+ status: snapshot.status,
263
+ })),
264
+ bootstrapDegraded: degradedComponents,
265
+ safeMode: false,
266
+ driftDetected,
267
+ });
268
+ return {
269
+ status: rollupStatus,
270
+ mode,
271
+ pid: process.pid,
272
+ startedAt: daemonStartedAt,
273
+ uptimeSeconds: Math.floor(process.uptime()),
274
+ safeMode: null,
275
+ degraded,
276
+ drift,
277
+ agents: Object.fromEntries(snapshots.map((snapshot) => [
278
+ snapshot.name,
279
+ {
280
+ status: snapshot.status,
281
+ pid: snapshot.pid,
282
+ crashes: snapshot.restartCount,
283
+ },
284
+ ])),
285
+ habits: {},
286
+ };
287
+ }
288
+ function recordRecoverableBootstrapFailure(options) {
289
+ const errorMessage = options.error instanceof Error ? options.error.message : String(options.error);
290
+ const existing = degradedComponents.find((entry) => entry.component === options.component);
291
+ const reason = `${errorMessage}. ${options.guidance}`;
292
+ if (existing) {
293
+ existing.reason = reason;
294
+ }
295
+ else {
296
+ degradedComponents.push({
297
+ component: options.component,
298
+ reason,
299
+ since: new Date().toISOString(),
300
+ });
301
+ }
302
+ (0, runtime_1.emitNervesEvent)({
303
+ level: "warn",
304
+ component: "daemon",
305
+ event: "daemon.bootstrap_degraded",
306
+ message: "recoverable daemon bootstrap failure; daemon remains available in degraded mode",
307
+ meta: {
308
+ agent: options.agent,
309
+ component: options.component,
310
+ habitsDir: options.habitsDir,
311
+ error: errorMessage,
312
+ guidance: options.guidance,
313
+ },
314
+ });
315
+ }
316
+ function emitHabitSetupError(agent, error) {
317
+ const normalized = error instanceof Error ? error : new Error(String(error));
318
+ (0, runtime_1.emitNervesEvent)({
319
+ level: "error",
320
+ component: "daemon",
321
+ event: "daemon.habit_setup_error",
322
+ message: `habit setup failed for agent ${agent}`,
323
+ meta: { agent, error: normalized.message },
324
+ });
325
+ }
326
+ /* v8 ignore start — daemon health writer wiring, tested via daemon-health.test.ts @preserve */
327
+ const healthWriter = new daemon_health_1.DaemonHealthWriter((0, daemon_health_1.getDefaultHealthPath)());
328
+ const healthSink = (0, daemon_health_1.createHealthNervesSink)(healthWriter, buildDaemonHealthState);
329
+ (0, index_1.registerGlobalLogSink)(healthSink);
330
+ /* v8 ignore stop */
331
+ /* v8 ignore start -- habit wiring: lambdas delegate to processManager/fs; tested via HabitScheduler unit tests @preserve */
332
+ void daemon.start().then(() => {
333
+ const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
334
+ const ouroPath = (0, os_cron_deps_1.resolveOuroBinaryPath)();
335
+ const osCronDeps = (0, os_cron_deps_1.createRealOsCronDeps)();
336
+ for (const agent of managedAgents) {
337
+ const bundleRoot = path.join(bundlesRoot, `${agent}.ouro`);
338
+ const habitsDir = path.join(bundleRoot, "habits");
339
+ const degradedComponent = `habits:${agent}`;
340
+ try {
341
+ // Migrate old tasks/habits/ to habits/ at bundle root
342
+ (0, habit_migration_1.migrateHabitsFromTaskSystem)(bundleRoot);
343
+ const osCronManager = new os_cron_1.LaunchdCronManager(osCronDeps);
344
+ const scheduler = new habit_scheduler_1.HabitScheduler({
345
+ agent,
346
+ habitsDir,
347
+ osCronManager,
348
+ onHabitFire: (habitName) => {
349
+ processManager.sendToAgent(agent, { type: "habit", habitName });
350
+ },
351
+ deps: {
352
+ readdir: (dir) => fs.readdirSync(dir),
353
+ readFile: (p, enc) => fs.readFileSync(p, enc),
354
+ writeFile: (p, c, enc) => fs.writeFileSync(p, c, enc),
355
+ existsSync: (p) => fs.existsSync(p),
356
+ now: () => Date.now(),
357
+ ouroPath,
358
+ watch: (dir, cb) => fs.watch(dir, cb),
359
+ },
360
+ });
361
+ try {
362
+ scheduler.start();
363
+ scheduler.startPeriodicReconciliation();
364
+ scheduler.watchForChanges();
365
+ habitSchedulers.push(scheduler);
366
+ }
367
+ catch (error) {
368
+ try {
369
+ scheduler.stopWatching();
370
+ scheduler.stop();
371
+ }
372
+ catch {
373
+ // Cleanup is best-effort for partially initialized schedulers.
374
+ }
375
+ emitHabitSetupError(agent, error);
376
+ recordRecoverableBootstrapFailure({
377
+ agent,
378
+ component: degradedComponent,
379
+ habitsDir,
380
+ error,
381
+ guidance: `fix ${agent} habits or cron setup and rerun ouro up to restore habit automation`,
382
+ });
383
+ }
384
+ }
385
+ catch (err) {
386
+ const error = err instanceof Error ? err : new Error(String(err));
387
+ emitHabitSetupError(agent, error);
388
+ recordRecoverableBootstrapFailure({
389
+ agent,
390
+ component: degradedComponent,
391
+ habitsDir,
392
+ error,
393
+ guidance: `fix ${agent} habits or cron setup and rerun ouro up to restore habit automation`,
394
+ });
395
+ }
396
+ }
397
+ healthMonitor.startPeriodicChecks(60_000);
398
+ /* v8 ignore start -- startup failure + signal handlers: call process.exit, untestable in vitest @preserve */
399
+ }).catch(async (err) => {
400
+ const error = err instanceof Error ? err : new Error(String(err));
401
+ _tombstoneWritten = true;
402
+ (0, daemon_tombstone_1.writeDaemonTombstone)("startupFailure", error);
108
403
  (0, runtime_1.emitNervesEvent)({
109
404
  level: "error",
110
405
  component: "daemon",
111
406
  event: "daemon.entry_error",
112
407
  message: "daemon entrypoint failed",
113
- meta: {},
408
+ meta: { error: error.message },
114
409
  });
410
+ setTimeout(() => process.exit(1), 5_000).unref();
115
411
  await daemon.stop();
116
412
  process.exit(1);
117
413
  });
118
414
  process.on("SIGINT", () => {
415
+ // ALWAYS write a tombstone, even on signal-driven shutdown. The previous
416
+ // behavior was to set _gracefulShutdown=true and skip the tombstone, which
417
+ // meant ANY external SIGINT/SIGTERM (launchd policy, OOM killer, manual
418
+ // kill, killOrphanProcesses from a sibling daemon) silently disappeared
419
+ // from the death log. The user lost weeks of visibility into why their
420
+ // daemon kept dying. Tombstones are informational — having a "sigint"
421
+ // tombstone is strictly better than silence.
422
+ _tombstoneWritten = true;
423
+ (0, daemon_tombstone_1.writeDaemonTombstone)("sigint", new Error("daemon received SIGINT"));
424
+ stopEntryRuntime();
425
+ setTimeout(() => process.exit(1), 5_000).unref();
119
426
  void daemon.stop().then(() => process.exit(0));
120
427
  });
121
428
  process.on("SIGTERM", () => {
429
+ _tombstoneWritten = true;
430
+ (0, daemon_tombstone_1.writeDaemonTombstone)("sigterm", new Error("daemon received SIGTERM"));
431
+ stopEntryRuntime();
432
+ setTimeout(() => process.exit(1), 5_000).unref();
122
433
  void daemon.stop().then(() => process.exit(0));
123
434
  });
435
+ /* v8 ignore stop */
436
+ // Suppress EPIPE on stdout/stderr — normal when detached daemon's parent exits
437
+ /* v8 ignore start -- EPIPE suppression: only fires when parent process exits @preserve */
438
+ process.stdout?.on?.("error", () => { });
439
+ process.stderr?.on?.("error", () => { });
440
+ /* v8 ignore stop */
441
+ /* v8 ignore start -- global exception handlers: genuinely untestable in vitest; exercised by real daemon crashes @preserve */
442
+ let _uncaughtCount = 0;
443
+ let _tombstoneWritten = false;
444
+ let _lastKnownCause = null;
445
+ const CIRCUIT_BREAKER_WINDOW_MS = 60_000;
446
+ const CIRCUIT_BREAKER_MAX = 10;
447
+ process.on("uncaughtException", (error) => {
448
+ // EPIPE is normal for detached daemon processes — parent closed the pipe
449
+ if (error.code === "EPIPE")
450
+ return;
451
+ _uncaughtCount++;
452
+ _lastKnownCause = error;
453
+ setTimeout(() => { _uncaughtCount--; }, CIRCUIT_BREAKER_WINDOW_MS).unref();
454
+ _tombstoneWritten = true;
455
+ (0, daemon_tombstone_1.writeDaemonTombstone)("uncaughtException", error);
456
+ (0, runtime_1.emitNervesEvent)({
457
+ level: "error",
458
+ component: "daemon",
459
+ event: "daemon.uncaught_exception",
460
+ message: "uncaught exception in daemon process (continuing)",
461
+ meta: { error: error.message, stack: error.stack ?? null, uncaughtCount: _uncaughtCount },
462
+ });
463
+ // Circuit breaker: if too many exceptions in a short window, the process
464
+ // is in a bad state — exit so launchd/self-spawn can restart fresh.
465
+ if (_uncaughtCount >= CIRCUIT_BREAKER_MAX) {
466
+ (0, runtime_1.emitNervesEvent)({
467
+ level: "error",
468
+ component: "daemon",
469
+ event: "daemon.circuit_breaker_exit",
470
+ message: `daemon exiting: ${_uncaughtCount} uncaught exceptions in ${CIRCUIT_BREAKER_WINDOW_MS / 1000}s`,
471
+ meta: { uncaughtCount: _uncaughtCount },
472
+ });
473
+ setTimeout(() => process.exit(1), 5_000).unref();
474
+ void daemon.stop().then(() => process.exit(1));
475
+ }
476
+ });
477
+ process.on("unhandledRejection", (reason) => {
478
+ const error = reason instanceof Error ? reason : new Error(String(reason));
479
+ _lastKnownCause = error;
480
+ _tombstoneWritten = true;
481
+ (0, daemon_tombstone_1.writeDaemonTombstone)("unhandledRejection", error);
482
+ (0, runtime_1.emitNervesEvent)({
483
+ level: "error",
484
+ component: "daemon",
485
+ event: "daemon.unhandled_rejection",
486
+ message: "unhandled promise rejection in daemon process",
487
+ meta: { reason: error.message, stack: error.stack ?? null },
488
+ });
489
+ });
490
+ // Catch-all: write tombstone on any exit where we didn't already record the cause.
491
+ // process.on('exit') is synchronous-only — writeDaemonTombstone uses writeFileSync, so it works.
492
+ //
493
+ // Previously this skipped writing if `_gracefulShutdown` was true, which made
494
+ // SIGINT/SIGTERM-driven exits invisible in the death log. The signal handlers
495
+ // above now always write their own tombstone before exiting, so this catch-all
496
+ // only runs for exits the signal handlers didn't reach (e.g. process.exit
497
+ // called from somewhere unexpected).
498
+ process.on("exit", (code) => {
499
+ if (_tombstoneWritten)
500
+ return;
501
+ const reason = code === 0 ? "unexpectedCleanExit" : "unexpectedExit";
502
+ const error = _lastKnownCause ?? new Error(`daemon exited with code ${code} (no specific cause captured)`);
503
+ (0, daemon_tombstone_1.writeDaemonTombstone)(reason, error);
504
+ });
505
+ /* v8 ignore stop */
@@ -0,0 +1,183 @@
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.HEALTH_TRACKED_EVENTS = exports.DaemonHealthWriter = void 0;
37
+ exports.isRollupStatus = isRollupStatus;
38
+ exports.isDaemonStatus = isDaemonStatus;
39
+ exports.getDefaultHealthPath = getDefaultHealthPath;
40
+ exports.createHealthNervesSink = createHealthNervesSink;
41
+ exports.readHealth = readHealth;
42
+ const fs = __importStar(require("fs"));
43
+ const os = __importStar(require("os"));
44
+ const path = __importStar(require("path"));
45
+ const runtime_1 = require("../../nerves/runtime");
46
+ /**
47
+ * Daemon-wide rollup vocabulary — locked layer-1 contract.
48
+ *
49
+ * - `RollupStatus` is what `computeDaemonRollup` returns (post-inventory,
50
+ * four-state). The function never returns `"down"` because by the time
51
+ * the rollup is reachable the daemon has already started, opened its
52
+ * socket, and read its agent inventory — pre-inventory failure is the
53
+ * caller's domain.
54
+ * - `DaemonStatus` is what `DaemonHealthState.status` accepts. The caller
55
+ * widens the rollup result with `"down"` along the daemon-entry failure
56
+ * path (e.g. when the daemon process can't read inventory at all).
57
+ *
58
+ * `isRollupStatus` and `isDaemonStatus` are runtime guards used both by
59
+ * `readHealth` (validating cached health files on disk) and by render-side
60
+ * consumers that want to narrow `unknown` JSON into the typed union before
61
+ * branching on it.
62
+ */
63
+ // Single source of truth — the literal lists below are the runtime
64
+ // projection of the type unions. A future literal added to RollupStatus
65
+ // MUST also be added to ROLLUP_STATUS_LITERALS or `satisfies` blows up
66
+ // at tsc. That tightens the Layer 1 contract: producer + consumer +
67
+ // guard all stay in lockstep.
68
+ const ROLLUP_STATUS_LITERALS = ["healthy", "partial", "degraded", "safe-mode"];
69
+ const DAEMON_STATUS_LITERALS = [...ROLLUP_STATUS_LITERALS, "down"];
70
+ const ROLLUP_STATUS_VALUES = new Set(ROLLUP_STATUS_LITERALS);
71
+ const DAEMON_STATUS_VALUES = new Set(DAEMON_STATUS_LITERALS);
72
+ function isRollupStatus(value) {
73
+ return typeof value === "string" && ROLLUP_STATUS_VALUES.has(value);
74
+ }
75
+ function isDaemonStatus(value) {
76
+ return typeof value === "string" && DAEMON_STATUS_VALUES.has(value);
77
+ }
78
+ class DaemonHealthWriter {
79
+ healthPath;
80
+ constructor(healthPath) {
81
+ this.healthPath = healthPath;
82
+ }
83
+ writeHealth(state) {
84
+ try {
85
+ const dir = path.dirname(this.healthPath);
86
+ fs.mkdirSync(dir, { recursive: true });
87
+ const tmpPath = `${this.healthPath}.tmp.${process.pid}`;
88
+ fs.writeFileSync(tmpPath, JSON.stringify(state, null, 2) + "\n", "utf-8");
89
+ fs.renameSync(tmpPath, this.healthPath);
90
+ (0, runtime_1.emitNervesEvent)({
91
+ component: "daemon",
92
+ event: "daemon.health_written",
93
+ message: "daemon health file written",
94
+ meta: { path: this.healthPath, status: state.status },
95
+ });
96
+ }
97
+ catch {
98
+ // Best-effort: if we can't write, don't crash the daemon.
99
+ }
100
+ }
101
+ }
102
+ exports.DaemonHealthWriter = DaemonHealthWriter;
103
+ function getDefaultHealthPath() {
104
+ return path.join(os.homedir(), ".ouro-cli", "daemon-health.json");
105
+ }
106
+ /** Events that trigger a debounced health file write */
107
+ exports.HEALTH_TRACKED_EVENTS = new Set([
108
+ "daemon.habit_cron_verification_failed",
109
+ "daemon.habit_fire",
110
+ "daemon.agent_exit",
111
+ "daemon.agent_started",
112
+ "daemon.agent_config_invalid",
113
+ "daemon.agent_config_failure",
114
+ "daemon.agent_entry_missing",
115
+ "daemon.agent_spawn_failed",
116
+ "daemon.agent_startup_stale_recovered",
117
+ "daemon.agent_restart_exhausted",
118
+ "daemon.agent_permanent_failure",
119
+ "daemon.agent_cooldown_recovery",
120
+ "daemon.bootstrap_degraded",
121
+ "daemon.safe_mode_entered",
122
+ "daemon.habit_scheduler_start",
123
+ ]);
124
+ /**
125
+ * Creates a nerves LogSink that triggers debounced health writes on relevant events.
126
+ * Components don't know about the health writer — they just emit events.
127
+ */
128
+ function createHealthNervesSink(writer, getState) {
129
+ let debounceTimer = null;
130
+ return (entry) => {
131
+ if (!exports.HEALTH_TRACKED_EVENTS.has(entry.event)) {
132
+ return;
133
+ }
134
+ // Debounce: max once per second
135
+ if (debounceTimer !== null) {
136
+ clearTimeout(debounceTimer);
137
+ }
138
+ debounceTimer = setTimeout(() => {
139
+ debounceTimer = null;
140
+ const state = getState();
141
+ writer.writeHealth(state);
142
+ }, 1000);
143
+ };
144
+ }
145
+ function readHealth(healthPath) {
146
+ try {
147
+ const raw = fs.readFileSync(healthPath, "utf-8");
148
+ const parsed = JSON.parse(raw);
149
+ if (!isDaemonStatus(parsed.status) ||
150
+ typeof parsed.mode !== "string" ||
151
+ typeof parsed.pid !== "number" ||
152
+ typeof parsed.startedAt !== "string" ||
153
+ typeof parsed.uptimeSeconds !== "number" ||
154
+ !Array.isArray(parsed.degraded) ||
155
+ typeof parsed.agents !== "object" ||
156
+ parsed.agents === null ||
157
+ typeof parsed.habits !== "object" ||
158
+ parsed.habits === null) {
159
+ return null;
160
+ }
161
+ // `drift` is required in DaemonHealthState but absent from cached
162
+ // health files written by pre-Layer-4 daemons. Tolerate that legacy
163
+ // shape by defaulting to []; the rest of the file is still valid.
164
+ const drift = Array.isArray(parsed.drift)
165
+ ? parsed.drift
166
+ : [];
167
+ return {
168
+ status: parsed.status,
169
+ mode: parsed.mode,
170
+ pid: parsed.pid,
171
+ startedAt: parsed.startedAt,
172
+ uptimeSeconds: parsed.uptimeSeconds,
173
+ safeMode: parsed.safeMode,
174
+ degraded: parsed.degraded,
175
+ drift,
176
+ agents: parsed.agents,
177
+ habits: parsed.habits,
178
+ };
179
+ }
180
+ catch {
181
+ return null;
182
+ }
183
+ }
@@ -0,0 +1,58 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.computeDaemonRollup = computeDaemonRollup;
4
+ /**
5
+ * Pure rollup decision function — given the post-inventory daemon
6
+ * surface, returns the daemon-wide rollup state per the locked Layer 1
7
+ * vocabulary table:
8
+ *
9
+ * | rollup | when |
10
+ * | ---------- | --------------------------------------------------------- |
11
+ * | healthy | every enabled agent serving + no bootstrap-degraded + no safe-mode |
12
+ * | partial | (≥1 serving + ≥1 not serving) OR (all serving + ≥1 bootstrap-degraded) |
13
+ * | degraded | zero enabled agents serving (fresh install OR all unhealthy) |
14
+ * | safe-mode | `safeMode === true` overrides everything else |
15
+ *
16
+ * The function NEVER returns `"down"`. By the time `computeDaemonRollup`
17
+ * is reachable, the daemon process has started, opened its socket, and
18
+ * read its agent inventory — pre-inventory failure is the caller's
19
+ * domain. `daemon-entry.ts`'s startup-failure path assigns `"down"` to
20
+ * `DaemonHealthState.status` directly without consulting this function.
21
+ */
22
+ function computeDaemonRollup(input) {
23
+ // Safe mode wins, period. Crash-loop detection trumps everything —
24
+ // we want the human to see SAFE MODE, not a noisy partial/degraded.
25
+ if (input.safeMode) {
26
+ return "safe-mode";
27
+ }
28
+ // Count serving agents. "Serving" = "running" worker status.
29
+ // Anything else (crashed/stopped/starting/etc) is not serving.
30
+ let serving = 0;
31
+ let notServing = 0;
32
+ for (const agent of input.enabledAgents) {
33
+ if (agent.status === "running") {
34
+ serving++;
35
+ }
36
+ else {
37
+ notServing++;
38
+ }
39
+ }
40
+ // Zero-serving wins over bootstrap-degraded — we have no working
41
+ // agents to surface a "partially working" story about. This covers
42
+ // both fresh-install (`enabledAgents.length === 0`) and
43
+ // all-failed-live-check (`serving === 0` with `notServing > 0`).
44
+ // Render layer (cli-render.ts) splits the UX copy by inspecting the
45
+ // agents map; the rollup itself doesn't carry the distinction.
46
+ if (serving === 0) {
47
+ return "degraded";
48
+ }
49
+ // From here we have ≥1 serving agent. The remaining choice is
50
+ // healthy vs partial.
51
+ const hasUnhealthyAgent = notServing > 0;
52
+ const hasBootstrapDegraded = input.bootstrapDegraded.length > 0;
53
+ const hasDrift = input.driftDetected === true;
54
+ if (hasUnhealthyAgent || hasBootstrapDegraded || hasDrift) {
55
+ return "partial";
56
+ }
57
+ return "healthy";
58
+ }