@ouro.bot/cli 0.1.0-alpha.32 → 0.1.0-alpha.321
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.
- package/README.md +188 -190
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/agent.json +3 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/SOUL.md +1 -1
- package/changelog.json +1924 -0
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +140 -0
- package/dist/arc/episodes.js +117 -0
- package/dist/arc/intentions.js +133 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +237 -0
- package/dist/arc/packets.js +193 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +65 -0
- package/dist/heart/active-work.js +832 -0
- package/dist/heart/agent-entry.js +37 -2
- package/dist/heart/attachments/image-normalize.js +194 -0
- package/dist/heart/attachments/materialize.js +97 -0
- package/dist/heart/attachments/originals.js +88 -0
- package/dist/heart/attachments/render.js +29 -0
- package/dist/heart/attachments/sources/adapter.js +2 -0
- package/dist/heart/attachments/sources/bluebubbles.js +156 -0
- package/dist/heart/attachments/sources/cli-local-file.js +78 -0
- package/dist/heart/attachments/sources/index.js +16 -0
- package/dist/heart/attachments/store.js +103 -0
- package/dist/heart/attachments/types.js +93 -0
- package/dist/heart/auth/auth-flow.js +456 -0
- package/dist/heart/bridges/manager.js +358 -0
- package/dist/heart/bridges/state-machine.js +135 -0
- package/dist/heart/bridges/store.js +123 -0
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +111 -0
- package/dist/heart/config-registry.js +304 -0
- package/dist/heart/config.js +63 -30
- package/dist/heart/core.js +669 -195
- package/dist/heart/cross-chat-delivery.js +131 -0
- package/dist/heart/daemon/agent-config-check.js +149 -0
- package/dist/heart/daemon/agent-discovery.js +79 -3
- package/dist/heart/daemon/agent-service.js +360 -0
- package/dist/heart/daemon/agentic-repair.js +170 -0
- package/dist/heart/daemon/cadence.js +70 -0
- package/dist/heart/daemon/cli-defaults.js +596 -0
- package/dist/heart/daemon/cli-exec.js +2238 -0
- package/dist/heart/daemon/cli-help.js +306 -0
- package/dist/heart/daemon/cli-parse.js +824 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +506 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/daemon-cli.js +29 -1171
- package/dist/heart/daemon/daemon-entry.js +333 -3
- package/dist/heart/daemon/daemon-health.js +137 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +153 -12
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +751 -58
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +322 -0
- package/dist/heart/daemon/health-monitor.js +66 -0
- package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
- package/dist/heart/daemon/hooks/bundle-meta.js +115 -1
- package/dist/heart/daemon/http-health-probe.js +80 -0
- package/dist/heart/daemon/inner-status.js +89 -0
- package/dist/heart/daemon/interactive-repair.js +69 -0
- package/dist/heart/daemon/launchd.js +46 -9
- package/dist/heart/daemon/log-tailer.js +82 -12
- package/dist/heart/daemon/logs-prune.js +105 -0
- package/dist/heart/daemon/message-router.js +17 -8
- package/dist/heart/daemon/os-cron-deps.js +134 -0
- package/dist/heart/daemon/ouro-bot-entry.js +1 -1
- package/dist/heart/daemon/process-manager.js +201 -0
- package/dist/heart/daemon/provider-discovery.js +105 -0
- package/dist/heart/daemon/pulse.js +463 -0
- package/dist/heart/daemon/run-hooks.js +2 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +101 -0
- package/dist/heart/daemon/runtime-mode.js +67 -0
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +72 -3
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +94 -0
- package/dist/heart/daemon/socket-client.js +307 -0
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +227 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/thoughts.js +510 -0
- package/dist/heart/daemon/up-progress.js +135 -0
- package/dist/heart/delegation.js +62 -0
- package/dist/heart/habits/habit-migration.js +181 -0
- package/dist/heart/habits/habit-parser.js +140 -0
- package/dist/heart/habits/habit-scheduler.js +371 -0
- package/dist/heart/{daemon → hatch}/hatch-flow.js +30 -120
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
- package/dist/heart/{daemon → hatch}/specialist-prompt.js +10 -7
- package/dist/heart/{daemon → hatch}/specialist-tools.js +49 -3
- package/dist/heart/identity.js +163 -60
- package/dist/heart/kicks.js +2 -20
- package/dist/heart/mcp/mcp-server.js +653 -0
- package/dist/heart/migrate-config.js +127 -0
- package/dist/heart/model-capabilities.js +59 -0
- package/dist/heart/outlook/outlook-http.js +439 -0
- package/dist/heart/outlook/outlook-read.js +28 -0
- package/dist/heart/outlook/outlook-render.js +1032 -0
- package/dist/heart/outlook/outlook-types.js +27 -0
- package/dist/heart/outlook/outlook-view.js +194 -0
- package/dist/heart/outlook/readers/agent-machine.js +355 -0
- package/dist/heart/outlook/readers/continuity-readers.js +332 -0
- package/dist/heart/outlook/readers/runtime-readers.js +660 -0
- package/dist/heart/outlook/readers/sessions.js +231 -0
- package/dist/heart/outlook/readers/shared.js +111 -0
- package/dist/heart/progress-story.js +42 -0
- package/dist/heart/provider-failover.js +88 -0
- package/dist/heart/provider-ping.js +162 -0
- package/dist/heart/providers/anthropic-token.js +163 -0
- package/dist/heart/providers/anthropic.js +169 -46
- package/dist/heart/providers/azure.js +98 -11
- package/dist/heart/providers/error-classification.js +63 -0
- package/dist/heart/providers/github-copilot.js +136 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +23 -5
- package/dist/heart/providers/openai-codex.js +33 -22
- package/dist/heart/session-activity.js +190 -0
- package/dist/heart/session-events.js +726 -0
- package/dist/heart/session-recall.js +162 -0
- package/dist/heart/start-of-turn-packet.js +341 -0
- package/dist/heart/streaming.js +36 -27
- package/dist/heart/sync.js +332 -0
- package/dist/heart/target-resolution.js +127 -0
- package/dist/heart/tempo.js +93 -0
- package/dist/heart/temporal-view.js +41 -0
- package/dist/heart/tool-activity-callbacks.js +36 -0
- package/dist/heart/tool-description.js +135 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +358 -0
- package/dist/heart/turn-coordinator.js +28 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-global-installer.js +1 -1
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/{daemon → versioning}/ouro-path-installer.js +78 -35
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/{daemon → versioning}/staged-restart.js +40 -8
- package/dist/heart/{daemon → versioning}/update-checker.js +12 -2
- package/dist/heart/{daemon → versioning}/update-hooks.js +63 -59
- package/dist/mind/associative-recall.js +137 -66
- package/dist/mind/bundle-manifest.js +8 -1
- package/dist/mind/context.js +89 -93
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +84 -96
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/first-impressions.js +14 -1
- package/dist/mind/friends/channel.js +56 -0
- package/dist/mind/friends/group-context.js +144 -0
- package/dist/mind/friends/resolver.js +37 -0
- package/dist/mind/friends/store-file.js +58 -3
- package/dist/mind/friends/trust-explanation.js +74 -0
- package/dist/mind/friends/types.js +8 -0
- package/dist/mind/journal-index.js +161 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +76 -9
- package/dist/mind/prompt.js +950 -113
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/mind/token-estimate.js +8 -12
- package/dist/nerves/cli-logging.js +7 -1
- package/dist/nerves/coverage/audit.js +1 -1
- package/dist/nerves/coverage/file-completeness.js +76 -5
- package/dist/nerves/coverage/run-artifacts.js +1 -1
- package/dist/nerves/event-buffer.js +111 -0
- package/dist/nerves/index.js +224 -4
- package/dist/nerves/observation.js +20 -0
- package/dist/nerves/redact.js +79 -0
- package/dist/nerves/runtime.js +5 -1
- package/dist/outlook-ui/assets/index-IuR4F6y6.js +61 -0
- package/dist/outlook-ui/assets/index-LwChZTgL.css +1 -0
- package/dist/outlook-ui/index.html +15 -0
- package/dist/repertoire/ado-client.js +15 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +319 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +79 -0
- package/dist/repertoire/coding/codex-jsonl.js +64 -0
- package/dist/repertoire/coding/context-pack.js +330 -0
- package/dist/repertoire/coding/feedback.js +197 -30
- package/dist/repertoire/coding/manager.js +159 -11
- package/dist/repertoire/coding/spawner.js +55 -9
- package/dist/repertoire/coding/tools.js +170 -7
- package/dist/repertoire/commerce-errors.js +109 -0
- package/dist/repertoire/commerce-self-test.js +156 -0
- package/dist/repertoire/credential-access.js +527 -0
- package/dist/repertoire/duffel-client.js +185 -0
- package/dist/repertoire/github-client.js +14 -55
- package/dist/repertoire/graph-client.js +11 -52
- package/dist/repertoire/guardrails.js +375 -0
- package/dist/repertoire/mcp-client.js +255 -0
- package/dist/repertoire/mcp-manager.js +305 -0
- package/dist/repertoire/mcp-tools.js +63 -0
- package/dist/repertoire/shell-sessions.js +133 -0
- package/dist/repertoire/skills.js +14 -23
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tasks/board.js +43 -5
- package/dist/repertoire/tasks/fix.js +182 -0
- package/dist/repertoire/tasks/index.js +28 -10
- package/dist/repertoire/tasks/lifecycle.js +2 -2
- package/dist/repertoire/tasks/parser.js +3 -2
- package/dist/repertoire/tasks/scanner.js +194 -37
- package/dist/repertoire/tasks/transitions.js +16 -79
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +316 -0
- package/dist/repertoire/tools-base.js +45 -771
- package/dist/repertoire/tools-bluebubbles.js +1 -0
- package/dist/repertoire/tools-bridge.js +141 -0
- package/dist/repertoire/tools-bundle.js +984 -0
- package/dist/repertoire/tools-config.js +185 -0
- package/dist/repertoire/tools-continuity.js +248 -0
- package/dist/repertoire/tools-credential.js +182 -0
- package/dist/repertoire/tools-files.js +342 -0
- package/dist/repertoire/tools-flight.js +224 -0
- package/dist/repertoire/tools-flow.js +105 -0
- package/dist/repertoire/tools-github.js +1 -7
- package/dist/repertoire/tools-memory.js +376 -0
- package/dist/repertoire/tools-session.js +739 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +180 -0
- package/dist/repertoire/tools-surface.js +243 -0
- package/dist/repertoire/tools-teams.js +12 -62
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +110 -0
- package/dist/repertoire/tools.js +144 -138
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +118 -0
- package/dist/repertoire/vault-setup.js +241 -0
- package/dist/scripts/claude-code-hook.js +41 -0
- package/dist/scripts/claude-code-stop-hook.js +47 -0
- package/dist/senses/attention-queue.js +116 -0
- package/dist/senses/bluebubbles/attachment-cache.js +53 -0
- package/dist/senses/bluebubbles/attachment-download.js +137 -0
- package/dist/senses/{bluebubbles-client.js → bluebubbles/client.js} +143 -9
- package/dist/senses/bluebubbles/entry.js +13 -0
- package/dist/senses/bluebubbles/inbound-log.js +113 -0
- package/dist/senses/bluebubbles/index.js +1436 -0
- package/dist/senses/{bluebubbles-media.js → bluebubbles/media.js} +121 -70
- package/dist/senses/{bluebubbles-model.js → bluebubbles/model.js} +43 -12
- package/dist/senses/{bluebubbles-mutation-log.js → bluebubbles/mutation-log.js} +46 -6
- package/dist/senses/bluebubbles/replay.js +129 -0
- package/dist/senses/bluebubbles/runtime-state.js +109 -0
- package/dist/senses/{bluebubbles-session-cleanup.js → bluebubbles/session-cleanup.js} +1 -1
- package/dist/senses/cli/bracketed-paste.js +82 -0
- package/dist/senses/cli/image-paste.js +287 -0
- package/dist/senses/cli/image-ref-navigation.js +75 -0
- package/dist/senses/cli/ink-app.js +156 -0
- package/dist/senses/cli/inline-diff.js +64 -0
- package/dist/senses/cli/input-keys.js +174 -0
- package/dist/senses/cli/kill-ring.js +86 -0
- package/dist/senses/cli/message-list.js +51 -0
- package/dist/senses/cli/ouro-tui.js +605 -0
- package/dist/senses/cli/spinner-imperative.js +135 -0
- package/dist/senses/cli/spinner.js +101 -0
- package/dist/senses/cli/status-line.js +60 -0
- package/dist/senses/cli/streaming-markdown.js +526 -0
- package/dist/senses/cli/tool-display.js +83 -0
- package/dist/senses/cli/tool-render.js +85 -0
- package/dist/senses/cli/tui-store.js +240 -0
- package/dist/senses/cli/virtual-list.js +35 -0
- package/dist/senses/cli-entry.js +1 -1
- package/dist/senses/cli-layout.js +187 -0
- package/dist/senses/cli.js +595 -246
- package/dist/senses/commands.js +65 -1
- package/dist/senses/continuity.js +94 -0
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +112 -19
- package/dist/senses/inner-dialog.js +633 -86
- package/dist/senses/pipeline.js +565 -0
- package/dist/senses/shared-turn.js +199 -0
- package/dist/senses/surface-tool.js +68 -0
- package/dist/senses/teams.js +666 -166
- package/dist/senses/trust-gate.js +112 -2
- package/package.json +27 -7
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +110 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +81 -0
- package/skills/travel-planning.md +138 -0
- package/dist/heart/daemon/subagent-installer.js +0 -134
- package/dist/senses/bluebubbles-entry.js +0 -11
- package/dist/senses/bluebubbles.js +0 -544
- package/dist/senses/debug-activity.js +0 -108
- package/subagents/README.md +0 -73
- package/subagents/work-doer.md +0 -235
- package/subagents/work-merger.md +0 -618
- package/subagents/work-planner.md +0 -382
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
- /package/dist/heart/{daemon → hatch}/hatch-animation.js +0 -0
- /package/dist/heart/{daemon → hatch}/specialist-orchestrator.js +0 -0
- /package/dist/heart/{daemon → versioning}/ouro-uti.js +0 -0
- /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,136 @@
|
|
|
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) {
|
|
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
|
+
const config = (0, config_1.getGithubCopilotConfig)();
|
|
25
|
+
if (!config.githubToken) {
|
|
26
|
+
throw new Error("provider 'github-copilot' is selected in agent.json but providers.github-copilot.githubToken is missing in secrets.json.");
|
|
27
|
+
}
|
|
28
|
+
if (!config.baseUrl) {
|
|
29
|
+
throw new Error("provider 'github-copilot' is selected in agent.json but providers.github-copilot.baseUrl is missing in secrets.json.");
|
|
30
|
+
}
|
|
31
|
+
const isCompletionsModel = model.startsWith("claude");
|
|
32
|
+
const modelCaps = (0, model_capabilities_1.getModelCapabilities)(model);
|
|
33
|
+
const capabilities = new Set();
|
|
34
|
+
/* v8 ignore next -- branch: capability detection tested via unit test @preserve */
|
|
35
|
+
if (modelCaps.reasoningEffort)
|
|
36
|
+
capabilities.add("reasoning-effort");
|
|
37
|
+
const client = new openai_1.default({
|
|
38
|
+
apiKey: config.githubToken,
|
|
39
|
+
baseURL: config.baseUrl,
|
|
40
|
+
maxRetries: 0,
|
|
41
|
+
});
|
|
42
|
+
if (isCompletionsModel) {
|
|
43
|
+
// Chat completions path (Claude models via Copilot)
|
|
44
|
+
return {
|
|
45
|
+
id: "github-copilot",
|
|
46
|
+
model,
|
|
47
|
+
client,
|
|
48
|
+
capabilities,
|
|
49
|
+
supportedReasoningEfforts: modelCaps.reasoningEffort,
|
|
50
|
+
resetTurnState(_messages) {
|
|
51
|
+
// No provider-owned turn state for chat-completions path.
|
|
52
|
+
},
|
|
53
|
+
appendToolOutput(_callId, _output) {
|
|
54
|
+
// Chat-completions providers rely on canonical messages only.
|
|
55
|
+
},
|
|
56
|
+
/* v8 ignore start -- streamTurn: tested via mock assertions in github-copilot.test.ts @preserve */
|
|
57
|
+
async streamTurn(request) {
|
|
58
|
+
const params = {
|
|
59
|
+
messages: request.messages,
|
|
60
|
+
tools: request.activeTools,
|
|
61
|
+
stream: true,
|
|
62
|
+
};
|
|
63
|
+
if (this.model)
|
|
64
|
+
params.model = this.model;
|
|
65
|
+
if (request.traceId)
|
|
66
|
+
params.metadata = { trace_id: request.traceId };
|
|
67
|
+
if (request.toolChoiceRequired)
|
|
68
|
+
params.tool_choice = "required";
|
|
69
|
+
try {
|
|
70
|
+
return await (0, streaming_1.streamChatCompletion)(this.client, params, request.callbacks, request.signal, request.eagerSettleStreaming);
|
|
71
|
+
}
|
|
72
|
+
catch (error) {
|
|
73
|
+
throw error instanceof Error ? error : new Error(String(error));
|
|
74
|
+
}
|
|
75
|
+
},
|
|
76
|
+
/* v8 ignore stop */
|
|
77
|
+
/* v8 ignore next 3 -- delegation: classification logic tested via classifyGithubCopilotError @preserve */
|
|
78
|
+
classifyError(error) {
|
|
79
|
+
return classifyGithubCopilotError(error);
|
|
80
|
+
},
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
// Responses API path (GPT models via Copilot)
|
|
84
|
+
let nativeInput = null;
|
|
85
|
+
let nativeInstructions = "";
|
|
86
|
+
return {
|
|
87
|
+
id: "github-copilot",
|
|
88
|
+
model,
|
|
89
|
+
client,
|
|
90
|
+
capabilities,
|
|
91
|
+
supportedReasoningEfforts: modelCaps.reasoningEffort,
|
|
92
|
+
/* v8 ignore start -- responses path: tested via mock assertions in github-copilot.test.ts @preserve */
|
|
93
|
+
resetTurnState(messages) {
|
|
94
|
+
const { instructions, input } = (0, streaming_1.toResponsesInput)(messages);
|
|
95
|
+
nativeInput = input;
|
|
96
|
+
nativeInstructions = instructions;
|
|
97
|
+
},
|
|
98
|
+
appendToolOutput(callId, output) {
|
|
99
|
+
if (!nativeInput)
|
|
100
|
+
return;
|
|
101
|
+
nativeInput.push({ type: "function_call_output", call_id: callId, output });
|
|
102
|
+
},
|
|
103
|
+
async streamTurn(request) {
|
|
104
|
+
if (!nativeInput)
|
|
105
|
+
this.resetTurnState(request.messages);
|
|
106
|
+
const params = {
|
|
107
|
+
model: this.model,
|
|
108
|
+
input: nativeInput,
|
|
109
|
+
instructions: nativeInstructions,
|
|
110
|
+
tools: (0, streaming_1.toResponsesTools)(request.activeTools),
|
|
111
|
+
reasoning: { effort: request.reasoningEffort ?? "medium", summary: "detailed" },
|
|
112
|
+
stream: true,
|
|
113
|
+
store: false,
|
|
114
|
+
include: ["reasoning.encrypted_content"],
|
|
115
|
+
};
|
|
116
|
+
if (request.traceId)
|
|
117
|
+
params.metadata = { trace_id: request.traceId };
|
|
118
|
+
if (request.toolChoiceRequired)
|
|
119
|
+
params.tool_choice = "required";
|
|
120
|
+
try {
|
|
121
|
+
const result = await (0, streaming_1.streamResponsesApi)(this.client, params, request.callbacks, request.signal, request.eagerSettleStreaming);
|
|
122
|
+
for (const item of result.outputItems)
|
|
123
|
+
nativeInput.push(item);
|
|
124
|
+
return result;
|
|
125
|
+
}
|
|
126
|
+
catch (error) {
|
|
127
|
+
throw error instanceof Error ? error : new Error(String(error));
|
|
128
|
+
}
|
|
129
|
+
},
|
|
130
|
+
/* v8 ignore stop */
|
|
131
|
+
/* v8 ignore next 3 -- delegation: classification logic tested via classifyGithubCopilotError @preserve */
|
|
132
|
+
classifyError(error) {
|
|
133
|
+
return classifyGithubCopilotError(error);
|
|
134
|
+
},
|
|
135
|
+
};
|
|
136
|
+
}
|
|
@@ -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,12 +3,25 @@ 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
|
-
|
|
23
|
+
const model_capabilities_1 = require("../model-capabilities");
|
|
24
|
+
function createMinimaxProviderRuntime(model) {
|
|
12
25
|
(0, runtime_1.emitNervesEvent)({
|
|
13
26
|
component: "engine",
|
|
14
27
|
event: "engine.provider_init",
|
|
@@ -19,16 +32,18 @@ function createMinimaxProviderRuntime() {
|
|
|
19
32
|
if (!minimaxConfig.apiKey) {
|
|
20
33
|
throw new Error("provider 'minimax' is selected in agent.json but providers.minimax.apiKey is missing in secrets.json.");
|
|
21
34
|
}
|
|
35
|
+
// Registry consulted; MiniMax models return empty defaults (no capabilities to derive)
|
|
36
|
+
(0, model_capabilities_1.getModelCapabilities)(model);
|
|
22
37
|
const client = new openai_1.default({
|
|
23
38
|
apiKey: minimaxConfig.apiKey,
|
|
24
|
-
baseURL:
|
|
25
|
-
timeout: 30000,
|
|
39
|
+
baseURL: exports.MINIMAX_PROVIDER_BASE_URL,
|
|
26
40
|
maxRetries: 0,
|
|
27
41
|
});
|
|
28
42
|
return {
|
|
29
43
|
id: "minimax",
|
|
30
|
-
model
|
|
44
|
+
model,
|
|
31
45
|
client,
|
|
46
|
+
capabilities: new Set(),
|
|
32
47
|
resetTurnState(_messages) {
|
|
33
48
|
// No provider-owned turn state for chat-completions providers.
|
|
34
49
|
},
|
|
@@ -47,7 +62,10 @@ function createMinimaxProviderRuntime() {
|
|
|
47
62
|
params.metadata = { trace_id: request.traceId };
|
|
48
63
|
if (request.toolChoiceRequired)
|
|
49
64
|
params.tool_choice = "required";
|
|
50
|
-
return (0, streaming_1.streamChatCompletion)(this.client, params, request.callbacks, request.signal);
|
|
65
|
+
return (0, streaming_1.streamChatCompletion)(this.client, params, request.callbacks, request.signal, request.eagerSettleStreaming);
|
|
66
|
+
},
|
|
67
|
+
classifyError(error) {
|
|
68
|
+
return classifyMinimaxError(error);
|
|
51
69
|
},
|
|
52
70
|
};
|
|
53
71
|
}
|
|
@@ -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 \`
|
|
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,7 +87,7 @@ function getChatGPTAccountIdFromToken(token) {
|
|
|
87
87
|
return "";
|
|
88
88
|
return accountId.trim();
|
|
89
89
|
}
|
|
90
|
-
function createOpenAICodexProviderRuntime() {
|
|
90
|
+
function createOpenAICodexProviderRuntime(model) {
|
|
91
91
|
(0, runtime_1.emitNervesEvent)({
|
|
92
92
|
component: "engine",
|
|
93
93
|
event: "engine.provider_init",
|
|
@@ -95,8 +95,8 @@ function createOpenAICodexProviderRuntime() {
|
|
|
95
95
|
meta: { provider: "openai-codex" },
|
|
96
96
|
});
|
|
97
97
|
const codexConfig = (0, config_1.getOpenAICodexConfig)();
|
|
98
|
-
if (!
|
|
99
|
-
throw new Error(getOpenAICodexReauthGuidance("provider 'openai-codex' is selected in agent.json but providers.openai-codex.
|
|
98
|
+
if (!codexConfig.oauthAccessToken) {
|
|
99
|
+
throw new Error(getOpenAICodexReauthGuidance("provider 'openai-codex' is selected in agent.json but providers.openai-codex.oauthAccessToken is missing in secrets.json."));
|
|
100
100
|
}
|
|
101
101
|
const token = codexConfig.oauthAccessToken.trim();
|
|
102
102
|
if (!token) {
|
|
@@ -106,6 +106,12 @@ function createOpenAICodexProviderRuntime() {
|
|
|
106
106
|
if (!chatgptAccountId) {
|
|
107
107
|
throw new Error(getOpenAICodexReauthGuidance("OpenAI Codex OAuth access token is missing a chatgpt_account_id claim required for chatgpt.com/backend-api/codex."));
|
|
108
108
|
}
|
|
109
|
+
const modelCaps = (0, model_capabilities_1.getModelCapabilities)(model);
|
|
110
|
+
const capabilities = new Set();
|
|
111
|
+
if (modelCaps.reasoningEffort)
|
|
112
|
+
capabilities.add("reasoning-effort");
|
|
113
|
+
if (modelCaps.phase)
|
|
114
|
+
capabilities.add("phase-annotation");
|
|
109
115
|
const client = new openai_1.default({
|
|
110
116
|
apiKey: token,
|
|
111
117
|
baseURL: OPENAI_CODEX_BACKEND_BASE_URL,
|
|
@@ -114,15 +120,16 @@ function createOpenAICodexProviderRuntime() {
|
|
|
114
120
|
"OpenAI-Beta": "responses=experimental",
|
|
115
121
|
originator: "ouroboros",
|
|
116
122
|
},
|
|
117
|
-
timeout: 30000,
|
|
118
123
|
maxRetries: 0,
|
|
119
124
|
});
|
|
120
125
|
let nativeInput = null;
|
|
121
126
|
let nativeInstructions = "";
|
|
122
127
|
return {
|
|
123
128
|
id: "openai-codex",
|
|
124
|
-
model
|
|
129
|
+
model,
|
|
125
130
|
client,
|
|
131
|
+
capabilities,
|
|
132
|
+
supportedReasoningEfforts: modelCaps.reasoningEffort,
|
|
126
133
|
resetTurnState(messages) {
|
|
127
134
|
const { instructions, input } = (0, streaming_1.toResponsesInput)(messages);
|
|
128
135
|
nativeInput = input;
|
|
@@ -141,7 +148,7 @@ function createOpenAICodexProviderRuntime() {
|
|
|
141
148
|
input: nativeInput,
|
|
142
149
|
instructions: nativeInstructions,
|
|
143
150
|
tools: (0, streaming_1.toResponsesTools)(request.activeTools),
|
|
144
|
-
reasoning: { effort: "medium", summary: "detailed" },
|
|
151
|
+
reasoning: { effort: request.reasoningEffort ?? "medium", summary: "detailed" },
|
|
145
152
|
stream: true,
|
|
146
153
|
store: false,
|
|
147
154
|
include: ["reasoning.encrypted_content"],
|
|
@@ -149,14 +156,18 @@ function createOpenAICodexProviderRuntime() {
|
|
|
149
156
|
if (request.toolChoiceRequired)
|
|
150
157
|
params.tool_choice = "required";
|
|
151
158
|
try {
|
|
152
|
-
const result = await (0, streaming_1.streamResponsesApi)(this.client, params, request.callbacks, request.signal);
|
|
159
|
+
const result = await (0, streaming_1.streamResponsesApi)(this.client, params, request.callbacks, request.signal, request.eagerSettleStreaming);
|
|
153
160
|
for (const item of result.outputItems)
|
|
154
161
|
nativeInput.push(item);
|
|
155
162
|
return result;
|
|
156
163
|
}
|
|
157
164
|
catch (error) {
|
|
158
|
-
throw
|
|
165
|
+
throw error instanceof Error ? error : new Error(String(error));
|
|
159
166
|
}
|
|
160
167
|
},
|
|
168
|
+
/* v8 ignore next 3 -- delegation: classification logic tested via classifyOpenAICodexError @preserve */
|
|
169
|
+
classifyError(error) {
|
|
170
|
+
return classifyOpenAICodexError(error);
|
|
171
|
+
},
|
|
161
172
|
};
|
|
162
173
|
}
|