fix: restore implicit providers + copilot auth choice

This commit is contained in:
Peter Steinberger
2026-01-13 04:04:11 +00:00
parent b41e75a15d
commit ea5597b483
2 changed files with 185 additions and 103 deletions

View File

@@ -2,64 +2,78 @@ import fs from "node:fs/promises";
import path from "node:path"; import path from "node:path";
import { type ClawdbotConfig, loadConfig } from "../config/config.js"; import { type ClawdbotConfig, loadConfig } from "../config/config.js";
import {
DEFAULT_COPILOT_API_BASE_URL,
resolveCopilotApiToken,
} from "../providers/github-copilot-token.js";
import { resolveClawdbotAgentDir } from "./agent-paths.js"; import { resolveClawdbotAgentDir } from "./agent-paths.js";
import { import {
type AuthProfileStore,
ensureAuthProfileStore, ensureAuthProfileStore,
listProfilesForProvider, listProfilesForProvider,
} from "./auth-profiles.js"; } from "./auth-profiles.js";
import { resolveEnvApiKey } from "./model-auth.js"; import {
normalizeProviders,
type ProviderConfig,
resolveImplicitProviders,
} from "./models-config.providers.js";
type ModelsConfig = NonNullable<ClawdbotConfig["models"]>; type ModelsConfig = NonNullable<ClawdbotConfig["models"]>;
type ProviderConfig = NonNullable<ModelsConfig["providers"]>[string];
const DEFAULT_MODE: NonNullable<ModelsConfig["mode"]> = "merge"; const DEFAULT_MODE: NonNullable<ModelsConfig["mode"]> = "merge";
const MINIMAX_API_BASE_URL = "https://api.minimax.io/anthropic";
const MINIMAX_DEFAULT_MODEL_ID = "MiniMax-M2.1";
const MINIMAX_DEFAULT_CONTEXT_WINDOW = 200000;
const MINIMAX_DEFAULT_MAX_TOKENS = 8192;
// Pricing: MiniMax doesn't publish public rates. Override in models.json for accurate costs.
const MINIMAX_API_COST = {
input: 15,
output: 60,
cacheRead: 2,
cacheWrite: 10,
};
function isRecord(value: unknown): value is Record<string, unknown> { function isRecord(value: unknown): value is Record<string, unknown> {
return Boolean(value && typeof value === "object" && !Array.isArray(value)); return Boolean(value && typeof value === "object" && !Array.isArray(value));
} }
function normalizeGoogleModelId(id: string): string { function mergeProviderModels(
if (id === "gemini-3-pro") return "gemini-3-pro-preview"; implicit: ProviderConfig,
if (id === "gemini-3-flash") return "gemini-3-flash-preview"; explicit: ProviderConfig,
return id; ): ProviderConfig {
const implicitModels = Array.isArray(implicit.models) ? implicit.models : [];
const explicitModels = Array.isArray(explicit.models) ? explicit.models : [];
if (implicitModels.length === 0) return { ...implicit, ...explicit };
const getId = (model: unknown): string => {
if (!model || typeof model !== "object") return "";
const id = (model as { id?: unknown }).id;
return typeof id === "string" ? id.trim() : "";
};
const seen = new Set(explicitModels.map(getId).filter(Boolean));
const mergedModels = [
...explicitModels,
...implicitModels.filter((model) => {
const id = getId(model);
if (!id) return false;
if (seen.has(id)) return false;
seen.add(id);
return true;
}),
];
return {
...implicit,
...explicit,
models: mergedModels,
};
} }
function normalizeGoogleProvider(provider: ProviderConfig): ProviderConfig { function mergeProviders(params: {
let mutated = false; implicit?: Record<string, ProviderConfig> | null;
const models = provider.models.map((model) => { explicit?: Record<string, ProviderConfig> | null;
const nextId = normalizeGoogleModelId(model.id); }): Record<string, ProviderConfig> {
if (nextId === model.id) return model; const out: Record<string, ProviderConfig> = params.implicit
mutated = true; ? { ...params.implicit }
return { ...model, id: nextId }; : {};
}); for (const [key, explicit] of Object.entries(params.explicit ?? {})) {
return mutated ? { ...provider, models } : provider; const providerKey = key.trim();
} if (!providerKey) continue;
const implicit = out[providerKey];
function normalizeProviders( out[providerKey] = implicit
providers: ModelsConfig["providers"], ? mergeProviderModels(implicit, explicit)
): ModelsConfig["providers"] { : explicit;
if (!providers) return providers;
let mutated = false;
const next: Record<string, ProviderConfig> = {};
for (const [key, provider] of Object.entries(providers)) {
const normalized =
key === "google" ? normalizeGoogleProvider(provider) : provider;
if (normalized !== provider) mutated = true;
next[key] = normalized;
} }
return mutated ? next : providers; return out;
} }
async function readJson(pathname: string): Promise<unknown> { async function readJson(pathname: string): Promise<unknown> {
@@ -71,67 +85,62 @@ async function readJson(pathname: string): Promise<unknown> {
} }
} }
function buildMinimaxApiProvider(apiKey?: string): ProviderConfig { async function maybeBuildCopilotProvider(params: {
return {
baseUrl: MINIMAX_API_BASE_URL,
...(apiKey ? { apiKey } : {}),
api: "anthropic-messages",
models: [
{
id: MINIMAX_DEFAULT_MODEL_ID,
name: "MiniMax M2.1",
reasoning: false,
input: ["text"],
cost: MINIMAX_API_COST,
contextWindow: MINIMAX_DEFAULT_CONTEXT_WINDOW,
maxTokens: MINIMAX_DEFAULT_MAX_TOKENS,
},
],
};
}
function resolveMinimaxApiKeyFromStore(
store: AuthProfileStore,
): string | undefined {
const profileIds = listProfilesForProvider(store, "minimax");
for (const profileId of profileIds) {
const cred = store.profiles[profileId];
if (!cred) continue;
if (cred.type === "api_key") {
const key = cred.key?.trim();
if (key) return key;
continue;
}
if (cred.type === "token") {
const token = cred.token?.trim();
if (!token) continue;
if (
typeof cred.expires === "number" &&
Number.isFinite(cred.expires) &&
cred.expires > 0 &&
Date.now() >= cred.expires
) {
continue;
}
return token;
}
}
return undefined;
}
function resolveImplicitProviders(params: {
cfg: ClawdbotConfig;
agentDir: string; agentDir: string;
}): ModelsConfig["providers"] { env?: NodeJS.ProcessEnv;
const providers: Record<string, ProviderConfig> = {}; }): Promise<ProviderConfig | null> {
const minimaxEnv = resolveEnvApiKey("minimax"); const env = params.env ?? process.env;
const authStore = ensureAuthProfileStore(params.agentDir); const authStore = ensureAuthProfileStore(params.agentDir);
const minimaxKey = const hasProfile =
minimaxEnv?.apiKey ?? resolveMinimaxApiKeyFromStore(authStore); listProfilesForProvider(authStore, "github-copilot").length > 0;
if (minimaxKey) { const envToken = env.COPILOT_GITHUB_TOKEN ?? env.GH_TOKEN ?? env.GITHUB_TOKEN;
providers.minimax = buildMinimaxApiProvider(minimaxKey); const githubToken = (envToken ?? "").trim();
if (!hasProfile && !githubToken) return null;
let selectedGithubToken = githubToken;
if (!selectedGithubToken && hasProfile) {
// Use the first available profile as a default for discovery (it will be
// re-resolved per-run by the embedded runner).
const profileId = listProfilesForProvider(authStore, "github-copilot")[0];
const profile = profileId ? authStore.profiles[profileId] : undefined;
if (profile && profile.type === "token") {
selectedGithubToken = profile.token;
}
} }
return providers;
let baseUrl = DEFAULT_COPILOT_API_BASE_URL;
if (selectedGithubToken) {
try {
const token = await resolveCopilotApiToken({
githubToken: selectedGithubToken,
env,
});
baseUrl = token.baseUrl;
} catch {
baseUrl = DEFAULT_COPILOT_API_BASE_URL;
}
}
// pi-coding-agent's ModelRegistry marks a model "available" only if its
// `AuthStorage` has auth configured for that provider (via auth.json/env/etc).
// Our Copilot auth lives in Clawdbot's auth-profiles store instead, so we also
// write a runtime-only auth.json entry for pi-coding-agent to pick up.
//
// This is safe because it's (1) within Clawdbot's agent dir, (2) contains the
// GitHub token (not the exchanged Copilot token), and (3) matches existing
// patterns for OAuth-like providers in pi-coding-agent.
// Note: we deliberately do not write pi-coding-agent's `auth.json` here.
// Clawdbot uses its own auth store and exchanges tokens at runtime.
// `models list` uses Clawdbot's auth heuristics for availability.
// We intentionally do NOT define custom models for Copilot in models.json.
// pi-coding-agent treats providers with models as replacements requiring apiKey.
// We only override baseUrl; the model list comes from pi-ai built-ins.
return {
baseUrl,
models: [],
} satisfies ProviderConfig;
} }
export async function ensureClawdbotModelsJson( export async function ensureClawdbotModelsJson(
@@ -142,9 +151,21 @@ export async function ensureClawdbotModelsJson(
const agentDir = agentDirOverride?.trim() const agentDir = agentDirOverride?.trim()
? agentDirOverride.trim() ? agentDirOverride.trim()
: resolveClawdbotAgentDir(); : resolveClawdbotAgentDir();
const configuredProviders = cfg.models?.providers ?? {};
const implicitProviders = resolveImplicitProviders({ cfg, agentDir }); const explicitProviders = (cfg.models?.providers ?? {}) as Record<
const providers = { ...implicitProviders, ...configuredProviders }; string,
ProviderConfig
>;
const implicitProviders = resolveImplicitProviders({ agentDir });
const providers: Record<string, ProviderConfig> = mergeProviders({
implicit: implicitProviders,
explicit: explicitProviders,
});
const implicitCopilot = await maybeBuildCopilotProvider({ agentDir });
if (implicitCopilot && !providers["github-copilot"]) {
providers["github-copilot"] = implicitCopilot;
}
if (Object.keys(providers).length === 0) { if (Object.keys(providers).length === 0) {
return { agentDir, wrote: false }; return { agentDir, wrote: false };
} }
@@ -165,7 +186,10 @@ export async function ensureClawdbotModelsJson(
} }
} }
const normalizedProviders = normalizeProviders(mergedProviders); const normalizedProviders = normalizeProviders({
providers: mergedProviders,
agentDir,
});
const next = `${JSON.stringify({ providers: normalizedProviders }, null, 2)}\n`; const next = `${JSON.stringify({ providers: normalizedProviders }, null, 2)}\n`;
try { try {
existingRaw = await fs.readFile(targetPath, "utf8"); existingRaw = await fs.readFile(targetPath, "utf8");

View File

@@ -21,6 +21,7 @@ import { loadModelCatalog } from "../agents/model-catalog.js";
import { resolveConfiguredModelRef } from "../agents/model-selection.js"; import { resolveConfiguredModelRef } from "../agents/model-selection.js";
import type { ClawdbotConfig } from "../config/config.js"; import type { ClawdbotConfig } from "../config/config.js";
import { upsertSharedEnvVar } from "../infra/env-file.js"; import { upsertSharedEnvVar } from "../infra/env-file.js";
import { githubCopilotLoginCommand } from "../providers/github-copilot-auth.js";
import type { RuntimeEnv } from "../runtime.js"; import type { RuntimeEnv } from "../runtime.js";
import type { WizardPrompter } from "../wizard/prompts.js"; import type { WizardPrompter } from "../wizard/prompts.js";
import { import {
@@ -931,6 +932,61 @@ export async function applyAuthChoice(params: {
agentModelOverride = modelRef; agentModelOverride = modelRef;
await noteAgentModel(modelRef); await noteAgentModel(modelRef);
} }
} else if (params.authChoice === "github-copilot") {
await params.prompter.note(
[
"This will open a GitHub device login to authorize Copilot.",
"Requires an active GitHub Copilot subscription.",
].join("\n"),
"GitHub Copilot",
);
if (!process.stdin.isTTY) {
await params.prompter.note(
"GitHub Copilot login requires an interactive TTY.",
"GitHub Copilot",
);
return { config: nextConfig, agentModelOverride };
}
try {
await githubCopilotLoginCommand({ yes: true }, params.runtime);
} catch (err) {
await params.prompter.note(
`GitHub Copilot login failed: ${String(err)}`,
"GitHub Copilot",
);
return { config: nextConfig, agentModelOverride };
}
nextConfig = applyAuthProfileConfig(nextConfig, {
profileId: "github-copilot:github",
provider: "github-copilot",
mode: "token",
});
if (params.setDefaultModel) {
const model = "github-copilot/gpt-4o";
nextConfig = {
...nextConfig,
agents: {
...nextConfig.agents,
defaults: {
...nextConfig.agents?.defaults,
model: {
...(typeof nextConfig.agents?.defaults?.model === "object"
? nextConfig.agents.defaults.model
: undefined),
primary: model,
},
},
},
};
await params.prompter.note(
`Default model set to ${model}`,
"Model configured",
);
}
} else if (params.authChoice === "minimax") { } else if (params.authChoice === "minimax") {
if (params.setDefaultModel) { if (params.setDefaultModel) {
nextConfig = applyMinimaxConfig(nextConfig); nextConfig = applyMinimaxConfig(nextConfig);
@@ -1018,6 +1074,8 @@ export function resolvePreferredProviderForAuthChoice(
return "google-antigravity"; return "google-antigravity";
case "synthetic-api-key": case "synthetic-api-key":
return "synthetic"; return "synthetic";
case "github-copilot":
return "github-copilot";
case "minimax-cloud": case "minimax-cloud":
case "minimax-api": case "minimax-api":
case "minimax-api-lightning": case "minimax-api-lightning":