@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
@@ -35,19 +35,21 @@ var __importStar = (this && this.__importStar) || (function () {
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
36
  exports.getSpecialistTools = getSpecialistTools;
37
37
  exports.createSpecialistExecTool = createSpecialistExecTool;
38
- const crypto = __importStar(require("crypto"));
39
38
  const fs = __importStar(require("fs"));
40
39
  const path = __importStar(require("path"));
41
40
  const tools_base_1 = require("../../repertoire/tools-base");
42
41
  const hatch_flow_1 = require("./hatch-flow");
43
42
  const hatch_animation_1 = require("./hatch-animation");
44
43
  const bundle_manifest_1 = require("../../mind/bundle-manifest");
44
+ const identity_1 = require("../identity");
45
45
  const runtime_1 = require("../../nerves/runtime");
46
+ const vault_setup_1 = require("../../repertoire/vault-setup");
47
+ const vault_unlock_1 = require("../../repertoire/vault-unlock");
46
48
  const completeAdoptionTool = {
47
49
  type: "function",
48
50
  function: {
49
51
  name: "complete_adoption",
50
- description: "finalize the agent bundle and hatch the new agent. call this only when you have written all 5 psyche files and agent.json to the temp directory, and the human has approved the bundle.",
52
+ description: "finalize the agent bundle and hatch the new agent. call this only when you have written all 5 psyche files and agent.json to the temp directory, and the human has approved the bundle. tool execution asks the human for the hatchling vault unlock secret through a hidden terminal prompt; do not ask for or include vault unlock secrets in chat or tool args.",
51
53
  parameters: {
52
54
  type: "object",
53
55
  properties: {
@@ -90,7 +92,7 @@ const listDirToolSchema = {
90
92
  * Returns the specialist's tool schema array.
91
93
  */
92
94
  function getSpecialistTools() {
93
- return [completeAdoptionTool, tools_base_1.finalAnswerTool, readFileTool.tool, writeFileTool.tool, listDirToolSchema];
95
+ return [completeAdoptionTool, tools_base_1.settleTool, readFileTool.tool, writeFileTool.tool, listDirToolSchema];
94
96
  }
95
97
  const PSYCHE_FILES = ["SOUL.md", "IDENTITY.md", "LORE.md", "TACIT.md", "ASPIRATIONS.md"];
96
98
  function isPascalCase(name) {
@@ -105,21 +107,21 @@ function writeReadme(dir, purpose) {
105
107
  }
106
108
  }
107
109
  function scaffoldBundle(bundleRoot) {
108
- writeReadme(path.join(bundleRoot, "memory"), "Persistent memory store.");
109
- writeReadme(path.join(bundleRoot, "memory", "daily"), "Daily memory entries.");
110
- writeReadme(path.join(bundleRoot, "memory", "archive"), "Archived memory.");
110
+ writeReadme(path.join(bundleRoot, "notes"), "Persistent notes store.");
111
+ writeReadme(path.join(bundleRoot, "notes", "daily"), "Daily note entries.");
112
+ writeReadme(path.join(bundleRoot, "notes", "archive"), "Archived notes.");
111
113
  writeReadme(path.join(bundleRoot, "friends"), "Known friend records.");
112
114
  writeReadme(path.join(bundleRoot, "tasks"), "Task files.");
113
- writeReadme(path.join(bundleRoot, "tasks", "habits"), "Recurring tasks.");
114
115
  writeReadme(path.join(bundleRoot, "tasks", "one-shots"), "One-shot tasks.");
116
+ writeReadme(path.join(bundleRoot, "habits"), "Recurring habits and autonomous rhythms.");
115
117
  writeReadme(path.join(bundleRoot, "tasks", "ongoing"), "Ongoing tasks.");
116
118
  writeReadme(path.join(bundleRoot, "skills"), "Local skill files.");
117
119
  writeReadme(path.join(bundleRoot, "senses"), "Sense-specific config.");
118
120
  writeReadme(path.join(bundleRoot, "senses", "teams"), "Teams sense config.");
119
- // Memory scaffold files
120
- const memoryRoot = path.join(bundleRoot, "memory");
121
- const factsPath = path.join(memoryRoot, "facts.jsonl");
122
- const entitiesPath = path.join(memoryRoot, "entities.json");
121
+ // Notes scaffold files
122
+ const notesRoot = path.join(bundleRoot, "notes");
123
+ const factsPath = path.join(notesRoot, "facts.jsonl");
124
+ const entitiesPath = path.join(notesRoot, "entities.json");
123
125
  /* v8 ignore next -- defensive: guard against re-scaffold on existing bundle @preserve */
124
126
  if (!fs.existsSync(factsPath))
125
127
  fs.writeFileSync(factsPath, "", "utf-8");
@@ -166,13 +168,34 @@ async function execCompleteAdoption(args, deps) {
166
168
  if (fs.existsSync(targetBundle)) {
167
169
  return `error: bundle '${name}.ouro' already exists at ${deps.bundlesRoot}. choose a different name.`;
168
170
  }
171
+ if (!deps.promptSecret) {
172
+ return "error: complete_adoption requires an interactive vault unlock secret prompt. Re-run `ouro hatch` in a terminal so the human can enter a hatchling vault unlock secret without echoing it.";
173
+ }
174
+ const vault = (0, identity_1.resolveVaultConfig)(name);
175
+ let vaultUnlockSecret;
176
+ try {
177
+ vaultUnlockSecret = await (0, vault_unlock_1.promptConfirmedVaultUnlockSecret)({
178
+ promptSecret: deps.promptSecret,
179
+ question: `Choose Ouro vault unlock secret for ${vault.email}: `,
180
+ confirmQuestion: `Confirm Ouro vault unlock secret for ${vault.email}: `,
181
+ emptyError: "hatchling vault creation requires an unlock secret. Re-run `ouro hatch` in an interactive terminal and enter a human-chosen unlock secret.",
182
+ });
183
+ }
184
+ catch (error) {
185
+ return `error: failed to read hatchling vault unlock secret: ${error instanceof Error ? error.message : /* v8 ignore next -- defensive: non-Error catch branch @preserve */ String(error)}`;
186
+ }
169
187
  // Scaffold structural dirs into tempDir
170
188
  scaffoldBundle(deps.tempDir);
171
189
  // Move tempDir -> final bundle location
172
190
  moveDir(deps.tempDir, targetBundle);
173
191
  // Write secrets
174
192
  try {
175
- (0, hatch_flow_1.writeSecretsFile)(name, deps.provider, deps.credentials, deps.secretsRoot);
193
+ const vaultResult = await (0, vault_setup_1.createVaultAccount)(name, vault.serverUrl, vault.email, vaultUnlockSecret);
194
+ if (!vaultResult.success) {
195
+ throw new Error(`failed to create vault: ${vaultResult.error}`);
196
+ }
197
+ (0, vault_unlock_1.storeVaultUnlockSecret)({ agentName: name, email: vault.email, serverUrl: vault.serverUrl }, vaultUnlockSecret);
198
+ await (0, hatch_flow_1.storeHatchlingProviderCredentials)(name, deps.provider, deps.credentials);
176
199
  }
177
200
  catch (e) {
178
201
  // Rollback: remove the moved bundle
@@ -33,14 +33,25 @@ var __importStar = (this && this.__importStar) || (function () {
33
33
  };
34
34
  })();
35
35
  Object.defineProperty(exports, "__esModule", { value: true });
36
- exports.DEFAULT_AGENT_SENSES = exports.DEFAULT_AGENT_PHRASES = exports.DEFAULT_AGENT_CONTEXT = void 0;
36
+ exports.HARNESS_CANONICAL_REPO_URL = exports.DEFAULT_AGENT_SENSES = exports.LEGACY_VAULT_SERVER_URL_ALIASES = exports.DEFAULT_VAULT_SERVER_URL = exports.DEFAULT_AGENT_PHRASES = exports.DEFAULT_AGENT_CONTEXT = exports.PROVIDER_CREDENTIALS = void 0;
37
+ exports.normalizeVaultServerUrl = normalizeVaultServerUrl;
38
+ exports.getVaultServerUrlCandidates = getVaultServerUrlCandidates;
39
+ exports.defaultStableVaultEmail = defaultStableVaultEmail;
40
+ exports.resolveVaultConfig = resolveVaultConfig;
41
+ exports.normalizeSenses = normalizeSenses;
37
42
  exports.buildDefaultAgentTemplate = buildDefaultAgentTemplate;
38
43
  exports.getAgentName = getAgentName;
39
44
  exports.getRepoRoot = getRepoRoot;
40
45
  exports.getAgentBundlesRoot = getAgentBundlesRoot;
41
46
  exports.getAgentRoot = getAgentRoot;
42
47
  exports.getAgentStateRoot = getAgentStateRoot;
43
- exports.getAgentSecretsPath = getAgentSecretsPath;
48
+ exports.getAgentRepoWorkspacesRoot = getAgentRepoWorkspacesRoot;
49
+ exports.getAgentDaemonStateRoot = getAgentDaemonStateRoot;
50
+ exports.getAgentDaemonLogsDir = getAgentDaemonLogsDir;
51
+ exports.getAgentDaemonLoggingConfigPath = getAgentDaemonLoggingConfigPath;
52
+ exports.getAgentMessagesRoot = getAgentMessagesRoot;
53
+ exports.getAgentMailroomRoot = getAgentMailroomRoot;
54
+ exports.getAgentToolsRoot = getAgentToolsRoot;
44
55
  exports.loadAgentConfig = loadAgentConfig;
45
56
  exports.setAgentName = setAgentName;
46
57
  exports.setAgentConfigOverride = setAgentConfigOverride;
@@ -50,6 +61,15 @@ const fs = __importStar(require("fs"));
50
61
  const os = __importStar(require("os"));
51
62
  const path = __importStar(require("path"));
52
63
  const runtime_1 = require("../nerves/runtime");
64
+ const migrate_config_1 = require("./migrate-config");
65
+ /** Single source of truth for per-provider credential field names, env var mappings, and prompt labels. */
66
+ exports.PROVIDER_CREDENTIALS = {
67
+ anthropic: { required: ["setupToken"], envVars: { ANTHROPIC_API_KEY: "setupToken" }, promptLabels: { setupToken: "Anthropic setup-token" } },
68
+ "openai-codex": { required: ["oauthAccessToken"], envVars: { OPENAI_API_KEY: "oauthAccessToken" }, promptLabels: { oauthAccessToken: "OpenAI Codex OAuth token" } },
69
+ azure: { required: ["apiKey", "endpoint", "deployment"], envVars: { AZURE_OPENAI_API_KEY: "apiKey", AZURE_OPENAI_KEY: "apiKey", AZURE_OPENAI_ENDPOINT: "endpoint", AZURE_OPENAI_DEPLOYMENT: "deployment" }, promptLabels: { apiKey: "Azure API key", endpoint: "Azure endpoint", deployment: "Azure deployment" } },
70
+ minimax: { required: ["apiKey"], envVars: { MINIMAX_API_KEY: "apiKey" }, promptLabels: { apiKey: "MiniMax API key" } },
71
+ "github-copilot": { required: ["githubToken", "baseUrl"], envVars: { GH_TOKEN: "githubToken", GITHUB_TOKEN: "githubToken" }, promptLabels: { githubToken: "GitHub token" } },
72
+ };
53
73
  exports.DEFAULT_AGENT_CONTEXT = {
54
74
  maxTokens: 80000,
55
75
  contextMargin: 20,
@@ -59,16 +79,72 @@ exports.DEFAULT_AGENT_PHRASES = {
59
79
  tool: ["running tool"],
60
80
  followup: ["processing"],
61
81
  };
82
+ exports.DEFAULT_VAULT_SERVER_URL = "https://vault.ouroboros.bot";
83
+ exports.LEGACY_VAULT_SERVER_URL_ALIASES = [
84
+ "https://vault.ouro.bot",
85
+ "https://ouro-vault.gentleflower-74452a1e.eastus2.azurecontainerapps.io",
86
+ ];
87
+ function normalizeVaultServerUrl(serverUrl) {
88
+ const trimmed = serverUrl.trim();
89
+ const withoutTrailingSlash = trimmed.replace(/\/+$/, "");
90
+ if (!withoutTrailingSlash) {
91
+ return exports.DEFAULT_VAULT_SERVER_URL;
92
+ }
93
+ if (exports.LEGACY_VAULT_SERVER_URL_ALIASES.includes(withoutTrailingSlash)) {
94
+ return exports.DEFAULT_VAULT_SERVER_URL;
95
+ }
96
+ return withoutTrailingSlash;
97
+ }
98
+ function getVaultServerUrlCandidates(serverUrl) {
99
+ const raw = serverUrl.trim();
100
+ const withoutTrailingSlash = raw.replace(/\/+$/, "");
101
+ const normalized = normalizeVaultServerUrl(serverUrl);
102
+ const candidates = [normalized];
103
+ for (const candidate of [withoutTrailingSlash, raw]) {
104
+ if (candidate && !candidates.includes(candidate)) {
105
+ candidates.push(candidate);
106
+ }
107
+ }
108
+ if (normalized === exports.DEFAULT_VAULT_SERVER_URL) {
109
+ for (const alias of exports.LEGACY_VAULT_SERVER_URL_ALIASES) {
110
+ if (!candidates.includes(alias)) {
111
+ candidates.push(alias);
112
+ }
113
+ }
114
+ }
115
+ return candidates;
116
+ }
117
+ function defaultStableVaultEmail(agentName) {
118
+ const local = agentName
119
+ .toLowerCase()
120
+ .replace(/[^a-z0-9._-]+/g, "-")
121
+ .replace(/^-+|-+$/g, "") || "agent";
122
+ return `${local}@ouro.bot`;
123
+ }
124
+ /**
125
+ * Resolve the vault config for an agent, applying defaults.
126
+ * If vault is not configured in agent.json, returns default values.
127
+ */
128
+ function resolveVaultConfig(agentName, config) {
129
+ return {
130
+ email: config?.email ?? defaultStableVaultEmail(agentName),
131
+ serverUrl: normalizeVaultServerUrl(config?.serverUrl ?? exports.DEFAULT_VAULT_SERVER_URL),
132
+ };
133
+ }
62
134
  exports.DEFAULT_AGENT_SENSES = {
63
135
  cli: { enabled: true },
64
136
  teams: { enabled: false },
65
137
  bluebubbles: { enabled: false },
138
+ mail: { enabled: false },
139
+ voice: { enabled: false },
66
140
  };
67
141
  function normalizeSenses(value, configFile) {
68
142
  const defaults = {
69
143
  cli: { ...exports.DEFAULT_AGENT_SENSES.cli },
70
144
  teams: { ...exports.DEFAULT_AGENT_SENSES.teams },
71
145
  bluebubbles: { ...exports.DEFAULT_AGENT_SENSES.bluebubbles },
146
+ mail: { ...exports.DEFAULT_AGENT_SENSES.mail },
147
+ voice: { ...exports.DEFAULT_AGENT_SENSES.voice },
72
148
  };
73
149
  if (value === undefined) {
74
150
  return defaults;
@@ -84,7 +160,7 @@ function normalizeSenses(value, configFile) {
84
160
  throw new Error(`agent.json at ${configFile} must include senses as an object when present.`);
85
161
  }
86
162
  const raw = value;
87
- const senseNames = ["cli", "teams", "bluebubbles"];
163
+ const senseNames = ["cli", "teams", "bluebubbles", "mail", "voice"];
88
164
  for (const senseName of senseNames) {
89
165
  const rawSense = raw[senseName];
90
166
  if (rawSense === undefined) {
@@ -117,14 +193,17 @@ function normalizeSenses(value, configFile) {
117
193
  }
118
194
  function buildDefaultAgentTemplate(_agentName) {
119
195
  return {
120
- version: 1,
196
+ version: 2,
121
197
  enabled: true,
122
- provider: "anthropic",
198
+ humanFacing: { provider: "anthropic", model: "claude-opus-4-6" },
199
+ agentFacing: { provider: "anthropic", model: "claude-opus-4-6" },
123
200
  context: { ...exports.DEFAULT_AGENT_CONTEXT },
124
201
  senses: {
125
202
  cli: { ...exports.DEFAULT_AGENT_SENSES.cli },
126
203
  teams: { ...exports.DEFAULT_AGENT_SENSES.teams },
127
204
  bluebubbles: { ...exports.DEFAULT_AGENT_SENSES.bluebubbles },
205
+ mail: { ...exports.DEFAULT_AGENT_SENSES.mail },
206
+ voice: { ...exports.DEFAULT_AGENT_SENSES.voice },
128
207
  },
129
208
  phrases: {
130
209
  thinking: [...exports.DEFAULT_AGENT_PHRASES.thinking],
@@ -142,12 +221,6 @@ let _agentConfigOverride = null;
142
221
  */
143
222
  function getAgentName() {
144
223
  if (_cachedAgentName) {
145
- (0, runtime_1.emitNervesEvent)({
146
- event: "identity.resolve",
147
- component: "config/identity",
148
- message: "resolved agent name from cache",
149
- meta: { source: "cache" },
150
- });
151
224
  return _cachedAgentName;
152
225
  }
153
226
  const idx = process.argv.indexOf("--agent");
@@ -184,29 +257,49 @@ function getAgentBundlesRoot() {
184
257
  function getAgentRoot(agentName = getAgentName()) {
185
258
  return path.join(getAgentBundlesRoot(), `${agentName}.ouro`);
186
259
  }
260
+ function resolveOptionalAgentName(agentName) {
261
+ if (agentName && agentName.trim().length > 0)
262
+ return agentName.trim();
263
+ try {
264
+ return getAgentName();
265
+ }
266
+ catch {
267
+ return "slugger";
268
+ }
269
+ }
187
270
  /**
188
271
  * Returns the bundle-local runtime state directory: `~/AgentBundles/<agentName>.ouro/state/`
189
272
  */
190
- function getAgentStateRoot(agentName = getAgentName()) {
191
- return path.join(getAgentRoot(agentName), "state");
273
+ function getAgentStateRoot(agentName) {
274
+ return path.join(getAgentRoot(resolveOptionalAgentName(agentName)), "state");
192
275
  }
193
- /**
194
- * Returns the conventional secrets path: `~/.agentsecrets/<agentName>/secrets.json`
195
- */
196
- function getAgentSecretsPath(agentName = getAgentName()) {
197
- return path.join(os.homedir(), ".agentsecrets", agentName, "secrets.json");
276
+ exports.HARNESS_CANONICAL_REPO_URL = "https://github.com/ouroborosbot/ouroboros.git";
277
+ function getAgentRepoWorkspacesRoot(agentName) {
278
+ return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "workspaces");
198
279
  }
199
- /**
200
- * Load and parse `<agentRoot>/agent.json`.
201
- * Reads the file fresh on each call unless an override is set.
202
- * Throws descriptive error if file is missing or contains invalid JSON.
203
- */
204
- function loadAgentConfig() {
205
- if (_agentConfigOverride) {
206
- return _agentConfigOverride;
207
- }
208
- const agentRoot = getAgentRoot();
209
- const configFile = path.join(agentRoot, "agent.json");
280
+ function getAgentDaemonStateRoot(agentName) {
281
+ return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "daemon");
282
+ }
283
+ function getAgentDaemonLogsDir(agentName) {
284
+ return path.join(getAgentDaemonStateRoot(resolveOptionalAgentName(agentName)), "logs");
285
+ }
286
+ function getAgentDaemonLoggingConfigPath(agentName) {
287
+ return path.join(getAgentDaemonStateRoot(resolveOptionalAgentName(agentName)), "logging.json");
288
+ }
289
+ function getAgentMessagesRoot(agentName) {
290
+ return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "messages");
291
+ }
292
+ function getAgentMailroomRoot(agentName) {
293
+ return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "mailroom");
294
+ }
295
+ function getAgentToolsRoot(agentName) {
296
+ return path.join(getAgentStateRoot(resolveOptionalAgentName(agentName)), "tools");
297
+ }
298
+ const VALID_PROVIDERS = ["azure", "minimax", "anthropic", "openai-codex", "github-copilot"];
299
+ function isValidProvider(value) {
300
+ return typeof value === "string" && VALID_PROVIDERS.includes(value);
301
+ }
302
+ function readAndParseAgentJson(configFile) {
210
303
  let raw;
211
304
  try {
212
305
  raw = fs.readFileSync(configFile, "utf-8");
@@ -224,9 +317,8 @@ function loadAgentConfig() {
224
317
  });
225
318
  throw new Error(`Cannot read agent.json at ${configFile}. Does the agent directory exist?`);
226
319
  }
227
- let parsed;
228
320
  try {
229
- parsed = JSON.parse(raw);
321
+ return JSON.parse(raw);
230
322
  }
231
323
  catch (error) {
232
324
  (0, runtime_1.emitNervesEvent)({
@@ -241,6 +333,62 @@ function loadAgentConfig() {
241
333
  });
242
334
  throw new Error(`Invalid JSON in agent.json at ${configFile}. Check syntax.`);
243
335
  }
336
+ }
337
+ function validateFacingConfig(parsed, facingName, configFile) {
338
+ const raw = parsed[facingName];
339
+ if (!raw || typeof raw !== "object" || Array.isArray(raw)) {
340
+ (0, runtime_1.emitNervesEvent)({
341
+ level: "error",
342
+ event: "config_identity.error",
343
+ component: "config/identity",
344
+ message: `agent config missing or invalid ${facingName}`,
345
+ meta: { path: configFile, [facingName]: raw ?? null },
346
+ });
347
+ throw new Error(`agent.json at ${configFile} must include ${facingName} as { provider, model }.`);
348
+ }
349
+ const facing = raw;
350
+ if (!isValidProvider(facing.provider)) {
351
+ (0, runtime_1.emitNervesEvent)({
352
+ level: "error",
353
+ event: "config_identity.error",
354
+ component: "config/identity",
355
+ message: `agent config has invalid provider in ${facingName}`,
356
+ meta: { path: configFile, provider: facing.provider ?? null },
357
+ });
358
+ throw new Error(`agent.json at ${configFile} ${facingName}.provider must be one of: ${VALID_PROVIDERS.join(", ")}.`);
359
+ }
360
+ if (typeof facing.model !== "string") {
361
+ (0, runtime_1.emitNervesEvent)({
362
+ level: "error",
363
+ event: "config_identity.error",
364
+ component: "config/identity",
365
+ message: `agent config has invalid model in ${facingName}`,
366
+ meta: { path: configFile, model: facing.model ?? null },
367
+ });
368
+ throw new Error(`agent.json at ${configFile} ${facingName}.model must be a string.`);
369
+ }
370
+ return { provider: facing.provider, model: facing.model };
371
+ }
372
+ /**
373
+ * Load and parse `<agentRoot>/agent.json`.
374
+ * Reads the file fresh on each call unless an override is set.
375
+ * If the config is v1, auto-migrates to v2 via migrateAgentConfigV1ToV2 and re-reads.
376
+ * Throws descriptive error if file is missing or contains invalid JSON.
377
+ */
378
+ function loadAgentConfig() {
379
+ if (_agentConfigOverride) {
380
+ return _agentConfigOverride;
381
+ }
382
+ const agentRoot = getAgentRoot();
383
+ const configFile = path.join(agentRoot, "agent.json");
384
+ let parsed = readAndParseAgentJson(configFile);
385
+ // Inline migration: v1 -> v2
386
+ const rawVersion = parsed.version;
387
+ const initialVersion = typeof rawVersion === "number" ? rawVersion : 1;
388
+ if (initialVersion < 2) {
389
+ (0, migrate_config_1.migrateAgentConfigV1ToV2)(agentRoot);
390
+ parsed = readAndParseAgentJson(configFile);
391
+ }
244
392
  const existingPhrases = parsed.phrases;
245
393
  const needsFill = !existingPhrases ||
246
394
  !existingPhrases.thinking ||
@@ -262,28 +410,11 @@ function loadAgentConfig() {
262
410
  });
263
411
  fs.writeFileSync(configFile, JSON.stringify(parsed, null, 2) + "\n", "utf-8");
264
412
  }
265
- const rawProvider = parsed.provider;
266
- if (rawProvider !== "azure" &&
267
- rawProvider !== "minimax" &&
268
- rawProvider !== "anthropic" &&
269
- rawProvider !== "openai-codex") {
270
- (0, runtime_1.emitNervesEvent)({
271
- level: "error",
272
- event: "config_identity.error",
273
- component: "config/identity",
274
- message: "agent config missing or invalid provider",
275
- meta: {
276
- path: configFile,
277
- provider: rawProvider,
278
- },
279
- });
280
- throw new Error(`agent.json at ${configFile} must include provider: "azure", "minimax", "anthropic", or "openai-codex".`);
281
- }
282
- const provider = rawProvider;
283
- const rawVersion = parsed.version;
284
- const version = rawVersion === undefined ? 1 : rawVersion;
285
- if (typeof version !== "number" ||
286
- !Number.isInteger(version) ||
413
+ // Validate v2 facing configs
414
+ const humanFacing = validateFacingConfig(parsed, "humanFacing", configFile);
415
+ const agentFacing = validateFacingConfig(parsed, "agentFacing", configFile);
416
+ const version = typeof parsed.version === "number" ? parsed.version : 1;
417
+ if (!Number.isInteger(version) ||
287
418
  version < 1) {
288
419
  (0, runtime_1.emitNervesEvent)({
289
420
  level: "error",
@@ -292,7 +423,7 @@ function loadAgentConfig() {
292
423
  message: "agent config missing or invalid version",
293
424
  meta: {
294
425
  path: configFile,
295
- version: rawVersion,
426
+ version: parsed.version,
296
427
  },
297
428
  });
298
429
  throw new Error(`agent.json at ${configFile} must include version as integer >= 1.`);
@@ -312,15 +443,30 @@ function loadAgentConfig() {
312
443
  });
313
444
  throw new Error(`agent.json at ${configFile} must include enabled as boolean.`);
314
445
  }
446
+ // Tolerate deprecated provider field for backward compatibility
447
+ const rawProvider = parsed.provider;
448
+ const provider = isValidProvider(rawProvider) ? rawProvider : undefined;
449
+ // Spread parsed first so any field present in AgentConfig is carried
450
+ // through by default, then explicitly override the fields that need
451
+ // validation or normalization. This eliminates the field-drop bug class
452
+ // that caused the `sync` block (and previously `shell`) to be silently
453
+ // omitted from the returned config. Regression-guarded by the
454
+ // Required<AgentConfig> contract test in identity-contract.test.ts.
315
455
  const config = {
456
+ ...parsed,
316
457
  version,
317
458
  enabled,
318
- provider,
319
- context: parsed.context,
320
- logging: parsed.logging,
459
+ humanFacing,
460
+ agentFacing,
321
461
  senses: normalizeSenses(parsed.senses, configFile),
322
462
  phrases: parsed.phrases,
323
463
  };
464
+ if (provider !== undefined) {
465
+ config.provider = provider;
466
+ }
467
+ else {
468
+ delete config.provider;
469
+ }
324
470
  (0, runtime_1.emitNervesEvent)({
325
471
  event: "identity.resolve",
326
472
  component: "config/identity",