@ouro.bot/cli 0.1.0-alpha.36 → 0.1.0-alpha.360

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 (326) hide show
  1. package/README.md +194 -184
  2. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
  3. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +1 -1
  4. package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
  5. package/changelog.json +2149 -0
  6. package/dist/arc/attention-types.js +8 -0
  7. package/dist/arc/cares.js +140 -0
  8. package/dist/arc/episodes.js +117 -0
  9. package/dist/arc/intentions.js +133 -0
  10. package/dist/arc/json-store.js +117 -0
  11. package/dist/arc/obligations.js +237 -0
  12. package/dist/arc/packets.js +193 -0
  13. package/dist/arc/presence.js +185 -0
  14. package/dist/arc/task-lifecycle.js +65 -0
  15. package/dist/heart/active-work.js +832 -0
  16. package/dist/heart/agent-entry.js +37 -2
  17. package/dist/heart/attachments/image-normalize.js +194 -0
  18. package/dist/heart/attachments/materialize.js +97 -0
  19. package/dist/heart/attachments/originals.js +88 -0
  20. package/dist/heart/attachments/render.js +29 -0
  21. package/dist/heart/attachments/sources/adapter.js +2 -0
  22. package/dist/heart/attachments/sources/bluebubbles.js +156 -0
  23. package/dist/heart/attachments/sources/cli-local-file.js +78 -0
  24. package/dist/heart/attachments/sources/index.js +16 -0
  25. package/dist/heart/attachments/store.js +103 -0
  26. package/dist/heart/attachments/types.js +93 -0
  27. package/dist/heart/auth/auth-flow.js +463 -0
  28. package/dist/heart/bridges/manager.js +358 -0
  29. package/dist/heart/bridges/state-machine.js +135 -0
  30. package/dist/heart/bridges/store.js +123 -0
  31. package/dist/heart/bundle-state.js +168 -0
  32. package/dist/heart/commitments.js +111 -0
  33. package/dist/heart/config-registry.js +304 -0
  34. package/dist/heart/config.js +53 -21
  35. package/dist/heart/core.js +743 -252
  36. package/dist/heart/cross-chat-delivery.js +131 -0
  37. package/dist/heart/daemon/agent-config-check.js +561 -0
  38. package/dist/heart/daemon/agent-discovery.js +79 -3
  39. package/dist/heart/daemon/agent-service.js +360 -0
  40. package/dist/heart/daemon/agentic-repair.js +185 -0
  41. package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
  42. package/dist/heart/daemon/cadence.js +70 -0
  43. package/dist/heart/daemon/cli-defaults.js +591 -0
  44. package/dist/heart/daemon/cli-exec.js +2649 -0
  45. package/dist/heart/daemon/cli-help.js +306 -0
  46. package/dist/heart/daemon/cli-parse.js +913 -0
  47. package/dist/heart/daemon/cli-render-doctor.js +57 -0
  48. package/dist/heart/daemon/cli-render.js +560 -0
  49. package/dist/heart/daemon/cli-types.js +8 -0
  50. package/dist/heart/daemon/daemon-cli.js +30 -1171
  51. package/dist/heart/daemon/daemon-entry.js +358 -3
  52. package/dist/heart/daemon/daemon-health.js +141 -0
  53. package/dist/heart/daemon/daemon-runtime-sync.js +157 -12
  54. package/dist/heart/daemon/daemon-tombstone.js +236 -0
  55. package/dist/heart/daemon/daemon.js +757 -58
  56. package/dist/heart/daemon/doctor-types.js +8 -0
  57. package/dist/heart/daemon/doctor.js +465 -0
  58. package/dist/heart/daemon/health-monitor.js +79 -1
  59. package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
  60. package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
  61. package/dist/heart/daemon/http-health-probe.js +80 -0
  62. package/dist/heart/daemon/inner-status.js +89 -0
  63. package/dist/heart/daemon/interactive-repair.js +91 -0
  64. package/dist/heart/daemon/launchd.js +46 -9
  65. package/dist/heart/daemon/log-tailer.js +82 -12
  66. package/dist/heart/daemon/logs-prune.js +105 -0
  67. package/dist/heart/daemon/message-router.js +17 -8
  68. package/dist/heart/daemon/os-cron-deps.js +134 -0
  69. package/dist/heart/daemon/ouro-bot-entry.js +1 -1
  70. package/dist/heart/daemon/process-manager.js +201 -0
  71. package/dist/heart/daemon/provider-discovery.js +140 -0
  72. package/dist/heart/daemon/pulse.js +475 -0
  73. package/dist/heart/daemon/run-hooks.js +2 -0
  74. package/dist/heart/daemon/runtime-logging.js +67 -16
  75. package/dist/heart/daemon/runtime-metadata.js +101 -0
  76. package/dist/heart/daemon/runtime-mode.js +67 -0
  77. package/dist/heart/daemon/safe-mode.js +161 -0
  78. package/dist/heart/daemon/sense-manager.js +72 -3
  79. package/dist/heart/daemon/session-id-resolver.js +131 -0
  80. package/dist/heart/daemon/skill-management-installer.js +94 -0
  81. package/dist/heart/daemon/socket-client.js +307 -0
  82. package/dist/heart/daemon/stale-bundle-prune.js +96 -0
  83. package/dist/heart/daemon/startup-tui.js +237 -0
  84. package/dist/heart/daemon/task-scheduler.js +3 -25
  85. package/dist/heart/daemon/thoughts.js +510 -0
  86. package/dist/heart/daemon/up-progress.js +135 -0
  87. package/dist/heart/delegation.js +62 -0
  88. package/dist/heart/habits/habit-migration.js +181 -0
  89. package/dist/heart/habits/habit-parser.js +140 -0
  90. package/dist/heart/habits/habit-scheduler.js +371 -0
  91. package/dist/heart/{daemon → hatch}/hatch-flow.js +52 -120
  92. package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
  93. package/dist/heart/{daemon → hatch}/specialist-prompt.js +10 -7
  94. package/dist/heart/{daemon → hatch}/specialist-tools.js +56 -10
  95. package/dist/heart/identity.js +154 -59
  96. package/dist/heart/kept-notes.js +357 -0
  97. package/dist/heart/kicks.js +2 -20
  98. package/dist/heart/machine-identity.js +161 -0
  99. package/dist/heart/mcp/mcp-server.js +653 -0
  100. package/dist/heart/migrate-config.js +127 -0
  101. package/dist/heart/model-capabilities.js +59 -0
  102. package/dist/heart/outlook/outlook-http-hooks.js +64 -0
  103. package/dist/heart/outlook/outlook-http-response.js +7 -0
  104. package/dist/heart/outlook/outlook-http-routes.js +232 -0
  105. package/dist/heart/outlook/outlook-http-static.js +99 -0
  106. package/dist/heart/outlook/outlook-http-transport.js +116 -0
  107. package/dist/heart/outlook/outlook-http.js +99 -0
  108. package/dist/heart/outlook/outlook-read.js +28 -0
  109. package/dist/heart/outlook/outlook-types.js +27 -0
  110. package/dist/heart/outlook/outlook-view.js +195 -0
  111. package/dist/heart/outlook/readers/agent-machine.js +359 -0
  112. package/dist/heart/outlook/readers/continuity-readers.js +332 -0
  113. package/dist/heart/outlook/readers/runtime-readers.js +660 -0
  114. package/dist/heart/outlook/readers/sessions.js +232 -0
  115. package/dist/heart/outlook/readers/shared.js +111 -0
  116. package/dist/heart/progress-story.js +42 -0
  117. package/dist/heart/provider-attempt.js +133 -0
  118. package/dist/heart/provider-binding-resolver.js +240 -0
  119. package/dist/heart/provider-credential-pool.js +395 -0
  120. package/dist/heart/provider-failover.js +274 -0
  121. package/dist/heart/provider-models.js +81 -0
  122. package/dist/heart/provider-ping.js +227 -0
  123. package/dist/heart/provider-state.js +208 -0
  124. package/dist/heart/provider-visibility.js +183 -0
  125. package/dist/heart/providers/anthropic-token.js +163 -0
  126. package/dist/heart/providers/anthropic.js +177 -50
  127. package/dist/heart/providers/azure.js +102 -11
  128. package/dist/heart/providers/error-classification.js +63 -0
  129. package/dist/heart/providers/github-copilot.js +145 -0
  130. package/dist/heart/providers/minimax-vlm.js +189 -0
  131. package/dist/heart/providers/minimax.js +28 -6
  132. package/dist/heart/providers/openai-codex.js +38 -23
  133. package/dist/heart/session-activity.js +190 -0
  134. package/dist/heart/session-events.js +855 -0
  135. package/dist/heart/session-transcript.js +167 -0
  136. package/dist/heart/start-of-turn-packet.js +345 -0
  137. package/dist/heart/streaming.js +36 -27
  138. package/dist/heart/sync.js +332 -0
  139. package/dist/heart/target-resolution.js +127 -0
  140. package/dist/heart/tempo.js +93 -0
  141. package/dist/heart/temporal-view.js +41 -0
  142. package/dist/heart/tool-activity-callbacks.js +36 -0
  143. package/dist/heart/tool-description.js +135 -0
  144. package/dist/heart/tool-friction.js +55 -0
  145. package/dist/heart/tool-loop.js +200 -0
  146. package/dist/heart/turn-context.js +362 -0
  147. package/dist/heart/turn-coordinator.js +28 -0
  148. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  149. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  150. package/dist/heart/versioning/ouro-path-installer.js +296 -0
  151. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  152. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  153. package/dist/heart/{daemon → versioning}/update-checker.js +12 -2
  154. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  155. package/dist/mind/bundle-manifest.js +7 -1
  156. package/dist/mind/context.js +141 -94
  157. package/dist/mind/diary-integrity.js +60 -0
  158. package/dist/mind/{memory.js → diary.js} +84 -96
  159. package/dist/mind/embedding-provider.js +60 -0
  160. package/dist/mind/file-state.js +179 -0
  161. package/dist/mind/first-impressions.js +14 -1
  162. package/dist/mind/friends/channel.js +56 -0
  163. package/dist/mind/friends/group-context.js +144 -0
  164. package/dist/mind/friends/resolver.js +38 -1
  165. package/dist/mind/friends/store-file.js +58 -3
  166. package/dist/mind/friends/trust-explanation.js +74 -0
  167. package/dist/mind/friends/types.js +9 -1
  168. package/dist/mind/journal-index.js +161 -0
  169. package/dist/mind/note-search.js +268 -0
  170. package/dist/mind/obligation-steering.js +221 -0
  171. package/dist/mind/pending.js +74 -7
  172. package/dist/mind/prompt.js +1013 -112
  173. package/dist/mind/provenance-trust.js +26 -0
  174. package/dist/mind/scrutiny.js +173 -0
  175. package/dist/mind/token-estimate.js +8 -12
  176. package/dist/nerves/cli-logging.js +7 -1
  177. package/dist/nerves/coverage/audit-rules.js +15 -6
  178. package/dist/nerves/coverage/audit.js +28 -2
  179. package/dist/nerves/coverage/cli.js +1 -1
  180. package/dist/nerves/coverage/file-completeness.js +83 -5
  181. package/dist/nerves/coverage/run-artifacts.js +1 -1
  182. package/dist/nerves/event-buffer.js +111 -0
  183. package/dist/nerves/index.js +224 -4
  184. package/dist/nerves/observation.js +20 -0
  185. package/dist/nerves/redact.js +79 -0
  186. package/dist/nerves/runtime.js +5 -1
  187. package/dist/outlook-ui/assets/index-LwChZTgL.css +1 -0
  188. package/dist/outlook-ui/assets/index-xTdv64BV.js +61 -0
  189. package/dist/outlook-ui/index.html +15 -0
  190. package/dist/repertoire/ado-client.js +15 -56
  191. package/dist/repertoire/ado-semantic.js +11 -10
  192. package/dist/repertoire/api-client.js +97 -0
  193. package/dist/repertoire/bitwarden-store.js +319 -0
  194. package/dist/repertoire/bundle-templates.js +72 -0
  195. package/dist/repertoire/bw-installer.js +79 -0
  196. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  197. package/dist/repertoire/coding/context-pack.js +330 -0
  198. package/dist/repertoire/coding/feedback.js +197 -30
  199. package/dist/repertoire/coding/manager.js +158 -9
  200. package/dist/repertoire/coding/spawner.js +55 -9
  201. package/dist/repertoire/coding/tools.js +170 -7
  202. package/dist/repertoire/commerce-errors.js +109 -0
  203. package/dist/repertoire/commerce-self-test.js +156 -0
  204. package/dist/repertoire/credential-access.js +527 -0
  205. package/dist/repertoire/duffel-client.js +185 -0
  206. package/dist/repertoire/github-client.js +14 -55
  207. package/dist/repertoire/graph-client.js +11 -52
  208. package/dist/repertoire/guardrails.js +375 -0
  209. package/dist/repertoire/mcp-client.js +255 -0
  210. package/dist/repertoire/mcp-manager.js +305 -0
  211. package/dist/repertoire/mcp-tools.js +63 -0
  212. package/dist/repertoire/shell-sessions.js +133 -0
  213. package/dist/repertoire/skills.js +15 -24
  214. package/dist/repertoire/stripe-client.js +131 -0
  215. package/dist/repertoire/tasks/board.js +43 -5
  216. package/dist/repertoire/tasks/fix.js +182 -0
  217. package/dist/repertoire/tasks/index.js +28 -10
  218. package/dist/repertoire/tasks/lifecycle.js +2 -2
  219. package/dist/repertoire/tasks/parser.js +3 -2
  220. package/dist/repertoire/tasks/scanner.js +194 -37
  221. package/dist/repertoire/tasks/transitions.js +16 -79
  222. package/dist/repertoire/tool-results.js +29 -0
  223. package/dist/repertoire/tools-attachments.js +316 -0
  224. package/dist/repertoire/tools-base.js +45 -771
  225. package/dist/repertoire/tools-bluebubbles.js +1 -0
  226. package/dist/repertoire/tools-bridge.js +141 -0
  227. package/dist/repertoire/tools-bundle.js +984 -0
  228. package/dist/repertoire/tools-config.js +185 -0
  229. package/dist/repertoire/tools-continuity.js +248 -0
  230. package/dist/repertoire/tools-credential.js +182 -0
  231. package/dist/repertoire/tools-files.js +342 -0
  232. package/dist/repertoire/tools-flight.js +224 -0
  233. package/dist/repertoire/tools-flow.js +105 -0
  234. package/dist/repertoire/tools-github.js +1 -7
  235. package/dist/repertoire/tools-notes.js +376 -0
  236. package/dist/repertoire/tools-session.js +739 -0
  237. package/dist/repertoire/tools-shell.js +120 -0
  238. package/dist/repertoire/tools-stripe.js +180 -0
  239. package/dist/repertoire/tools-surface.js +243 -0
  240. package/dist/repertoire/tools-teams.js +12 -62
  241. package/dist/repertoire/tools-travel.js +125 -0
  242. package/dist/repertoire/tools-user-profile.js +144 -0
  243. package/dist/repertoire/tools-vault.js +110 -0
  244. package/dist/repertoire/tools.js +144 -138
  245. package/dist/repertoire/travel-api-client.js +360 -0
  246. package/dist/repertoire/user-profile.js +118 -0
  247. package/dist/repertoire/vault-setup.js +241 -0
  248. package/dist/scripts/claude-code-hook.js +41 -0
  249. package/dist/scripts/claude-code-stop-hook.js +47 -0
  250. package/dist/senses/attention-queue.js +116 -0
  251. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  252. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  253. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +225 -9
  254. package/dist/senses/bluebubbles/entry.js +13 -0
  255. package/dist/senses/bluebubbles/inbound-log.js +113 -0
  256. package/dist/senses/bluebubbles/index.js +1616 -0
  257. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  258. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +43 -12
  259. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
  260. package/dist/senses/bluebubbles/replay.js +129 -0
  261. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  262. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  263. package/dist/senses/cli/bracketed-paste.js +82 -0
  264. package/dist/senses/cli/image-paste.js +287 -0
  265. package/dist/senses/cli/image-ref-navigation.js +75 -0
  266. package/dist/senses/cli/ink-app.js +156 -0
  267. package/dist/senses/cli/inline-diff.js +64 -0
  268. package/dist/senses/cli/input-keys.js +174 -0
  269. package/dist/senses/cli/kill-ring.js +86 -0
  270. package/dist/senses/cli/message-list.js +51 -0
  271. package/dist/senses/cli/ouro-tui.js +605 -0
  272. package/dist/senses/cli/spinner-imperative.js +135 -0
  273. package/dist/senses/cli/spinner.js +101 -0
  274. package/dist/senses/cli/status-line.js +60 -0
  275. package/dist/senses/cli/streaming-markdown.js +526 -0
  276. package/dist/senses/cli/tool-display.js +83 -0
  277. package/dist/senses/cli/tool-render.js +85 -0
  278. package/dist/senses/cli/tui-store.js +240 -0
  279. package/dist/senses/cli/virtual-list.js +35 -0
  280. package/dist/senses/cli-entry.js +1 -1
  281. package/dist/senses/cli-layout.js +187 -0
  282. package/dist/senses/cli.js +587 -249
  283. package/dist/senses/commands.js +66 -3
  284. package/dist/senses/continuity.js +94 -0
  285. package/dist/senses/habit-turn-message.js +108 -0
  286. package/dist/senses/inner-dialog-worker.js +112 -19
  287. package/dist/senses/inner-dialog.js +633 -86
  288. package/dist/senses/pipeline.js +603 -0
  289. package/dist/senses/proactive-content-guard.js +51 -0
  290. package/dist/senses/shared-turn.js +199 -0
  291. package/dist/senses/surface-tool.js +68 -0
  292. package/dist/senses/teams.js +690 -160
  293. package/dist/senses/trust-gate.js +112 -2
  294. package/package.json +29 -7
  295. package/skills/agent-commerce.md +106 -0
  296. package/skills/browser-navigation.md +110 -0
  297. package/skills/commerce-setup-guide.md +116 -0
  298. package/skills/commerce-setup.md +84 -0
  299. package/skills/configure-dev-tools.md +81 -0
  300. package/skills/travel-planning.md +138 -0
  301. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  302. package/dist/heart/daemon/subagent-installer.js +0 -134
  303. package/dist/mind/associative-recall.js +0 -197
  304. package/dist/senses/bluebubbles-entry.js +0 -11
  305. package/dist/senses/bluebubbles.js +0 -558
  306. package/dist/senses/debug-activity.js +0 -127
  307. package/subagents/README.md +0 -73
  308. package/subagents/work-doer.md +0 -235
  309. package/subagents/work-merger.md +0 -618
  310. package/subagents/work-planner.md +0 -382
  311. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  312. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  313. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  314. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  315. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  316. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  317. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  318. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  319. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  320. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  321. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  322. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  323. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  324. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  325. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  326. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -0,0 +1,463 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.readAgentConfigForAgent = readAgentConfigForAgent;
37
+ exports.writeAgentProviderSelection = writeAgentProviderSelection;
38
+ exports.loadAgentSecrets = loadAgentSecrets;
39
+ exports.writeProviderCredentials = writeProviderCredentials;
40
+ exports.writeAgentModel = writeAgentModel;
41
+ exports.collectRuntimeAuthCredentials = collectRuntimeAuthCredentials;
42
+ exports.resolveHatchCredentials = resolveHatchCredentials;
43
+ exports.runRuntimeAuthFlow = runRuntimeAuthFlow;
44
+ const child_process_1 = require("child_process");
45
+ const fs = __importStar(require("fs"));
46
+ const os = __importStar(require("os"));
47
+ const path = __importStar(require("path"));
48
+ const runtime_1 = require("../../nerves/runtime");
49
+ const identity_1 = require("../identity");
50
+ const migrate_config_1 = require("../migrate-config");
51
+ const provider_models_1 = require("../provider-models");
52
+ const ANTHROPIC_SETUP_TOKEN_PREFIX = "sk-ant-oat01-";
53
+ const ANTHROPIC_SETUP_TOKEN_MIN_LENGTH = 80;
54
+ const DEFAULT_SECRETS_TEMPLATE = {
55
+ providers: {
56
+ azure: {
57
+ apiKey: "",
58
+ endpoint: "",
59
+ deployment: "",
60
+ apiVersion: "2025-04-01-preview",
61
+ },
62
+ minimax: {
63
+ apiKey: "",
64
+ },
65
+ anthropic: {
66
+ setupToken: "",
67
+ refreshToken: "",
68
+ expiresAt: 0,
69
+ },
70
+ "openai-codex": {
71
+ oauthAccessToken: "",
72
+ },
73
+ "github-copilot": {
74
+ githubToken: "",
75
+ baseUrl: "",
76
+ },
77
+ },
78
+ teams: {
79
+ clientId: "",
80
+ clientSecret: "",
81
+ tenantId: "",
82
+ },
83
+ oauth: {
84
+ graphConnectionName: "graph",
85
+ adoConnectionName: "ado",
86
+ githubConnectionName: "",
87
+ },
88
+ teamsChannel: {
89
+ skipConfirmation: true,
90
+ port: 3978,
91
+ },
92
+ vault: {
93
+ masterPassword: "",
94
+ },
95
+ integrations: {
96
+ perplexityApiKey: "",
97
+ openaiEmbeddingsApiKey: "",
98
+ },
99
+ };
100
+ function deepMerge(defaults, partial) {
101
+ const result = { ...defaults };
102
+ for (const key of Object.keys(partial)) {
103
+ const left = result[key];
104
+ const right = partial[key];
105
+ if (right !== null &&
106
+ typeof right === "object" &&
107
+ !Array.isArray(right) &&
108
+ left !== null &&
109
+ typeof left === "object" &&
110
+ !Array.isArray(left)) {
111
+ result[key] = deepMerge(left, right);
112
+ continue;
113
+ }
114
+ result[key] = right;
115
+ }
116
+ return result;
117
+ }
118
+ function readJsonRecord(filePath, label) {
119
+ try {
120
+ const raw = fs.readFileSync(filePath, "utf8");
121
+ const parsed = JSON.parse(raw);
122
+ if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
123
+ throw new Error("expected object");
124
+ }
125
+ return parsed;
126
+ }
127
+ catch (error) {
128
+ throw new Error(`Failed to read ${label} at ${filePath}: ${String(error)}`);
129
+ }
130
+ }
131
+ function readAgentConfigForAgent(agentName, bundlesRoot = (0, identity_1.getAgentBundlesRoot)()) {
132
+ const agentRoot = path.join(bundlesRoot, `${agentName}.ouro`);
133
+ const configPath = path.join(agentRoot, "agent.json");
134
+ let parsed = readJsonRecord(configPath, "agent config");
135
+ // Inline migration: v1 -> v2
136
+ const version = typeof parsed.version === "number" ? parsed.version : 1;
137
+ if (version < 2) {
138
+ (0, migrate_config_1.migrateAgentConfigV1ToV2)(agentRoot);
139
+ parsed = readJsonRecord(configPath, "agent config");
140
+ }
141
+ // Validate v2 required facing fields
142
+ const humanFacing = parsed.humanFacing;
143
+ const agentFacing = parsed.agentFacing;
144
+ if (!humanFacing || typeof humanFacing !== "object") {
145
+ throw new Error(`agent.json at ${configPath} has unsupported provider '${String(parsed.provider)}'`);
146
+ }
147
+ const provider = humanFacing.provider;
148
+ if (provider !== "azure" &&
149
+ provider !== "anthropic" &&
150
+ provider !== "minimax" &&
151
+ provider !== "openai-codex" &&
152
+ provider !== "github-copilot") {
153
+ throw new Error(`agent.json at ${configPath} has unsupported provider '${String(provider)}'`);
154
+ }
155
+ if (!agentFacing || typeof agentFacing !== "object") {
156
+ throw new Error(`agent.json at ${configPath} has unsupported provider '${String(parsed.provider)}'`);
157
+ }
158
+ // Spread-with-validation: same pattern as loadAgentConfig to eliminate
159
+ // the unvalidated-pass-through bug class. The spread carries through
160
+ // every field present in parsed; senses goes through the same
161
+ // normalization as loadAgentConfig so the two entry points return
162
+ // equivalent configs for the same file.
163
+ const config = {
164
+ ...parsed,
165
+ senses: (0, identity_1.normalizeSenses)(parsed.senses, configPath),
166
+ };
167
+ return {
168
+ configPath,
169
+ config,
170
+ };
171
+ }
172
+ function writeAgentProviderSelection(agentName, facing, provider, bundlesRoot = (0, identity_1.getAgentBundlesRoot)()) {
173
+ const { configPath, config } = readAgentConfigForAgent(agentName, bundlesRoot);
174
+ const facingKey = facing === "human" ? "humanFacing" : "agentFacing";
175
+ const previousFacing = config[facingKey];
176
+ const resolved = (0, provider_models_1.resolveModelForProviderSelection)(provider, previousFacing.model);
177
+ const nextConfig = {
178
+ ...config,
179
+ [facingKey]: { ...previousFacing, provider, model: resolved.model },
180
+ };
181
+ fs.writeFileSync(configPath, `${JSON.stringify(nextConfig, null, 2)}\n`, "utf8");
182
+ (0, runtime_1.emitNervesEvent)({
183
+ component: "daemon",
184
+ event: "daemon.auth_provider_selected",
185
+ message: "updated agent provider selection after auth flow",
186
+ meta: {
187
+ agentName,
188
+ facing,
189
+ provider,
190
+ previousProvider: previousFacing.provider,
191
+ previousModel: previousFacing.model,
192
+ model: resolved.model,
193
+ preservedModel: resolved.preserved,
194
+ configPath,
195
+ },
196
+ });
197
+ return configPath;
198
+ }
199
+ function resolveAgentSecretsPath(agentName, deps = {}) {
200
+ if (deps.secretsRoot)
201
+ return path.join(deps.secretsRoot, agentName, "secrets.json");
202
+ const homeDir = deps.homeDir ?? os.homedir();
203
+ return (0, identity_1.getAgentSecretsPath)(agentName).replace(os.homedir(), homeDir);
204
+ }
205
+ function loadAgentSecrets(agentName, deps = {}) {
206
+ const secretsPath = resolveAgentSecretsPath(agentName, deps);
207
+ const secretsDir = path.dirname(secretsPath);
208
+ fs.mkdirSync(secretsDir, { recursive: true });
209
+ let onDisk = {};
210
+ try {
211
+ onDisk = readJsonRecord(secretsPath, "secrets config");
212
+ }
213
+ catch (error) {
214
+ const message = error.message;
215
+ if (!message.includes("ENOENT"))
216
+ throw error;
217
+ }
218
+ return {
219
+ secretsPath,
220
+ secrets: deepMerge(DEFAULT_SECRETS_TEMPLATE, onDisk),
221
+ };
222
+ }
223
+ function writeSecrets(secretsPath, secrets) {
224
+ fs.writeFileSync(secretsPath, `${JSON.stringify(secrets, null, 2)}\n`, "utf8");
225
+ }
226
+ function writeProviderCredentials(agentName, provider, credentials, deps = {}) {
227
+ const { secretsPath, secrets } = loadAgentSecrets(agentName, deps);
228
+ applyCredentials(secrets, provider, credentials);
229
+ writeSecrets(secretsPath, secrets);
230
+ return { secretsPath, secrets };
231
+ }
232
+ function writeAgentModel(agentName, facing, modelName, deps = {}) {
233
+ const { configPath, config } = readAgentConfigForAgent(agentName, deps.bundlesRoot);
234
+ const facingKey = facing === "human" ? "humanFacing" : "agentFacing";
235
+ const facingBlock = config[facingKey];
236
+ const previousModel = facingBlock.model;
237
+ const provider = facingBlock.provider;
238
+ const nextConfig = {
239
+ ...config,
240
+ [facingKey]: { ...facingBlock, model: modelName },
241
+ };
242
+ fs.writeFileSync(configPath, `${JSON.stringify(nextConfig, null, 2)}\n`, "utf8");
243
+ (0, runtime_1.emitNervesEvent)({
244
+ component: "daemon",
245
+ event: "daemon.config_model_updated",
246
+ message: "updated agent model in agent.json",
247
+ meta: { agentName, facing, provider, modelName, previousModel, configPath },
248
+ });
249
+ return { configPath, provider, previousModel };
250
+ }
251
+ function readCodexAccessToken(homeDir) {
252
+ const authPath = path.join(homeDir, ".codex", "auth.json");
253
+ try {
254
+ const raw = fs.readFileSync(authPath, "utf8");
255
+ const parsed = JSON.parse(raw);
256
+ const token = parsed?.tokens?.access_token;
257
+ return typeof token === "string" ? token.trim() : /* v8 ignore next -- defensive: codex login always writes a string token @preserve */ "";
258
+ }
259
+ catch {
260
+ return "";
261
+ }
262
+ }
263
+ function ensurePromptInput(promptInput, provider) {
264
+ if (promptInput)
265
+ return promptInput;
266
+ throw new Error(`No prompt input is available for ${provider} authentication.`);
267
+ }
268
+ function validateAnthropicToken(token) {
269
+ const trimmed = token.trim();
270
+ if (!trimmed) {
271
+ throw new Error("No Anthropic setup token was provided.");
272
+ }
273
+ if (!trimmed.startsWith(ANTHROPIC_SETUP_TOKEN_PREFIX)) {
274
+ throw new Error(`Invalid Anthropic setup token format. Expected prefix ${ANTHROPIC_SETUP_TOKEN_PREFIX}.`);
275
+ }
276
+ if (trimmed.length < ANTHROPIC_SETUP_TOKEN_MIN_LENGTH) {
277
+ throw new Error("Anthropic setup token looks too short.");
278
+ }
279
+ return trimmed;
280
+ }
281
+ async function collectRuntimeAuthCredentials(input, deps) {
282
+ const spawnSync = deps.spawnSync ?? child_process_1.spawnSync;
283
+ const homeDir = deps.homeDir ?? os.homedir();
284
+ if (input.provider === "github-copilot") {
285
+ let token = process.env.GH_TOKEN?.trim() || process.env.GITHUB_TOKEN?.trim() || "";
286
+ if (!token) {
287
+ const result = spawnSync("gh", ["auth", "token"], { encoding: "utf8" });
288
+ token = (result.status === 0 && result.stdout ? result.stdout.trim() : "");
289
+ }
290
+ if (!token) {
291
+ (0, runtime_1.emitNervesEvent)({
292
+ component: "daemon",
293
+ event: "daemon.auth_gh_login_start",
294
+ message: "starting gh auth login for runtime auth",
295
+ meta: { agentName: input.agentName },
296
+ });
297
+ const loginResult = spawnSync("gh", ["auth", "login"], { stdio: "inherit" });
298
+ if (loginResult.status !== 0) {
299
+ throw new Error("'gh auth login' failed. Install the GitHub CLI (gh) and try again.");
300
+ }
301
+ const retryResult = spawnSync("gh", ["auth", "token"], { encoding: "utf8" });
302
+ /* v8 ignore next -- branch: retry after login always succeeds in tests @preserve */
303
+ token = (retryResult.status === 0 && retryResult.stdout ? retryResult.stdout.trim() : "");
304
+ /* v8 ignore next -- defensive: gh auth login succeeded but token still missing @preserve */
305
+ if (!token) {
306
+ throw new Error("gh auth login completed but no token was found. Run `gh auth login` and try again.");
307
+ }
308
+ }
309
+ const response = await fetch("https://api.github.com/copilot_internal/user", {
310
+ headers: { Authorization: `Bearer ${token}` },
311
+ });
312
+ if (!response.ok) {
313
+ throw new Error(`GitHub Copilot endpoint discovery failed (HTTP ${response.status}). Ensure your GitHub account has Copilot access.`);
314
+ }
315
+ const body = await response.json();
316
+ const baseUrl = body?.endpoints?.api;
317
+ /* v8 ignore next -- defensive: valid response but missing endpoints field @preserve */
318
+ if (!baseUrl) {
319
+ throw new Error("GitHub Copilot endpoint discovery returned no endpoints.api. Ensure your GitHub account has Copilot access.");
320
+ }
321
+ return { githubToken: token, baseUrl };
322
+ }
323
+ if (input.provider === "openai-codex") {
324
+ // Always run codex login when auth is explicitly requested — stale tokens
325
+ // are indistinguishable from valid ones without an API call, and the user
326
+ // is asking to re-authenticate.
327
+ (0, runtime_1.emitNervesEvent)({
328
+ component: "daemon",
329
+ event: "daemon.auth_codex_login_start",
330
+ message: "starting codex login for runtime auth",
331
+ meta: { agentName: input.agentName },
332
+ });
333
+ const result = spawnSync("codex", ["login"], { stdio: "inherit" });
334
+ if (result.error) {
335
+ throw new Error(`Failed to run 'codex login': ${result.error.message}`);
336
+ }
337
+ if (result.status !== 0) {
338
+ throw new Error(`'codex login' exited with status ${result.status}.`);
339
+ }
340
+ const token = readCodexAccessToken(homeDir);
341
+ if (!token) {
342
+ throw new Error("Codex login completed but no token was found in ~/.codex/auth.json. Re-run `codex login` and try again.");
343
+ }
344
+ return { oauthAccessToken: token };
345
+ }
346
+ if (input.provider === "anthropic") {
347
+ (0, runtime_1.emitNervesEvent)({
348
+ component: "daemon",
349
+ event: "daemon.auth_claude_setup_start",
350
+ message: "starting claude setup-token for runtime auth",
351
+ meta: { agentName: input.agentName },
352
+ });
353
+ const result = spawnSync("claude", ["setup-token"], { stdio: "inherit" });
354
+ if (result.error) {
355
+ throw new Error(`Failed to run 'claude setup-token': ${result.error.message}`);
356
+ }
357
+ if (result.status !== 0) {
358
+ throw new Error(`'claude setup-token' exited with status ${result.status}.`);
359
+ }
360
+ const prompt = ensurePromptInput(input.promptInput, input.provider);
361
+ const setupToken = validateAnthropicToken(await prompt("Paste the setup token from `claude setup-token`: "));
362
+ // Exchange the setup token for an access+refresh token pair so auto-refresh works.
363
+ // The setup token IS the initial access token — we use it as a refresh token to
364
+ // get back a proper token pair from the OAuth endpoint.
365
+ /* v8 ignore start -- token exchange: requires live Anthropic OAuth endpoint @preserve */
366
+ try {
367
+ const { refreshAnthropicToken } = await Promise.resolve().then(() => __importStar(require("../providers/anthropic-token")));
368
+ const tokenState = await refreshAnthropicToken(setupToken);
369
+ if (tokenState) {
370
+ return {
371
+ setupToken: tokenState.accessToken,
372
+ refreshToken: tokenState.refreshToken,
373
+ expiresAt: tokenState.expiresAt,
374
+ };
375
+ }
376
+ }
377
+ catch {
378
+ // Exchange failed — use the raw setup token as-is (it'll work until expiry)
379
+ }
380
+ /* v8 ignore stop */
381
+ return { setupToken };
382
+ }
383
+ // Generic prompt-for-fields fallback (minimax, azure, any future simple providers)
384
+ const prompt = ensurePromptInput(input.promptInput, input.provider);
385
+ const desc = identity_1.PROVIDER_CREDENTIALS[input.provider];
386
+ const creds = {};
387
+ for (const field of desc.required) {
388
+ /* v8 ignore next -- fallback: all current providers define promptLabels for required fields @preserve */
389
+ const label = desc.promptLabels[field] ?? field;
390
+ const value = (await prompt(`${label}: `)).trim();
391
+ if (!value)
392
+ throw new Error(`${label} is required.`);
393
+ creds[field] = value;
394
+ }
395
+ return creds;
396
+ }
397
+ async function resolveHatchCredentials(input) {
398
+ const credentials = { ...(input.credentials ?? {}) };
399
+ // If all required fields are already provided, return as-is
400
+ const cred = credentials;
401
+ const missing = identity_1.PROVIDER_CREDENTIALS[input.provider].required.some((key) => !cred[key]);
402
+ if (!missing)
403
+ return credentials;
404
+ // Try the full auth flow (wraps collectRuntimeAuthCredentials + writes secrets)
405
+ if (input.runAuthFlow) {
406
+ const result = await input.runAuthFlow({
407
+ agentName: input.agentName,
408
+ provider: input.provider,
409
+ promptInput: input.promptInput,
410
+ });
411
+ Object.assign(credentials, result.credentials);
412
+ /* v8 ignore next 3 -- branch: auth flow always fills all required fields in production @preserve */
413
+ if (!identity_1.PROVIDER_CREDENTIALS[input.provider].required.some((key) => !credentials[key])) {
414
+ return credentials;
415
+ }
416
+ }
417
+ // Prompt for any still-missing required fields
418
+ /* v8 ignore next -- guard: no promptInput means we can't collect remaining fields @preserve */
419
+ if (input.promptInput) {
420
+ const desc = identity_1.PROVIDER_CREDENTIALS[input.provider];
421
+ for (const field of desc.required) {
422
+ if (!cred[field]) {
423
+ const label = desc.promptLabels[field] ?? field;
424
+ cred[field] = await input.promptInput(`${label}: `);
425
+ }
426
+ }
427
+ }
428
+ return credentials;
429
+ }
430
+ function applyCredentials(secrets, provider, credentials) {
431
+ const target = secrets.providers[provider];
432
+ // Copy all non-empty credential fields to the provider's secrets block
433
+ for (const [key, value] of Object.entries(credentials)) {
434
+ /* v8 ignore next -- guard: skip null/empty fields from partial credential objects @preserve */
435
+ if (value != null && value !== "") {
436
+ target[key] = typeof value === "string" ? value.trim() : value;
437
+ }
438
+ }
439
+ }
440
+ async function runRuntimeAuthFlow(input, deps = {}) {
441
+ (0, runtime_1.emitNervesEvent)({
442
+ component: "daemon",
443
+ event: "daemon.auth_flow_start",
444
+ message: "starting runtime auth flow",
445
+ meta: { agentName: input.agentName, provider: input.provider },
446
+ });
447
+ const homeDir = deps.homeDir ?? os.homedir();
448
+ const credentials = await collectRuntimeAuthCredentials(input, deps);
449
+ const { secretsPath } = writeProviderCredentials(input.agentName, input.provider, credentials, { homeDir });
450
+ (0, runtime_1.emitNervesEvent)({
451
+ component: "daemon",
452
+ event: "daemon.auth_flow_end",
453
+ message: "completed runtime auth flow",
454
+ meta: { agentName: input.agentName, provider: input.provider, secretsPath },
455
+ });
456
+ return {
457
+ agentName: input.agentName,
458
+ provider: input.provider,
459
+ secretsPath,
460
+ message: `authenticated ${input.agentName} with ${input.provider}`,
461
+ credentials,
462
+ };
463
+ }