@ouro.bot/cli 0.1.0-alpha.35 → 0.1.0-alpha.351

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 (325) hide show
  1. package/README.md +188 -187
  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 +2098 -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 +744 -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 +170 -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 +2633 -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 +512 -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 +751 -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 +105 -0
  72. package/dist/heart/daemon/pulse.js +463 -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 +194 -0
  111. package/dist/heart/outlook/readers/agent-machine.js +355 -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 +231 -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 +135 -0
  121. package/dist/heart/provider-models.js +81 -0
  122. package/dist/heart/provider-ping.js +258 -0
  123. package/dist/heart/provider-state.js +208 -0
  124. package/dist/heart/providers/anthropic-token.js +163 -0
  125. package/dist/heart/providers/anthropic.js +171 -50
  126. package/dist/heart/providers/azure.js +97 -11
  127. package/dist/heart/providers/error-classification.js +63 -0
  128. package/dist/heart/providers/github-copilot.js +135 -0
  129. package/dist/heart/providers/minimax-vlm.js +189 -0
  130. package/dist/heart/providers/minimax.js +23 -6
  131. package/dist/heart/providers/openai-codex.js +33 -23
  132. package/dist/heart/session-activity.js +190 -0
  133. package/dist/heart/session-events.js +727 -0
  134. package/dist/heart/session-transcript.js +162 -0
  135. package/dist/heart/start-of-turn-packet.js +341 -0
  136. package/dist/heart/streaming.js +36 -27
  137. package/dist/heart/sync.js +332 -0
  138. package/dist/heart/target-resolution.js +127 -0
  139. package/dist/heart/tempo.js +93 -0
  140. package/dist/heart/temporal-view.js +41 -0
  141. package/dist/heart/tool-activity-callbacks.js +36 -0
  142. package/dist/heart/tool-description.js +135 -0
  143. package/dist/heart/tool-friction.js +55 -0
  144. package/dist/heart/tool-loop.js +200 -0
  145. package/dist/heart/turn-context.js +358 -0
  146. package/dist/heart/turn-coordinator.js +28 -0
  147. package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
  148. package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
  149. package/dist/heart/versioning/ouro-path-installer.js +296 -0
  150. package/dist/heart/versioning/ouro-version-manager.js +295 -0
  151. package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
  152. package/dist/heart/{daemon → versioning}/update-checker.js +12 -2
  153. package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
  154. package/dist/mind/bundle-manifest.js +7 -1
  155. package/dist/mind/context.js +140 -94
  156. package/dist/mind/diary-integrity.js +60 -0
  157. package/dist/mind/{memory.js → diary.js} +84 -96
  158. package/dist/mind/embedding-provider.js +60 -0
  159. package/dist/mind/file-state.js +179 -0
  160. package/dist/mind/first-impressions.js +14 -1
  161. package/dist/mind/friends/channel.js +56 -0
  162. package/dist/mind/friends/group-context.js +144 -0
  163. package/dist/mind/friends/resolver.js +38 -1
  164. package/dist/mind/friends/store-file.js +58 -3
  165. package/dist/mind/friends/trust-explanation.js +74 -0
  166. package/dist/mind/friends/types.js +9 -1
  167. package/dist/mind/journal-index.js +161 -0
  168. package/dist/mind/note-search.js +268 -0
  169. package/dist/mind/obligation-steering.js +221 -0
  170. package/dist/mind/pending.js +74 -7
  171. package/dist/mind/prompt.js +1001 -112
  172. package/dist/mind/provenance-trust.js +26 -0
  173. package/dist/mind/scrutiny.js +173 -0
  174. package/dist/mind/token-estimate.js +8 -12
  175. package/dist/nerves/cli-logging.js +7 -1
  176. package/dist/nerves/coverage/audit-rules.js +15 -6
  177. package/dist/nerves/coverage/audit.js +28 -2
  178. package/dist/nerves/coverage/cli.js +1 -1
  179. package/dist/nerves/coverage/file-completeness.js +83 -5
  180. package/dist/nerves/coverage/run-artifacts.js +1 -1
  181. package/dist/nerves/event-buffer.js +111 -0
  182. package/dist/nerves/index.js +224 -4
  183. package/dist/nerves/observation.js +20 -0
  184. package/dist/nerves/redact.js +79 -0
  185. package/dist/nerves/runtime.js +5 -1
  186. package/dist/outlook-ui/assets/index-LwChZTgL.css +1 -0
  187. package/dist/outlook-ui/assets/index-xTdv64BV.js +61 -0
  188. package/dist/outlook-ui/index.html +15 -0
  189. package/dist/repertoire/ado-client.js +15 -56
  190. package/dist/repertoire/ado-semantic.js +11 -10
  191. package/dist/repertoire/api-client.js +97 -0
  192. package/dist/repertoire/bitwarden-store.js +319 -0
  193. package/dist/repertoire/bundle-templates.js +72 -0
  194. package/dist/repertoire/bw-installer.js +79 -0
  195. package/dist/repertoire/coding/codex-jsonl.js +64 -0
  196. package/dist/repertoire/coding/context-pack.js +330 -0
  197. package/dist/repertoire/coding/feedback.js +197 -30
  198. package/dist/repertoire/coding/manager.js +158 -9
  199. package/dist/repertoire/coding/spawner.js +55 -9
  200. package/dist/repertoire/coding/tools.js +170 -7
  201. package/dist/repertoire/commerce-errors.js +109 -0
  202. package/dist/repertoire/commerce-self-test.js +156 -0
  203. package/dist/repertoire/credential-access.js +527 -0
  204. package/dist/repertoire/duffel-client.js +185 -0
  205. package/dist/repertoire/github-client.js +14 -55
  206. package/dist/repertoire/graph-client.js +11 -52
  207. package/dist/repertoire/guardrails.js +375 -0
  208. package/dist/repertoire/mcp-client.js +255 -0
  209. package/dist/repertoire/mcp-manager.js +305 -0
  210. package/dist/repertoire/mcp-tools.js +63 -0
  211. package/dist/repertoire/shell-sessions.js +133 -0
  212. package/dist/repertoire/skills.js +15 -24
  213. package/dist/repertoire/stripe-client.js +131 -0
  214. package/dist/repertoire/tasks/board.js +43 -5
  215. package/dist/repertoire/tasks/fix.js +182 -0
  216. package/dist/repertoire/tasks/index.js +28 -10
  217. package/dist/repertoire/tasks/lifecycle.js +2 -2
  218. package/dist/repertoire/tasks/parser.js +3 -2
  219. package/dist/repertoire/tasks/scanner.js +194 -37
  220. package/dist/repertoire/tasks/transitions.js +16 -79
  221. package/dist/repertoire/tool-results.js +29 -0
  222. package/dist/repertoire/tools-attachments.js +316 -0
  223. package/dist/repertoire/tools-base.js +45 -771
  224. package/dist/repertoire/tools-bluebubbles.js +1 -0
  225. package/dist/repertoire/tools-bridge.js +141 -0
  226. package/dist/repertoire/tools-bundle.js +984 -0
  227. package/dist/repertoire/tools-config.js +185 -0
  228. package/dist/repertoire/tools-continuity.js +248 -0
  229. package/dist/repertoire/tools-credential.js +182 -0
  230. package/dist/repertoire/tools-files.js +342 -0
  231. package/dist/repertoire/tools-flight.js +224 -0
  232. package/dist/repertoire/tools-flow.js +105 -0
  233. package/dist/repertoire/tools-github.js +1 -7
  234. package/dist/repertoire/tools-notes.js +376 -0
  235. package/dist/repertoire/tools-session.js +739 -0
  236. package/dist/repertoire/tools-shell.js +120 -0
  237. package/dist/repertoire/tools-stripe.js +180 -0
  238. package/dist/repertoire/tools-surface.js +243 -0
  239. package/dist/repertoire/tools-teams.js +12 -62
  240. package/dist/repertoire/tools-travel.js +125 -0
  241. package/dist/repertoire/tools-user-profile.js +144 -0
  242. package/dist/repertoire/tools-vault.js +110 -0
  243. package/dist/repertoire/tools.js +144 -138
  244. package/dist/repertoire/travel-api-client.js +360 -0
  245. package/dist/repertoire/user-profile.js +118 -0
  246. package/dist/repertoire/vault-setup.js +241 -0
  247. package/dist/scripts/claude-code-hook.js +41 -0
  248. package/dist/scripts/claude-code-stop-hook.js +47 -0
  249. package/dist/senses/attention-queue.js +116 -0
  250. package/dist/senses/bluebubbles/attachment-cache.js +53 -0
  251. package/dist/senses/bluebubbles/attachment-download.js +137 -0
  252. package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +225 -9
  253. package/dist/senses/bluebubbles/entry.js +13 -0
  254. package/dist/senses/bluebubbles/inbound-log.js +113 -0
  255. package/dist/senses/bluebubbles/index.js +1616 -0
  256. package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
  257. package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +43 -12
  258. package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
  259. package/dist/senses/bluebubbles/replay.js +129 -0
  260. package/dist/senses/bluebubbles/runtime-state.js +109 -0
  261. package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
  262. package/dist/senses/cli/bracketed-paste.js +82 -0
  263. package/dist/senses/cli/image-paste.js +287 -0
  264. package/dist/senses/cli/image-ref-navigation.js +75 -0
  265. package/dist/senses/cli/ink-app.js +156 -0
  266. package/dist/senses/cli/inline-diff.js +64 -0
  267. package/dist/senses/cli/input-keys.js +174 -0
  268. package/dist/senses/cli/kill-ring.js +86 -0
  269. package/dist/senses/cli/message-list.js +51 -0
  270. package/dist/senses/cli/ouro-tui.js +605 -0
  271. package/dist/senses/cli/spinner-imperative.js +135 -0
  272. package/dist/senses/cli/spinner.js +101 -0
  273. package/dist/senses/cli/status-line.js +60 -0
  274. package/dist/senses/cli/streaming-markdown.js +526 -0
  275. package/dist/senses/cli/tool-display.js +83 -0
  276. package/dist/senses/cli/tool-render.js +85 -0
  277. package/dist/senses/cli/tui-store.js +240 -0
  278. package/dist/senses/cli/virtual-list.js +35 -0
  279. package/dist/senses/cli-entry.js +1 -1
  280. package/dist/senses/cli-layout.js +187 -0
  281. package/dist/senses/cli.js +603 -246
  282. package/dist/senses/commands.js +65 -1
  283. package/dist/senses/continuity.js +94 -0
  284. package/dist/senses/habit-turn-message.js +108 -0
  285. package/dist/senses/inner-dialog-worker.js +112 -19
  286. package/dist/senses/inner-dialog.js +633 -86
  287. package/dist/senses/pipeline.js +567 -0
  288. package/dist/senses/proactive-content-guard.js +51 -0
  289. package/dist/senses/shared-turn.js +199 -0
  290. package/dist/senses/surface-tool.js +68 -0
  291. package/dist/senses/teams.js +690 -160
  292. package/dist/senses/trust-gate.js +112 -2
  293. package/package.json +29 -7
  294. package/skills/agent-commerce.md +106 -0
  295. package/skills/browser-navigation.md +110 -0
  296. package/skills/commerce-setup-guide.md +116 -0
  297. package/skills/commerce-setup.md +84 -0
  298. package/skills/configure-dev-tools.md +81 -0
  299. package/skills/travel-planning.md +138 -0
  300. package/dist/heart/daemon/ouro-path-installer.js +0 -178
  301. package/dist/heart/daemon/subagent-installer.js +0 -134
  302. package/dist/mind/associative-recall.js +0 -197
  303. package/dist/senses/bluebubbles-entry.js +0 -11
  304. package/dist/senses/bluebubbles.js +0 -548
  305. package/dist/senses/debug-activity.js +0 -127
  306. package/subagents/README.md +0 -73
  307. package/subagents/work-doer.md +0 -235
  308. package/subagents/work-merger.md +0 -618
  309. package/subagents/work-planner.md +0 -382
  310. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
  311. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
  312. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
  313. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
  314. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
  315. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
  316. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
  317. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
  318. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
  319. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
  320. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
  321. /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
  322. /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
  323. /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
  324. /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
  325. /package/dist/heart/{daemon → versioning}/wrapper-publish-guard.js +0 -0
@@ -0,0 +1,63 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.isNetworkError = isNetworkError;
4
+ exports.classifyHttpError = classifyHttpError;
5
+ const runtime_1 = require("../../nerves/runtime");
6
+ // Node socket / DNS error codes that indicate a transient network failure.
7
+ const NETWORK_ERROR_CODES = new Set([
8
+ "ECONNRESET",
9
+ "ECONNREFUSED",
10
+ "ENOTFOUND",
11
+ "ETIMEDOUT",
12
+ "EPIPE",
13
+ "EAI_AGAIN",
14
+ "EHOSTUNREACH",
15
+ "ENETUNREACH",
16
+ "ECONNABORTED",
17
+ ]);
18
+ // Substrings the OpenAI/Anthropic SDKs use when wrapping fetch/socket failures
19
+ // into Error.message instead of an err.code.
20
+ const NETWORK_ERROR_MESSAGE_PATTERNS = [
21
+ "fetch failed",
22
+ "socket hang up",
23
+ "getaddrinfo",
24
+ "request timed out", // OpenAI SDK timeout — see SDK source
25
+ "request timeout",
26
+ "connection error",
27
+ ];
28
+ // True if the error looks like a transient network issue (no HTTP status, just
29
+ // a socket/DNS/timeout failure from the underlying transport).
30
+ function isNetworkError(error) {
31
+ const code = error.code || "";
32
+ if (NETWORK_ERROR_CODES.has(code))
33
+ return true;
34
+ const msg = (error.message || "").toLowerCase();
35
+ return NETWORK_ERROR_MESSAGE_PATTERNS.some((pat) => msg.includes(pat));
36
+ }
37
+ // Standard HTTP error → ProviderErrorClassification mapping. Providers wrap
38
+ // this with their own overrides.
39
+ function classifyHttpError(error, overrides) {
40
+ const status = error.status;
41
+ if (overrides?.isAuthFailure?.(error) || status === 401 || status === 403) {
42
+ return "auth-failure";
43
+ }
44
+ if (status === 429) {
45
+ if (overrides?.isUsageLimit?.(error))
46
+ return "usage-limit";
47
+ return "rate-limit";
48
+ }
49
+ if (overrides?.isServerError?.(error) || (status !== undefined && status >= 500)) {
50
+ return "server-error";
51
+ }
52
+ if (isNetworkError(error))
53
+ return "network-error";
54
+ return "unknown";
55
+ }
56
+ /* v8 ignore start — module-level observability event */
57
+ (0, runtime_1.emitNervesEvent)({
58
+ component: "engine",
59
+ event: "engine.error_classification_loaded",
60
+ message: "shared provider error classification loaded",
61
+ meta: {},
62
+ });
63
+ /* v8 ignore stop */
@@ -0,0 +1,135 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.classifyGithubCopilotError = classifyGithubCopilotError;
7
+ exports.createGithubCopilotProviderRuntime = createGithubCopilotProviderRuntime;
8
+ const openai_1 = __importDefault(require("openai"));
9
+ const config_1 = require("../config");
10
+ const runtime_1 = require("../../nerves/runtime");
11
+ const streaming_1 = require("../streaming");
12
+ const model_capabilities_1 = require("../model-capabilities");
13
+ const error_classification_1 = require("./error-classification");
14
+ function classifyGithubCopilotError(error) {
15
+ return (0, error_classification_1.classifyHttpError)(error);
16
+ }
17
+ function createGithubCopilotProviderRuntime(model, config = (0, config_1.getGithubCopilotConfig)()) {
18
+ (0, runtime_1.emitNervesEvent)({
19
+ component: "engine",
20
+ event: "engine.provider_init",
21
+ message: "github-copilot provider init",
22
+ meta: { provider: "github-copilot" },
23
+ });
24
+ if (!config.githubToken) {
25
+ throw new Error("provider 'github-copilot' is selected in agent.json but providers.github-copilot.githubToken is missing in secrets.json.");
26
+ }
27
+ if (!config.baseUrl) {
28
+ throw new Error("provider 'github-copilot' is selected in agent.json but providers.github-copilot.baseUrl is missing in secrets.json.");
29
+ }
30
+ const isCompletionsModel = model.startsWith("claude");
31
+ const modelCaps = (0, model_capabilities_1.getModelCapabilities)(model);
32
+ const capabilities = new Set();
33
+ /* v8 ignore next -- branch: capability detection tested via unit test @preserve */
34
+ if (modelCaps.reasoningEffort)
35
+ capabilities.add("reasoning-effort");
36
+ const client = new openai_1.default({
37
+ apiKey: config.githubToken,
38
+ baseURL: config.baseUrl,
39
+ maxRetries: 0,
40
+ });
41
+ if (isCompletionsModel) {
42
+ // Chat completions path (Claude models via Copilot)
43
+ return {
44
+ id: "github-copilot",
45
+ model,
46
+ client,
47
+ capabilities,
48
+ supportedReasoningEfforts: modelCaps.reasoningEffort,
49
+ resetTurnState(_messages) {
50
+ // No provider-owned turn state for chat-completions path.
51
+ },
52
+ appendToolOutput(_callId, _output) {
53
+ // Chat-completions providers rely on canonical messages only.
54
+ },
55
+ /* v8 ignore start -- streamTurn: tested via mock assertions in github-copilot.test.ts @preserve */
56
+ async streamTurn(request) {
57
+ const params = {
58
+ messages: request.messages,
59
+ tools: request.activeTools,
60
+ stream: true,
61
+ };
62
+ if (this.model)
63
+ params.model = this.model;
64
+ if (request.traceId)
65
+ params.metadata = { trace_id: request.traceId };
66
+ if (request.toolChoiceRequired)
67
+ params.tool_choice = "required";
68
+ try {
69
+ return await (0, streaming_1.streamChatCompletion)(this.client, params, request.callbacks, request.signal, request.eagerSettleStreaming);
70
+ }
71
+ catch (error) {
72
+ throw error instanceof Error ? error : new Error(String(error));
73
+ }
74
+ },
75
+ /* v8 ignore stop */
76
+ /* v8 ignore next 3 -- delegation: classification logic tested via classifyGithubCopilotError @preserve */
77
+ classifyError(error) {
78
+ return classifyGithubCopilotError(error);
79
+ },
80
+ };
81
+ }
82
+ // Responses API path (GPT models via Copilot)
83
+ let nativeInput = null;
84
+ let nativeInstructions = "";
85
+ return {
86
+ id: "github-copilot",
87
+ model,
88
+ client,
89
+ capabilities,
90
+ supportedReasoningEfforts: modelCaps.reasoningEffort,
91
+ /* v8 ignore start -- responses path: tested via mock assertions in github-copilot.test.ts @preserve */
92
+ resetTurnState(messages) {
93
+ const { instructions, input } = (0, streaming_1.toResponsesInput)(messages);
94
+ nativeInput = input;
95
+ nativeInstructions = instructions;
96
+ },
97
+ appendToolOutput(callId, output) {
98
+ if (!nativeInput)
99
+ return;
100
+ nativeInput.push({ type: "function_call_output", call_id: callId, output });
101
+ },
102
+ async streamTurn(request) {
103
+ if (!nativeInput)
104
+ this.resetTurnState(request.messages);
105
+ const params = {
106
+ model: this.model,
107
+ input: nativeInput,
108
+ instructions: nativeInstructions,
109
+ tools: (0, streaming_1.toResponsesTools)(request.activeTools),
110
+ reasoning: { effort: request.reasoningEffort ?? "medium", summary: "detailed" },
111
+ stream: true,
112
+ store: false,
113
+ include: ["reasoning.encrypted_content"],
114
+ };
115
+ if (request.traceId)
116
+ params.metadata = { trace_id: request.traceId };
117
+ if (request.toolChoiceRequired)
118
+ params.tool_choice = "required";
119
+ try {
120
+ const result = await (0, streaming_1.streamResponsesApi)(this.client, params, request.callbacks, request.signal, request.eagerSettleStreaming);
121
+ for (const item of result.outputItems)
122
+ nativeInput.push(item);
123
+ return result;
124
+ }
125
+ catch (error) {
126
+ throw error instanceof Error ? error : new Error(String(error));
127
+ }
128
+ },
129
+ /* v8 ignore stop */
130
+ /* v8 ignore next 3 -- delegation: classification logic tested via classifyGithubCopilotError @preserve */
131
+ classifyError(error) {
132
+ return classifyGithubCopilotError(error);
133
+ },
134
+ };
135
+ }
@@ -0,0 +1,189 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.minimaxVlmDescribe = minimaxVlmDescribe;
4
+ const runtime_1 = require("../../nerves/runtime");
5
+ const error_classification_1 = require("./error-classification");
6
+ // No default timeout — if the caller doesn't pass `timeoutMs`, we don't
7
+ // layer an AbortSignal on top of fetch, which means undici's own defaults
8
+ // (headersTimeout and bodyTimeout, both 5 minutes in the current Node) are
9
+ // the ceiling. That's the right ceiling: long enough to tolerate any
10
+ // legitimate MiniMax VLM slow window, short enough to eventually give up on
11
+ // a dead endpoint, and — unlike a number we picked — it's the stack's
12
+ // actual default rather than a harness-imposed fiction.
13
+ //
14
+ // Tests that want to exercise the timeout branch pass an explicit
15
+ // `timeoutMs`; production callers omit it and get undici's behavior.
16
+ //
17
+ // Same reasoning as PR #322 (dropped the 30s SDK timeout from the LLM
18
+ // providers): the harness shouldn't impose artificially tight request
19
+ // ceilings when the underlying stack already has correct defaults.
20
+ const MODEL_NAME = "minimax-vlm";
21
+ const SUPPORTED_DATA_URL_PATTERN = /^data:image\/(png|jpeg|webp);base64,/i;
22
+ function buildVlmUrl(baseURL) {
23
+ const stripped = baseURL.replace(/\/v1\/?$/, "").replace(/\/$/, "");
24
+ return `${stripped}/v1/coding_plan/vlm`;
25
+ }
26
+ function extractMimeType(imageDataUrl) {
27
+ const match = /^data:([^;,]+)[;,]/.exec(imageDataUrl);
28
+ return match?.[1]?.toLowerCase();
29
+ }
30
+ function emitStart(params) {
31
+ (0, runtime_1.emitNervesEvent)({
32
+ level: "warn",
33
+ component: "senses",
34
+ event: "senses.bluebubbles_vlm_describe_start",
35
+ message: "minimax vlm describe start",
36
+ meta: {
37
+ attachmentGuid: params.attachmentGuid,
38
+ mimeType: params.mimeType,
39
+ promptLength: params.prompt.length,
40
+ model: MODEL_NAME,
41
+ chatModel: params.chatModel,
42
+ },
43
+ });
44
+ }
45
+ function emitEnd(params, descriptionLength, latencyMs) {
46
+ (0, runtime_1.emitNervesEvent)({
47
+ level: "warn",
48
+ component: "senses",
49
+ event: "senses.bluebubbles_vlm_describe_end",
50
+ message: "minimax vlm describe end",
51
+ meta: {
52
+ attachmentGuid: params.attachmentGuid,
53
+ descriptionLength,
54
+ latencyMs,
55
+ model: MODEL_NAME,
56
+ },
57
+ });
58
+ }
59
+ function emitError(params, reason, status, traceId) {
60
+ (0, runtime_1.emitNervesEvent)({
61
+ level: "warn",
62
+ component: "senses",
63
+ event: "senses.bluebubbles_vlm_describe_error",
64
+ message: "minimax vlm describe error",
65
+ meta: {
66
+ attachmentGuid: params.attachmentGuid,
67
+ mimeType: params.mimeType,
68
+ model: MODEL_NAME,
69
+ reason,
70
+ status,
71
+ traceId,
72
+ },
73
+ });
74
+ }
75
+ function throwAndEmit(params, message, status, traceId) {
76
+ emitError(params, message, status, traceId);
77
+ const err = new Error(message);
78
+ if (status !== undefined)
79
+ err.status = status;
80
+ if (traceId)
81
+ err.traceId = traceId;
82
+ throw err;
83
+ }
84
+ async function minimaxVlmDescribe(params) {
85
+ if (!params.apiKey) {
86
+ // We deliberately do NOT emit _start for param-validation errors — there's
87
+ // no meaningful "started a request" to pair with. Only the _error fires.
88
+ emitError(params, "minimax VLM: API key is empty — re-run credential setup or add a minimax key to secrets.json");
89
+ throw new Error("minimax VLM: API key is empty — re-run credential setup or add a minimax key to secrets.json");
90
+ }
91
+ if (!params.prompt) {
92
+ emitError(params, "minimax VLM: missing prompt — supply a targeted question (e.g. 'what's the flight number in the bottom-right?') and retry");
93
+ throw new Error("minimax VLM: missing prompt — supply a targeted question (e.g. 'what's the flight number in the bottom-right?') and retry");
94
+ }
95
+ if (!params.imageDataUrl) {
96
+ emitError(params, "minimax VLM: missing image data URL — verify the attachment downloaded OK or ask the user to resend the image");
97
+ throw new Error("minimax VLM: missing image data URL — verify the attachment downloaded OK or ask the user to resend the image");
98
+ }
99
+ if (!SUPPORTED_DATA_URL_PATTERN.test(params.imageDataUrl)) {
100
+ const mime = extractMimeType(params.imageDataUrl) ?? "unknown";
101
+ const msg = `minimax VLM: image must be a data:image/(png|jpeg|webp);base64,... URL but got ${mime} — ask the user to resend as a screenshot (png or jpeg)`;
102
+ emitError({ ...params, mimeType: params.mimeType ?? mime }, msg);
103
+ throw new Error(msg);
104
+ }
105
+ const fetchImpl = params.fetchImpl ?? fetch;
106
+ const url = buildVlmUrl(params.baseURL);
107
+ emitStart(params);
108
+ const startTs = Date.now();
109
+ // Only layer an AbortSignal if the caller explicitly asked for one.
110
+ // Omitting it lets undici's defaults (headersTimeout + bodyTimeout = 5min
111
+ // each in current Node) act as the ceiling — see the long comment on
112
+ // DEFAULT_TIMEOUT_MS above for the rationale.
113
+ const fetchOptions = {
114
+ method: "POST",
115
+ headers: {
116
+ "Authorization": `Bearer ${params.apiKey}`,
117
+ "Content-Type": "application/json",
118
+ "MM-API-Source": "Ouroboros",
119
+ },
120
+ body: JSON.stringify({
121
+ prompt: params.prompt,
122
+ image_url: params.imageDataUrl,
123
+ }),
124
+ };
125
+ if (typeof params.timeoutMs === "number" && params.timeoutMs > 0) {
126
+ fetchOptions.signal = AbortSignal.timeout(params.timeoutMs);
127
+ }
128
+ let response;
129
+ try {
130
+ response = await fetchImpl(url, fetchOptions);
131
+ }
132
+ catch (error) {
133
+ const err = (error instanceof Error ? error : new Error(String(error)));
134
+ if (err.name === "AbortError" || err.name === "TimeoutError") {
135
+ const seconds = typeof params.timeoutMs === "number" ? Math.round(params.timeoutMs / 1000) : null;
136
+ const hint = seconds !== null
137
+ ? `request timed out after ${seconds}s — retry or ask the user to resend the image`
138
+ : `request timed out (underlying stack default) — retry or ask the user to resend the image`;
139
+ throwAndEmit(params, `minimax VLM: ${hint}`);
140
+ }
141
+ if ((0, error_classification_1.isNetworkError)(err)) {
142
+ throwAndEmit(params, `minimax VLM: network error talking to the vlm endpoint (${err.message}) — retry in a moment`);
143
+ }
144
+ throwAndEmit(params, `minimax VLM: unexpected transport error (${err.message}) — retry, and if it persists surface 'image understanding is unavailable' to the user`);
145
+ }
146
+ const traceId = response.headers.get("Trace-Id") ?? response.headers.get("trace-id") ?? undefined;
147
+ const status = response.status;
148
+ if (status === 401 || status === 403) {
149
+ throwAndEmit(params, `minimax VLM: rejected credentials (HTTP ${status}) — the minimax key for this agent is invalid, re-run credential setup or rotate the key`, status, traceId);
150
+ }
151
+ if (status === 429) {
152
+ throwAndEmit(params, `minimax VLM: rate limited (HTTP 429) — wait and retry in a moment`, status, traceId);
153
+ }
154
+ if (status >= 500) {
155
+ throwAndEmit(params, `minimax VLM: server error (HTTP ${status}) — retry, and if it persists surface 'image understanding is unavailable' to the user`, status, traceId);
156
+ }
157
+ if (!response.ok) {
158
+ throwAndEmit(params, `minimax VLM: unexpected HTTP ${status} — retry, and if it persists surface 'image understanding is unavailable' to the user`, status, traceId);
159
+ }
160
+ let body;
161
+ try {
162
+ body = await response.json();
163
+ }
164
+ catch (error) {
165
+ const err = error;
166
+ throwAndEmit(params, `minimax VLM: malformed response (not JSON: ${err.message}) — this is a provider bug, retry or surface 'image understanding is unavailable' to the user`, status, traceId);
167
+ }
168
+ const parsed = body;
169
+ const providerStatus = parsed?.base_resp?.status_code ?? 0;
170
+ if (providerStatus !== 0) {
171
+ const providerMsg = parsed?.base_resp?.status_msg ?? "unknown";
172
+ throwAndEmit(params, `minimax VLM: provider returned status_code=${providerStatus} (${providerMsg}) — check the minimax account or retry, and if it persists surface 'image understanding is unavailable' to the user`, status, traceId);
173
+ }
174
+ const content = parsed?.content;
175
+ if (typeof content !== "string" || content.length === 0) {
176
+ throwAndEmit(params, `minimax VLM: returned no description (empty content field) — this is a provider bug, retry or surface 'image understanding is unavailable' to the user`, status, traceId);
177
+ }
178
+ const latencyMs = Date.now() - startTs;
179
+ emitEnd(params, content.length, latencyMs);
180
+ return content;
181
+ }
182
+ /* v8 ignore start — module-level observability event */
183
+ (0, runtime_1.emitNervesEvent)({
184
+ component: "engine",
185
+ event: "engine.minimax_vlm_loaded",
186
+ message: "minimax vlm client loaded",
187
+ meta: {},
188
+ });
189
+ /* v8 ignore stop */
@@ -3,32 +3,46 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
3
3
  return (mod && mod.__esModule) ? mod : { "default": mod };
4
4
  };
5
5
  Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.MINIMAX_PROVIDER_BASE_URL = void 0;
7
+ exports.classifyMinimaxError = classifyMinimaxError;
6
8
  exports.createMinimaxProviderRuntime = createMinimaxProviderRuntime;
7
9
  const openai_1 = __importDefault(require("openai"));
8
10
  const config_1 = require("../config");
9
11
  const runtime_1 = require("../../nerves/runtime");
12
+ const error_classification_1 = require("./error-classification");
13
+ /**
14
+ * Canonical MiniMax chat-completions base URL. Exported so the BlueBubbles
15
+ * VLM fallback (which targets a sibling endpoint under the same host) can
16
+ * derive its URL from one source of truth instead of hard-coding.
17
+ */
18
+ exports.MINIMAX_PROVIDER_BASE_URL = "https://api.minimaxi.chat/v1";
19
+ function classifyMinimaxError(error) {
20
+ return (0, error_classification_1.classifyHttpError)(error);
21
+ }
10
22
  const streaming_1 = require("../streaming");
11
- function createMinimaxProviderRuntime() {
23
+ const model_capabilities_1 = require("../model-capabilities");
24
+ function createMinimaxProviderRuntime(model, minimaxConfig = (0, config_1.getMinimaxConfig)()) {
12
25
  (0, runtime_1.emitNervesEvent)({
13
26
  component: "engine",
14
27
  event: "engine.provider_init",
15
28
  message: "minimax provider init",
16
29
  meta: { provider: "minimax" },
17
30
  });
18
- const minimaxConfig = (0, config_1.getMinimaxConfig)();
19
31
  if (!minimaxConfig.apiKey) {
20
32
  throw new Error("provider 'minimax' is selected in agent.json but providers.minimax.apiKey is missing in secrets.json.");
21
33
  }
34
+ // Registry consulted; MiniMax models return empty defaults (no capabilities to derive)
35
+ (0, model_capabilities_1.getModelCapabilities)(model);
22
36
  const client = new openai_1.default({
23
37
  apiKey: minimaxConfig.apiKey,
24
- baseURL: "https://api.minimaxi.chat/v1",
25
- timeout: 30000,
38
+ baseURL: exports.MINIMAX_PROVIDER_BASE_URL,
26
39
  maxRetries: 0,
27
40
  });
28
41
  return {
29
42
  id: "minimax",
30
- model: minimaxConfig.model,
43
+ model,
31
44
  client,
45
+ capabilities: new Set(),
32
46
  resetTurnState(_messages) {
33
47
  // No provider-owned turn state for chat-completions providers.
34
48
  },
@@ -47,7 +61,10 @@ function createMinimaxProviderRuntime() {
47
61
  params.metadata = { trace_id: request.traceId };
48
62
  if (request.toolChoiceRequired)
49
63
  params.tool_choice = "required";
50
- return (0, streaming_1.streamChatCompletion)(this.client, params, request.callbacks, request.signal);
64
+ return (0, streaming_1.streamChatCompletion)(this.client, params, request.callbacks, request.signal, request.eagerSettleStreaming);
65
+ },
66
+ classifyError(error) {
67
+ return classifyMinimaxError(error);
51
68
  },
52
69
  };
53
70
  }
@@ -3,12 +3,15 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
3
3
  return (mod && mod.__esModule) ? mod : { "default": mod };
4
4
  };
5
5
  Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.classifyOpenAICodexError = classifyOpenAICodexError;
6
7
  exports.createOpenAICodexProviderRuntime = createOpenAICodexProviderRuntime;
7
8
  const openai_1 = __importDefault(require("openai"));
8
9
  const config_1 = require("../config");
9
10
  const identity_1 = require("../identity");
10
11
  const runtime_1 = require("../../nerves/runtime");
11
12
  const streaming_1 = require("../streaming");
13
+ const model_capabilities_1 = require("../model-capabilities");
14
+ const error_classification_1 = require("./error-classification");
12
15
  const OPENAI_CODEX_AUTH_FAILURE_MARKERS = [
13
16
  "authentication failed",
14
17
  "unauthorized",
@@ -27,11 +30,11 @@ function getOpenAICodexOAuthInstructions() {
27
30
  const agentName = getOpenAICodexAgentNameForGuidance();
28
31
  return [
29
32
  "Fix:",
30
- ` 1. Run \`npm run auth:openai-codex -- --agent ${agentName}\``,
31
- " (or run `codex login` and set the OAuth token manually)",
33
+ ` 1. Run \`ouro auth --agent ${agentName}\``,
32
34
  ` 2. Open ${getOpenAICodexSecretsPathForGuidance()}`,
33
35
  " 3. Confirm providers.openai-codex.oauthAccessToken is set",
34
36
  " 4. This provider uses chatgpt.com/backend-api/codex/responses (not api.openai.com/responses).",
37
+ " 5. After reauth, retry the failed ouro command or reconnect this session.",
35
38
  ].join("\n");
36
39
  }
37
40
  function getOpenAICodexReauthGuidance(reason) {
@@ -41,22 +44,19 @@ function getOpenAICodexReauthGuidance(reason) {
41
44
  getOpenAICodexOAuthInstructions(),
42
45
  ].join("\n");
43
46
  }
47
+ function classifyOpenAICodexError(error) {
48
+ return (0, error_classification_1.classifyHttpError)(error, {
49
+ isAuthFailure: isOpenAICodexAuthFailure,
50
+ isUsageLimit: (e) => {
51
+ const lower = e.message.toLowerCase();
52
+ return lower.includes("usage") || lower.includes("quota") || lower.includes("exceeded your");
53
+ },
54
+ });
55
+ }
44
56
  function isOpenAICodexAuthFailure(error) {
45
- if (!(error instanceof Error))
46
- return false;
47
- const status = error.status;
48
- if (status === 401 || status === 403)
49
- return true;
50
57
  const lower = error.message.toLowerCase();
51
58
  return OPENAI_CODEX_AUTH_FAILURE_MARKERS.some((marker) => lower.includes(marker));
52
59
  }
53
- function withOpenAICodexAuthGuidance(error) {
54
- const base = error instanceof Error ? error.message : String(error);
55
- if (isOpenAICodexAuthFailure(error)) {
56
- return new Error(getOpenAICodexReauthGuidance(`OpenAI Codex authentication failed (${base}).`));
57
- }
58
- return error instanceof Error ? error : new Error(String(error));
59
- }
60
60
  function decodeJwtPayload(token) {
61
61
  const parts = token.split(".");
62
62
  if (parts.length < 2)
@@ -87,16 +87,15 @@ function getChatGPTAccountIdFromToken(token) {
87
87
  return "";
88
88
  return accountId.trim();
89
89
  }
90
- function createOpenAICodexProviderRuntime() {
90
+ function createOpenAICodexProviderRuntime(model, codexConfig = (0, config_1.getOpenAICodexConfig)()) {
91
91
  (0, runtime_1.emitNervesEvent)({
92
92
  component: "engine",
93
93
  event: "engine.provider_init",
94
94
  message: "openai-codex provider init",
95
95
  meta: { provider: "openai-codex" },
96
96
  });
97
- const codexConfig = (0, config_1.getOpenAICodexConfig)();
98
- if (!(codexConfig.model && codexConfig.oauthAccessToken)) {
99
- throw new Error(getOpenAICodexReauthGuidance("provider 'openai-codex' is selected in agent.json but providers.openai-codex.model/oauthAccessToken is incomplete in secrets.json."));
97
+ if (!codexConfig.oauthAccessToken) {
98
+ throw new Error(getOpenAICodexReauthGuidance("provider 'openai-codex' is selected in agent.json but providers.openai-codex.oauthAccessToken is missing in secrets.json."));
100
99
  }
101
100
  const token = codexConfig.oauthAccessToken.trim();
102
101
  if (!token) {
@@ -106,6 +105,12 @@ function createOpenAICodexProviderRuntime() {
106
105
  if (!chatgptAccountId) {
107
106
  throw new Error(getOpenAICodexReauthGuidance("OpenAI Codex OAuth access token is missing a chatgpt_account_id claim required for chatgpt.com/backend-api/codex."));
108
107
  }
108
+ const modelCaps = (0, model_capabilities_1.getModelCapabilities)(model);
109
+ const capabilities = new Set();
110
+ if (modelCaps.reasoningEffort)
111
+ capabilities.add("reasoning-effort");
112
+ if (modelCaps.phase)
113
+ capabilities.add("phase-annotation");
109
114
  const client = new openai_1.default({
110
115
  apiKey: token,
111
116
  baseURL: OPENAI_CODEX_BACKEND_BASE_URL,
@@ -114,15 +119,16 @@ function createOpenAICodexProviderRuntime() {
114
119
  "OpenAI-Beta": "responses=experimental",
115
120
  originator: "ouroboros",
116
121
  },
117
- timeout: 30000,
118
122
  maxRetries: 0,
119
123
  });
120
124
  let nativeInput = null;
121
125
  let nativeInstructions = "";
122
126
  return {
123
127
  id: "openai-codex",
124
- model: codexConfig.model,
128
+ model,
125
129
  client,
130
+ capabilities,
131
+ supportedReasoningEfforts: modelCaps.reasoningEffort,
126
132
  resetTurnState(messages) {
127
133
  const { instructions, input } = (0, streaming_1.toResponsesInput)(messages);
128
134
  nativeInput = input;
@@ -141,7 +147,7 @@ function createOpenAICodexProviderRuntime() {
141
147
  input: nativeInput,
142
148
  instructions: nativeInstructions,
143
149
  tools: (0, streaming_1.toResponsesTools)(request.activeTools),
144
- reasoning: { effort: "medium", summary: "detailed" },
150
+ reasoning: { effort: request.reasoningEffort ?? "medium", summary: "detailed" },
145
151
  stream: true,
146
152
  store: false,
147
153
  include: ["reasoning.encrypted_content"],
@@ -149,14 +155,18 @@ function createOpenAICodexProviderRuntime() {
149
155
  if (request.toolChoiceRequired)
150
156
  params.tool_choice = "required";
151
157
  try {
152
- const result = await (0, streaming_1.streamResponsesApi)(this.client, params, request.callbacks, request.signal);
158
+ const result = await (0, streaming_1.streamResponsesApi)(this.client, params, request.callbacks, request.signal, request.eagerSettleStreaming);
153
159
  for (const item of result.outputItems)
154
160
  nativeInput.push(item);
155
161
  return result;
156
162
  }
157
163
  catch (error) {
158
- throw withOpenAICodexAuthGuidance(error);
164
+ throw error instanceof Error ? error : new Error(String(error));
159
165
  }
160
166
  },
167
+ /* v8 ignore next 3 -- delegation: classification logic tested via classifyOpenAICodexError @preserve */
168
+ classifyError(error) {
169
+ return classifyOpenAICodexError(error);
170
+ },
161
171
  };
162
172
  }