/** * @license * Copyright 2025 Google LLC * SPDX-License-Identifier: Apache-2.0 */ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest'; import { AuthDialog } from './AuthDialog.js'; import { LoadedSettings, SettingScope } from '../../config/settings.js'; import { AuthType } from '@qwen-code/qwen-code-core'; import { renderWithProviders } from '../../test-utils/render.js'; import { UIStateContext } from '../contexts/UIStateContext.js'; import { UIActionsContext } from '../contexts/UIActionsContext.js'; import type { UIState } from '../contexts/UIStateContext.js'; import type { UIActions } from '../contexts/UIActionsContext.js'; const createMockUIState = (overrides: Partial = {}): UIState => { // AuthDialog only uses authError and pendingAuthType const baseState = { authError: null, pendingAuthType: undefined, } as Partial; return { ...baseState, ...overrides, } as UIState; }; const createMockUIActions = (overrides: Partial = {}): UIActions => { // AuthDialog only uses handleAuthSelect const baseActions = { handleAuthSelect: vi.fn(), } as Partial; return { ...baseActions, ...overrides, } as UIActions; }; const renderAuthDialog = ( settings: LoadedSettings, uiStateOverrides: Partial = {}, uiActionsOverrides: Partial = {}, ) => { const uiState = createMockUIState(uiStateOverrides); const uiActions = createMockUIActions(uiActionsOverrides); return renderWithProviders( , { settings }, ); }; describe('AuthDialog', () => { const wait = (ms = 50) => new Promise((resolve) => setTimeout(resolve, ms)); let originalEnv: NodeJS.ProcessEnv; beforeEach(() => { originalEnv = { ...process.env }; process.env['GEMINI_API_KEY'] = ''; process.env['QWEN_DEFAULT_AUTH_TYPE'] = ''; vi.clearAllMocks(); }); afterEach(() => { process.env = originalEnv; }); it('should show an error if the initial auth type is invalid', () => { process.env['GEMINI_API_KEY'] = ''; const settings: LoadedSettings = new LoadedSettings( { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: {}, originalSettings: {}, path: '', }, { settings: { security: { auth: { selectedType: AuthType.USE_GEMINI, }, }, }, originalSettings: { security: { auth: { selectedType: AuthType.USE_GEMINI, }, }, }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, true, new Set(), ); const { lastFrame } = renderAuthDialog(settings, { authError: 'GEMINI_API_KEY environment variable not found', }); expect(lastFrame()).toContain( 'GEMINI_API_KEY environment variable not found', ); }); describe('GEMINI_API_KEY environment variable', () => { it('should detect GEMINI_API_KEY environment variable', () => { process.env['GEMINI_API_KEY'] = 'foobar'; const settings: LoadedSettings = new LoadedSettings( { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, originalSettings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: {}, originalSettings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, true, new Set(), ); const { lastFrame } = renderAuthDialog(settings); // Since the auth dialog only shows OpenAI option now, // it won't show GEMINI_API_KEY messages expect(lastFrame()).toContain('OpenAI'); }); it('should not show the GEMINI_API_KEY message if QWEN_DEFAULT_AUTH_TYPE is set to something else', () => { process.env['GEMINI_API_KEY'] = 'foobar'; process.env['QWEN_DEFAULT_AUTH_TYPE'] = AuthType.USE_OPENAI; const settings: LoadedSettings = new LoadedSettings( { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, originalSettings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: {}, originalSettings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, true, new Set(), ); const { lastFrame } = renderAuthDialog(settings); expect(lastFrame()).not.toContain( 'Existing API key detected (GEMINI_API_KEY)', ); }); it('should show the GEMINI_API_KEY message if QWEN_DEFAULT_AUTH_TYPE is set to use api key', () => { process.env['GEMINI_API_KEY'] = 'foobar'; process.env['QWEN_DEFAULT_AUTH_TYPE'] = AuthType.USE_OPENAI; const settings: LoadedSettings = new LoadedSettings( { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, originalSettings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: {}, originalSettings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, true, new Set(), ); const { lastFrame } = renderAuthDialog(settings); // Since the auth dialog only shows OpenAI option now, // it won't show GEMINI_API_KEY messages expect(lastFrame()).toContain('OpenAI'); }); }); describe('QWEN_DEFAULT_AUTH_TYPE environment variable', () => { it('should select the auth type specified by QWEN_DEFAULT_AUTH_TYPE', () => { process.env['QWEN_DEFAULT_AUTH_TYPE'] = AuthType.USE_OPENAI; const settings: LoadedSettings = new LoadedSettings( { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, originalSettings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: {}, originalSettings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, true, new Set(), ); const { lastFrame } = renderAuthDialog(settings); // This is a bit brittle, but it's the best way to check which item is selected. expect(lastFrame()).toContain('● 2. OpenAI'); }); it('should fall back to default if QWEN_DEFAULT_AUTH_TYPE is not set', () => { const settings: LoadedSettings = new LoadedSettings( { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, originalSettings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: {}, originalSettings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, true, new Set(), ); const { lastFrame } = renderAuthDialog(settings); // Default is Qwen OAuth (first option) expect(lastFrame()).toContain('● 1. Qwen OAuth'); }); it('should show an error and fall back to default if QWEN_DEFAULT_AUTH_TYPE is invalid', () => { process.env['QWEN_DEFAULT_AUTH_TYPE'] = 'invalid-auth-type'; const settings: LoadedSettings = new LoadedSettings( { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, originalSettings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: {}, originalSettings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, true, new Set(), ); const { lastFrame } = renderAuthDialog(settings); // Since the auth dialog doesn't show QWEN_DEFAULT_AUTH_TYPE errors anymore, // it will just show the default Qwen OAuth option expect(lastFrame()).toContain('● 1. Qwen OAuth'); }); }); it('should prevent exiting when no auth method is selected and show error message', async () => { const handleAuthSelect = vi.fn(); const settings: LoadedSettings = new LoadedSettings( { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: {}, originalSettings: {}, path: '', }, { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, originalSettings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, true, new Set(), ); const { lastFrame, stdin, unmount } = renderAuthDialog( settings, {}, { handleAuthSelect }, ); await wait(); // Simulate pressing escape key stdin.write('\u001b'); // ESC key await wait(); // Should show error message instead of calling handleAuthSelect expect(lastFrame()).toContain( 'You must select an auth method to proceed. Press Ctrl+C again to exit.', ); expect(handleAuthSelect).not.toHaveBeenCalled(); unmount(); }); it('should not exit if there is already an error message', async () => { const handleAuthSelect = vi.fn(); const settings: LoadedSettings = new LoadedSettings( { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: {}, originalSettings: {}, path: '', }, { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, originalSettings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, true, new Set(), ); const { lastFrame, stdin, unmount } = renderAuthDialog( settings, { authError: 'Initial error' }, { handleAuthSelect }, ); await wait(); expect(lastFrame()).toContain('Initial error'); // Simulate pressing escape key stdin.write('\u001b'); // ESC key await wait(); // Should not call handleAuthSelect expect(handleAuthSelect).not.toHaveBeenCalled(); unmount(); }); it('should allow exiting when auth method is already selected', async () => { const handleAuthSelect = vi.fn(); const settings: LoadedSettings = new LoadedSettings( { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: {}, originalSettings: {}, path: '', }, { settings: { security: { auth: { selectedType: AuthType.USE_OPENAI } }, ui: { customThemes: {} }, mcpServers: {}, }, originalSettings: { security: { auth: { selectedType: AuthType.USE_OPENAI } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, originalSettings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, true, new Set(), ); const { stdin, unmount } = renderAuthDialog( settings, {}, { handleAuthSelect }, ); await wait(); // Simulate pressing escape key stdin.write('\u001b'); // ESC key await wait(); // Should call handleAuthSelect with undefined to exit expect(handleAuthSelect).toHaveBeenCalledWith(undefined, SettingScope.User); unmount(); }); });