/** * @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'; 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: {} }, path: '', }, { settings: {}, path: '', }, { settings: { security: { auth: { selectedType: AuthType.USE_GEMINI, }, }, }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, [], true, new Set(), ); const { lastFrame } = renderWithProviders( {}} settings={settings} initialErrorMessage="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: {}, }, path: '', }, { settings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, [], true, new Set(), ); const { lastFrame } = renderWithProviders( {}} settings={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.LOGIN_WITH_GOOGLE; const settings: LoadedSettings = new LoadedSettings( { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, [], true, new Set(), ); const { lastFrame } = renderWithProviders( {}} settings={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_GEMINI; const settings: LoadedSettings = new LoadedSettings( { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, [], true, new Set(), ); const { lastFrame } = renderWithProviders( {}} settings={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: {}, }, path: '', }, { settings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, [], true, new Set(), ); const { lastFrame } = renderWithProviders( {}} settings={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: {}, }, path: '', }, { settings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, [], true, new Set(), ); const { lastFrame } = renderWithProviders( {}} settings={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: {}, }, path: '', }, { settings: {}, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, [], true, new Set(), ); const { lastFrame } = renderWithProviders( {}} settings={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 onSelect = vi.fn(); const settings: LoadedSettings = new LoadedSettings( { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: {}, path: '', }, { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, [], true, new Set(), ); const { lastFrame, stdin, unmount } = renderWithProviders( , ); await wait(); // Simulate pressing escape key stdin.write('\u001b'); // ESC key await wait(); // Should show error message instead of calling onSelect expect(lastFrame()).toContain( 'You must select an auth method to proceed. Press Ctrl+C again to exit.', ); expect(onSelect).not.toHaveBeenCalled(); unmount(); }); it('should not exit if there is already an error message', async () => { const onSelect = vi.fn(); const settings: LoadedSettings = new LoadedSettings( { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: {}, path: '', }, { settings: { security: { auth: { selectedType: undefined } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, [], true, new Set(), ); const { lastFrame, stdin, unmount } = renderWithProviders( , ); await wait(); expect(lastFrame()).toContain('Initial error'); // Simulate pressing escape key stdin.write('\u001b'); // ESC key await wait(); // Should not call onSelect expect(onSelect).not.toHaveBeenCalled(); unmount(); }); it('should allow exiting when auth method is already selected', async () => { const onSelect = vi.fn(); const settings: LoadedSettings = new LoadedSettings( { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, { settings: {}, path: '', }, { settings: { security: { auth: { selectedType: AuthType.LOGIN_WITH_GOOGLE } }, ui: { customThemes: {} }, mcpServers: {}, }, path: '', }, { settings: { ui: { customThemes: {} }, mcpServers: {} }, path: '', }, [], true, new Set(), ); const { stdin, unmount } = renderWithProviders( , ); await wait(); // Simulate pressing escape key stdin.write('\u001b'); // ESC key await wait(); // Should call onSelect with undefined to exit expect(onSelect).toHaveBeenCalledWith(undefined, SettingScope.User); unmount(); }); });