codex-multi-auth 0.1.0
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/LICENSE +1 -0
- package/README.md +162 -0
- package/assets/opencode-logo-ornate-dark.svg +18 -0
- package/assets/readme-hero.svg +31 -0
- package/config/README.md +87 -0
- package/config/minimal-opencode.json +13 -0
- package/config/opencode-legacy.json +571 -0
- package/config/opencode-modern.json +239 -0
- package/dist/index.d.ts +45 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +3160 -0
- package/dist/index.js.map +1 -0
- package/dist/lib/accounts/rate-limits.d.ts +22 -0
- package/dist/lib/accounts/rate-limits.d.ts.map +1 -0
- package/dist/lib/accounts/rate-limits.js +63 -0
- package/dist/lib/accounts/rate-limits.js.map +1 -0
- package/dist/lib/accounts.d.ts +95 -0
- package/dist/lib/accounts.d.ts.map +1 -0
- package/dist/lib/accounts.js +668 -0
- package/dist/lib/accounts.js.map +1 -0
- package/dist/lib/audit.d.ts +45 -0
- package/dist/lib/audit.d.ts.map +1 -0
- package/dist/lib/audit.js +131 -0
- package/dist/lib/audit.js.map +1 -0
- package/dist/lib/auth/auth.d.ts +56 -0
- package/dist/lib/auth/auth.d.ts.map +1 -0
- package/dist/lib/auth/auth.js +214 -0
- package/dist/lib/auth/auth.js.map +1 -0
- package/dist/lib/auth/browser.d.ts +34 -0
- package/dist/lib/auth/browser.d.ts.map +1 -0
- package/dist/lib/auth/browser.js +185 -0
- package/dist/lib/auth/browser.js.map +1 -0
- package/dist/lib/auth/server.d.ts +24 -0
- package/dist/lib/auth/server.d.ts.map +1 -0
- package/dist/lib/auth/server.js +116 -0
- package/dist/lib/auth/server.js.map +1 -0
- package/dist/lib/auth/token-utils.d.ts +59 -0
- package/dist/lib/auth/token-utils.d.ts.map +1 -0
- package/dist/lib/auth/token-utils.js +331 -0
- package/dist/lib/auth/token-utils.js.map +1 -0
- package/dist/lib/auth-rate-limit.d.ts +20 -0
- package/dist/lib/auth-rate-limit.d.ts.map +1 -0
- package/dist/lib/auth-rate-limit.js +91 -0
- package/dist/lib/auth-rate-limit.js.map +1 -0
- package/dist/lib/auto-update-checker.d.ts +10 -0
- package/dist/lib/auto-update-checker.d.ts.map +1 -0
- package/dist/lib/auto-update-checker.js +216 -0
- package/dist/lib/auto-update-checker.js.map +1 -0
- package/dist/lib/capability-policy.d.ts +18 -0
- package/dist/lib/capability-policy.d.ts.map +1 -0
- package/dist/lib/capability-policy.js +150 -0
- package/dist/lib/capability-policy.js.map +1 -0
- package/dist/lib/circuit-breaker.d.ts +34 -0
- package/dist/lib/circuit-breaker.d.ts.map +1 -0
- package/dist/lib/circuit-breaker.js +124 -0
- package/dist/lib/circuit-breaker.js.map +1 -0
- package/dist/lib/cli.d.ts +64 -0
- package/dist/lib/cli.d.ts.map +1 -0
- package/dist/lib/cli.js +274 -0
- package/dist/lib/cli.js.map +1 -0
- package/dist/lib/codex-cli/observability.d.ts +22 -0
- package/dist/lib/codex-cli/observability.d.ts.map +1 -0
- package/dist/lib/codex-cli/observability.js +36 -0
- package/dist/lib/codex-cli/observability.js.map +1 -0
- package/dist/lib/codex-cli/state.d.ts +86 -0
- package/dist/lib/codex-cli/state.d.ts.map +1 -0
- package/dist/lib/codex-cli/state.js +470 -0
- package/dist/lib/codex-cli/state.js.map +1 -0
- package/dist/lib/codex-cli/sync.d.ts +27 -0
- package/dist/lib/codex-cli/sync.d.ts.map +1 -0
- package/dist/lib/codex-cli/sync.js +325 -0
- package/dist/lib/codex-cli/sync.js.map +1 -0
- package/dist/lib/codex-cli/writer.d.ts +12 -0
- package/dist/lib/codex-cli/writer.d.ts.map +1 -0
- package/dist/lib/codex-cli/writer.js +388 -0
- package/dist/lib/codex-cli/writer.js.map +1 -0
- package/dist/lib/codex-manager.d.ts +2 -0
- package/dist/lib/codex-manager.d.ts.map +1 -0
- package/dist/lib/codex-manager.js +4841 -0
- package/dist/lib/codex-manager.js.map +1 -0
- package/dist/lib/config.d.ts +269 -0
- package/dist/lib/config.d.ts.map +1 -0
- package/dist/lib/config.js +789 -0
- package/dist/lib/config.js.map +1 -0
- package/dist/lib/constants.d.ts +78 -0
- package/dist/lib/constants.d.ts.map +1 -0
- package/dist/lib/constants.js +78 -0
- package/dist/lib/constants.js.map +1 -0
- package/dist/lib/context-overflow.d.ts +27 -0
- package/dist/lib/context-overflow.d.ts.map +1 -0
- package/dist/lib/context-overflow.js +124 -0
- package/dist/lib/context-overflow.js.map +1 -0
- package/dist/lib/dashboard-settings.d.ts +90 -0
- package/dist/lib/dashboard-settings.d.ts.map +1 -0
- package/dist/lib/dashboard-settings.js +327 -0
- package/dist/lib/dashboard-settings.js.map +1 -0
- package/dist/lib/entitlement-cache.d.ts +41 -0
- package/dist/lib/entitlement-cache.d.ts.map +1 -0
- package/dist/lib/entitlement-cache.js +137 -0
- package/dist/lib/entitlement-cache.js.map +1 -0
- package/dist/lib/errors.d.ts +113 -0
- package/dist/lib/errors.d.ts.map +1 -0
- package/dist/lib/errors.js +103 -0
- package/dist/lib/errors.js.map +1 -0
- package/dist/lib/forecast.d.ts +42 -0
- package/dist/lib/forecast.d.ts.map +1 -0
- package/dist/lib/forecast.js +256 -0
- package/dist/lib/forecast.js.map +1 -0
- package/dist/lib/health.d.ts +33 -0
- package/dist/lib/health.d.ts.map +1 -0
- package/dist/lib/health.js +70 -0
- package/dist/lib/health.js.map +1 -0
- package/dist/lib/index.d.ts +32 -0
- package/dist/lib/index.d.ts.map +1 -0
- package/dist/lib/index.js +32 -0
- package/dist/lib/index.js.map +1 -0
- package/dist/lib/live-account-sync.d.ts +39 -0
- package/dist/lib/live-account-sync.d.ts.map +1 -0
- package/dist/lib/live-account-sync.js +196 -0
- package/dist/lib/live-account-sync.js.map +1 -0
- package/dist/lib/logger.d.ts +40 -0
- package/dist/lib/logger.d.ts.map +1 -0
- package/dist/lib/logger.js +364 -0
- package/dist/lib/logger.js.map +1 -0
- package/dist/lib/oauth-success.html +338 -0
- package/dist/lib/parallel-probe.d.ts +28 -0
- package/dist/lib/parallel-probe.d.ts.map +1 -0
- package/dist/lib/parallel-probe.js +97 -0
- package/dist/lib/parallel-probe.js.map +1 -0
- package/dist/lib/preemptive-quota-scheduler.d.ts +53 -0
- package/dist/lib/preemptive-quota-scheduler.d.ts.map +1 -0
- package/dist/lib/preemptive-quota-scheduler.js +220 -0
- package/dist/lib/preemptive-quota-scheduler.js.map +1 -0
- package/dist/lib/proactive-refresh.d.ts +66 -0
- package/dist/lib/proactive-refresh.d.ts.map +1 -0
- package/dist/lib/proactive-refresh.js +143 -0
- package/dist/lib/proactive-refresh.js.map +1 -0
- package/dist/lib/prompts/codex-opencode-bridge.d.ts +19 -0
- package/dist/lib/prompts/codex-opencode-bridge.d.ts.map +1 -0
- package/dist/lib/prompts/codex-opencode-bridge.js +169 -0
- package/dist/lib/prompts/codex-opencode-bridge.js.map +1 -0
- package/dist/lib/prompts/codex.d.ts +41 -0
- package/dist/lib/prompts/codex.d.ts.map +1 -0
- package/dist/lib/prompts/codex.js +383 -0
- package/dist/lib/prompts/codex.js.map +1 -0
- package/dist/lib/prompts/opencode-codex.d.ts +25 -0
- package/dist/lib/prompts/opencode-codex.d.ts.map +1 -0
- package/dist/lib/prompts/opencode-codex.js +270 -0
- package/dist/lib/prompts/opencode-codex.js.map +1 -0
- package/dist/lib/quota-cache.d.ts +68 -0
- package/dist/lib/quota-cache.d.ts.map +1 -0
- package/dist/lib/quota-cache.js +224 -0
- package/dist/lib/quota-cache.js.map +1 -0
- package/dist/lib/quota-probe.d.ts +49 -0
- package/dist/lib/quota-probe.d.ts.map +1 -0
- package/dist/lib/quota-probe.js +368 -0
- package/dist/lib/quota-probe.js.map +1 -0
- package/dist/lib/recovery/constants.d.ts +12 -0
- package/dist/lib/recovery/constants.d.ts.map +1 -0
- package/dist/lib/recovery/constants.js +31 -0
- package/dist/lib/recovery/constants.js.map +1 -0
- package/dist/lib/recovery/index.d.ts +12 -0
- package/dist/lib/recovery/index.d.ts.map +1 -0
- package/dist/lib/recovery/index.js +12 -0
- package/dist/lib/recovery/index.js.map +1 -0
- package/dist/lib/recovery/storage.d.ts +24 -0
- package/dist/lib/recovery/storage.d.ts.map +1 -0
- package/dist/lib/recovery/storage.js +362 -0
- package/dist/lib/recovery/storage.js.map +1 -0
- package/dist/lib/recovery/types.d.ts +116 -0
- package/dist/lib/recovery/types.d.ts.map +1 -0
- package/dist/lib/recovery/types.js +7 -0
- package/dist/lib/recovery/types.js.map +1 -0
- package/dist/lib/recovery.d.ts +31 -0
- package/dist/lib/recovery.d.ts.map +1 -0
- package/dist/lib/recovery.js +313 -0
- package/dist/lib/recovery.js.map +1 -0
- package/dist/lib/refresh-guardian.d.ts +31 -0
- package/dist/lib/refresh-guardian.d.ts.map +1 -0
- package/dist/lib/refresh-guardian.js +151 -0
- package/dist/lib/refresh-guardian.js.map +1 -0
- package/dist/lib/refresh-lease.d.ts +37 -0
- package/dist/lib/refresh-lease.d.ts.map +1 -0
- package/dist/lib/refresh-lease.js +335 -0
- package/dist/lib/refresh-lease.js.map +1 -0
- package/dist/lib/refresh-queue.d.ts +117 -0
- package/dist/lib/refresh-queue.d.ts.map +1 -0
- package/dist/lib/refresh-queue.js +297 -0
- package/dist/lib/refresh-queue.js.map +1 -0
- package/dist/lib/request/failure-policy.d.ts +42 -0
- package/dist/lib/request/failure-policy.d.ts.map +1 -0
- package/dist/lib/request/failure-policy.js +133 -0
- package/dist/lib/request/failure-policy.js.map +1 -0
- package/dist/lib/request/fetch-helpers.d.ts +152 -0
- package/dist/lib/request/fetch-helpers.d.ts.map +1 -0
- package/dist/lib/request/fetch-helpers.js +704 -0
- package/dist/lib/request/fetch-helpers.js.map +1 -0
- package/dist/lib/request/helpers/input-utils.d.ts +7 -0
- package/dist/lib/request/helpers/input-utils.d.ts.map +1 -0
- package/dist/lib/request/helpers/input-utils.js +214 -0
- package/dist/lib/request/helpers/input-utils.js.map +1 -0
- package/dist/lib/request/helpers/model-map.d.ts +28 -0
- package/dist/lib/request/helpers/model-map.d.ts.map +1 -0
- package/dist/lib/request/helpers/model-map.js +133 -0
- package/dist/lib/request/helpers/model-map.js.map +1 -0
- package/dist/lib/request/helpers/tool-utils.d.ts +29 -0
- package/dist/lib/request/helpers/tool-utils.d.ts.map +1 -0
- package/dist/lib/request/helpers/tool-utils.js +117 -0
- package/dist/lib/request/helpers/tool-utils.js.map +1 -0
- package/dist/lib/request/rate-limit-backoff.d.ts +17 -0
- package/dist/lib/request/rate-limit-backoff.d.ts.map +1 -0
- package/dist/lib/request/rate-limit-backoff.js +83 -0
- package/dist/lib/request/rate-limit-backoff.js.map +1 -0
- package/dist/lib/request/request-transformer.d.ts +107 -0
- package/dist/lib/request/request-transformer.d.ts.map +1 -0
- package/dist/lib/request/request-transformer.js +814 -0
- package/dist/lib/request/request-transformer.js.map +1 -0
- package/dist/lib/request/response-handler.d.ts +23 -0
- package/dist/lib/request/response-handler.d.ts.map +1 -0
- package/dist/lib/request/response-handler.js +155 -0
- package/dist/lib/request/response-handler.js.map +1 -0
- package/dist/lib/request/stream-failover.d.ts +21 -0
- package/dist/lib/request/stream-failover.d.ts.map +1 -0
- package/dist/lib/request/stream-failover.js +204 -0
- package/dist/lib/request/stream-failover.js.map +1 -0
- package/dist/lib/rotation.d.ts +146 -0
- package/dist/lib/rotation.d.ts.map +1 -0
- package/dist/lib/rotation.js +321 -0
- package/dist/lib/rotation.js.map +1 -0
- package/dist/lib/runtime-paths.d.ts +58 -0
- package/dist/lib/runtime-paths.d.ts.map +1 -0
- package/dist/lib/runtime-paths.js +164 -0
- package/dist/lib/runtime-paths.js.map +1 -0
- package/dist/lib/schemas.d.ts +435 -0
- package/dist/lib/schemas.d.ts.map +1 -0
- package/dist/lib/schemas.js +268 -0
- package/dist/lib/schemas.js.map +1 -0
- package/dist/lib/session-affinity.d.ts +23 -0
- package/dist/lib/session-affinity.d.ts.map +1 -0
- package/dist/lib/session-affinity.js +127 -0
- package/dist/lib/session-affinity.js.map +1 -0
- package/dist/lib/shutdown.d.ts +7 -0
- package/dist/lib/shutdown.d.ts.map +1 -0
- package/dist/lib/shutdown.js +43 -0
- package/dist/lib/shutdown.js.map +1 -0
- package/dist/lib/storage/migrations.d.ts +59 -0
- package/dist/lib/storage/migrations.d.ts.map +1 -0
- package/dist/lib/storage/migrations.js +41 -0
- package/dist/lib/storage/migrations.js.map +1 -0
- package/dist/lib/storage/paths.d.ts +51 -0
- package/dist/lib/storage/paths.d.ts.map +1 -0
- package/dist/lib/storage/paths.js +152 -0
- package/dist/lib/storage/paths.js.map +1 -0
- package/dist/lib/storage.d.ts +106 -0
- package/dist/lib/storage.d.ts.map +1 -0
- package/dist/lib/storage.js +896 -0
- package/dist/lib/storage.js.map +1 -0
- package/dist/lib/table-formatter.d.ts +32 -0
- package/dist/lib/table-formatter.d.ts.map +1 -0
- package/dist/lib/table-formatter.js +44 -0
- package/dist/lib/table-formatter.js.map +1 -0
- package/dist/lib/tools/hashline-tools.d.ts +51 -0
- package/dist/lib/tools/hashline-tools.d.ts.map +1 -0
- package/dist/lib/tools/hashline-tools.js +456 -0
- package/dist/lib/tools/hashline-tools.js.map +1 -0
- package/dist/lib/types.d.ts +130 -0
- package/dist/lib/types.d.ts.map +1 -0
- package/dist/lib/types.js +2 -0
- package/dist/lib/types.js.map +1 -0
- package/dist/lib/ui/ansi.d.ts +40 -0
- package/dist/lib/ui/ansi.d.ts.map +1 -0
- package/dist/lib/ui/ansi.js +68 -0
- package/dist/lib/ui/ansi.js.map +1 -0
- package/dist/lib/ui/auth-menu.d.ts +76 -0
- package/dist/lib/ui/auth-menu.d.ts.map +1 -0
- package/dist/lib/ui/auth-menu.js +590 -0
- package/dist/lib/ui/auth-menu.js.map +1 -0
- package/dist/lib/ui/confirm.d.ts +11 -0
- package/dist/lib/ui/confirm.d.ts.map +1 -0
- package/dist/lib/ui/confirm.js +29 -0
- package/dist/lib/ui/confirm.js.map +1 -0
- package/dist/lib/ui/copy.d.ts +123 -0
- package/dist/lib/ui/copy.d.ts.map +1 -0
- package/dist/lib/ui/copy.js +127 -0
- package/dist/lib/ui/copy.js.map +1 -0
- package/dist/lib/ui/format.d.ts +62 -0
- package/dist/lib/ui/format.d.ts.map +1 -0
- package/dist/lib/ui/format.js +205 -0
- package/dist/lib/ui/format.js.map +1 -0
- package/dist/lib/ui/runtime.d.ts +43 -0
- package/dist/lib/ui/runtime.d.ts.map +1 -0
- package/dist/lib/ui/runtime.js +69 -0
- package/dist/lib/ui/runtime.js.map +1 -0
- package/dist/lib/ui/select.d.ts +60 -0
- package/dist/lib/ui/select.d.ts.map +1 -0
- package/dist/lib/ui/select.js +467 -0
- package/dist/lib/ui/select.js.map +1 -0
- package/dist/lib/ui/theme.d.ts +56 -0
- package/dist/lib/ui/theme.d.ts.map +1 -0
- package/dist/lib/ui/theme.js +186 -0
- package/dist/lib/ui/theme.js.map +1 -0
- package/dist/lib/unified-settings.d.ts +71 -0
- package/dist/lib/unified-settings.d.ts.map +1 -0
- package/dist/lib/unified-settings.js +299 -0
- package/dist/lib/unified-settings.js.map +1 -0
- package/dist/lib/utils.d.ts +29 -0
- package/dist/lib/utils.d.ts.map +1 -0
- package/dist/lib/utils.js +54 -0
- package/dist/lib/utils.js.map +1 -0
- package/package.json +115 -0
- package/scripts/audit-dev-allowlist.js +128 -0
- package/scripts/bench-format/hashline-v2.mjs +642 -0
- package/scripts/bench-format/models.mjs +105 -0
- package/scripts/bench-format/opencode.mjs +205 -0
- package/scripts/bench-format/render.mjs +496 -0
- package/scripts/bench-format/stats.mjs +54 -0
- package/scripts/bench-format/tasks.mjs +151 -0
- package/scripts/benchmark-edit-formats.mjs +1161 -0
- package/scripts/benchmark-render-dashboard.mjs +49 -0
- package/scripts/codex-multi-auth.js +6 -0
- package/scripts/codex-routing.js +34 -0
- package/scripts/codex.js +122 -0
- package/scripts/copy-oauth-success.js +37 -0
- package/scripts/install-opencode-codex-auth.js +193 -0
- package/scripts/test-all-models.sh +7 -0
- package/scripts/test-model-matrix.js +424 -0
- package/scripts/validate-model-map.sh +7 -0
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
import { spawnSync } from "node:child_process";
|
|
2
|
+
import { resolveOpencodeExecutable } from "./opencode.mjs";
|
|
3
|
+
|
|
4
|
+
const FALLBACK_OPENAI_CODEX_STABLE = [
|
|
5
|
+
"openai/gpt-5-codex",
|
|
6
|
+
"openai/gpt-5.1-codex",
|
|
7
|
+
"openai/gpt-5.1-codex-mini",
|
|
8
|
+
"openai/gpt-5.1-codex-max",
|
|
9
|
+
"openai/gpt-5.2-codex",
|
|
10
|
+
"openai/gpt-5.3-codex",
|
|
11
|
+
];
|
|
12
|
+
|
|
13
|
+
const OPENAI_CODEX_PREFIXES = ["openai/", "openai-multi/"];
|
|
14
|
+
|
|
15
|
+
export function isOpenAiCodexProviderPrefix(modelId) {
|
|
16
|
+
return OPENAI_CODEX_PREFIXES.some((prefix) => modelId.startsWith(prefix));
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function codexTail(modelId) {
|
|
20
|
+
const slash = modelId.indexOf("/");
|
|
21
|
+
return slash >= 0 ? modelId.slice(slash + 1) : modelId;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export function canonicalCodexModelId(modelId) {
|
|
25
|
+
return `openai/${codexTail(modelId)}`;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export function aliasCandidatesForCodexModel(modelId) {
|
|
29
|
+
if (!isOpenAiCodexProviderPrefix(modelId)) {
|
|
30
|
+
return [modelId];
|
|
31
|
+
}
|
|
32
|
+
const tail = codexTail(modelId);
|
|
33
|
+
const candidates = [`openai/${tail}`, `openai-multi/${tail}`];
|
|
34
|
+
return [...new Set([modelId, ...candidates])];
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export function isStableOpenAiCodexModel(modelId) {
|
|
38
|
+
if (!isOpenAiCodexProviderPrefix(modelId)) {
|
|
39
|
+
return false;
|
|
40
|
+
}
|
|
41
|
+
if (!modelId.includes("codex")) {
|
|
42
|
+
return false;
|
|
43
|
+
}
|
|
44
|
+
if (modelId.includes("-spark") || modelId.includes("-latest")) {
|
|
45
|
+
return false;
|
|
46
|
+
}
|
|
47
|
+
return true;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
export function listOpencodeModels() {
|
|
51
|
+
const executable = resolveOpencodeExecutable();
|
|
52
|
+
const child = spawnSync(executable.command, ["models"], {
|
|
53
|
+
encoding: "utf8",
|
|
54
|
+
windowsHide: true,
|
|
55
|
+
shell: executable.shell,
|
|
56
|
+
maxBuffer: 10 * 1024 * 1024,
|
|
57
|
+
});
|
|
58
|
+
const text = `${child.stdout ?? ""}\n${child.stderr ?? ""}`;
|
|
59
|
+
if ((child.status ?? 1) !== 0) {
|
|
60
|
+
throw new Error(`Failed to list OpenCode models (exit=${child.status ?? 1})`);
|
|
61
|
+
}
|
|
62
|
+
return text
|
|
63
|
+
.split(/\r?\n/)
|
|
64
|
+
.map((line) => line.trim())
|
|
65
|
+
.filter((line) => line.length > 0)
|
|
66
|
+
.filter((line) => !line.startsWith("Using Node "));
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
function dedupeCodexModels(models) {
|
|
70
|
+
const byCanonical = new Map();
|
|
71
|
+
for (const modelId of models) {
|
|
72
|
+
const canonical = canonicalCodexModelId(modelId);
|
|
73
|
+
const existing = byCanonical.get(canonical);
|
|
74
|
+
if (!existing) {
|
|
75
|
+
byCanonical.set(canonical, modelId);
|
|
76
|
+
continue;
|
|
77
|
+
}
|
|
78
|
+
// Prefer the shorter/default provider form when both exist.
|
|
79
|
+
if (existing.startsWith("openai-multi/") && modelId.startsWith("openai/")) {
|
|
80
|
+
byCanonical.set(canonical, modelId);
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
return [...byCanonical.values()].sort((a, b) => canonicalCodexModelId(a).localeCompare(canonicalCodexModelId(b)));
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
export function resolveModelPreset(presetName, explicitModels) {
|
|
87
|
+
if (explicitModels && explicitModels.length > 0) {
|
|
88
|
+
return explicitModels;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
if (presetName !== "codex-core") {
|
|
92
|
+
throw new Error(`Unsupported preset: ${presetName}`);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
try {
|
|
96
|
+
const models = dedupeCodexModels(listOpencodeModels().filter(isStableOpenAiCodexModel));
|
|
97
|
+
if (models.length > 0) {
|
|
98
|
+
return models;
|
|
99
|
+
}
|
|
100
|
+
} catch {
|
|
101
|
+
// Fall back to static list.
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
return [...FALLBACK_OPENAI_CODEX_STABLE];
|
|
105
|
+
}
|
|
@@ -0,0 +1,205 @@
|
|
|
1
|
+
import { spawnSync } from "node:child_process";
|
|
2
|
+
import { dirname, resolve } from "node:path";
|
|
3
|
+
import { fileURLToPath } from "node:url";
|
|
4
|
+
|
|
5
|
+
const scriptDir = dirname(fileURLToPath(import.meta.url));
|
|
6
|
+
const repoRoot = resolve(scriptDir, "..", "..");
|
|
7
|
+
|
|
8
|
+
export function getRepoRoot() {
|
|
9
|
+
return repoRoot;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export function resolveOpencodeExecutable() {
|
|
13
|
+
const envOverride = process.env.OPENCODE_BIN;
|
|
14
|
+
if (envOverride && envOverride.trim().length > 0) {
|
|
15
|
+
const command = envOverride.trim();
|
|
16
|
+
return { command, shell: /\.cmd$/i.test(command) };
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
if (process.platform !== "win32") {
|
|
20
|
+
return { command: "opencode", shell: false };
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const whereResult = spawnSync("where", ["opencode"], {
|
|
24
|
+
encoding: "utf8",
|
|
25
|
+
windowsHide: true,
|
|
26
|
+
});
|
|
27
|
+
const candidates = `${whereResult.stdout ?? ""}\n${whereResult.stderr ?? ""}`
|
|
28
|
+
.split(/\r?\n/)
|
|
29
|
+
.map((line) => line.trim())
|
|
30
|
+
.filter(Boolean);
|
|
31
|
+
|
|
32
|
+
if (candidates.length === 0) {
|
|
33
|
+
return { command: "opencode", shell: false };
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
const exactExe = candidates.find((candidate) => /npm\\opencode\.exe$/i.test(candidate));
|
|
37
|
+
if (exactExe) {
|
|
38
|
+
return { command: exactExe, shell: false };
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
const exactCmd = candidates.find((candidate) => /npm\\opencode\.cmd$/i.test(candidate));
|
|
42
|
+
if (exactCmd) {
|
|
43
|
+
return { command: exactCmd, shell: true };
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
const anyCmd = candidates.find((candidate) => /\.cmd$/i.test(candidate));
|
|
47
|
+
if (anyCmd) {
|
|
48
|
+
return { command: anyCmd, shell: true };
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
return { command: candidates[0], shell: false };
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
export function parseNdjson(text) {
|
|
55
|
+
const events = [];
|
|
56
|
+
for (const line of text.split(/\r?\n/)) {
|
|
57
|
+
const trimmed = line.trim();
|
|
58
|
+
if (!trimmed) {
|
|
59
|
+
continue;
|
|
60
|
+
}
|
|
61
|
+
try {
|
|
62
|
+
events.push(JSON.parse(trimmed));
|
|
63
|
+
} catch {
|
|
64
|
+
// Ignore non-JSON lines emitted by wrappers (bun install, warnings, etc.).
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
return events;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
export function getToolEvents(events) {
|
|
71
|
+
return events
|
|
72
|
+
.filter((event) => event?.type === "tool_use" && event?.part?.type === "tool")
|
|
73
|
+
.map((event) => ({
|
|
74
|
+
tool: event.part.tool,
|
|
75
|
+
input: event.part.state?.input ?? {},
|
|
76
|
+
output: event.part.state?.output,
|
|
77
|
+
status: event.part.state?.status,
|
|
78
|
+
start: event.part.state?.time?.start,
|
|
79
|
+
end: event.part.state?.time?.end,
|
|
80
|
+
durationMs:
|
|
81
|
+
typeof event.part.state?.time?.start === "number" &&
|
|
82
|
+
typeof event.part.state?.time?.end === "number"
|
|
83
|
+
? event.part.state.time.end - event.part.state.time.start
|
|
84
|
+
: null,
|
|
85
|
+
}));
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
export function getSessionDuration(events) {
|
|
89
|
+
const starts = events
|
|
90
|
+
.filter((event) => event?.type === "step_start" && typeof event.timestamp === "number")
|
|
91
|
+
.map((event) => event.timestamp);
|
|
92
|
+
const finishes = events
|
|
93
|
+
.filter((event) => event?.type === "step_finish" && typeof event.timestamp === "number")
|
|
94
|
+
.map((event) => event.timestamp);
|
|
95
|
+
if (starts.length === 0 || finishes.length === 0) {
|
|
96
|
+
return null;
|
|
97
|
+
}
|
|
98
|
+
return Math.max(...finishes) - Math.min(...starts);
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
export function getTokenTotals(events) {
|
|
102
|
+
const stepFinishes = events.filter((event) => event?.type === "step_finish" && event?.part?.tokens);
|
|
103
|
+
if (stepFinishes.length === 0) {
|
|
104
|
+
return null;
|
|
105
|
+
}
|
|
106
|
+
const total = {
|
|
107
|
+
total: 0,
|
|
108
|
+
input: 0,
|
|
109
|
+
output: 0,
|
|
110
|
+
reasoning: 0,
|
|
111
|
+
cacheRead: 0,
|
|
112
|
+
cacheWrite: 0,
|
|
113
|
+
};
|
|
114
|
+
for (const event of stepFinishes) {
|
|
115
|
+
const tokens = event.part.tokens ?? {};
|
|
116
|
+
const input = Number(tokens.input ?? 0);
|
|
117
|
+
const output = Number(tokens.output ?? 0);
|
|
118
|
+
const reasoning = Number(tokens.reasoning ?? 0);
|
|
119
|
+
const explicitTotal = Number(tokens.total ?? NaN);
|
|
120
|
+
total.total += Number.isFinite(explicitTotal) ? explicitTotal : input + output + reasoning;
|
|
121
|
+
total.input += input;
|
|
122
|
+
total.output += output;
|
|
123
|
+
total.reasoning += reasoning;
|
|
124
|
+
total.cacheRead += Number(tokens.cache?.read ?? 0);
|
|
125
|
+
total.cacheWrite += Number(tokens.cache?.write ?? 0);
|
|
126
|
+
}
|
|
127
|
+
return total;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
export function getTextOutput(events) {
|
|
131
|
+
return events
|
|
132
|
+
.filter((event) => event?.type === "text" && typeof event?.part?.text === "string")
|
|
133
|
+
.map((event) => event.part.text)
|
|
134
|
+
.join("\n");
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
export function getEventError(events) {
|
|
138
|
+
const errorEvent = events.find((event) => event?.type === "error");
|
|
139
|
+
if (!errorEvent) {
|
|
140
|
+
return null;
|
|
141
|
+
}
|
|
142
|
+
return {
|
|
143
|
+
name: errorEvent.error?.name ?? "UnknownError",
|
|
144
|
+
message: errorEvent.error?.data?.message ?? errorEvent.error?.message ?? "Unknown error",
|
|
145
|
+
};
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
export function runOpencodeJson({
|
|
149
|
+
executable,
|
|
150
|
+
prompt,
|
|
151
|
+
model,
|
|
152
|
+
variant,
|
|
153
|
+
agent,
|
|
154
|
+
cwd,
|
|
155
|
+
homeDir,
|
|
156
|
+
timeoutMs,
|
|
157
|
+
extraEnv,
|
|
158
|
+
}) {
|
|
159
|
+
const startWall = Date.now();
|
|
160
|
+
const args = ["run", "--format", "json", "--agent", agent, "--model", model];
|
|
161
|
+
if (variant) {
|
|
162
|
+
args.push("--variant", variant);
|
|
163
|
+
}
|
|
164
|
+
args.push(prompt);
|
|
165
|
+
|
|
166
|
+
const child = spawnSync(executable.command, args, {
|
|
167
|
+
cwd: cwd ?? repoRoot,
|
|
168
|
+
encoding: "utf8",
|
|
169
|
+
windowsHide: true,
|
|
170
|
+
shell: executable.shell,
|
|
171
|
+
timeout: timeoutMs,
|
|
172
|
+
maxBuffer: 30 * 1024 * 1024,
|
|
173
|
+
env: {
|
|
174
|
+
...process.env,
|
|
175
|
+
...(homeDir ? { HOME: homeDir, USERPROFILE: homeDir } : {}),
|
|
176
|
+
...extraEnv,
|
|
177
|
+
},
|
|
178
|
+
});
|
|
179
|
+
|
|
180
|
+
const wallMs = Date.now() - startWall;
|
|
181
|
+
const stdout = child.stdout ?? "";
|
|
182
|
+
const stderr = child.stderr ?? "";
|
|
183
|
+
const events = parseNdjson(stdout);
|
|
184
|
+
const eventError = getEventError(events);
|
|
185
|
+
const timedOut =
|
|
186
|
+
child.error?.code === "ETIMEDOUT" ||
|
|
187
|
+
child.signal === "SIGTERM" ||
|
|
188
|
+
/timed out/i.test(String(child.error?.message ?? ""));
|
|
189
|
+
if (child.error && !timedOut) {
|
|
190
|
+
throw child.error;
|
|
191
|
+
}
|
|
192
|
+
const modelNotFound = /Model not found|ProviderModelNotFoundError/i.test(`${stdout}\n${stderr}`) || /Model not found/i.test(eventError?.message ?? "");
|
|
193
|
+
|
|
194
|
+
return {
|
|
195
|
+
status: child.status ?? 1,
|
|
196
|
+
signal: child.signal ?? null,
|
|
197
|
+
stdout,
|
|
198
|
+
stderr,
|
|
199
|
+
wallMs,
|
|
200
|
+
events,
|
|
201
|
+
eventError,
|
|
202
|
+
timedOut,
|
|
203
|
+
modelNotFound,
|
|
204
|
+
};
|
|
205
|
+
}
|