diff options
| author | Abhi <[email protected]> | 2025-06-17 22:17:16 -0400 |
|---|---|---|
| committer | GitHub <[email protected]> | 2025-06-17 22:17:16 -0400 |
| commit | f3c1cbbabfe31510d15c979fc4669e7ece3eab55 (patch) | |
| tree | 0b7b6611d8913484eb00a3b22435180abb1b5ff9 /packages/cli/src/ui/components/InputPrompt.test.tsx | |
| parent | 443465a8051c89e83b48d1aa2273dfe13b8e2e94 (diff) | |
feat: shell history (#1169)
Diffstat (limited to 'packages/cli/src/ui/components/InputPrompt.test.tsx')
| -rw-r--r-- | packages/cli/src/ui/components/InputPrompt.test.tsx | 187 |
1 files changed, 187 insertions, 0 deletions
diff --git a/packages/cli/src/ui/components/InputPrompt.test.tsx b/packages/cli/src/ui/components/InputPrompt.test.tsx new file mode 100644 index 00000000..d0133e53 --- /dev/null +++ b/packages/cli/src/ui/components/InputPrompt.test.tsx @@ -0,0 +1,187 @@ +/** + * @license + * Copyright 2025 Google LLC + * SPDX-License-Identifier: Apache-2.0 + */ + +import { render } from 'ink-testing-library'; +import { InputPrompt, InputPromptProps } from './InputPrompt.js'; +import type { TextBuffer } from './shared/text-buffer.js'; +import { Config } from '@gemini-cli/core'; +import { vi } from 'vitest'; +import { useShellHistory } from '../hooks/useShellHistory.js'; +import { useCompletion } from '../hooks/useCompletion.js'; +import { useInputHistory } from '../hooks/useInputHistory.js'; + +vi.mock('../hooks/useShellHistory.js'); +vi.mock('../hooks/useCompletion.js'); +vi.mock('../hooks/useInputHistory.js'); + +type MockedUseShellHistory = ReturnType<typeof useShellHistory>; +type MockedUseCompletion = ReturnType<typeof useCompletion>; +type MockedUseInputHistory = ReturnType<typeof useInputHistory>; + +describe('InputPrompt', () => { + let props: InputPromptProps; + let mockShellHistory: MockedUseShellHistory; + let mockCompletion: MockedUseCompletion; + let mockInputHistory: MockedUseInputHistory; + let mockBuffer: TextBuffer; + + const mockedUseShellHistory = vi.mocked(useShellHistory); + const mockedUseCompletion = vi.mocked(useCompletion); + const mockedUseInputHistory = vi.mocked(useInputHistory); + + beforeEach(() => { + vi.resetAllMocks(); + + mockBuffer = { + text: '', + cursor: [0, 0], + lines: [''], + setText: vi.fn((newText: string) => { + mockBuffer.text = newText; + mockBuffer.lines = [newText]; + mockBuffer.cursor = [0, newText.length]; + mockBuffer.viewportVisualLines = [newText]; + mockBuffer.allVisualLines = [newText]; + }), + viewportVisualLines: [''], + allVisualLines: [''], + visualCursor: [0, 0], + visualScrollRow: 0, + handleInput: vi.fn(), + move: vi.fn(), + moveToOffset: vi.fn(), + killLineRight: vi.fn(), + killLineLeft: vi.fn(), + openInExternalEditor: vi.fn(), + newline: vi.fn(), + replaceRangeByOffset: vi.fn(), + } as unknown as TextBuffer; + + mockShellHistory = { + addCommandToHistory: vi.fn(), + getPreviousCommand: vi.fn().mockReturnValue(null), + getNextCommand: vi.fn().mockReturnValue(null), + resetHistoryPosition: vi.fn(), + }; + mockedUseShellHistory.mockReturnValue(mockShellHistory); + + mockCompletion = { + suggestions: [], + activeSuggestionIndex: -1, + isLoadingSuggestions: false, + showSuggestions: false, + visibleStartIndex: 0, + navigateUp: vi.fn(), + navigateDown: vi.fn(), + resetCompletionState: vi.fn(), + setActiveSuggestionIndex: vi.fn(), + setShowSuggestions: vi.fn(), + }; + mockedUseCompletion.mockReturnValue(mockCompletion); + + mockInputHistory = { + navigateUp: vi.fn(), + navigateDown: vi.fn(), + handleSubmit: vi.fn(), + }; + mockedUseInputHistory.mockReturnValue(mockInputHistory); + + props = { + buffer: mockBuffer, + onSubmit: vi.fn(), + userMessages: [], + onClearScreen: vi.fn(), + config: { + getProjectRoot: () => '/test/project', + getTargetDir: () => '/test/project/src', + } as unknown as Config, + slashCommands: [], + shellModeActive: false, + setShellModeActive: vi.fn(), + inputWidth: 80, + suggestionsWidth: 80, + focus: true, + }; + }); + + const wait = (ms = 50) => new Promise((resolve) => setTimeout(resolve, ms)); + + it('should call shellHistory.getPreviousCommand on up arrow in shell mode', async () => { + props.shellModeActive = true; + const { stdin, unmount } = render(<InputPrompt {...props} />); + await wait(); + + stdin.write('\u001B[A'); + await wait(); + + expect(mockShellHistory.getPreviousCommand).toHaveBeenCalled(); + unmount(); + }); + + it('should call shellHistory.getNextCommand on down arrow in shell mode', async () => { + props.shellModeActive = true; + const { stdin, unmount } = render(<InputPrompt {...props} />); + await wait(); + + stdin.write('\u001B[B'); + await wait(); + + expect(mockShellHistory.getNextCommand).toHaveBeenCalled(); + unmount(); + }); + + it('should set the buffer text when a shell history command is retrieved', async () => { + props.shellModeActive = true; + vi.mocked(mockShellHistory.getPreviousCommand).mockReturnValue( + 'previous command', + ); + const { stdin, unmount } = render(<InputPrompt {...props} />); + await wait(); + + stdin.write('\u001B[A'); + await wait(); + + expect(mockShellHistory.getPreviousCommand).toHaveBeenCalled(); + expect(props.buffer.setText).toHaveBeenCalledWith('previous command'); + unmount(); + }); + + it('should call shellHistory.addCommandToHistory on submit in shell mode', async () => { + props.shellModeActive = true; + props.buffer.setText('ls -l'); + const { stdin, unmount } = render(<InputPrompt {...props} />); + await wait(); + + stdin.write('\r'); + await wait(); + + expect(mockShellHistory.addCommandToHistory).toHaveBeenCalledWith('ls -l'); + expect(props.onSubmit).toHaveBeenCalledWith('ls -l'); + unmount(); + }); + + it('should NOT call shell history methods when not in shell mode', async () => { + props.buffer.setText('some text'); + const { stdin, unmount } = render(<InputPrompt {...props} />); + await wait(); + + stdin.write('\u001B[A'); // Up arrow + await wait(); + stdin.write('\u001B[B'); // Down arrow + await wait(); + stdin.write('\r'); // Enter + await wait(); + + expect(mockShellHistory.getPreviousCommand).not.toHaveBeenCalled(); + expect(mockShellHistory.getNextCommand).not.toHaveBeenCalled(); + expect(mockShellHistory.addCommandToHistory).not.toHaveBeenCalled(); + + expect(mockInputHistory.navigateUp).toHaveBeenCalled(); + expect(mockInputHistory.navigateDown).toHaveBeenCalled(); + expect(props.onSubmit).toHaveBeenCalledWith('some text'); + unmount(); + }); +}); |
