@ouro.bot/cli 0.1.0-alpha.52 → 0.1.0-alpha.520

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 (375) hide show
  1. package/README.md +133 -19
  2. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
  3. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
  4. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  5. package/changelog.json +3353 -0
  6. package/dist/arc/attention-types.js +8 -0
  7. package/dist/arc/cares.js +140 -0
  8. package/dist/arc/episodes.js +117 -0
  9. package/dist/arc/intentions.js +133 -0
  10. package/dist/arc/json-store.js +117 -0
  11. package/dist/arc/obligations.js +237 -0
  12. package/dist/arc/packets.js +193 -0
  13. package/dist/arc/presence.js +185 -0
  14. package/dist/arc/task-lifecycle.js +65 -0
  15. package/dist/heart/active-work.js +837 -26
  16. package/dist/heart/agent-entry.js +58 -3
  17. package/dist/heart/attachments/image-normalize.js +194 -0
  18. package/dist/heart/attachments/materialize.js +97 -0
  19. package/dist/heart/attachments/originals.js +88 -0
  20. package/dist/heart/attachments/render.js +29 -0
  21. package/dist/heart/attachments/sources/adapter.js +2 -0
  22. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  23. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  24. package/dist/heart/attachments/sources/index.js +16 -0
  25. package/dist/heart/attachments/store.js +103 -0
  26. package/dist/heart/attachments/types.js +93 -0
  27. package/dist/heart/auth/auth-flow.js +427 -0
  28. package/dist/heart/background-operations.js +281 -0
  29. package/dist/heart/bundle-state.js +168 -0
  30. package/dist/heart/commitments.js +111 -0
  31. package/dist/heart/config-registry.js +304 -0
  32. package/dist/heart/config.js +119 -129
  33. package/dist/heart/core.js +948 -243
  34. package/dist/heart/cross-chat-delivery.js +3 -18
  35. package/dist/heart/daemon/agent-config-check.js +512 -0
  36. package/dist/heart/daemon/agent-discovery.js +102 -3
  37. package/dist/heart/daemon/agent-service.js +360 -0
  38. package/dist/heart/daemon/agentic-repair.js +554 -0
  39. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  40. package/dist/heart/daemon/boot-sync-probe.js +197 -0
  41. package/dist/heart/daemon/cadence.js +70 -0
  42. package/dist/heart/daemon/cli-defaults.js +643 -0
  43. package/dist/heart/daemon/cli-exec.js +7476 -0
  44. package/dist/heart/daemon/cli-help.js +493 -0
  45. package/dist/heart/daemon/cli-parse.js +1557 -0
  46. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  47. package/dist/heart/daemon/cli-render.js +649 -0
  48. package/dist/heart/daemon/cli-types.js +8 -0
  49. package/dist/heart/daemon/connect-bay.js +323 -0
  50. package/dist/heart/daemon/daemon-cli.js +29 -1631
  51. package/dist/heart/daemon/daemon-entry.js +404 -3
  52. package/dist/heart/daemon/daemon-health.js +183 -0
  53. package/dist/heart/daemon/daemon-rollup.js +58 -0
  54. package/dist/heart/daemon/daemon-runtime-sync.js +190 -12
  55. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  56. package/dist/heart/daemon/daemon.js +758 -60
  57. package/dist/heart/daemon/dns-workflow.js +394 -0
  58. package/dist/heart/daemon/doctor-types.js +8 -0
  59. package/dist/heart/daemon/doctor.js +837 -0
  60. package/dist/heart/daemon/drift-detection.js +146 -0
  61. package/dist/heart/daemon/health-monitor.js +92 -1
  62. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  63. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  64. package/dist/heart/daemon/http-health-probe.js +80 -0
  65. package/dist/heart/daemon/human-command-screens.js +234 -0
  66. package/dist/heart/daemon/human-readiness.js +114 -0
  67. package/dist/heart/daemon/inner-status.js +102 -0
  68. package/dist/heart/daemon/interactive-repair.js +394 -0
  69. package/dist/heart/daemon/launchd.js +25 -5
  70. package/dist/heart/daemon/log-tailer.js +82 -12
  71. package/dist/heart/daemon/logs-prune.js +110 -0
  72. package/dist/heart/daemon/message-router.js +2 -2
  73. package/dist/heart/daemon/os-cron-deps.js +134 -0
  74. package/dist/heart/daemon/ouro-bot-entry.js +4 -2
  75. package/dist/heart/daemon/ouro-entry.js +3 -1
  76. package/dist/heart/daemon/process-manager.js +381 -26
  77. package/dist/heart/daemon/provider-discovery.js +137 -0
  78. package/dist/heart/daemon/provider-ping-progress.js +83 -0
  79. package/dist/heart/daemon/pulse.js +475 -0
  80. package/dist/heart/daemon/readiness-repair.js +365 -0
  81. package/dist/heart/daemon/run-hooks.js +2 -0
  82. package/dist/heart/daemon/runtime-logging.js +67 -16
  83. package/dist/heart/daemon/runtime-metadata.js +73 -0
  84. package/dist/heart/daemon/runtime-mode.js +67 -0
  85. package/dist/heart/daemon/safe-mode.js +161 -0
  86. package/dist/heart/daemon/sense-manager.js +259 -37
  87. package/dist/heart/daemon/session-id-resolver.js +131 -0
  88. package/dist/heart/daemon/skill-management-installer.js +94 -0
  89. package/dist/heart/daemon/socket-client.js +109 -4
  90. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  91. package/dist/heart/daemon/startup-tui.js +330 -0
  92. package/dist/heart/daemon/task-scheduler.js +3 -25
  93. package/dist/heart/daemon/terminal-ui.js +499 -0
  94. package/dist/heart/daemon/thoughts.js +162 -17
  95. package/dist/heart/daemon/up-progress.js +366 -0
  96. package/dist/heart/daemon/vault-items.js +56 -0
  97. package/dist/heart/delegation.js +1 -1
  98. package/dist/heart/habits/habit-migration.js +189 -0
  99. package/dist/heart/habits/habit-parser.js +140 -0
  100. package/dist/heart/habits/habit-runtime-state.js +100 -0
  101. package/dist/heart/habits/habit-scheduler.js +372 -0
  102. package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -117
  103. package/dist/heart/{daemon → hatch}/hatch-specialist.js +6 -8
  104. package/dist/heart/{daemon → hatch}/specialist-prompt.js +12 -9
  105. package/dist/heart/{daemon → hatch}/specialist-tools.js +35 -12
  106. package/dist/heart/identity.js +205 -66
  107. package/dist/heart/kept-notes.js +357 -0
  108. package/dist/heart/kicks.js +1 -1
  109. package/dist/heart/machine-identity.js +161 -0
  110. package/dist/heart/mail-import-discovery.js +353 -0
  111. package/dist/heart/mcp/mcp-server.js +653 -0
  112. package/dist/heart/migrate-config.js +100 -0
  113. package/dist/heart/model-capabilities.js +19 -0
  114. package/dist/heart/outlook/outlook-http-hooks.js +66 -0
  115. package/dist/heart/outlook/outlook-http-response.js +7 -0
  116. package/dist/heart/outlook/outlook-http-routes.js +244 -0
  117. package/dist/heart/outlook/outlook-http-static.js +103 -0
  118. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  119. package/dist/heart/outlook/outlook-http.js +99 -0
  120. package/dist/heart/outlook/outlook-read.js +31 -0
  121. package/dist/heart/outlook/outlook-types.js +27 -0
  122. package/dist/heart/outlook/outlook-view.js +195 -0
  123. package/dist/heart/outlook/readers/agent-machine.js +382 -0
  124. package/dist/heart/outlook/readers/continuity-readers.js +336 -0
  125. package/dist/heart/outlook/readers/mail.js +362 -0
  126. package/dist/heart/outlook/readers/runtime-readers.js +650 -0
  127. package/dist/heart/outlook/readers/sessions.js +232 -0
  128. package/dist/heart/outlook/readers/shared.js +111 -0
  129. package/dist/heart/platform.js +81 -0
  130. package/dist/heart/provider-attempt.js +134 -0
  131. package/dist/heart/provider-binding-resolver.js +255 -0
  132. package/dist/heart/provider-credentials.js +425 -0
  133. package/dist/heart/provider-failover.js +301 -0
  134. package/dist/heart/provider-models.js +81 -0
  135. package/dist/heart/provider-ping.js +262 -0
  136. package/dist/heart/provider-state.js +216 -0
  137. package/dist/heart/provider-visibility.js +188 -0
  138. package/dist/heart/providers/anthropic-token.js +131 -0
  139. package/dist/heart/providers/anthropic.js +139 -52
  140. package/dist/heart/providers/azure.js +97 -13
  141. package/dist/heart/providers/error-classification.js +127 -0
  142. package/dist/heart/providers/github-copilot.js +145 -0
  143. package/dist/heart/providers/minimax-vlm.js +189 -0
  144. package/dist/heart/providers/minimax.js +26 -8
  145. package/dist/heart/providers/openai-codex.js +55 -40
  146. package/dist/heart/runtime-capability-check.js +170 -0
  147. package/dist/heart/runtime-credentials.js +367 -0
  148. package/dist/heart/sense-truth.js +11 -4
  149. package/dist/heart/session-activity.js +43 -22
  150. package/dist/heart/session-events.js +1149 -0
  151. package/dist/heart/session-playback-cli-main.js +5 -0
  152. package/dist/heart/session-playback-cli.js +36 -0
  153. package/dist/heart/session-playback.js +231 -0
  154. package/dist/heart/session-stats-cli-main.js +5 -0
  155. package/dist/heart/session-stats.js +182 -0
  156. package/dist/heart/session-transcript.js +243 -0
  157. package/dist/heart/start-of-turn-packet.js +345 -0
  158. package/dist/heart/streaming.js +44 -27
  159. package/dist/heart/sync-classification.js +176 -0
  160. package/dist/heart/sync.js +449 -0
  161. package/dist/heart/target-resolution.js +9 -5
  162. package/dist/heart/tempo.js +93 -0
  163. package/dist/heart/temporal-view.js +41 -0
  164. package/dist/heart/timeouts.js +101 -0
  165. package/dist/heart/tool-activity-callbacks.js +36 -0
  166. package/dist/heart/tool-description.js +139 -0
  167. package/dist/heart/tool-friction.js +55 -0
  168. package/dist/heart/tool-loop.js +200 -0
  169. package/dist/heart/turn-context.js +381 -0
  170. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  171. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  172. package/dist/heart/versioning/ouro-path-installer.js +425 -0
  173. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  174. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  175. package/dist/heart/{daemon → versioning}/update-checker.js +5 -1
  176. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  177. package/dist/mailroom/attention.js +167 -0
  178. package/dist/mailroom/autonomy.js +209 -0
  179. package/dist/mailroom/blob-store.js +606 -0
  180. package/dist/mailroom/body-cache.js +61 -0
  181. package/dist/mailroom/core.js +672 -0
  182. package/dist/mailroom/entry.js +160 -0
  183. package/dist/mailroom/file-store.js +426 -0
  184. package/dist/mailroom/mbox-import.js +382 -0
  185. package/dist/mailroom/outbound.js +380 -0
  186. package/dist/mailroom/policy.js +263 -0
  187. package/dist/mailroom/reader.js +228 -0
  188. package/dist/mailroom/search-cache.js +182 -0
  189. package/dist/mailroom/search-relevance.js +319 -0
  190. package/dist/mailroom/smtp-ingress.js +176 -0
  191. package/dist/mailroom/source-state.js +176 -0
  192. package/dist/mailroom/thread.js +109 -0
  193. package/dist/mailroom/travel-extract.js +89 -0
  194. package/dist/mind/bundle-manifest.js +7 -1
  195. package/dist/mind/context.js +165 -101
  196. package/dist/mind/diary-integrity.js +60 -0
  197. package/dist/mind/{memory.js → diary.js} +74 -93
  198. package/dist/mind/embedding-provider.js +60 -0
  199. package/dist/mind/file-state.js +179 -0
  200. package/dist/mind/friends/channel.js +30 -0
  201. package/dist/mind/friends/resolver.js +54 -2
  202. package/dist/mind/friends/store-file.js +39 -3
  203. package/dist/mind/friends/types.js +2 -2
  204. package/dist/mind/journal-index.js +161 -0
  205. package/dist/mind/note-search.js +268 -0
  206. package/dist/mind/obligation-steering.js +221 -0
  207. package/dist/mind/pending.js +4 -0
  208. package/dist/mind/prompt-refresh.js +3 -2
  209. package/dist/mind/prompt.js +942 -122
  210. package/dist/mind/provenance-trust.js +26 -0
  211. package/dist/mind/scrutiny.js +173 -0
  212. package/dist/nerves/cli-logging.js +7 -1
  213. package/dist/nerves/coverage/audit-rules.js +15 -6
  214. package/dist/nerves/coverage/audit.js +28 -2
  215. package/dist/nerves/coverage/cli.js +1 -1
  216. package/dist/nerves/coverage/contract.js +5 -5
  217. package/dist/nerves/coverage/file-completeness.js +139 -5
  218. package/dist/nerves/coverage/run-artifacts.js +1 -1
  219. package/dist/nerves/event-buffer.js +111 -0
  220. package/dist/nerves/index.js +224 -4
  221. package/dist/nerves/observation.js +20 -0
  222. package/dist/nerves/redact.js +79 -0
  223. package/dist/nerves/review/cli-main.js +5 -0
  224. package/dist/nerves/review/cli.js +156 -0
  225. package/dist/nerves/review/core.js +152 -0
  226. package/dist/nerves/runtime.js +5 -1
  227. package/dist/outlook-ui/assets/index-BPr5vNuM.css +1 -0
  228. package/dist/outlook-ui/assets/index-Cm51CY9W.js +61 -0
  229. package/dist/outlook-ui/index.html +15 -0
  230. package/dist/repertoire/ado-client.js +15 -56
  231. package/dist/repertoire/ado-semantic.js +11 -10
  232. package/dist/repertoire/api-client.js +97 -0
  233. package/dist/repertoire/bitwarden-store.js +816 -0
  234. package/dist/repertoire/bundle-templates.js +72 -0
  235. package/dist/repertoire/bw-installer.js +180 -0
  236. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  237. package/dist/repertoire/coding/context-pack.js +330 -0
  238. package/dist/repertoire/coding/feedback.js +197 -30
  239. package/dist/repertoire/coding/manager.js +158 -9
  240. package/dist/repertoire/coding/spawner.js +55 -9
  241. package/dist/repertoire/coding/tools.js +170 -7
  242. package/dist/repertoire/commerce-errors.js +109 -0
  243. package/dist/repertoire/commerce-self-test.js +156 -0
  244. package/dist/repertoire/credential-access.js +111 -0
  245. package/dist/repertoire/duffel-client.js +185 -0
  246. package/dist/repertoire/github-client.js +14 -55
  247. package/dist/repertoire/graph-client.js +11 -52
  248. package/dist/repertoire/guardrails.js +396 -0
  249. package/dist/repertoire/mcp-client.js +255 -0
  250. package/dist/repertoire/mcp-manager.js +305 -0
  251. package/dist/repertoire/mcp-tools.js +63 -0
  252. package/dist/repertoire/shell-sessions.js +133 -0
  253. package/dist/repertoire/skills.js +15 -24
  254. package/dist/repertoire/stripe-client.js +131 -0
  255. package/dist/repertoire/tasks/board.js +31 -5
  256. package/dist/repertoire/tasks/fix.js +182 -0
  257. package/dist/repertoire/tasks/index.js +16 -4
  258. package/dist/repertoire/tasks/lifecycle.js +2 -2
  259. package/dist/repertoire/tasks/parser.js +3 -2
  260. package/dist/repertoire/tasks/scanner.js +194 -37
  261. package/dist/repertoire/tasks/transitions.js +16 -78
  262. package/dist/repertoire/tool-results.js +29 -0
  263. package/dist/repertoire/tools-attachments.js +317 -0
  264. package/dist/repertoire/tools-base.js +47 -1075
  265. package/dist/repertoire/tools-bluebubbles.js +1 -0
  266. package/dist/repertoire/tools-bridge.js +142 -0
  267. package/dist/repertoire/tools-bundle.js +984 -0
  268. package/dist/repertoire/tools-config.js +185 -0
  269. package/dist/repertoire/tools-continuity.js +248 -0
  270. package/dist/repertoire/tools-credential.js +381 -0
  271. package/dist/repertoire/tools-files.js +342 -0
  272. package/dist/repertoire/tools-flight.js +224 -0
  273. package/dist/repertoire/tools-flow.js +119 -0
  274. package/dist/repertoire/tools-github.js +1 -7
  275. package/dist/repertoire/tools-mail.js +1477 -0
  276. package/dist/repertoire/tools-notes.js +421 -0
  277. package/dist/repertoire/tools-session.js +750 -0
  278. package/dist/repertoire/tools-shell.js +120 -0
  279. package/dist/repertoire/tools-stripe.js +180 -0
  280. package/dist/repertoire/tools-surface.js +243 -0
  281. package/dist/repertoire/tools-teams.js +9 -39
  282. package/dist/repertoire/tools-travel.js +125 -0
  283. package/dist/repertoire/tools-trip.js +422 -0
  284. package/dist/repertoire/tools-user-profile.js +144 -0
  285. package/dist/repertoire/tools-vault.js +40 -0
  286. package/dist/repertoire/tools.js +108 -100
  287. package/dist/repertoire/travel-api-client.js +360 -0
  288. package/dist/repertoire/user-profile.js +131 -0
  289. package/dist/repertoire/vault-setup.js +246 -0
  290. package/dist/repertoire/vault-unlock.js +561 -0
  291. package/dist/scripts/claude-code-hook.js +41 -0
  292. package/dist/scripts/claude-code-stop-hook.js +47 -0
  293. package/dist/senses/attention-queue.js +116 -0
  294. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  295. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  296. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
  297. package/dist/senses/bluebubbles/entry.js +77 -0
  298. package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
  299. package/dist/senses/bluebubbles/index.js +1947 -0
  300. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  301. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  302. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
  303. package/dist/senses/bluebubbles/processed-log.js +111 -0
  304. package/dist/senses/bluebubbles/replay.js +129 -0
  305. package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +2 -2
  306. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  307. package/dist/senses/cli/bracketed-paste.js +82 -0
  308. package/dist/senses/cli/image-paste.js +287 -0
  309. package/dist/senses/cli/image-ref-navigation.js +75 -0
  310. package/dist/senses/cli/ink-app.js +156 -0
  311. package/dist/senses/cli/inline-diff.js +64 -0
  312. package/dist/senses/cli/input-keys.js +174 -0
  313. package/dist/senses/cli/kill-ring.js +86 -0
  314. package/dist/senses/cli/message-list.js +51 -0
  315. package/dist/senses/cli/ouro-tui.js +607 -0
  316. package/dist/senses/cli/spinner-imperative.js +135 -0
  317. package/dist/senses/cli/spinner.js +101 -0
  318. package/dist/senses/cli/status-line.js +60 -0
  319. package/dist/senses/cli/streaming-markdown.js +526 -0
  320. package/dist/senses/cli/tool-display.js +85 -0
  321. package/dist/senses/cli/tool-render.js +85 -0
  322. package/dist/senses/cli/tui-store.js +240 -0
  323. package/dist/senses/cli/virtual-list.js +35 -0
  324. package/dist/senses/cli-entry.js +60 -8
  325. package/dist/senses/cli-layout.js +187 -0
  326. package/dist/senses/cli.js +520 -209
  327. package/dist/senses/commands.js +66 -3
  328. package/dist/senses/habit-turn-message.js +108 -0
  329. package/dist/senses/inner-dialog-worker.js +175 -21
  330. package/dist/senses/inner-dialog.js +330 -27
  331. package/dist/senses/mail-entry.js +66 -0
  332. package/dist/senses/mail.js +379 -0
  333. package/dist/senses/pipeline.js +569 -182
  334. package/dist/senses/proactive-content-guard.js +51 -0
  335. package/dist/senses/shared-turn.js +248 -0
  336. package/dist/senses/surface-tool.js +68 -0
  337. package/dist/senses/teams-entry.js +60 -8
  338. package/dist/senses/teams.js +387 -98
  339. package/dist/senses/trust-gate.js +100 -5
  340. package/dist/trips/core.js +138 -0
  341. package/dist/trips/store.js +146 -0
  342. package/package.json +37 -7
  343. package/skills/agent-commerce.md +106 -0
  344. package/skills/browser-navigation.md +117 -0
  345. package/skills/commerce-setup-guide.md +116 -0
  346. package/skills/commerce-setup.md +84 -0
  347. package/skills/configure-dev-tools.md +101 -0
  348. package/skills/travel-planning.md +138 -0
  349. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  350. package/dist/heart/daemon/subagent-installer.js +0 -166
  351. package/dist/heart/session-recall.js +0 -116
  352. package/dist/mind/associative-recall.js +0 -209
  353. package/dist/senses/bluebubbles-entry.js +0 -13
  354. package/dist/senses/bluebubbles.js +0 -1177
  355. package/dist/senses/debug-activity.js +0 -148
  356. package/subagents/README.md +0 -86
  357. package/subagents/work-doer.md +0 -237
  358. package/subagents/work-merger.md +0 -618
  359. package/subagents/work-planner.md +0 -390
  360. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  361. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  362. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  363. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  364. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  365. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  366. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  367. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  368. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  369. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  370. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  371. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  372. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  373. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  374. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  375. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -34,20 +34,303 @@ var __importStar = (this && this.__importStar) || (function () {
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
36
  exports.OuroDaemon = void 0;
37
+ exports.parseOrphanPidsFromPs = parseOrphanPidsFromPs;
38
+ exports.filterPidfilePidsToActualOrphans = filterPidfilePidsToActualOrphans;
39
+ exports.mergeUniqueOrphanPids = mergeUniqueOrphanPids;
40
+ exports.killOrphanProcesses = killOrphanProcesses;
41
+ exports.writePidfile = writePidfile;
42
+ exports.handleAgentSenseTurn = handleAgentSenseTurn;
37
43
  const fs = __importStar(require("fs"));
38
44
  const net = __importStar(require("net"));
45
+ const os = __importStar(require("os"));
39
46
  const path = __importStar(require("path"));
40
47
  const identity_1 = require("../identity");
48
+ const agent_discovery_1 = require("./agent-discovery");
41
49
  const runtime_1 = require("../../nerves/runtime");
42
50
  const runtime_metadata_1 = require("./runtime-metadata");
43
- const update_hooks_1 = require("./update-hooks");
51
+ const runtime_mode_1 = require("./runtime-mode");
52
+ const update_hooks_1 = require("../versioning/update-hooks");
44
53
  const bundle_meta_1 = require("./hooks/bundle-meta");
54
+ const agent_config_v2_1 = require("./hooks/agent-config-v2");
45
55
  const bundle_manifest_1 = require("../../mind/bundle-manifest");
46
- const update_checker_1 = require("./update-checker");
47
- const staged_restart_1 = require("./staged-restart");
56
+ const update_checker_1 = require("../versioning/update-checker");
57
+ const staged_restart_1 = require("../versioning/staged-restart");
58
+ const ouro_version_manager_1 = require("../versioning/ouro-version-manager");
48
59
  const child_process_1 = require("child_process");
49
60
  const pending_1 = require("../../mind/pending");
61
+ const agent_service_1 = require("./agent-service");
50
62
  const channel_1 = require("../../mind/friends/channel");
63
+ const mcp_manager_1 = require("../../repertoire/mcp-manager");
64
+ const outlook_http_1 = require("../outlook/outlook-http");
65
+ const outlook_types_1 = require("../outlook/outlook-types");
66
+ const outlook_read_1 = require("../outlook/outlook-read");
67
+ const outlook_view_1 = require("../outlook/outlook-view");
68
+ const provider_visibility_1 = require("../provider-visibility");
69
+ const socket_client_1 = require("./socket-client");
70
+ const PIDFILE_PATH = path.join(os.homedir(), ".ouro-cli", "daemon.pids");
71
+ /**
72
+ * Defense-in-depth: detect if we're running under vitest. The pidfile lives
73
+ * at a hardcoded path under the user's real ~/.ouro-cli/ — there's no DI
74
+ * seam to redirect it. So when a test creates a real OuroDaemon and calls
75
+ * start(), the daemon's killOrphanProcesses() reads the REAL pidfile,
76
+ * ps-verifies the PIDs, and SIGTERMs the production daemon. We saw this
77
+ * cause an outage on 2026-04-08 (alpha.265 daemon killed 93s after startup
78
+ * by a vitest test that called daemon.start()).
79
+ *
80
+ * Both killOrphanProcesses() and writePidfile() short-circuit under vitest
81
+ * to make the production pidfile sacred. Tests that need to verify these
82
+ * functions' behavior should use the extracted pure helpers
83
+ * (parseOrphanPidsFromPs, filterPidfilePidsToActualOrphans).
84
+ */
85
+ function isVitestProcess() {
86
+ /* v8 ignore next -- defensive: process and process.argv always exist in node @preserve */
87
+ if (typeof process === "undefined" || !Array.isArray(process.argv))
88
+ return false;
89
+ return process.argv.some((arg) => typeof arg === "string" && arg.includes("vitest"));
90
+ }
91
+ /**
92
+ * Scan `ps -eo pid,ppid,command` output for daemon-owned entry points whose
93
+ * parent has died (PPID reparented to init/PID 1). Returns the list of PIDs
94
+ * that are safe to SIGTERM — true orphans, not children of live sibling
95
+ * daemons running from worktrees, test suites, or other users of the harness.
96
+ *
97
+ * Exported so unit tests can exercise the filter without shelling out.
98
+ */
99
+ function parseOrphanPidsFromPs(psOutput, selfPid) {
100
+ const orphans = [];
101
+ for (const line of psOutput.split("\n")) {
102
+ // Explicitly exclude MCP server processes — they share a harness entry
103
+ // point but are not daemon children and must never be killed.
104
+ if (line.includes("mcp-serve") || line.includes("mcp serve"))
105
+ continue;
106
+ // Match only daemon-owned JS entry points.
107
+ if (!line.includes("agent-entry.js")
108
+ && !line.includes("daemon-entry.js")
109
+ && !line.includes("bluebubbles/entry.js")
110
+ && !line.includes("teams-entry.js"))
111
+ continue;
112
+ // Parse `<pid> <ppid> <command...>`. ps pads these with leading spaces.
113
+ // Regex guarantees both groups are \d+ so parseInt can't produce NaN.
114
+ const match = line.trim().match(/^(\d+)\s+(\d+)\s/);
115
+ if (!match)
116
+ continue;
117
+ const pid = parseInt(match[1], 10);
118
+ const ppid = parseInt(match[2], 10);
119
+ if (pid === selfPid)
120
+ continue;
121
+ // CRITICAL: only kill processes whose parent is init (PID 1). A live
122
+ // PPID means the process belongs to another daemon instance (parallel
123
+ // test run, sibling worktree, another user of /tmp/ouroboros-daemon.sock).
124
+ // Killing those will crash unrelated harnesses — we saw this in B6
125
+ // when a vitest worker's daemon killed slugger's production children.
126
+ if (ppid !== 1)
127
+ continue;
128
+ orphans.push(pid);
129
+ }
130
+ return orphans;
131
+ }
132
+ /**
133
+ * Given a list of PIDs from the pidfile, return only those that are actual
134
+ * orphans (PPID reparented to init/PID 1). Protects against a polluted
135
+ * pidfile killing a PID that the OS has reassigned to an unrelated process.
136
+ *
137
+ * Implementation: shells out to `ps -p <csv> -o pid,ppid` for a batch lookup.
138
+ * Returns the empty list if ps fails — safer to skip cleanup than to
139
+ * wildcard-kill on a bad read.
140
+ *
141
+ * Exported for direct unit coverage.
142
+ */
143
+ function filterPidfilePidsToActualOrphans(candidatePids, psRunner = runPsCheck) {
144
+ if (candidatePids.length === 0)
145
+ return [];
146
+ const psOutput = psRunner(candidatePids);
147
+ if (psOutput === null)
148
+ return [];
149
+ const survivingOrphans = [];
150
+ // `ps -p x,y,z -o pid,ppid` emits a header line then one row per found PID.
151
+ // PIDs not found (already exited) are silently omitted — which is the
152
+ // correct behavior for us: we only want to kill live orphans.
153
+ for (const line of psOutput.split("\n")) {
154
+ const match = line.trim().match(/^(\d+)\s+(\d+)$/);
155
+ if (!match)
156
+ continue;
157
+ const pid = parseInt(match[1], 10);
158
+ const ppid = parseInt(match[2], 10);
159
+ if (ppid !== 1)
160
+ continue;
161
+ if (!candidatePids.includes(pid))
162
+ continue;
163
+ survivingOrphans.push(pid);
164
+ }
165
+ return survivingOrphans;
166
+ }
167
+ function mergeUniqueOrphanPids(...sources) {
168
+ const merged = [];
169
+ const seen = new Set();
170
+ for (const source of sources) {
171
+ for (const pid of source) {
172
+ if (seen.has(pid))
173
+ continue;
174
+ seen.add(pid);
175
+ merged.push(pid);
176
+ }
177
+ }
178
+ return merged;
179
+ }
180
+ /* v8 ignore start -- shells out to ps; covered by filterPidfilePidsToActualOrphans unit tests via injected runner @preserve */
181
+ function runPsCheck(pids) {
182
+ try {
183
+ const csv = pids.join(",");
184
+ return (0, child_process_1.execSync)(`ps -p ${csv} -o pid=,ppid=`, { encoding: "utf-8", timeout: 5000 });
185
+ }
186
+ catch {
187
+ // ps returns non-zero when none of the requested PIDs exist. Treat as
188
+ // "no survivors" rather than an error.
189
+ return "";
190
+ }
191
+ }
192
+ /* v8 ignore stop */
193
+ /**
194
+ * Kill all ouro processes from the previous daemon instance using the pidfile.
195
+ * On startup, reads PIDs from ~/.ouro-cli/daemon.pids, kills them all, then
196
+ * deletes the file. The new daemon writes its own PIDs after spawning.
197
+ *
198
+ * Safety: pidfile contents are verified before being killed — each PID must
199
+ * be an actual orphan (PPID reparented to init/PID 1) via
200
+ * `filterPidfilePidsToActualOrphans`. Otherwise a polluted pidfile (written
201
+ * by a test, or a crashed daemon whose PIDs have since been reused by the
202
+ * OS) could SIGTERM unrelated processes.
203
+ *
204
+ * Falls back to ps-based scanning scoped to true orphans (PPID=1) if the
205
+ * pidfile doesn't exist (first run, previous daemon crashed before writing,
206
+ * manual cleanup). The scope is narrow on purpose — see parseOrphanPidsFromPs.
207
+ */
208
+ /* v8 ignore start -- process lifecycle: uses kill/ps, tested via deployment @preserve */
209
+ function isProductionDaemonSocketPath(socketPath) {
210
+ return socketPath === socket_client_1.DEFAULT_DAEMON_SOCKET_PATH;
211
+ }
212
+ function killOrphanProcesses(socketPath = socket_client_1.DEFAULT_DAEMON_SOCKET_PATH) {
213
+ if (!isProductionDaemonSocketPath(socketPath)) {
214
+ (0, runtime_1.emitNervesEvent)({
215
+ level: "warn",
216
+ component: "daemon",
217
+ event: "daemon.orphan_cleanup_nonproduction_blocked",
218
+ message: "blocked orphan cleanup for non-production daemon socket",
219
+ meta: { socketPath, pidfilePath: PIDFILE_PATH },
220
+ });
221
+ return;
222
+ }
223
+ if (isVitestProcess()) {
224
+ (0, runtime_1.emitNervesEvent)({
225
+ level: "warn",
226
+ component: "daemon",
227
+ event: "daemon.orphan_cleanup_test_blocked",
228
+ message: "blocked killOrphanProcesses from touching real pidfile under vitest",
229
+ meta: { pidfilePath: PIDFILE_PATH },
230
+ });
231
+ return;
232
+ }
233
+ try {
234
+ let pidfileOrphans = [];
235
+ let scanOrphans = [];
236
+ // Primary: read pidfile from previous daemon
237
+ try {
238
+ const raw = fs.readFileSync(PIDFILE_PATH, "utf-8");
239
+ const candidates = raw.split("\n")
240
+ .map((s) => parseInt(s.trim(), 10))
241
+ .filter((n) => !isNaN(n) && n !== process.pid);
242
+ // Verify each candidate is an actual live orphan before killing. See
243
+ // docstring above for why this matters.
244
+ pidfileOrphans = filterPidfilePidsToActualOrphans(candidates);
245
+ fs.unlinkSync(PIDFILE_PATH);
246
+ }
247
+ catch {
248
+ // No pidfile — the ps scan below still covers true orphans.
249
+ }
250
+ // Always supplement the pidfile with the scoped ps scan. A stale or
251
+ // partial pidfile can otherwise kill one old daemon while leaving a
252
+ // sibling PPID=1 daemon alive without a socket.
253
+ try {
254
+ const result = (0, child_process_1.execSync)("ps -eo pid,ppid,command", { encoding: "utf-8", timeout: 5000 });
255
+ scanOrphans = parseOrphanPidsFromPs(result, process.pid);
256
+ }
257
+ catch { /* ps failed — best effort */ }
258
+ const pidsToKill = mergeUniqueOrphanPids(pidfileOrphans, scanOrphans);
259
+ if (pidsToKill.length > 0) {
260
+ for (const pid of pidsToKill) {
261
+ try {
262
+ process.kill(pid, "SIGTERM");
263
+ }
264
+ catch { /* already exited */ }
265
+ }
266
+ (0, runtime_1.emitNervesEvent)({
267
+ component: "daemon",
268
+ event: "daemon.orphan_cleanup",
269
+ message: `killed ${pidsToKill.length} orphaned ouro processes`,
270
+ meta: { pids: pidsToKill },
271
+ });
272
+ }
273
+ }
274
+ catch (error) {
275
+ (0, runtime_1.emitNervesEvent)({
276
+ level: "warn",
277
+ component: "daemon",
278
+ event: "daemon.orphan_cleanup_error",
279
+ message: "failed to clean up orphaned ouro processes",
280
+ meta: { error: error instanceof Error ? error.message : String(error) },
281
+ });
282
+ }
283
+ }
284
+ /**
285
+ * Write all managed PIDs (daemon + children) to the pidfile.
286
+ * Called after all agents and senses are spawned.
287
+ */
288
+ function writePidfile(extraPids = [], socketPath = socket_client_1.DEFAULT_DAEMON_SOCKET_PATH) {
289
+ if (!isProductionDaemonSocketPath(socketPath)) {
290
+ (0, runtime_1.emitNervesEvent)({
291
+ level: "warn",
292
+ component: "daemon",
293
+ event: "daemon.write_pidfile_nonproduction_blocked",
294
+ message: "blocked production pidfile write for non-production daemon socket",
295
+ meta: { socketPath, pidfilePath: PIDFILE_PATH, attemptedPids: extraPids.length },
296
+ });
297
+ return;
298
+ }
299
+ if (isVitestProcess()) {
300
+ (0, runtime_1.emitNervesEvent)({
301
+ level: "warn",
302
+ component: "daemon",
303
+ event: "daemon.write_pidfile_test_blocked",
304
+ message: "blocked writePidfile from clobbering real pidfile under vitest",
305
+ meta: { pidfilePath: PIDFILE_PATH, attemptedPids: extraPids.length },
306
+ });
307
+ return;
308
+ }
309
+ try {
310
+ const pids = [process.pid, ...extraPids].filter(Boolean);
311
+ fs.mkdirSync(path.dirname(PIDFILE_PATH), { recursive: true });
312
+ fs.writeFileSync(PIDFILE_PATH, pids.join("\n") + "\n", "utf-8");
313
+ }
314
+ catch { /* best effort */ }
315
+ }
316
+ function readSocketIdentity(socketPath) {
317
+ try {
318
+ const stats = fs.lstatSync(socketPath);
319
+ return {
320
+ dev: stats.dev,
321
+ ino: stats.ino,
322
+ ctimeMs: stats.ctimeMs,
323
+ };
324
+ }
325
+ catch {
326
+ return null;
327
+ }
328
+ }
329
+ function sameSocketIdentity(left, right) {
330
+ if (!left || !right)
331
+ return false;
332
+ return left.dev === right.dev && left.ino === right.ino && left.ctimeMs === right.ctimeMs;
333
+ }
51
334
  function buildWorkerRows(snapshots) {
52
335
  return snapshots.map((snapshot) => ({
53
336
  agent: snapshot.name,
@@ -56,6 +339,10 @@ function buildWorkerRows(snapshots) {
56
339
  pid: snapshot.pid,
57
340
  restartCount: snapshot.restartCount,
58
341
  startedAt: snapshot.startedAt,
342
+ lastExitCode: snapshot.lastExitCode ?? null,
343
+ lastSignal: snapshot.lastSignal ?? null,
344
+ errorReason: snapshot.errorReason ?? null,
345
+ fixHint: snapshot.fixHint ?? null,
59
346
  }));
60
347
  }
61
348
  function formatStatusSummary(payload) {
@@ -88,6 +375,35 @@ function parseIncomingCommand(raw) {
88
375
  }
89
376
  return parsed;
90
377
  }
378
+ /**
379
+ * Handle agent.senseTurn command: runs a full agent turn via the daemon process.
380
+ * Dynamic import lazy-loads shared-turn. Hot-reload works because ouro dev
381
+ * restarts the daemon process (fresh module cache).
382
+ */
383
+ async function handleAgentSenseTurn(command) {
384
+ try {
385
+ const { setAgentName } = await Promise.resolve().then(() => __importStar(require("../identity")));
386
+ setAgentName(command.agent);
387
+ const { runSenseTurn } = await Promise.resolve().then(() => __importStar(require("../../senses/shared-turn")));
388
+ const result = await runSenseTurn({
389
+ agentName: command.agent,
390
+ channel: command.channel,
391
+ sessionKey: command.sessionKey,
392
+ friendId: command.friendId,
393
+ userMessage: command.message,
394
+ });
395
+ return {
396
+ ok: true,
397
+ message: result.response,
398
+ data: { ponderDeferred: result.ponderDeferred },
399
+ };
400
+ }
401
+ catch (error) {
402
+ /* v8 ignore next -- branch: String(error) fallback only for non-Error throws @preserve */
403
+ const errorMessage = error instanceof Error ? error.message : String(error);
404
+ return { ok: false, error: `sense turn failed: ${errorMessage}` };
405
+ }
406
+ }
91
407
  class OuroDaemon {
92
408
  socketPath;
93
409
  processManager;
@@ -96,7 +412,12 @@ class OuroDaemon {
96
412
  router;
97
413
  senseManager;
98
414
  bundlesRoot;
415
+ mode;
99
416
  server = null;
417
+ outlookServer = null;
418
+ socketIdentity = null;
419
+ senseAutostartTimer = null;
420
+ outlookServerFactory;
100
421
  constructor(options) {
101
422
  this.socketPath = options.socketPath;
102
423
  this.processManager = options.processManager;
@@ -105,6 +426,72 @@ class OuroDaemon {
105
426
  this.router = options.router;
106
427
  this.senseManager = options.senseManager ?? null;
107
428
  this.bundlesRoot = options.bundlesRoot ?? (0, identity_1.getAgentBundlesRoot)();
429
+ this.mode = options.mode ?? "production";
430
+ this.outlookServerFactory = options.outlookServerFactory ?? this.createDefaultOutlookServer.bind(this);
431
+ }
432
+ /* v8 ignore start -- default outlook server wiring: production-only path, tests inject outlookServerFactory stub instead. startOutlookHttpServer itself has full coverage in outlook-http.test.ts @preserve */
433
+ createDefaultOutlookServer() {
434
+ return (0, outlook_http_1.startOutlookHttpServer)({
435
+ host: "127.0.0.1",
436
+ port: outlook_types_1.OUTLOOK_DEFAULT_PORT,
437
+ bundlesRoot: this.bundlesRoot,
438
+ readMachineState: () => (0, outlook_read_1.readOutlookMachineState)({ bundlesRoot: this.bundlesRoot }),
439
+ readMachineView: ({ machine }) => {
440
+ const overview = this.buildStatusPayload().overview;
441
+ return (0, outlook_view_1.buildOutlookMachineView)({
442
+ machine,
443
+ daemon: {
444
+ status: overview.daemon,
445
+ health: overview.health,
446
+ mode: overview.mode,
447
+ socketPath: overview.socketPath,
448
+ outlookUrl: overview.outlookUrl,
449
+ entryPath: overview.entryPath,
450
+ workerCount: overview.workerCount,
451
+ senseCount: overview.senseCount,
452
+ },
453
+ });
454
+ },
455
+ readAgentState: (agentName) => (0, outlook_read_1.readOutlookAgentState)(agentName, { bundlesRoot: this.bundlesRoot }),
456
+ readAgentView: (agentName) => {
457
+ const agent = (0, outlook_read_1.readOutlookAgentState)(agentName, { bundlesRoot: this.bundlesRoot });
458
+ return (0, outlook_view_1.buildOutlookAgentView)({
459
+ agent,
460
+ viewer: { kind: "human" },
461
+ });
462
+ },
463
+ });
464
+ }
465
+ /* v8 ignore stop */
466
+ buildStatusPayload() {
467
+ const snapshots = this.processManager.listAgentSnapshots();
468
+ const workers = buildWorkerRows(snapshots);
469
+ const senses = this.senseManager?.listSenseRows() ?? [];
470
+ const repoRoot = (0, identity_1.getRepoRoot)();
471
+ const sync = (0, agent_discovery_1.listBundleSyncRows)({ bundlesRoot: this.bundlesRoot });
472
+ const agents = (0, agent_discovery_1.listAllBundleAgents)({ bundlesRoot: this.bundlesRoot });
473
+ const providers = agents.flatMap((agent) => (0, provider_visibility_1.providerVisibilityStatusRows)((0, provider_visibility_1.buildAgentProviderVisibility)({
474
+ agentName: agent.name,
475
+ agentRoot: path.join(this.bundlesRoot, `${agent.name}.ouro`),
476
+ })));
477
+ return {
478
+ overview: {
479
+ daemon: "running",
480
+ health: workers.every((worker) => worker.status === "running") ? "ok" : "warn",
481
+ socketPath: this.socketPath,
482
+ outlookUrl: this.outlookServer?.origin ?? "http://127.0.0.1:0",
483
+ ...(0, runtime_metadata_1.getRuntimeMetadata)(),
484
+ workerCount: workers.length,
485
+ senseCount: senses.length,
486
+ entryPath: path.join(repoRoot, "dist", "heart", "daemon", "daemon-entry.js"),
487
+ mode: (0, runtime_mode_1.detectRuntimeMode)(repoRoot),
488
+ },
489
+ workers,
490
+ senses,
491
+ sync,
492
+ agents,
493
+ ...(providers.length > 0 ? { providers } : {}),
494
+ };
108
495
  }
109
496
  async start() {
110
497
  if (this.server)
@@ -115,62 +502,231 @@ class OuroDaemon {
115
502
  message: "starting daemon server",
116
503
  meta: { socketPath: this.socketPath },
117
504
  });
505
+ try {
506
+ await this.startInner();
507
+ }
508
+ catch (err) {
509
+ // Emit a paired terminating event (`_error`) so the nerves audit's
510
+ // start_end_pairing rule is satisfied when startup throws mid-sequence
511
+ // and `stop()` (which emits `server_end`) is never called.
512
+ (0, runtime_1.emitNervesEvent)({
513
+ level: "error",
514
+ component: "daemon",
515
+ event: "daemon.server_error",
516
+ message: "daemon start failed",
517
+ meta: {
518
+ error: err instanceof Error ? err.message : /* v8 ignore next -- defensive: non-Error catch branch @preserve */ String(err),
519
+ },
520
+ });
521
+ throw err;
522
+ }
523
+ }
524
+ async startInner() {
118
525
  // Register update hooks and apply pending updates before starting agents
119
526
  (0, update_hooks_1.registerUpdateHook)(bundle_meta_1.bundleMetaHook);
527
+ (0, update_hooks_1.registerUpdateHook)(agent_config_v2_1.agentConfigV2Hook);
120
528
  const currentVersion = (0, bundle_manifest_1.getPackageVersion)();
121
529
  await (0, update_hooks_1.applyPendingUpdates)(this.bundlesRoot, currentVersion);
122
530
  // Start periodic update checker (polls npm registry every 30 minutes)
531
+ // Skip in dev mode — dev builds should not auto-update from npm
123
532
  const bundlesRoot = this.bundlesRoot;
124
- const daemon = this;
125
- (0, update_checker_1.startUpdateChecker)({
126
- currentVersion,
127
- deps: {
128
- distTag: "alpha",
129
- fetchRegistryJson: /* v8 ignore next -- integration: real HTTP fetch @preserve */ async () => {
130
- const res = await fetch("https://registry.npmjs.org/@ouro.bot/cli");
131
- return res.json();
132
- },
133
- },
134
- onUpdate: /* v8 ignore start -- integration: real npm install + process spawn @preserve */ async (result) => {
135
- if (!result.latestVersion)
136
- return;
137
- await (0, staged_restart_1.performStagedRestart)(result.latestVersion, {
138
- execSync: (cmd) => (0, child_process_1.execSync)(cmd, { stdio: "inherit" }),
139
- spawnSync: child_process_1.spawnSync,
140
- resolveNewCodePath: (_version) => {
141
- try {
142
- const resolved = (0, child_process_1.execSync)(`node -e "console.log(require.resolve('@ouro.bot/cli/package.json'))"`, { encoding: "utf-8" }).trim();
143
- return resolved ? path.dirname(resolved) : null;
144
- }
145
- catch {
146
- return null;
147
- }
533
+ const daemonSocketPath = this.socketPath;
534
+ if (this.mode === "dev") {
535
+ (0, runtime_1.emitNervesEvent)({
536
+ component: "daemon",
537
+ event: "daemon.update_checker_skip",
538
+ message: "skipping update checker in dev mode",
539
+ meta: { reason: "dev mode" },
540
+ });
541
+ }
542
+ else {
543
+ const daemon = this;
544
+ (0, update_checker_1.startUpdateChecker)({
545
+ currentVersion,
546
+ deps: {
547
+ distTag: "latest",
548
+ fetchRegistryJson: /* v8 ignore next -- integration: real HTTP fetch @preserve */ async () => {
549
+ const res = await fetch("https://registry.npmjs.org/@ouro.bot/cli");
550
+ return res.json();
148
551
  },
149
- gracefulShutdown: () => daemon.stop(),
150
- nodePath: process.execPath,
151
- bundlesRoot,
152
- });
153
- },
154
- /* v8 ignore stop */
155
- });
156
- await this.processManager.startAutoStartAgents();
157
- await this.senseManager?.startAutoStartSenses();
552
+ },
553
+ onUpdate: /* v8 ignore start -- integration: real npm install + process spawn @preserve */ async (result) => {
554
+ if (!result.latestVersion)
555
+ return;
556
+ // Install via the version manager (NOT `npm install -g`). The
557
+ // global install path doesn't end up on the daemon process's
558
+ // NODE_PATH, so the previous `require.resolve('@ouro.bot/cli')`
559
+ // -based path lookup always returned null and the staged restart
560
+ // never actually completed. Verified live on 2026-04-08:
561
+ // alpha.268 daemon detected alpha.270 was available, ran the
562
+ // staged restart, and bailed at `staged_restart_path_failed` —
563
+ // meaning the daemon could never auto-update itself and required
564
+ // manual `ouro up` to pick up new versions.
565
+ //
566
+ // Switch to the version-managed layout the CLI itself uses:
567
+ // installVersion(version) puts files at
568
+ // ~/.ouro-cli/versions/{version}/node_modules/@ouro.bot/cli
569
+ // which is a known path we can compute deterministically.
570
+ // Then activateVersion(version) flips the CurrentVersion symlink
571
+ // so the next `ouro up` from the user sees the same version
572
+ // the daemon is running.
573
+ const cliHome = (0, ouro_version_manager_1.getOuroCliHome)();
574
+ await (0, staged_restart_1.performStagedRestart)(result.latestVersion, {
575
+ execSync: (cmd) => (0, child_process_1.execSync)(cmd, { stdio: "inherit" }),
576
+ spawnSync: child_process_1.spawnSync,
577
+ installNewVersion: (version) => {
578
+ (0, ouro_version_manager_1.installVersion)(version, {});
579
+ (0, ouro_version_manager_1.activateVersion)(version, {});
580
+ },
581
+ resolveNewCodePath: (version) => {
582
+ const versionPath = path.join(cliHome, "versions", version, "node_modules", "@ouro.bot", "cli");
583
+ return fs.existsSync(versionPath) ? versionPath : null;
584
+ },
585
+ gracefulShutdown: () => daemon.stop(),
586
+ spawnNewDaemon: (entryPath, sock) => {
587
+ const outFd = fs.openSync(os.devNull, "w");
588
+ const errFd = fs.openSync(os.devNull, "w");
589
+ const child = (0, child_process_1.spawn)(process.execPath, [entryPath, "--socket", sock], {
590
+ detached: true,
591
+ stdio: ["ignore", outFd, errFd],
592
+ });
593
+ child.unref();
594
+ return { pid: child.pid ?? null };
595
+ },
596
+ nodePath: process.execPath,
597
+ bundlesRoot,
598
+ socketPath: daemonSocketPath,
599
+ });
600
+ },
601
+ /* v8 ignore stop */
602
+ });
603
+ }
604
+ // MCP connections are lazily initialized per-agent during senseTurn
605
+ // (daemon manages multiple agents; agent identity must be set before loading MCP config)
606
+ /* v8 ignore start -- orphan cleanup + pidfile: calls process management functions @preserve */
607
+ killOrphanProcesses(this.socketPath);
608
+ /* v8 ignore stop */
609
+ await this.openCommandSocket();
610
+ this.triggerAutoStartAgents();
611
+ this.triggerAutoStartSensesWhenAgentsSettled();
612
+ // Write all managed PIDs to disk so the next daemon can clean up
613
+ /* v8 ignore start -- pidfile write: collects PIDs from process managers @preserve */
614
+ const agentPids = this.processManager.listAgentSnapshots().map((s) => s.pid).filter((p) => p !== null);
615
+ const sensePids = this.senseManager?.listManagedPids?.() ?? [];
616
+ writePidfile([...agentPids, ...sensePids], this.socketPath);
617
+ /* v8 ignore stop */
158
618
  this.scheduler.start?.();
159
619
  await this.scheduler.reconcile?.();
160
620
  await this.drainPendingBundleMessages();
161
621
  await this.drainPendingSenseMessages();
622
+ // startInner is only reachable when this.server is null (guarded in
623
+ // start()), and stop() nulls out this.outlookServer alongside this.server,
624
+ // so outlookServer is guaranteed unset here — no need for a guard.
625
+ try {
626
+ this.outlookServer = await this.outlookServerFactory();
627
+ }
628
+ catch (error) {
629
+ (0, runtime_1.emitNervesEvent)({
630
+ level: "warn",
631
+ component: "daemon",
632
+ event: "daemon.outlook_start_failed",
633
+ message: `Outlook server failed to start: ${String(error)}`,
634
+ meta: { port: outlook_types_1.OUTLOOK_DEFAULT_PORT },
635
+ });
636
+ }
637
+ }
638
+ triggerAutoStartAgents() {
639
+ if (this.processManager.triggerAutoStartAgents) {
640
+ this.processManager.triggerAutoStartAgents();
641
+ return;
642
+ }
643
+ void this.processManager.startAutoStartAgents().catch((error) => {
644
+ (0, runtime_1.emitNervesEvent)({
645
+ level: "error",
646
+ component: "daemon",
647
+ event: "daemon.agent_autostart_error",
648
+ message: "agent autostart failed after daemon socket opened",
649
+ meta: { error: error instanceof Error ? error.message : String(error) },
650
+ });
651
+ });
652
+ }
653
+ triggerAutoStartSenses() {
654
+ /* v8 ignore next -- defensive: callers already check senseManager before delegating here @preserve */
655
+ if (!this.senseManager)
656
+ return;
657
+ if (this.senseManager.triggerAutoStartSenses) {
658
+ this.senseManager.triggerAutoStartSenses();
659
+ return;
660
+ }
661
+ void this.senseManager.startAutoStartSenses().catch((error) => {
662
+ (0, runtime_1.emitNervesEvent)({
663
+ level: "error",
664
+ component: "daemon",
665
+ event: "daemon.sense_autostart_error",
666
+ message: "sense autostart failed after daemon socket opened",
667
+ meta: { error: error instanceof Error ? error.message : String(error) },
668
+ });
669
+ });
670
+ }
671
+ triggerAutoStartSensesWhenAgentsSettled() {
672
+ if (!this.senseManager)
673
+ return;
674
+ const waitingOnAgents = this.processManager.listAgentSnapshots()
675
+ .some((snapshot) => snapshot.status === "starting");
676
+ if (!waitingOnAgents) {
677
+ this.triggerAutoStartSenses();
678
+ return;
679
+ }
680
+ this.senseAutostartTimer = setTimeout(() => {
681
+ this.senseAutostartTimer = null;
682
+ this.triggerAutoStartSensesWhenAgentsSettled();
683
+ }, 250);
684
+ }
685
+ async openCommandSocket() {
162
686
  if (fs.existsSync(this.socketPath)) {
163
687
  fs.unlinkSync(this.socketPath);
164
688
  }
165
- this.server = net.createServer((connection) => {
689
+ // allowHalfOpen: true lets the server keep its writable side open after
690
+ // the client sends FIN. Without this, when a client calls `client.end()`
691
+ // after writing a command, node closes the server's writable side
692
+ // automatically — so a long-running response (like an agent.senseTurn
693
+ // LLM turn that takes 5+ seconds) never reaches the client. The
694
+ // socket-client fix in #303/#334 also removed client.end() on the
695
+ // sending side, but this option is defense in depth: even if a future
696
+ // caller half-closes, the server still writes its response correctly.
697
+ this.server = net.createServer({ allowHalfOpen: true }, (connection) => {
166
698
  let raw = "";
167
699
  let responded = false;
700
+ /* v8 ignore start — connection error handler requires real socket error @preserve */
701
+ connection.on("error", (err) => {
702
+ (0, runtime_1.emitNervesEvent)({
703
+ level: "warn",
704
+ component: "daemon",
705
+ event: "daemon.connection_error",
706
+ message: "socket connection error",
707
+ meta: { error: err.message, code: err.code ?? null },
708
+ });
709
+ });
710
+ /* v8 ignore stop */
168
711
  const flushResponse = async () => {
169
712
  if (responded)
170
713
  return;
171
714
  responded = true;
172
715
  const response = await this.handleRawPayload(raw);
173
- connection.end(response);
716
+ try {
717
+ connection.end(response);
718
+ /* v8 ignore start — EPIPE catch requires real socket disconnect @preserve */
719
+ }
720
+ catch (err) {
721
+ (0, runtime_1.emitNervesEvent)({
722
+ level: "warn",
723
+ component: "daemon",
724
+ event: "daemon.connection_end_error",
725
+ message: "failed to send response to client (EPIPE)",
726
+ meta: { error: err instanceof Error ? err.message : String(err) },
727
+ });
728
+ }
729
+ /* v8 ignore stop */
174
730
  };
175
731
  connection.on("data", (chunk) => {
176
732
  raw += chunk.toString("utf-8");
@@ -183,7 +739,23 @@ class OuroDaemon {
183
739
  const server = this.server;
184
740
  await new Promise((resolve, reject) => {
185
741
  server.once("error", reject);
186
- server.listen(this.socketPath, () => resolve());
742
+ server.listen(this.socketPath, () => {
743
+ // Replace the one-time error listener with a persistent one after successful listen
744
+ server.removeAllListeners("error");
745
+ this.socketIdentity = readSocketIdentity(this.socketPath);
746
+ /* v8 ignore start — server error after listen requires real socket race condition @preserve */
747
+ server.on("error", (err) => {
748
+ (0, runtime_1.emitNervesEvent)({
749
+ level: "error",
750
+ component: "daemon",
751
+ event: "daemon.server_error",
752
+ message: "daemon server error after listen",
753
+ meta: { error: err.message, code: err.code ?? null },
754
+ });
755
+ });
756
+ /* v8 ignore stop */
757
+ resolve();
758
+ });
187
759
  });
188
760
  }
189
761
  async drainPendingBundleMessages() {
@@ -322,25 +894,86 @@ class OuroDaemon {
322
894
  }
323
895
  }
324
896
  async stop() {
897
+ // Must be named `_end` (not `_stop`) to satisfy the nerves audit's
898
+ // start/end pairing rule — see src/nerves/coverage/audit-rules.ts.
899
+ // This is the counterpart to `daemon.server_start` emitted at line 480.
325
900
  (0, runtime_1.emitNervesEvent)({
326
901
  component: "daemon",
327
- event: "daemon.server_stop",
902
+ event: "daemon.server_end",
328
903
  message: "stopping daemon server",
329
904
  meta: { socketPath: this.socketPath },
330
905
  });
331
906
  (0, update_checker_1.stopUpdateChecker)();
907
+ (0, mcp_manager_1.shutdownSharedMcpManager)();
332
908
  this.scheduler.stop?.();
909
+ if (this.senseAutostartTimer) {
910
+ clearTimeout(this.senseAutostartTimer);
911
+ this.senseAutostartTimer = null;
912
+ }
333
913
  await this.processManager.stopAll();
334
914
  await this.senseManager?.stopAll();
335
915
  if (this.server) {
336
- await new Promise((resolve) => {
337
- this.server?.close(() => resolve());
338
- });
916
+ // DO NOT `await` server.close() here. server.close() resolves only
917
+ // after every open connection has closed. When stop() is invoked
918
+ // from the daemon.stop command handler, the calling client's
919
+ // connection is STILL open — its flushResponse() is currently
920
+ // awaiting THIS function. Awaiting close() creates a deadlock:
921
+ //
922
+ // client → flushResponse → handleRawPayload → daemon.stop case
923
+ // → stop() → await server.close() (waits for client's connection)
924
+ // → client's connection waits for flushResponse to call
925
+ // connection.end() → DEADLOCK
926
+ //
927
+ // Both processes sit in kevent forever. Verified live on
928
+ // 2026-04-08: alpha.268 daemon hung at `daemon.server_end` log
929
+ // line for 5+ minutes after a client sent daemon.stop, while the
930
+ // client (alpha.270 ouro up) hung waiting for the response.
931
+ //
932
+ // This regressed when #303/#334/#339 stopped half-closing the
933
+ // client socket and switched the server to allowHalfOpen: true.
934
+ // Previously, the client called .end() after writing its command,
935
+ // which (with allowHalfOpen: false) caused node to auto-tear-down
936
+ // the server's writable side — incidentally unblocking
937
+ // server.close() before the response was sent. The half-close
938
+ // breakage masked this deadlock; the fix exposed it.
939
+ //
940
+ // Solution: fire close() and let it complete asynchronously. Once
941
+ // stop() returns, the daemon.stop case returns its response,
942
+ // flushResponse() calls connection.end(response), the connection
943
+ // closes, and server.close()'s pending callback fires. The event
944
+ // loop drains and the daemon exits cleanly.
945
+ this.server.close();
339
946
  this.server = null;
340
947
  }
341
- if (fs.existsSync(this.socketPath)) {
948
+ if (this.outlookServer) {
949
+ await this.outlookServer.stop();
950
+ this.outlookServer = null;
951
+ }
952
+ const socketPathExists = fs.existsSync(this.socketPath);
953
+ const currentSocketIdentity = socketPathExists ? readSocketIdentity(this.socketPath) : null;
954
+ if (sameSocketIdentity(this.socketIdentity, currentSocketIdentity)) {
342
955
  fs.unlinkSync(this.socketPath);
343
956
  }
957
+ else if (socketPathExists) {
958
+ const expectedSocketIdentity = { dev: null, ino: null, ctimeMs: null, ...this.socketIdentity };
959
+ const actualSocketIdentity = { dev: null, ino: null, ctimeMs: null, ...currentSocketIdentity };
960
+ (0, runtime_1.emitNervesEvent)({
961
+ level: "warn",
962
+ component: "daemon",
963
+ event: "daemon.socket_cleanup_skipped",
964
+ message: "skipped daemon socket cleanup because the socket path no longer belongs to this daemon",
965
+ meta: {
966
+ socketPath: this.socketPath,
967
+ expectedDev: expectedSocketIdentity.dev,
968
+ expectedIno: expectedSocketIdentity.ino,
969
+ expectedCtimeMs: expectedSocketIdentity.ctimeMs,
970
+ actualDev: actualSocketIdentity.dev,
971
+ actualIno: actualSocketIdentity.ino,
972
+ actualCtimeMs: actualSocketIdentity.ctimeMs,
973
+ },
974
+ });
975
+ }
976
+ this.socketIdentity = null;
344
977
  }
345
978
  async handleRawPayload(raw) {
346
979
  try {
@@ -362,6 +995,27 @@ class OuroDaemon {
362
995
  message: "handling daemon command",
363
996
  meta: { kind: command.kind },
364
997
  });
998
+ try {
999
+ return await this.handleCommandInner(command);
1000
+ /* v8 ignore start — command error catch tested in daemon-command-error.test; instanceof branches defensive @preserve */
1001
+ }
1002
+ catch (error) {
1003
+ (0, runtime_1.emitNervesEvent)({
1004
+ level: "error",
1005
+ component: "daemon",
1006
+ event: "daemon.command_error",
1007
+ message: "unexpected error handling daemon command",
1008
+ meta: {
1009
+ kind: command.kind,
1010
+ error: error instanceof Error ? error.message : String(error),
1011
+ stack: error instanceof Error ? error.stack ?? null : null,
1012
+ },
1013
+ });
1014
+ throw error;
1015
+ }
1016
+ /* v8 ignore stop */
1017
+ }
1018
+ async handleCommandInner(command) {
365
1019
  switch (command.kind) {
366
1020
  case "daemon.start":
367
1021
  await this.start();
@@ -370,21 +1024,7 @@ class OuroDaemon {
370
1024
  await this.stop();
371
1025
  return { ok: true, message: "daemon stopped" };
372
1026
  case "daemon.status": {
373
- const snapshots = this.processManager.listAgentSnapshots();
374
- const workers = buildWorkerRows(snapshots);
375
- const senses = this.senseManager?.listSenseRows() ?? [];
376
- const data = {
377
- overview: {
378
- daemon: "running",
379
- health: workers.every((worker) => worker.status === "running") ? "ok" : "warn",
380
- socketPath: this.socketPath,
381
- ...(0, runtime_metadata_1.getRuntimeMetadata)(),
382
- workerCount: workers.length,
383
- senseCount: senses.length,
384
- },
385
- workers,
386
- senses,
387
- };
1027
+ const data = this.buildStatusPayload();
388
1028
  return {
389
1029
  ok: true,
390
1030
  summary: formatStatusSummary(data),
@@ -401,7 +1041,7 @@ class OuroDaemon {
401
1041
  ok: true,
402
1042
  summary: "logs: use `ouro logs` to tail daemon and agent output",
403
1043
  message: "log streaming available via ouro logs",
404
- data: { logDir: "~/.agentstate/daemon/logs" },
1044
+ data: { logDir: "~/AgentBundles/<agent>.ouro/state/daemon/logs" },
405
1045
  };
406
1046
  case "agent.start":
407
1047
  await this.processManager.startAgent(command.agent);
@@ -412,6 +1052,35 @@ class OuroDaemon {
412
1052
  case "agent.restart":
413
1053
  await this.processManager.restartAgent?.(command.agent);
414
1054
  return { ok: true, message: `restarted ${command.agent}` };
1055
+ case "agent.ask":
1056
+ return (0, agent_service_1.handleAgentAsk)(command);
1057
+ case "agent.status":
1058
+ return (0, agent_service_1.handleAgentStatus)(command);
1059
+ case "agent.catchup":
1060
+ return (0, agent_service_1.handleAgentCatchup)(command);
1061
+ case "agent.delegate":
1062
+ return (0, agent_service_1.handleAgentDelegate)(command);
1063
+ case "agent.getContext":
1064
+ return (0, agent_service_1.handleAgentGetContext)(command);
1065
+ case "agent.searchNotes":
1066
+ return (0, agent_service_1.handleAgentSearchNotes)(command);
1067
+ case "agent.getTask":
1068
+ return (0, agent_service_1.handleAgentGetTask)(command);
1069
+ case "agent.checkScope":
1070
+ return (0, agent_service_1.handleAgentCheckScope)(command);
1071
+ case "agent.requestDecision":
1072
+ return (0, agent_service_1.handleAgentRequestDecision)(command);
1073
+ case "agent.checkGuidance":
1074
+ return (0, agent_service_1.handleAgentCheckGuidance)(command);
1075
+ case "agent.reportProgress":
1076
+ return (0, agent_service_1.handleAgentReportProgress)(command);
1077
+ case "agent.reportBlocker":
1078
+ return (0, agent_service_1.handleAgentReportBlocker)(command);
1079
+ case "agent.reportComplete":
1080
+ return (0, agent_service_1.handleAgentReportComplete)(command);
1081
+ case "agent.senseTurn":
1082
+ return handleAgentSenseTurn(command);
1083
+ /* v8 ignore stop */
415
1084
  case "cron.list": {
416
1085
  const jobs = this.scheduler.listJobs();
417
1086
  const summary = jobs.length === 0
@@ -472,6 +1141,35 @@ class OuroDaemon {
472
1141
  data: receipt,
473
1142
  };
474
1143
  }
1144
+ case "habit.poke": {
1145
+ this.processManager.sendToAgent?.(command.agent, { type: "habit", habitName: command.habitName });
1146
+ return {
1147
+ ok: true,
1148
+ message: `poked habit ${command.habitName} for ${command.agent}`,
1149
+ };
1150
+ }
1151
+ case "mcp.list": {
1152
+ const mcpManager = await (0, mcp_manager_1.getSharedMcpManager)();
1153
+ if (!mcpManager) {
1154
+ return { ok: true, data: [], message: "no MCP servers configured" };
1155
+ }
1156
+ return { ok: true, data: mcpManager.listAllTools() };
1157
+ }
1158
+ case "mcp.call": {
1159
+ const mcpCallManager = await (0, mcp_manager_1.getSharedMcpManager)();
1160
+ if (!mcpCallManager) {
1161
+ return { ok: false, error: "no MCP servers configured" };
1162
+ }
1163
+ try {
1164
+ const parsedArgs = command.args ? JSON.parse(command.args) : {};
1165
+ const result = await mcpCallManager.callTool(command.server, command.tool, parsedArgs);
1166
+ return { ok: true, data: result };
1167
+ }
1168
+ catch (error) {
1169
+ /* v8 ignore next -- defensive: callTool errors are always Error instances @preserve */
1170
+ return { ok: false, error: error instanceof Error ? error.message : String(error) };
1171
+ }
1172
+ }
475
1173
  case "hatch.start":
476
1174
  return {
477
1175
  ok: true,