import { CLAUDE_CLI_PROFILE_ID, type OpenClawConfig, type ProviderAuthResult, } from "openclaw/plugin-sdk/provider-auth"; import { readClaudeCliCredentialsForSetup, readClaudeCliCredentialsForSetupNonInteractive, } from "./cli-auth-seam.js"; import { CLAUDE_CLI_BACKEND_ID, CLAUDE_CLI_DEFAULT_ALLOWLIST_REFS, CLAUDE_CLI_DEFAULT_MODEL_REF, } from "./cli-shared.js"; type AgentDefaultsModel = NonNullable["defaults"]>["model"]; type AgentDefaultsModels = NonNullable["defaults"]>["models"]; type ClaudeCliCredential = NonNullable>; function toClaudeCliModelRef(raw: string): string | null { const trimmed = raw.trim(); if (!trimmed.toLowerCase().startsWith("anthropic/")) { return null; } const modelId = trimmed.slice("anthropic/".length).trim(); if (!modelId.toLowerCase().startsWith("claude-")) { return null; } return `claude-cli/${modelId}`; } function rewriteModelSelection(model: AgentDefaultsModel): { value: AgentDefaultsModel; primary?: string; changed: boolean; } { if (typeof model === "string") { const converted = toClaudeCliModelRef(model); return converted ? { value: converted, primary: converted, changed: true } : { value: model, changed: false }; } if (!model || typeof model !== "object" || Array.isArray(model)) { return { value: model, changed: false }; } const current = model as Record; const next: Record = { ...current }; let changed = false; let primary: string | undefined; if (typeof current.primary === "string") { const converted = toClaudeCliModelRef(current.primary); if (converted) { next.primary = converted; primary = converted; changed = true; } } const currentFallbacks = current.fallbacks; if (Array.isArray(currentFallbacks)) { const nextFallbacks = currentFallbacks.map((entry) => typeof entry === "string" ? (toClaudeCliModelRef(entry) ?? entry) : entry, ); if (nextFallbacks.some((entry, index) => entry !== currentFallbacks[index])) { next.fallbacks = nextFallbacks; changed = true; } } return { value: changed ? next : model, ...(primary ? { primary } : {}), changed, }; } function rewriteModelEntryMap(models: Record | undefined): { value: Record | undefined; migrated: string[]; } { if (!models) { return { value: models, migrated: [] }; } const next = { ...models }; const migrated: string[] = []; for (const [rawKey, value] of Object.entries(models)) { const converted = toClaudeCliModelRef(rawKey); if (!converted) { continue; } if (!(converted in next)) { next[converted] = value; } delete next[rawKey]; migrated.push(converted); } return { value: migrated.length > 0 ? next : models, migrated, }; } function seedClaudeCliAllowlist( models: NonNullable, ): NonNullable { const next = { ...models }; for (const ref of CLAUDE_CLI_DEFAULT_ALLOWLIST_REFS) { next[ref] = next[ref] ?? {}; } return next; } export function hasClaudeCliAuth(options?: { allowKeychainPrompt?: boolean }): boolean { return Boolean( options?.allowKeychainPrompt === false ? readClaudeCliCredentialsForSetupNonInteractive() : readClaudeCliCredentialsForSetup(), ); } function buildClaudeCliAuthProfiles( credential?: ClaudeCliCredential | null, ): ProviderAuthResult["profiles"] { if (!credential) { return []; } if (credential.type === "oauth") { return [ { profileId: CLAUDE_CLI_PROFILE_ID, credential: { type: "oauth", provider: CLAUDE_CLI_BACKEND_ID, access: credential.access, refresh: credential.refresh, expires: credential.expires, }, }, ]; } return [ { profileId: CLAUDE_CLI_PROFILE_ID, credential: { type: "token", provider: CLAUDE_CLI_BACKEND_ID, token: credential.token, expires: credential.expires, }, }, ]; } export function buildAnthropicCliMigrationResult( config: OpenClawConfig, credential?: ClaudeCliCredential | null, ): ProviderAuthResult { const defaults = config.agents?.defaults; const rewrittenModel = rewriteModelSelection(defaults?.model); const rewrittenModels = rewriteModelEntryMap(defaults?.models); const existingModels = (rewrittenModels.value ?? defaults?.models ?? {}) as NonNullable; const nextModels = seedClaudeCliAllowlist(existingModels); const defaultModel = rewrittenModel.primary ?? CLAUDE_CLI_DEFAULT_MODEL_REF; return { profiles: buildClaudeCliAuthProfiles(credential), configPatch: { agents: { defaults: { ...(rewrittenModel.changed ? { model: rewrittenModel.value } : {}), models: nextModels, }, }, }, defaultModel, notes: [ "Claude CLI auth detected; switched Anthropic model selection to the local Claude CLI backend.", "Existing Anthropic auth profiles are kept for rollback.", ...(rewrittenModels.migrated.length > 0 ? [`Migrated allowlist entries: ${rewrittenModels.migrated.join(", ")}.`] : []), ], }; }