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

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