@ouro.bot/cli 0.1.0-alpha.56 → 0.1.0-alpha.560

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 (392) hide show
  1. package/README.md +127 -23
  2. package/RepairGuide.ouro/agent.json +5 -0
  3. package/RepairGuide.ouro/psyche/IDENTITY.md +19 -0
  4. package/RepairGuide.ouro/psyche/SOUL.md +55 -0
  5. package/RepairGuide.ouro/skills/diagnose-broken-remote.md +63 -0
  6. package/RepairGuide.ouro/skills/diagnose-stacked-typed-issues.md +35 -0
  7. package/RepairGuide.ouro/skills/diagnose-sync-blocked.md +54 -0
  8. package/RepairGuide.ouro/skills/diagnose-vault-expired.md +60 -0
  9. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +4 -2
  10. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +2 -2
  11. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  12. package/changelog.json +3596 -0
  13. package/dist/arc/attention-types.js +8 -0
  14. package/dist/arc/cares.js +140 -0
  15. package/dist/arc/episodes.js +117 -0
  16. package/dist/arc/intentions.js +133 -0
  17. package/dist/arc/json-store.js +117 -0
  18. package/dist/arc/obligations.js +237 -0
  19. package/dist/arc/packets.js +193 -0
  20. package/dist/arc/presence.js +185 -0
  21. package/dist/arc/task-lifecycle.js +65 -0
  22. package/dist/heart/active-work.js +837 -26
  23. package/dist/heart/agent-entry.js +58 -3
  24. package/dist/heart/attachments/image-normalize.js +194 -0
  25. package/dist/heart/attachments/materialize.js +97 -0
  26. package/dist/heart/attachments/originals.js +88 -0
  27. package/dist/heart/attachments/render.js +29 -0
  28. package/dist/heart/attachments/sources/adapter.js +2 -0
  29. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  30. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  31. package/dist/heart/attachments/sources/index.js +16 -0
  32. package/dist/heart/attachments/store.js +103 -0
  33. package/dist/heart/attachments/types.js +93 -0
  34. package/dist/heart/auth/auth-flow.js +479 -0
  35. package/dist/heart/background-operations.js +281 -0
  36. package/dist/heart/bundle-state.js +168 -0
  37. package/dist/heart/commitments.js +111 -0
  38. package/dist/heart/config-registry.js +322 -0
  39. package/dist/heart/config.js +114 -118
  40. package/dist/heart/core.js +913 -246
  41. package/dist/heart/cross-chat-delivery.js +3 -18
  42. package/dist/heart/daemon/agent-config-check.js +419 -0
  43. package/dist/heart/daemon/agent-discovery.js +102 -3
  44. package/dist/heart/daemon/agent-service.js +522 -0
  45. package/dist/heart/daemon/agentic-repair.js +547 -0
  46. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  47. package/dist/heart/daemon/boot-sync-probe.js +197 -0
  48. package/dist/heart/daemon/cadence.js +70 -0
  49. package/dist/heart/daemon/cli-defaults.js +776 -0
  50. package/dist/heart/daemon/cli-exec.js +7457 -0
  51. package/dist/heart/daemon/cli-help.js +498 -0
  52. package/dist/heart/daemon/cli-parse.js +1592 -0
  53. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  54. package/dist/heart/daemon/cli-render.js +763 -0
  55. package/dist/heart/daemon/cli-types.js +8 -0
  56. package/dist/heart/daemon/connect-bay.js +323 -0
  57. package/dist/heart/daemon/daemon-cli.js +29 -1698
  58. package/dist/heart/daemon/daemon-entry.js +387 -2
  59. package/dist/heart/daemon/daemon-health.js +176 -0
  60. package/dist/heart/daemon/daemon-rollup.js +57 -0
  61. package/dist/heart/daemon/daemon-runtime-sync.js +88 -13
  62. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  63. package/dist/heart/daemon/daemon.js +796 -71
  64. package/dist/heart/daemon/dns-workflow.js +394 -0
  65. package/dist/heart/daemon/doctor-types.js +8 -0
  66. package/dist/heart/daemon/doctor.js +826 -0
  67. package/dist/heart/daemon/health-monitor.js +122 -1
  68. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  69. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  70. package/dist/heart/daemon/http-health-probe.js +80 -0
  71. package/dist/heart/daemon/human-command-screens.js +234 -0
  72. package/dist/heart/daemon/human-readiness.js +114 -0
  73. package/dist/heart/daemon/inner-status.js +89 -0
  74. package/dist/heart/daemon/interactive-repair.js +394 -0
  75. package/dist/heart/daemon/launchd.js +37 -8
  76. package/dist/heart/daemon/log-tailer.js +82 -12
  77. package/dist/heart/daemon/logs-prune.js +110 -0
  78. package/dist/heart/daemon/mcp-canary.js +297 -0
  79. package/dist/heart/daemon/message-router.js +2 -2
  80. package/dist/heart/daemon/os-cron-deps.js +135 -0
  81. package/dist/heart/daemon/os-cron.js +14 -12
  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 +3 -31
  92. package/dist/heart/daemon/safe-mode.js +161 -0
  93. package/dist/heart/daemon/sense-manager.js +389 -38
  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 +158 -11
  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 +32 -56
  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 +203 -57
  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 +683 -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 +267 -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-readiness-cache.js +40 -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/runtime-cwd.js +87 -0
  156. package/dist/heart/sense-truth.js +13 -4
  157. package/dist/heart/session-activity.js +43 -22
  158. package/dist/heart/session-events.js +1149 -0
  159. package/dist/heart/session-playback-cli-main.js +5 -0
  160. package/dist/heart/session-playback-cli.js +36 -0
  161. package/dist/heart/session-playback.js +231 -0
  162. package/dist/heart/session-stats-cli-main.js +5 -0
  163. package/dist/heart/session-stats.js +182 -0
  164. package/dist/heart/session-transcript.js +243 -0
  165. package/dist/heart/start-of-turn-packet.js +345 -0
  166. package/dist/heart/streaming.js +44 -27
  167. package/dist/heart/sync-classification.js +176 -0
  168. package/dist/heart/sync.js +449 -0
  169. package/dist/heart/target-resolution.js +9 -5
  170. package/dist/heart/tempo.js +93 -0
  171. package/dist/heart/temporal-view.js +41 -0
  172. package/dist/heart/timeouts.js +101 -0
  173. package/dist/heart/tool-activity-callbacks.js +59 -0
  174. package/dist/heart/tool-description.js +139 -0
  175. package/dist/heart/tool-friction.js +55 -0
  176. package/dist/heart/tool-loop.js +200 -0
  177. package/dist/heart/turn-context.js +389 -0
  178. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +6 -5
  179. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  180. package/dist/heart/versioning/ouro-path-installer.js +426 -0
  181. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  182. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  183. package/dist/heart/{daemon → versioning}/update-checker.js +6 -1
  184. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  185. package/dist/mailbox-ui/assets/index-B-461hes.js +61 -0
  186. package/dist/mailbox-ui/assets/index-BPr5vNuM.css +1 -0
  187. package/dist/mailbox-ui/index.html +15 -0
  188. package/dist/mailroom/attention.js +167 -0
  189. package/dist/mailroom/autonomy.js +209 -0
  190. package/dist/mailroom/blob-store.js +674 -0
  191. package/dist/mailroom/body-cache.js +61 -0
  192. package/dist/mailroom/core.js +720 -0
  193. package/dist/mailroom/entry.js +160 -0
  194. package/dist/mailroom/file-store.js +430 -0
  195. package/dist/mailroom/mbox-import.js +383 -0
  196. package/dist/mailroom/outbound.js +380 -0
  197. package/dist/mailroom/policy.js +263 -0
  198. package/dist/mailroom/reader.js +233 -0
  199. package/dist/mailroom/search-cache.js +256 -0
  200. package/dist/mailroom/search-relevance.js +319 -0
  201. package/dist/mailroom/smtp-ingress.js +176 -0
  202. package/dist/mailroom/source-state.js +176 -0
  203. package/dist/mailroom/thread.js +109 -0
  204. package/dist/mailroom/travel-extract.js +89 -0
  205. package/dist/mind/bundle-manifest.js +7 -1
  206. package/dist/mind/context.js +165 -101
  207. package/dist/mind/diary-integrity.js +60 -0
  208. package/dist/mind/{memory.js → diary.js} +62 -75
  209. package/dist/mind/embedding-provider.js +60 -0
  210. package/dist/mind/file-state.js +179 -0
  211. package/dist/mind/friends/channel.js +39 -0
  212. package/dist/mind/friends/resolver.js +54 -2
  213. package/dist/mind/friends/store-file.js +39 -3
  214. package/dist/mind/friends/types.js +2 -2
  215. package/dist/mind/journal-index.js +161 -0
  216. package/dist/mind/note-search.js +268 -0
  217. package/dist/mind/obligation-steering.js +221 -0
  218. package/dist/mind/pending.js +4 -0
  219. package/dist/mind/prompt-refresh.js +3 -2
  220. package/dist/mind/prompt.js +1011 -123
  221. package/dist/mind/provenance-trust.js +26 -0
  222. package/dist/mind/scrutiny.js +173 -0
  223. package/dist/nerves/cli-logging.js +7 -1
  224. package/dist/nerves/coverage/audit-rules.js +15 -6
  225. package/dist/nerves/coverage/audit.js +28 -2
  226. package/dist/nerves/coverage/cli.js +1 -1
  227. package/dist/nerves/coverage/contract.js +5 -5
  228. package/dist/nerves/coverage/file-completeness.js +129 -5
  229. package/dist/nerves/coverage/run-artifacts.js +1 -1
  230. package/dist/nerves/event-buffer.js +111 -0
  231. package/dist/nerves/index.js +224 -4
  232. package/dist/nerves/observation.js +20 -0
  233. package/dist/nerves/redact.js +79 -0
  234. package/dist/nerves/review/cli-main.js +5 -0
  235. package/dist/nerves/review/cli.js +156 -0
  236. package/dist/nerves/review/core.js +152 -0
  237. package/dist/nerves/runtime.js +5 -1
  238. package/dist/repertoire/ado-client.js +15 -56
  239. package/dist/repertoire/ado-semantic.js +11 -10
  240. package/dist/repertoire/api-client.js +97 -0
  241. package/dist/repertoire/bitwarden-store.js +963 -0
  242. package/dist/repertoire/bundle-templates.js +72 -0
  243. package/dist/repertoire/bw-installer.js +180 -0
  244. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  245. package/dist/repertoire/coding/context-pack.js +330 -0
  246. package/dist/repertoire/coding/feedback.js +197 -30
  247. package/dist/repertoire/coding/manager.js +158 -9
  248. package/dist/repertoire/coding/spawner.js +55 -9
  249. package/dist/repertoire/coding/tools.js +170 -7
  250. package/dist/repertoire/commerce-errors.js +109 -0
  251. package/dist/repertoire/commerce-self-test.js +156 -0
  252. package/dist/repertoire/credential-access.js +178 -0
  253. package/dist/repertoire/duffel-client.js +185 -0
  254. package/dist/repertoire/github-client.js +14 -55
  255. package/dist/repertoire/graph-client.js +11 -52
  256. package/dist/repertoire/guardrails.js +396 -0
  257. package/dist/repertoire/mcp-client.js +295 -0
  258. package/dist/repertoire/mcp-manager.js +362 -0
  259. package/dist/repertoire/mcp-tools.js +63 -0
  260. package/dist/repertoire/shell-sessions.js +133 -0
  261. package/dist/repertoire/skills.js +15 -24
  262. package/dist/repertoire/stripe-client.js +131 -0
  263. package/dist/repertoire/tasks/board.js +31 -5
  264. package/dist/repertoire/tasks/fix.js +182 -0
  265. package/dist/repertoire/tasks/index.js +16 -4
  266. package/dist/repertoire/tasks/lifecycle.js +2 -2
  267. package/dist/repertoire/tasks/parser.js +3 -2
  268. package/dist/repertoire/tasks/scanner.js +194 -37
  269. package/dist/repertoire/tasks/transitions.js +16 -78
  270. package/dist/repertoire/tool-results.js +29 -0
  271. package/dist/repertoire/tools-attachments.js +317 -0
  272. package/dist/repertoire/tools-base.js +47 -1075
  273. package/dist/repertoire/tools-bluebubbles.js +1 -0
  274. package/dist/repertoire/tools-bridge.js +142 -0
  275. package/dist/repertoire/tools-bundle.js +984 -0
  276. package/dist/repertoire/tools-config.js +185 -0
  277. package/dist/repertoire/tools-continuity.js +248 -0
  278. package/dist/repertoire/tools-credential.js +381 -0
  279. package/dist/repertoire/tools-files.js +342 -0
  280. package/dist/repertoire/tools-flight.js +224 -0
  281. package/dist/repertoire/tools-flow.js +119 -0
  282. package/dist/repertoire/tools-github.js +1 -7
  283. package/dist/repertoire/tools-mail.js +1857 -0
  284. package/dist/repertoire/tools-notes.js +421 -0
  285. package/dist/repertoire/tools-session.js +750 -0
  286. package/dist/repertoire/tools-shell.js +120 -0
  287. package/dist/repertoire/tools-stripe.js +180 -0
  288. package/dist/repertoire/tools-surface.js +243 -0
  289. package/dist/repertoire/tools-teams.js +9 -39
  290. package/dist/repertoire/tools-travel.js +125 -0
  291. package/dist/repertoire/tools-trip.js +604 -0
  292. package/dist/repertoire/tools-user-profile.js +144 -0
  293. package/dist/repertoire/tools-vault.js +40 -0
  294. package/dist/repertoire/tools.js +108 -100
  295. package/dist/repertoire/travel-api-client.js +360 -0
  296. package/dist/repertoire/user-profile.js +131 -0
  297. package/dist/repertoire/vault-setup.js +246 -0
  298. package/dist/repertoire/vault-unlock.js +594 -0
  299. package/dist/scripts/claude-code-hook.js +41 -0
  300. package/dist/scripts/claude-code-stop-hook.js +47 -0
  301. package/dist/senses/attention-queue.js +116 -0
  302. package/dist/senses/bluebubbles/active-turns.js +216 -0
  303. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  304. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  305. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +219 -18
  306. package/dist/senses/bluebubbles/entry.js +77 -0
  307. package/dist/senses/{bluebubbles-inbound-log.js → bluebubbles/inbound-log.js} +20 -3
  308. package/dist/senses/bluebubbles/index.js +2305 -0
  309. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  310. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +33 -12
  311. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +3 -3
  312. package/dist/senses/bluebubbles/processed-log.js +133 -0
  313. package/dist/senses/bluebubbles/replay.js +137 -0
  314. package/dist/senses/{bluebubbles-runtime-state.js → bluebubbles/runtime-state.js} +30 -2
  315. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  316. package/dist/senses/cli/bracketed-paste.js +82 -0
  317. package/dist/senses/cli/image-paste.js +287 -0
  318. package/dist/senses/cli/image-ref-navigation.js +75 -0
  319. package/dist/senses/cli/ink-app.js +156 -0
  320. package/dist/senses/cli/inline-diff.js +64 -0
  321. package/dist/senses/cli/input-keys.js +174 -0
  322. package/dist/senses/cli/kill-ring.js +86 -0
  323. package/dist/senses/cli/message-list.js +51 -0
  324. package/dist/senses/cli/ouro-tui.js +607 -0
  325. package/dist/senses/cli/spinner-imperative.js +135 -0
  326. package/dist/senses/cli/spinner.js +101 -0
  327. package/dist/senses/cli/status-line.js +60 -0
  328. package/dist/senses/cli/streaming-markdown.js +526 -0
  329. package/dist/senses/cli/tool-display.js +85 -0
  330. package/dist/senses/cli/tool-render.js +85 -0
  331. package/dist/senses/cli/tui-store.js +240 -0
  332. package/dist/senses/cli/virtual-list.js +35 -0
  333. package/dist/senses/cli-entry.js +60 -8
  334. package/dist/senses/cli-layout.js +187 -0
  335. package/dist/senses/cli.js +520 -209
  336. package/dist/senses/commands.js +66 -3
  337. package/dist/senses/habit-turn-message.js +108 -0
  338. package/dist/senses/inner-dialog-worker.js +175 -21
  339. package/dist/senses/inner-dialog.js +330 -27
  340. package/dist/senses/mail-entry.js +66 -0
  341. package/dist/senses/mail.js +379 -0
  342. package/dist/senses/pipeline.js +549 -181
  343. package/dist/senses/proactive-content-guard.js +51 -0
  344. package/dist/senses/shared-turn.js +248 -0
  345. package/dist/senses/surface-tool.js +68 -0
  346. package/dist/senses/teams-entry.js +60 -8
  347. package/dist/senses/teams.js +387 -98
  348. package/dist/senses/trust-gate.js +100 -5
  349. package/dist/senses/voice/elevenlabs.js +125 -0
  350. package/dist/senses/voice/index.js +22 -0
  351. package/dist/senses/voice/transcript.js +70 -0
  352. package/dist/senses/voice/turn.js +85 -0
  353. package/dist/senses/voice/types.js +2 -0
  354. package/dist/senses/voice/whisper.js +133 -0
  355. package/dist/senses/voice-entry.js +80 -0
  356. package/dist/trips/core.js +138 -0
  357. package/dist/trips/store.js +146 -0
  358. package/package.json +38 -7
  359. package/skills/agent-commerce.md +106 -0
  360. package/skills/browser-navigation.md +117 -0
  361. package/skills/commerce-setup-guide.md +116 -0
  362. package/skills/commerce-setup.md +84 -0
  363. package/skills/configure-dev-tools.md +101 -0
  364. package/skills/travel-planning.md +138 -0
  365. package/dist/heart/daemon/auth-flow.js +0 -351
  366. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  367. package/dist/heart/daemon/subagent-installer.js +0 -166
  368. package/dist/heart/session-recall.js +0 -116
  369. package/dist/mind/associative-recall.js +0 -209
  370. package/dist/senses/bluebubbles-entry.js +0 -13
  371. package/dist/senses/bluebubbles.js +0 -1177
  372. package/dist/senses/debug-activity.js +0 -148
  373. package/subagents/README.md +0 -86
  374. package/subagents/work-doer.md +0 -237
  375. package/subagents/work-merger.md +0 -618
  376. package/subagents/work-planner.md +0 -390
  377. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  378. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  379. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  380. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  381. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  382. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  383. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  384. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  385. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  386. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  387. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  388. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  389. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  390. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  391. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  392. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -0,0 +1,776 @@
1
+ "use strict";
2
+ /**
3
+ * CLI default dependency wiring.
4
+ *
5
+ * Creates the production OuroCliDeps with real filesystem, socket, and
6
+ * process bindings. Tests inject mocks for all of these.
7
+ */
8
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
9
+ if (k2 === undefined) k2 = k;
10
+ var desc = Object.getOwnPropertyDescriptor(m, k);
11
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
12
+ desc = { enumerable: true, get: function() { return m[k]; } };
13
+ }
14
+ Object.defineProperty(o, k2, desc);
15
+ }) : (function(o, m, k, k2) {
16
+ if (k2 === undefined) k2 = k;
17
+ o[k2] = m[k];
18
+ }));
19
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
20
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
21
+ }) : function(o, v) {
22
+ o["default"] = v;
23
+ });
24
+ var __importStar = (this && this.__importStar) || (function () {
25
+ var ownKeys = function(o) {
26
+ ownKeys = Object.getOwnPropertyNames || function (o) {
27
+ var ar = [];
28
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
29
+ return ar;
30
+ };
31
+ return ownKeys(o);
32
+ };
33
+ return function (mod) {
34
+ if (mod && mod.__esModule) return mod;
35
+ var result = {};
36
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
37
+ __setModuleDefault(result, mod);
38
+ return result;
39
+ };
40
+ })();
41
+ Object.defineProperty(exports, "__esModule", { value: true });
42
+ exports.defaultStartDaemonProcess = defaultStartDaemonProcess;
43
+ exports.readFirstBundleMetaVersion = readFirstBundleMetaVersion;
44
+ exports.isDaemonLaunchAgentLoaded = isDaemonLaunchAgentLoaded;
45
+ exports.waitForBootstrappedDaemonSocket = waitForBootstrappedDaemonSocket;
46
+ exports.defaultListDiscoveredAgents = defaultListDiscoveredAgents;
47
+ exports.defaultRunSerpentGuide = defaultRunSerpentGuide;
48
+ exports.createDefaultOuroCliDeps = createDefaultOuroCliDeps;
49
+ const child_process_1 = require("child_process");
50
+ const fs = __importStar(require("fs"));
51
+ const os = __importStar(require("os"));
52
+ const path = __importStar(require("path"));
53
+ const identity_1 = require("../identity");
54
+ const runtime_1 = require("../../nerves/runtime");
55
+ const ouro_path_installer_1 = require("../versioning/ouro-path-installer");
56
+ const ouro_uti_1 = require("../versioning/ouro-uti");
57
+ const ouro_version_manager_1 = require("../versioning/ouro-version-manager");
58
+ const update_checker_1 = require("../versioning/update-checker");
59
+ const skill_management_installer_1 = require("./skill-management-installer");
60
+ const hatch_flow_1 = require("../hatch/hatch-flow");
61
+ const specialist_orchestrator_1 = require("../hatch/specialist-orchestrator");
62
+ const specialist_prompt_1 = require("../hatch/specialist-prompt");
63
+ const specialist_tools_1 = require("../hatch/specialist-tools");
64
+ const runtime_mode_1 = require("./runtime-mode");
65
+ const agent_discovery_1 = require("./agent-discovery");
66
+ const bundle_manifest_1 = require("../../mind/bundle-manifest");
67
+ const ouro_bot_global_installer_1 = require("../versioning/ouro-bot-global-installer");
68
+ const logs_prune_1 = require("./logs-prune");
69
+ const daemon_health_1 = require("./daemon-health");
70
+ const log_tailer_1 = require("./log-tailer");
71
+ const launchd_1 = require("./launchd");
72
+ const socket_client_1 = require("./socket-client");
73
+ const session_activity_1 = require("../session-activity");
74
+ const auth_flow_1 = require("../auth/auth-flow");
75
+ const provider_models_1 = require("../provider-models");
76
+ const cli_parse_1 = require("./cli-parse");
77
+ const provider_discovery_1 = require("./provider-discovery");
78
+ const provider_credentials_1 = require("../provider-credentials");
79
+ // ── Default implementations ──
80
+ async function defaultStartDaemonProcess(socketPath) {
81
+ const launchdStarted = await startDaemonProcessViaLaunchd(socketPath);
82
+ if (launchdStarted)
83
+ return launchdStarted;
84
+ const entry = path.join((0, identity_1.getRepoRoot)(), "dist", "heart", "daemon", "daemon-entry.js");
85
+ // Redirect stdio to /dev/null via file descriptors — using 'ignore' causes EPIPE
86
+ // when the daemon's logging system writes to stderr after the parent exits.
87
+ const outFd = fs.openSync(os.devNull, "w");
88
+ const errFd = fs.openSync(os.devNull, "w");
89
+ const child = (0, child_process_1.spawn)(process.execPath, [entry, "--socket", socketPath], {
90
+ detached: true,
91
+ stdio: ["ignore", outFd, errFd],
92
+ });
93
+ child.unref();
94
+ // Don't close fds — the child process needs them. They'll be cleaned up when the parent exits.
95
+ return { pid: child.pid ?? null };
96
+ }
97
+ function defaultWriteStdout(text) {
98
+ process.stdout.write(text.endsWith("\n") ? text : `${text}\n`);
99
+ }
100
+ /* v8 ignore start -- thin terminal adapter around process stdout @preserve */
101
+ function defaultWriteRaw(text) {
102
+ process.stdout.write(text);
103
+ }
104
+ /* v8 ignore stop */
105
+ function resolveDaemonBootEntryPath(homeDir) {
106
+ const repoRoot = (0, identity_1.getRepoRoot)();
107
+ const versionManagerRootMarker = `${path.sep}.ouro-cli${path.sep}versions${path.sep}`;
108
+ if ((0, runtime_mode_1.detectRuntimeMode)(repoRoot) === "production" && repoRoot.includes(versionManagerRootMarker)) {
109
+ return path.join((0, ouro_version_manager_1.getOuroCliHome)(homeDir), "CurrentVersion", "node_modules", "@ouro.bot", "cli", "dist", "heart", "daemon", "daemon-entry.js");
110
+ }
111
+ return path.join(repoRoot, "dist", "heart", "daemon", "daemon-entry.js");
112
+ }
113
+ /**
114
+ * Read the runtimeVersion from the first .ouro bundle's bundle-meta.json.
115
+ * Returns undefined if none found or unreadable.
116
+ */
117
+ function readFirstBundleMetaVersion(bundlesRoot) {
118
+ try {
119
+ if (!fs.existsSync(bundlesRoot))
120
+ return undefined;
121
+ const entries = fs.readdirSync(bundlesRoot, { withFileTypes: true });
122
+ for (const entry of entries) {
123
+ /* v8 ignore next -- skip non-.ouro dirs: tested via version-detect tests @preserve */
124
+ if (!entry.isDirectory() || !entry.name.endsWith(".ouro"))
125
+ continue;
126
+ const metaPath = path.join(bundlesRoot, entry.name, "bundle-meta.json");
127
+ if (!fs.existsSync(metaPath))
128
+ continue;
129
+ const raw = fs.readFileSync(metaPath, "utf-8");
130
+ const meta = JSON.parse(raw);
131
+ if (meta.runtimeVersion)
132
+ return meta.runtimeVersion;
133
+ }
134
+ }
135
+ catch {
136
+ // Best effort — return undefined on any error
137
+ }
138
+ return undefined;
139
+ }
140
+ function defaultCleanupStaleSocket(socketPath) {
141
+ if (fs.existsSync(socketPath)) {
142
+ fs.unlinkSync(socketPath);
143
+ }
144
+ }
145
+ function defaultReadHealthUpdatedAt(healthPath) {
146
+ try {
147
+ return fs.statSync(healthPath).mtimeMs;
148
+ }
149
+ catch {
150
+ return null;
151
+ }
152
+ }
153
+ function defaultReadRecentDaemonLogLines(lines = 10) {
154
+ const files = (0, log_tailer_1.discoverLogFiles)({});
155
+ const recentLines = [];
156
+ for (const file of files) {
157
+ recentLines.push(...(0, log_tailer_1.readLastLines)(file, lines, fs.readFileSync));
158
+ }
159
+ return recentLines.slice(-lines).map((line) => (0, log_tailer_1.formatLogLine)(line));
160
+ }
161
+ /* v8 ignore start -- CLI npm registry fetch wrapper: integration code @preserve */
162
+ async function defaultFetchCliRegistryJson(timeoutMs) {
163
+ const controller = new AbortController();
164
+ const timeoutId = setTimeout(() => {
165
+ controller.abort();
166
+ }, timeoutMs);
167
+ try {
168
+ const res = await fetch("https://registry.npmjs.org/@ouro.bot/cli", {
169
+ signal: controller.signal,
170
+ });
171
+ return res.json();
172
+ }
173
+ catch (error) {
174
+ if (error instanceof Error && error.name === "AbortError") {
175
+ throw new Error(`update check timed out after ${Math.max(1, Math.round(timeoutMs / 1000))}s`);
176
+ }
177
+ throw error;
178
+ }
179
+ finally {
180
+ clearTimeout(timeoutId);
181
+ }
182
+ }
183
+ /* v8 ignore stop */
184
+ function defaultSleep(ms) {
185
+ return new Promise((resolve) => setTimeout(resolve, ms));
186
+ }
187
+ function defaultSpawnBackgroundCli(argv) {
188
+ const child = (0, child_process_1.spawn)(process.execPath, argv, {
189
+ detached: true,
190
+ stdio: "ignore",
191
+ });
192
+ child.unref();
193
+ return Promise.resolve({ pid: child.pid ?? null });
194
+ }
195
+ function defaultFallbackPendingMessage(command) {
196
+ const inboxDir = path.join((0, identity_1.getAgentBundlesRoot)(), `${command.to}.ouro`, "inbox");
197
+ const pendingPath = path.join(inboxDir, "pending.jsonl");
198
+ const queuedAt = new Date().toISOString();
199
+ const payload = {
200
+ from: command.from,
201
+ to: command.to,
202
+ content: command.content,
203
+ priority: command.priority ?? "normal",
204
+ sessionId: command.sessionId,
205
+ taskRef: command.taskRef,
206
+ queuedAt,
207
+ };
208
+ fs.mkdirSync(inboxDir, { recursive: true });
209
+ fs.appendFileSync(pendingPath, `${JSON.stringify(payload)}\n`, "utf-8");
210
+ (0, runtime_1.emitNervesEvent)({
211
+ level: "warn",
212
+ component: "daemon",
213
+ event: "daemon.message_fallback_queued",
214
+ message: "queued message to pending fallback file",
215
+ meta: {
216
+ to: command.to,
217
+ path: pendingPath,
218
+ sessionId: command.sessionId ?? null,
219
+ taskRef: command.taskRef ?? null,
220
+ },
221
+ });
222
+ return pendingPath;
223
+ }
224
+ function currentUserUid() {
225
+ return process.getuid?.() ?? 0;
226
+ }
227
+ function launchAgentDomain(userUid = currentUserUid()) {
228
+ return `gui/${userUid}`;
229
+ }
230
+ function writeDaemonBootPlist(socketPath) {
231
+ const homeDir = os.homedir();
232
+ const writeDeps = {
233
+ writeFile: (filePath, content) => fs.writeFileSync(filePath, content, "utf-8"),
234
+ mkdirp: (dir) => fs.mkdirSync(dir, { recursive: true }),
235
+ homeDir,
236
+ };
237
+ const entryPath = resolveDaemonBootEntryPath(homeDir);
238
+ /* v8 ignore next -- covered via mock in daemon-cli-defaults.test.ts; v8 on CI attributes the real fs.existsSync branch to the non-mock load @preserve */
239
+ if (!fs.existsSync(entryPath)) {
240
+ (0, runtime_1.emitNervesEvent)({
241
+ level: "warn",
242
+ component: "daemon",
243
+ event: "daemon.entry_path_missing",
244
+ message: "entryPath does not exist on disk — plist may point to a stale location. Run 'ouro daemon install' from the correct location.",
245
+ meta: { entryPath },
246
+ });
247
+ }
248
+ return (0, launchd_1.writeLaunchAgentPlist)(writeDeps, {
249
+ nodePath: process.execPath,
250
+ entryPath,
251
+ socketPath,
252
+ logDir: (0, identity_1.getAgentDaemonLogsDir)(),
253
+ envPath: process.env.PATH,
254
+ });
255
+ }
256
+ function isDaemonLaunchAgentLoaded(deps) {
257
+ const userUid = deps?.userUid ?? currentUserUid();
258
+ const exec = deps?.exec ?? ((cmd) => { (0, child_process_1.execSync)(cmd, { stdio: "ignore" }); });
259
+ try {
260
+ exec(`launchctl print ${launchAgentDomain(userUid)}/${launchd_1.DAEMON_PLIST_LABEL}`);
261
+ return true;
262
+ }
263
+ catch {
264
+ return false;
265
+ }
266
+ }
267
+ function readDaemonLaunchAgentPid() {
268
+ try {
269
+ const output = (0, child_process_1.execSync)(`launchctl print ${launchAgentDomain()}/${launchd_1.DAEMON_PLIST_LABEL}`, { encoding: "utf-8" });
270
+ const match = output.match(/^\s*pid = (\d+)/m);
271
+ return match ? Number(match[1]) : null;
272
+ }
273
+ catch {
274
+ return null;
275
+ }
276
+ }
277
+ async function waitForBootstrappedDaemonSocket(socketPath, deps = {}) {
278
+ const checkSocketAlive = deps.checkSocketAlive ?? socket_client_1.checkDaemonSocketAlive;
279
+ const now = deps.now ?? Date.now;
280
+ const sleep = deps.sleep ?? ((ms) => new Promise((resolve) => setTimeout(resolve, ms)));
281
+ const timeoutMs = deps.timeoutMs ?? 30_000;
282
+ const initialSettleMs = deps.initialSettleMs ?? 1_000;
283
+ const pollIntervalMs = deps.pollIntervalMs ?? 250;
284
+ const requiredConsecutiveAliveChecks = deps.requiredConsecutiveAliveChecks ?? 2;
285
+ const deadline = now() + timeoutMs;
286
+ const firstTrustworthyCheckAt = now() + initialSettleMs;
287
+ let consecutiveAliveChecks = 0;
288
+ while (now() < deadline) {
289
+ await sleep(pollIntervalMs);
290
+ if (now() < firstTrustworthyCheckAt)
291
+ continue;
292
+ if (await checkSocketAlive(socketPath)) {
293
+ consecutiveAliveChecks += 1;
294
+ if (consecutiveAliveChecks >= requiredConsecutiveAliveChecks)
295
+ return;
296
+ }
297
+ else {
298
+ consecutiveAliveChecks = 0;
299
+ }
300
+ }
301
+ (0, runtime_1.emitNervesEvent)({
302
+ level: "warn",
303
+ component: "daemon",
304
+ event: "daemon.launchd_bootstrap_socket_wait_timeout",
305
+ message: "launchd bootstrap finished but daemon socket did not settle before timeout",
306
+ meta: { socketPath },
307
+ });
308
+ }
309
+ async function startDaemonProcessViaLaunchd(socketPath) {
310
+ if (process.platform !== "darwin") {
311
+ return null;
312
+ }
313
+ const plistPath = writeDaemonBootPlist(socketPath);
314
+ const userUid = currentUserUid();
315
+ const domain = launchAgentDomain(userUid);
316
+ try {
317
+ (0, runtime_1.emitNervesEvent)({
318
+ component: "daemon",
319
+ event: "daemon.launchd_bootstrap_start",
320
+ message: "starting daemon launch agent for current login session",
321
+ meta: { plistPath, label: launchd_1.DAEMON_PLIST_LABEL },
322
+ });
323
+ if (isDaemonLaunchAgentLoaded({ exec: (cmd) => { (0, child_process_1.execSync)(cmd, { stdio: "ignore" }); }, userUid })) {
324
+ (0, child_process_1.execSync)(`launchctl kickstart -k ${domain}/${launchd_1.DAEMON_PLIST_LABEL}`, { stdio: "ignore" });
325
+ }
326
+ else {
327
+ (0, child_process_1.execSync)(`launchctl bootstrap ${domain} "${plistPath}"`, { stdio: "ignore" });
328
+ }
329
+ (0, runtime_1.emitNervesEvent)({
330
+ component: "daemon",
331
+ event: "daemon.launchd_bootstrap_end",
332
+ message: "daemon launch agent started for current login session",
333
+ meta: { plistPath, label: launchd_1.DAEMON_PLIST_LABEL },
334
+ });
335
+ await waitForBootstrappedDaemonSocket(socketPath);
336
+ return { pid: readDaemonLaunchAgentPid() };
337
+ }
338
+ catch (error) {
339
+ (0, runtime_1.emitNervesEvent)({
340
+ level: "warn",
341
+ component: "daemon",
342
+ event: "daemon.launchd_bootstrap_error",
343
+ message: "failed to start daemon launch agent for current login session",
344
+ meta: { plistPath, label: launchd_1.DAEMON_PLIST_LABEL, error: error instanceof Error ? error.message : String(error) },
345
+ });
346
+ return null;
347
+ }
348
+ }
349
+ async function defaultEnsureDaemonBootPersistence(socketPath) {
350
+ if (process.platform !== "darwin") {
351
+ return;
352
+ }
353
+ const plistPath = writeDaemonBootPlist(socketPath);
354
+ const userUid = currentUserUid();
355
+ if (isDaemonLaunchAgentLoaded({ exec: (cmd) => { (0, child_process_1.execSync)(cmd, { stdio: "ignore" }); }, userUid })) {
356
+ (0, runtime_1.emitNervesEvent)({
357
+ component: "daemon",
358
+ event: "daemon.launchd_bootstrap_skipped_loaded",
359
+ message: "daemon launch agent already loaded",
360
+ meta: { plistPath, label: launchd_1.DAEMON_PLIST_LABEL },
361
+ });
362
+ return;
363
+ }
364
+ }
365
+ function defaultPrepareDaemonRuntimeReplacement() {
366
+ if (process.platform !== "darwin") {
367
+ return;
368
+ }
369
+ (0, launchd_1.bootoutLaunchAgentByLabel)({
370
+ exec: (cmd) => { (0, child_process_1.execSync)(cmd, { stdio: "ignore" }); },
371
+ userUid: process.getuid?.() ?? 0,
372
+ });
373
+ // launchctl bootout may return before the old daemon's SIGTERM cleanup has
374
+ // finished unlinking its Unix socket. Give that cleanup a short chance to
375
+ // settle before the replacement daemon binds the same socket path.
376
+ (0, child_process_1.execSync)("/bin/sleep 1", { stdio: "ignore" });
377
+ }
378
+ async function defaultPromptInput(question) {
379
+ const readline = await Promise.resolve().then(() => __importStar(require("readline/promises")));
380
+ const rl = readline.createInterface({
381
+ input: process.stdin,
382
+ output: process.stdout,
383
+ });
384
+ try {
385
+ const response = await rl.question(question);
386
+ return response.trim();
387
+ }
388
+ finally {
389
+ rl.close();
390
+ }
391
+ }
392
+ async function defaultPromptSecret(question) {
393
+ if (process.stdin.isTTY !== true || process.stdout.isTTY !== true) {
394
+ throw new Error("vault unlock secret entry requires an interactive terminal so the secret can be hidden. Re-run this command in a terminal and enter the human-chosen secret when prompted.");
395
+ }
396
+ const readline = await Promise.resolve().then(() => __importStar(require("readline")));
397
+ const rl = readline.createInterface({
398
+ input: process.stdin,
399
+ output: process.stdout,
400
+ terminal: true,
401
+ });
402
+ const mutableRl = rl;
403
+ const originalWriteToOutput = mutableRl._writeToOutput;
404
+ let muted = false;
405
+ if (originalWriteToOutput) {
406
+ mutableRl._writeToOutput = (stringToWrite) => {
407
+ if (!muted) {
408
+ originalWriteToOutput.call(rl, stringToWrite);
409
+ }
410
+ };
411
+ }
412
+ try {
413
+ const response = await new Promise((resolve) => {
414
+ rl.question(question, (answer) => {
415
+ process.stdout.write("\n");
416
+ resolve(answer);
417
+ });
418
+ muted = true;
419
+ });
420
+ return response.trim();
421
+ }
422
+ finally {
423
+ if (originalWriteToOutput) {
424
+ mutableRl._writeToOutput = originalWriteToOutput;
425
+ }
426
+ rl.close();
427
+ }
428
+ }
429
+ function defaultListDiscoveredAgents() {
430
+ return (0, agent_discovery_1.listEnabledBundleAgents)({
431
+ bundlesRoot: (0, identity_1.getAgentBundlesRoot)(),
432
+ readdirSync: fs.readdirSync,
433
+ readFileSync: fs.readFileSync,
434
+ });
435
+ }
436
+ // ── Serpent guide (interactive hatch) ──
437
+ /* v8 ignore start -- integration: interactive terminal specialist session @preserve */
438
+ async function defaultRunSerpentGuide() {
439
+ const { runCliSession } = await Promise.resolve().then(() => __importStar(require("../../senses/cli")));
440
+ const { setAgentName, setAgentConfigOverride } = await Promise.resolve().then(() => __importStar(require("../identity")));
441
+ const readlinePromises = await Promise.resolve().then(() => __importStar(require("readline/promises")));
442
+ const crypto = await Promise.resolve().then(() => __importStar(require("crypto")));
443
+ // Phase 1: cold CLI — collect provider/credentials with a simple readline
444
+ const coldRl = readlinePromises.createInterface({ input: process.stdin, output: process.stdout });
445
+ const coldPrompt = async (q) => {
446
+ const answer = await coldRl.question(q);
447
+ return answer.trim();
448
+ };
449
+ let providerRaw;
450
+ let credentials = {};
451
+ let providerConfig = {};
452
+ let selectedCredentialPayload = {};
453
+ const tempDir = path.join(os.tmpdir(), `ouro-hatch-${crypto.randomUUID()}`);
454
+ try {
455
+ const discovered = [];
456
+ const existingBundles = (0, specialist_orchestrator_1.listExistingBundles)((0, identity_1.getAgentBundlesRoot)());
457
+ const existingBundleCount = existingBundles.length;
458
+ const hatchVerb = existingBundleCount > 0 ? "let's hatch a new agent." : "let's hatch your first agent.";
459
+ // Default models per provider (used when entering new credentials)
460
+ const defaultModels = provider_models_1.DEFAULT_PROVIDER_MODELS;
461
+ const { pingProvider } = await Promise.resolve().then(() => __importStar(require("../provider-ping")));
462
+ const installedAgentCreds = await (0, provider_discovery_1.discoverInstalledAgentCredentials)(existingBundles);
463
+ for (const cred of installedAgentCreds) {
464
+ discovered.push({
465
+ ...cred,
466
+ providerConfig: { model: defaultModels[cred.provider], ...cred.providerConfig },
467
+ });
468
+ }
469
+ // Scan environment variables for API keys using the shared helper
470
+ const envCreds = (0, provider_discovery_1.scanEnvVarCredentials)(process.env);
471
+ const envDiscovered = [];
472
+ for (const cred of envCreds) {
473
+ // Enrich with default model and first matching env var name (for display)
474
+ const desc = identity_1.PROVIDER_CREDENTIALS[cred.provider];
475
+ const firstEnvVar = Object.entries(desc.envVars).find(([envVar]) => process.env[envVar])?.[0] ?? "";
476
+ const provCfg = { model: defaultModels[cred.provider], ...cred.providerConfig };
477
+ if (cred.provider === "azure" && cred.credentials.deployment)
478
+ provCfg.deployment = cred.credentials.deployment;
479
+ const enriched = { ...cred, providerConfig: provCfg };
480
+ envDiscovered.push({ ...enriched, envVar: firstEnvVar });
481
+ discovered.push(enriched);
482
+ }
483
+ let welcomed = false;
484
+ while (true) {
485
+ if (!welcomed) {
486
+ process.stdout.write(`\n\ud83d\udc0d welcome to ouroboros! ${hatchVerb}\n`);
487
+ welcomed = true;
488
+ }
489
+ if (discovered.length > 0) {
490
+ process.stdout.write("i found existing API credentials:\n\n");
491
+ const credentialOptions = discovered;
492
+ for (let i = 0; i < credentialOptions.length; i++) {
493
+ const model = credentialOptions[i].providerConfig.model || credentialOptions[i].providerConfig.deployment || "";
494
+ const modelLabel = model ? `, ${model}` : "";
495
+ const envMatch = envDiscovered.find((e) => e.provider === credentialOptions[i].provider && credentialOptions[i].agentName === "env");
496
+ const sourceLabel = (0, provider_discovery_1.describeDiscoveredCredentialSource)(credentialOptions[i], envMatch?.envVar);
497
+ process.stdout.write(` ${i + 1}. ${credentialOptions[i].provider}${modelLabel} (${sourceLabel})\n`);
498
+ }
499
+ process.stdout.write("\n");
500
+ const choice = await coldPrompt("use one of these? enter number, or 'new' for a different key, or 'q' to cancel: ");
501
+ if (["q", "quit", "cancel"].includes(choice.toLowerCase())) {
502
+ coldRl.close();
503
+ return null;
504
+ }
505
+ const idx = parseInt(choice, 10) - 1;
506
+ if (idx >= 0 && idx < credentialOptions.length) {
507
+ providerRaw = credentialOptions[idx].provider;
508
+ credentials = credentialOptions[idx].credentials;
509
+ providerConfig = credentialOptions[idx].providerConfig;
510
+ }
511
+ else {
512
+ const pRaw = await coldPrompt("provider (anthropic/azure/minimax/openai-codex/github-copilot): ");
513
+ if (!(0, cli_parse_1.isAgentProvider)(pRaw)) {
514
+ process.stdout.write("unknown provider. run `ouro hatch` to try again.\n");
515
+ coldRl.close();
516
+ return null;
517
+ }
518
+ providerRaw = pRaw;
519
+ providerConfig = { model: defaultModels[providerRaw] };
520
+ credentials = await (0, auth_flow_1.collectRuntimeAuthCredentials)({ agentName: "SerpentGuide", provider: providerRaw, promptInput: coldPrompt }, {});
521
+ }
522
+ }
523
+ else {
524
+ process.stdout.write("i need an API key to power our conversation.\n\n");
525
+ const pRaw = await coldPrompt("provider (anthropic/azure/minimax/openai-codex/github-copilot): ");
526
+ if (!(0, cli_parse_1.isAgentProvider)(pRaw)) {
527
+ process.stdout.write("unknown provider. run `ouro hatch` to try again.\n");
528
+ coldRl.close();
529
+ return null;
530
+ }
531
+ providerRaw = pRaw;
532
+ providerConfig = { model: defaultModels[providerRaw] };
533
+ credentials = await (0, auth_flow_1.collectRuntimeAuthCredentials)({ agentName: "SerpentGuide", provider: providerRaw, promptInput: coldPrompt }, {});
534
+ }
535
+ selectedCredentialPayload = { ...providerConfig, ...credentials };
536
+ const pingResult = await pingProvider(providerRaw, selectedCredentialPayload);
537
+ if (pingResult.ok) {
538
+ break;
539
+ }
540
+ process.stdout.write(`credentials didn't work (${pingResult.message}). `);
541
+ if (discovered.length > 0) {
542
+ process.stdout.write("choose another saved credential or enter 'new'.\n\n");
543
+ }
544
+ else {
545
+ process.stdout.write("let's try again.\n\n");
546
+ }
547
+ }
548
+ coldRl.close();
549
+ process.stdout.write("\n");
550
+ const split = (0, provider_credentials_1.splitProviderCredentialFields)(providerRaw, selectedCredentialPayload);
551
+ (0, provider_credentials_1.cacheProviderCredentialRecords)("SerpentGuide", [
552
+ (0, provider_credentials_1.createProviderCredentialRecord)({
553
+ provider: providerRaw,
554
+ credentials: split.credentials,
555
+ config: split.config,
556
+ provenance: { source: "manual" },
557
+ }),
558
+ ]);
559
+ // Phase 2: configure runtime for serpent guide
560
+ const bundleSourceDir = path.resolve(__dirname, "..", "..", "..", "SerpentGuide.ouro");
561
+ const bundlesRoot = (0, identity_1.getAgentBundlesRoot)();
562
+ // Suppress non-critical log noise during hatch.
563
+ const { setRuntimeLogger } = await Promise.resolve().then(() => __importStar(require("../../nerves/runtime")));
564
+ const { createLogger } = await Promise.resolve().then(() => __importStar(require("../../nerves")));
565
+ setRuntimeLogger(createLogger({ level: "error" }));
566
+ // Configure runtime: set agent identity + config override so runAgent
567
+ // doesn't try to read from ~/AgentBundles/SerpentGuide.ouro/. (As of
568
+ // Layer 3, SerpentGuide identities live in-repo only — the
569
+ // `~/AgentBundles/SerpentGuide.ouro/psyche/identities` override path
570
+ // was removed. The override path is no longer read or honored.)
571
+ setAgentName("SerpentGuide");
572
+ // Build specialist system prompt
573
+ const soulText = (0, specialist_orchestrator_1.loadSoulText)(bundleSourceDir);
574
+ const identitiesDir = path.join(bundleSourceDir, "psyche", "identities");
575
+ const identity = (0, specialist_orchestrator_1.pickRandomIdentity)(identitiesDir);
576
+ // Load identity-specific spinner phrases (falls back to DEFAULT_AGENT_PHRASES)
577
+ const { loadIdentityPhrases } = await Promise.resolve().then(() => __importStar(require("../hatch/specialist-orchestrator")));
578
+ const phrases = loadIdentityPhrases(bundleSourceDir, identity.fileName);
579
+ const resolvedModel = providerConfig.model || providerConfig.deployment || "";
580
+ setAgentConfigOverride({
581
+ version: 2,
582
+ enabled: true,
583
+ provider: providerRaw,
584
+ humanFacing: { provider: providerRaw, model: resolvedModel },
585
+ agentFacing: { provider: providerRaw, model: resolvedModel },
586
+ phrases,
587
+ });
588
+ const systemPrompt = (0, specialist_prompt_1.buildSpecialistSystemPrompt)(soulText, identity.content, existingBundles, {
589
+ tempDir,
590
+ provider: providerRaw,
591
+ model: providerConfig.model ?? "",
592
+ });
593
+ // Build specialist tools
594
+ const specialistTools = (0, specialist_tools_1.getSpecialistTools)();
595
+ const specialistExecTool = (0, specialist_tools_1.createSpecialistExecTool)({
596
+ tempDir,
597
+ credentials: selectedCredentialPayload,
598
+ provider: providerRaw,
599
+ bundlesRoot,
600
+ animationWriter: (text) => process.stdout.write(text),
601
+ promptSecret: defaultPromptSecret,
602
+ });
603
+ // Run the serpent guide session via runCliSession
604
+ const result = await runCliSession({
605
+ agentName: "SerpentGuide",
606
+ tools: specialistTools,
607
+ execTool: specialistExecTool,
608
+ exitOnToolCall: "complete_adoption",
609
+ autoFirstTurn: true,
610
+ banner: false,
611
+ disableCommands: true,
612
+ skipSystemPromptRefresh: true,
613
+ messages: [
614
+ { role: "system", content: systemPrompt },
615
+ { role: "user", content: "hi" },
616
+ ],
617
+ });
618
+ if (result.exitReason === "tool_exit" && result.toolResult) {
619
+ const parsed = typeof result.toolResult === "string" ? JSON.parse(result.toolResult) : result.toolResult;
620
+ if (parsed.success && parsed.agentName) {
621
+ return parsed.agentName;
622
+ }
623
+ }
624
+ return null;
625
+ }
626
+ catch (err) {
627
+ process.stderr.write(`\nouro hatch error: ${err instanceof Error ? err.stack ?? err.message : String(err)}\n`);
628
+ coldRl.close();
629
+ return null;
630
+ }
631
+ finally {
632
+ // Clear specialist config/identity so the hatched agent gets its own
633
+ setAgentConfigOverride(null);
634
+ const { resetProviderRuntime } = await Promise.resolve().then(() => __importStar(require("../core")));
635
+ resetProviderRuntime();
636
+ const { resetConfigCache } = await Promise.resolve().then(() => __importStar(require("../config")));
637
+ resetConfigCache();
638
+ // Restore default logging
639
+ const { setRuntimeLogger: restoreLogger } = await Promise.resolve().then(() => __importStar(require("../../nerves/runtime")));
640
+ restoreLogger(null);
641
+ // Clean up temp dir if it still exists
642
+ try {
643
+ if (fs.existsSync(tempDir)) {
644
+ fs.rmSync(tempDir, { recursive: true, force: true });
645
+ }
646
+ }
647
+ catch {
648
+ // Best effort cleanup
649
+ }
650
+ }
651
+ }
652
+ /* v8 ignore stop */
653
+ // ── Factory ──
654
+ function createDefaultOuroCliDeps(socketPath = socket_client_1.DEFAULT_DAEMON_SOCKET_PATH) {
655
+ return {
656
+ socketPath,
657
+ sendCommand: socket_client_1.sendDaemonCommand,
658
+ startDaemonProcess: defaultStartDaemonProcess,
659
+ writeStdout: defaultWriteStdout,
660
+ setExitCode: (code) => {
661
+ const current = typeof process.exitCode === "number" ? process.exitCode : 0;
662
+ process.exitCode = Math.max(current, code);
663
+ },
664
+ writeRaw: defaultWriteRaw,
665
+ isTTY: process.stdout.isTTY === true,
666
+ checkSocketAlive: socket_client_1.checkDaemonSocketAlive,
667
+ cleanupStaleSocket: defaultCleanupStaleSocket,
668
+ fallbackPendingMessage: defaultFallbackPendingMessage,
669
+ tailLogs: (options) => (0, log_tailer_1.tailLogs)(options),
670
+ healthFilePath: (0, daemon_health_1.getDefaultHealthPath)(),
671
+ readHealthState: daemon_health_1.readHealth,
672
+ readHealthUpdatedAt: defaultReadHealthUpdatedAt,
673
+ readRecentDaemonLogLines: defaultReadRecentDaemonLogLines,
674
+ sleep: defaultSleep,
675
+ spawnBackgroundCli: defaultSpawnBackgroundCli,
676
+ now: () => Date.now(),
677
+ updateCheckTimeoutMs: update_checker_1.CLI_UPDATE_CHECK_TIMEOUT_MS,
678
+ startupPollIntervalMs: 250,
679
+ startupStabilityWindowMs: 1_500,
680
+ startupTimeoutMs: 60_000,
681
+ startupRetryLimit: 1,
682
+ listDiscoveredAgents: defaultListDiscoveredAgents,
683
+ runHatchFlow: hatch_flow_1.runHatchFlow,
684
+ promptInput: defaultPromptInput,
685
+ promptSecret: defaultPromptSecret,
686
+ runSerpentGuide: defaultRunSerpentGuide,
687
+ runAuthFlow: auth_flow_1.runRuntimeAuthFlow,
688
+ registerOuroBundleType: ouro_uti_1.registerOuroBundleUti,
689
+ installOuroCommand: ouro_path_installer_1.installOuroCommand,
690
+ /* v8 ignore start -- self-healing: ensures active symlink matches running runtime version @preserve */
691
+ ensureCurrentVersionInstalled: () => {
692
+ const linkedVersion = (0, ouro_version_manager_1.getCurrentVersion)({});
693
+ const version = (0, bundle_manifest_1.getPackageVersion)();
694
+ if (linkedVersion === version)
695
+ return;
696
+ (0, ouro_version_manager_1.ensureLayout)({});
697
+ const cliHome = (0, ouro_version_manager_1.getOuroCliHome)();
698
+ const versionEntry = path.join(cliHome, "versions", version, "node_modules", "@ouro.bot", "cli", "dist", "heart", "daemon", "ouro-entry.js");
699
+ if (!fs.existsSync(versionEntry)) {
700
+ (0, ouro_version_manager_1.installVersion)(version, {});
701
+ }
702
+ (0, ouro_version_manager_1.activateVersion)(version, {});
703
+ // Self-prune: every successful version activation cleans up old
704
+ // versions outside the retention window. Without this, every CLI
705
+ // version ever installed accumulates indefinitely under
706
+ // ~/.ouro-cli/versions/ — the user observed installs going back to
707
+ // alpha.85 from March 20 (~100MB of dead node_modules trees).
708
+ // pruneOldVersions keeps the 5 most recent + the active + the
709
+ // previous version, so rollback stays one command away.
710
+ (0, ouro_version_manager_1.pruneOldVersions)(undefined, {});
711
+ },
712
+ /* v8 ignore stop */
713
+ /* v8 ignore start -- CLI version management defaults: integration code @preserve */
714
+ checkForCliUpdate: async () => {
715
+ const { checkForUpdate } = await Promise.resolve().then(() => __importStar(require("../versioning/update-checker")));
716
+ return checkForUpdate((0, bundle_manifest_1.getPackageVersion)(), {
717
+ fetchRegistryJson: () => defaultFetchCliRegistryJson(update_checker_1.CLI_UPDATE_CHECK_TIMEOUT_MS),
718
+ distTag: update_checker_1.CLI_UPDATE_DIST_TAG,
719
+ });
720
+ },
721
+ installCliVersion: async (version) => { (0, ouro_version_manager_1.installVersion)(version, {}); },
722
+ activateCliVersion: (version) => {
723
+ (0, ouro_version_manager_1.activateVersion)(version, {});
724
+ // Same self-prune as ensureCurrentVersionInstalled — fires from the
725
+ // checkForCliUpdate path 1 (in-process update detected → activate
726
+ // → re-exec). Without this, the path 1 code path never triggers
727
+ // a prune.
728
+ (0, ouro_version_manager_1.pruneOldVersions)(undefined, {});
729
+ },
730
+ getCurrentCliVersion: () => (0, ouro_version_manager_1.getCurrentVersion)({}),
731
+ getPreviousCliVersion: () => (0, ouro_version_manager_1.getPreviousVersion)({}),
732
+ listCliVersions: () => (0, ouro_version_manager_1.listInstalledVersions)({}),
733
+ reExecFromNewVersion: (reArgs) => {
734
+ const entry = path.join((0, ouro_version_manager_1.getOuroCliHome)(), "CurrentVersion", "node_modules", "@ouro.bot", "cli", "dist", "heart", "daemon", "ouro-entry.js");
735
+ require("child_process").execFileSync("node", [entry, ...reArgs], { stdio: "inherit" });
736
+ process.exit(0);
737
+ },
738
+ /* v8 ignore stop */
739
+ syncGlobalOuroBotWrapper: ouro_bot_global_installer_1.syncGlobalOuroBotWrapper,
740
+ pruneDaemonLogs: logs_prune_1.pruneDaemonLogs,
741
+ ensureSkillManagement: skill_management_installer_1.ensureSkillManagement,
742
+ prepareDaemonRuntimeReplacement: defaultPrepareDaemonRuntimeReplacement,
743
+ ensureDaemonBootPersistence: defaultEnsureDaemonBootPersistence,
744
+ /* v8 ignore start -- dev-mode defaults: tests inject mocks for mode detection and binary resolution @preserve */
745
+ detectMode: () => (0, runtime_mode_1.detectRuntimeMode)((0, identity_1.getRepoRoot)()),
746
+ getInstalledBinaryPath: () => {
747
+ const cliHome = (0, ouro_version_manager_1.getOuroCliHome)();
748
+ const binaryPath = path.join(cliHome, "bin", "ouro");
749
+ return fs.existsSync(binaryPath) ? binaryPath : null;
750
+ },
751
+ execInstalledBinary: (binaryPath, binArgs) => {
752
+ const { execFileSync } = require("child_process");
753
+ execFileSync(binaryPath, binArgs, { stdio: "inherit" });
754
+ process.exit(0);
755
+ },
756
+ /* v8 ignore stop */
757
+ /* v8 ignore next 3 -- integration: launches interactive CLI session @preserve */
758
+ startChat: async (agentName) => {
759
+ const { main } = await Promise.resolve().then(() => __importStar(require("../../senses/cli")));
760
+ await main(agentName);
761
+ },
762
+ scanSessions: async (agentName) => {
763
+ const agentRoot = (0, identity_1.getAgentRoot)(agentName);
764
+ return (0, session_activity_1.listSessionActivity)({
765
+ sessionsDir: path.join(agentRoot, "state", "sessions"),
766
+ friendsDir: path.join(agentRoot, "friends"),
767
+ agentName,
768
+ }).map((entry) => ({
769
+ friendId: entry.friendId,
770
+ friendName: entry.friendName,
771
+ channel: entry.channel,
772
+ lastActivity: entry.lastActivityAt,
773
+ }));
774
+ },
775
+ };
776
+ }