/** * Credential storage for API keys and OAuth tokens. * Handles loading, saving, and refreshing credentials from auth.json. */ import { getEnvApiKey, getOAuthApiKey, loginAnthropic, loginAntigravity, loginGeminiCli, loginGitHubCopilot, type OAuthCredentials, type OAuthProvider, } from "@mariozechner/pi-ai"; import { chmodSync, existsSync, mkdirSync, readFileSync, writeFileSync } from "fs"; import { dirname } from "path"; export type ApiKeyCredential = { type: "api_key"; key: string; }; export type OAuthCredential = { type: "oauth"; } & OAuthCredentials; export type AuthCredential = ApiKeyCredential | OAuthCredential; export type AuthStorageData = Record; /** * Credential storage backed by a JSON file. */ export class AuthStorage { private data: AuthStorageData = {}; private runtimeOverrides: Map = new Map(); private fallbackResolver?: (provider: string) => string | undefined; constructor(private authPath: string) { this.reload(); } /** * Set a runtime API key override (not persisted to disk). * Used for CLI --api-key flag. */ setRuntimeApiKey(provider: string, apiKey: string): void { this.runtimeOverrides.set(provider, apiKey); } /** * Remove a runtime API key override. */ removeRuntimeApiKey(provider: string): void { this.runtimeOverrides.delete(provider); } /** * Set a fallback resolver for API keys not found in auth.json or env vars. * Used for custom provider keys from models.json. */ setFallbackResolver(resolver: (provider: string) => string | undefined): void { this.fallbackResolver = resolver; } /** * Reload credentials from disk. */ reload(): void { if (!existsSync(this.authPath)) { this.data = {}; return; } try { this.data = JSON.parse(readFileSync(this.authPath, "utf-8")); } catch { this.data = {}; } } /** * Save credentials to disk. */ private save(): void { const dir = dirname(this.authPath); if (!existsSync(dir)) { mkdirSync(dir, { recursive: true, mode: 0o700 }); } writeFileSync(this.authPath, JSON.stringify(this.data, null, 2), "utf-8"); chmodSync(this.authPath, 0o600); } /** * Get credential for a provider. */ get(provider: string): AuthCredential | undefined { return this.data[provider] ?? undefined; } /** * Set credential for a provider. */ set(provider: string, credential: AuthCredential): void { this.data[provider] = credential; this.save(); } /** * Remove credential for a provider. */ remove(provider: string): void { delete this.data[provider]; this.save(); } /** * List all providers with credentials. */ list(): string[] { return Object.keys(this.data); } /** * Check if credentials exist for a provider. */ has(provider: string): boolean { return provider in this.data; } /** * Get all credentials (for passing to getOAuthApiKey). */ getAll(): AuthStorageData { return { ...this.data }; } /** * Login to an OAuth provider. */ async login( provider: OAuthProvider, callbacks: { onAuth: (info: { url: string; instructions?: string }) => void; onPrompt: (prompt: { message: string; placeholder?: string }) => Promise; onProgress?: (message: string) => void; }, ): Promise { let credentials: OAuthCredentials; switch (provider) { case "anthropic": credentials = await loginAnthropic( (url) => callbacks.onAuth({ url }), () => callbacks.onPrompt({ message: "Paste the authorization code:" }), ); break; case "github-copilot": credentials = await loginGitHubCopilot({ onAuth: (url, instructions) => callbacks.onAuth({ url, instructions }), onPrompt: callbacks.onPrompt, onProgress: callbacks.onProgress, }); break; case "google-gemini-cli": credentials = await loginGeminiCli(callbacks.onAuth, callbacks.onProgress); break; case "google-antigravity": credentials = await loginAntigravity(callbacks.onAuth, callbacks.onProgress); break; default: throw new Error(`Unknown OAuth provider: ${provider}`); } this.set(provider, { type: "oauth", ...credentials }); } /** * Logout from a provider. */ logout(provider: string): void { this.remove(provider); } /** * Get API key for a provider. * Priority: * 1. Runtime override (CLI --api-key) * 2. API key from auth.json * 3. OAuth token from auth.json (auto-refreshed) * 4. Environment variable * 5. Fallback resolver (models.json custom providers) */ async getApiKey(provider: string): Promise { // Runtime override takes highest priority const runtimeKey = this.runtimeOverrides.get(provider); if (runtimeKey) { return runtimeKey; } const cred = this.data[provider]; if (cred?.type === "api_key") { return cred.key; } if (cred?.type === "oauth") { // Filter to only oauth credentials for getOAuthApiKey const oauthCreds: Record = {}; for (const [key, value] of Object.entries(this.data)) { if (value.type === "oauth") { oauthCreds[key] = value; } } try { const result = await getOAuthApiKey(provider as OAuthProvider, oauthCreds); if (result) { this.data[provider] = { type: "oauth", ...result.newCredentials }; this.save(); return result.apiKey; } } catch { this.remove(provider); } } // Fall back to environment variable const envKey = getEnvApiKey(provider); if (envKey) return envKey; // Fall back to custom resolver (e.g., models.json custom providers) return this.fallbackResolver?.(provider) ?? undefined; } }