/** * @license * Copyright 2025 Google LLC * SPDX-License-Identifier: Apache-2.0 */ import { renderWithProviders } from '../../test-utils/render.js'; import { waitFor } from '../../test-utils/async.js'; import { act } from 'react'; import { vi, describe, it, expect, beforeEach } from 'vitest'; import { FolderTrustDialog } from './FolderTrustDialog.js'; import { ExitCodes } from '@google/gemini-cli-core'; import * as processUtils from '../../utils/processUtils.js'; vi.mock('../../utils/processUtils.js', () => ({ relaunchApp: vi.fn(), })); const mockedExit = vi.hoisted(() => vi.fn()); const mockedCwd = vi.hoisted(() => vi.fn()); vi.mock('node:process', async () => { const actual = await vi.importActual('node:process'); return { ...actual, exit: mockedExit, cwd: mockedCwd, }; }); describe('FolderTrustDialog', () => { beforeEach(() => { vi.clearAllMocks(); vi.useRealTimers(); mockedCwd.mockReturnValue('/home/user/project'); }); it('should render the dialog with title and description', () => { const { lastFrame, unmount } = renderWithProviders( , ); expect(lastFrame()).toContain('Do you trust this folder?'); expect(lastFrame()).toContain( 'Trusting a folder allows Gemini to execute commands it suggests.', ); unmount(); }); it('should display exit message and call process.exit and not call onSelect when escape is pressed', async () => { const onSelect = vi.fn(); const { lastFrame, stdin, unmount } = renderWithProviders( , ); act(() => { stdin.write('\u001b[27u'); // Press kitty escape key }); await waitFor(() => { expect(lastFrame()).toContain( 'A folder trust level must be selected to continue. Exiting since escape was pressed.', ); }); await waitFor(() => { expect(mockedExit).toHaveBeenCalledWith( ExitCodes.FATAL_CANCELLATION_ERROR, ); }); expect(onSelect).not.toHaveBeenCalled(); unmount(); }); it('should display restart message when isRestarting is true', () => { const { lastFrame, unmount } = renderWithProviders( , ); expect(lastFrame()).toContain('Gemini CLI is restarting'); unmount(); }); it('should call relaunchApp when isRestarting is true', async () => { vi.useFakeTimers(); const relaunchApp = vi.spyOn(processUtils, 'relaunchApp'); const { unmount } = renderWithProviders( , ); await vi.advanceTimersByTimeAsync(250); expect(relaunchApp).toHaveBeenCalled(); unmount(); vi.useRealTimers(); }); it('should not call relaunchApp if unmounted before timeout', async () => { vi.useFakeTimers(); const relaunchApp = vi.spyOn(processUtils, 'relaunchApp'); const { unmount } = renderWithProviders( , ); // Unmount immediately (before 250ms) unmount(); await vi.advanceTimersByTimeAsync(250); expect(relaunchApp).not.toHaveBeenCalled(); vi.useRealTimers(); }); it('should not call process.exit when "r" is pressed and isRestarting is false', async () => { const { stdin, unmount } = renderWithProviders( , ); act(() => { stdin.write('r'); }); await waitFor(() => { expect(mockedExit).not.toHaveBeenCalled(); }); unmount(); }); describe('directory display', () => { it('should correctly display the folder name for a nested directory', () => { mockedCwd.mockReturnValue('/home/user/project'); const { lastFrame, unmount } = renderWithProviders( , ); expect(lastFrame()).toContain('Trust folder (project)'); unmount(); }); it('should correctly display the parent folder name for a nested directory', () => { mockedCwd.mockReturnValue('/home/user/project'); const { lastFrame, unmount } = renderWithProviders( , ); expect(lastFrame()).toContain('Trust parent folder (user)'); unmount(); }); it('should correctly display an empty parent folder name for a directory directly under root', () => { mockedCwd.mockReturnValue('/project'); const { lastFrame, unmount } = renderWithProviders( , ); expect(lastFrame()).toContain('Trust parent folder ()'); unmount(); }); }); });