/** * @license * Copyright 2025 Google LLC * SPDX-License-Identifier: Apache-2.0 */ import React, { useState } from 'react'; import { Box, Text, useInput } from 'ink'; import { Colors } from '../colors.js'; import { RadioButtonSelect } from './shared/RadioButtonSelect.js'; import { LoadedSettings, SettingScope } from '../../config/settings.js'; import { AuthType } from '@google/gemini-cli-core'; import { validateAuthMethod } from '../../config/auth.js'; interface AuthDialogProps { onSelect: (authMethod: AuthType | undefined, scope: SettingScope) => void; settings: LoadedSettings; initialErrorMessage?: string | null; } function parseDefaultAuthType( defaultAuthType: string | undefined, ): AuthType | null { if ( defaultAuthType && Object.values(AuthType).includes(defaultAuthType as AuthType) ) { return defaultAuthType as AuthType; } return null; } export function AuthDialog({ onSelect, settings, initialErrorMessage, }: AuthDialogProps): React.JSX.Element { const [errorMessage, setErrorMessage] = useState(() => { if (initialErrorMessage) { return initialErrorMessage; } const defaultAuthType = parseDefaultAuthType( process.env.GEMINI_DEFAULT_AUTH_TYPE, ); if (process.env.GEMINI_DEFAULT_AUTH_TYPE && defaultAuthType === null) { return ( `Invalid value for GEMINI_DEFAULT_AUTH_TYPE: "${process.env.GEMINI_DEFAULT_AUTH_TYPE}". ` + `Valid values are: ${Object.values(AuthType).join(', ')}.` ); } if ( process.env.GEMINI_API_KEY && (!defaultAuthType || defaultAuthType === AuthType.USE_GEMINI) ) { return 'Existing API key detected (GEMINI_API_KEY). Select "Gemini API Key" option to use it.'; } return null; }); const items = [ { label: 'Login with Google', value: AuthType.LOGIN_WITH_GOOGLE, }, ...(process.env.CLOUD_SHELL === 'true' ? [ { label: 'Use Cloud Shell user credentials', value: AuthType.CLOUD_SHELL, }, ] : []), { label: 'Use Gemini API Key', value: AuthType.USE_GEMINI, }, { label: 'Vertex AI', value: AuthType.USE_VERTEX_AI }, ]; const initialAuthIndex = items.findIndex((item) => { if (settings.merged.selectedAuthType) { return item.value === settings.merged.selectedAuthType; } const defaultAuthType = parseDefaultAuthType( process.env.GEMINI_DEFAULT_AUTH_TYPE, ); if (defaultAuthType) { return item.value === defaultAuthType; } if (process.env.GEMINI_API_KEY) { return item.value === AuthType.USE_GEMINI; } return item.value === AuthType.LOGIN_WITH_GOOGLE; }); const handleAuthSelect = (authMethod: AuthType) => { const error = validateAuthMethod(authMethod); if (error) { setErrorMessage(error); } else { setErrorMessage(null); onSelect(authMethod, SettingScope.User); } }; useInput((_input, key) => { if (key.escape) { // Prevent exit if there is an error message. // This means they user is not authenticated yet. if (errorMessage) { return; } if (settings.merged.selectedAuthType === undefined) { // Prevent exiting if no auth method is set setErrorMessage( 'You must select an auth method to proceed. Press Ctrl+C twice to exit.', ); return; } onSelect(undefined, SettingScope.User); } }); return ( Get started How would you like to authenticate for this project? {errorMessage && ( {errorMessage} )} (Use Enter to select) Terms of Services and Privacy Notice for Gemini CLI { 'https://github.com/google-gemini/gemini-cli/blob/main/docs/tos-privacy.md' } ); }