mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-05-24 05:04:28 +00:00
No intentional different behavior aside for tweaks suggested from the code review of #506 Refactor: Extract console message logic to custom hook This commit refactors the console message handling from App.tsx into a new custom hook useConsoleMessages. This change improves the testability of the console message logic and declutters the main App component. Created useConsoleMessages.ts to encapsulate console message state and update logic. Updated App.tsx to utilize the new useConsoleMessages hook. Added unit tests for useConsoleMessages.ts to ensure its functionality. I deleted and started over on LoadingIndicator.test.tsx as I spent way too much time trying to fix it before just regenerating the tests as the code was easier to write tests for from scratch and the existing tests were not that good (I added them in the previous pull request).
162 lines
4.7 KiB
TypeScript
162 lines
4.7 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
import React from 'react';
|
|
import { render } from 'ink-testing-library';
|
|
import { Text } from 'ink';
|
|
import { LoadingIndicator } from './LoadingIndicator.js';
|
|
import {
|
|
StreamingContext,
|
|
StreamingContextType,
|
|
} from '../contexts/StreamingContext.js';
|
|
import { StreamingState } from '../types.js';
|
|
import { vi } from 'vitest';
|
|
|
|
// Mock ink-spinner
|
|
vi.mock('ink-spinner', () => ({
|
|
default: () => <Text>MockSpinner</Text>,
|
|
}));
|
|
|
|
const renderWithContext = (
|
|
ui: React.ReactElement,
|
|
streamingStateValue: StreamingState,
|
|
) => {
|
|
const contextValue: StreamingContextType = {
|
|
streamingState: streamingStateValue,
|
|
};
|
|
return render(
|
|
<StreamingContext.Provider value={contextValue}>
|
|
{ui}
|
|
</StreamingContext.Provider>,
|
|
);
|
|
};
|
|
|
|
describe('<LoadingIndicator />', () => {
|
|
const defaultProps = {
|
|
currentLoadingPhrase: 'Loading...',
|
|
elapsedTime: 5,
|
|
};
|
|
|
|
it('should not render when streamingState is Idle', () => {
|
|
const { lastFrame } = renderWithContext(
|
|
<LoadingIndicator {...defaultProps} />,
|
|
StreamingState.Idle,
|
|
);
|
|
expect(lastFrame()).toBe('');
|
|
});
|
|
|
|
it('should render spinner, phrase, and time when streamingState is Responding', () => {
|
|
const { lastFrame } = renderWithContext(
|
|
<LoadingIndicator {...defaultProps} />,
|
|
StreamingState.Responding,
|
|
);
|
|
const output = lastFrame();
|
|
expect(output).toContain('MockSpinner');
|
|
expect(output).toContain('Loading...');
|
|
expect(output).toContain('(esc to cancel, 5s)');
|
|
});
|
|
|
|
it('should render phrase and time but no spinner when streamingState is WaitingForConfirmation', () => {
|
|
const props = {
|
|
currentLoadingPhrase: 'Confirm action',
|
|
elapsedTime: 10,
|
|
};
|
|
const { lastFrame } = renderWithContext(
|
|
<LoadingIndicator {...props} />,
|
|
StreamingState.WaitingForConfirmation,
|
|
);
|
|
const output = lastFrame();
|
|
expect(output).not.toContain('MockSpinner');
|
|
expect(output).toContain('Confirm action');
|
|
expect(output).not.toContain('(esc to cancel)');
|
|
expect(output).not.toContain(', 10s');
|
|
});
|
|
|
|
it('should display the currentLoadingPhrase correctly', () => {
|
|
const props = {
|
|
currentLoadingPhrase: 'Processing data...',
|
|
elapsedTime: 3,
|
|
};
|
|
const { lastFrame } = renderWithContext(
|
|
<LoadingIndicator {...props} />,
|
|
StreamingState.Responding,
|
|
);
|
|
expect(lastFrame()).toContain('Processing data...');
|
|
});
|
|
|
|
it('should display the elapsedTime correctly when Responding', () => {
|
|
const props = {
|
|
currentLoadingPhrase: 'Working...',
|
|
elapsedTime: 8,
|
|
};
|
|
const { lastFrame } = renderWithContext(
|
|
<LoadingIndicator {...props} />,
|
|
StreamingState.Responding,
|
|
);
|
|
expect(lastFrame()).toContain('(esc to cancel, 8s)');
|
|
});
|
|
|
|
it('should render rightContent when provided', () => {
|
|
const rightContent = <Text>Extra Info</Text>;
|
|
const { lastFrame } = renderWithContext(
|
|
<LoadingIndicator {...defaultProps} rightContent={rightContent} />,
|
|
StreamingState.Responding,
|
|
);
|
|
expect(lastFrame()).toContain('Extra Info');
|
|
});
|
|
|
|
it('should transition correctly between states using rerender', () => {
|
|
const { lastFrame, rerender } = renderWithContext(
|
|
<LoadingIndicator {...defaultProps} />,
|
|
StreamingState.Idle,
|
|
);
|
|
expect(lastFrame()).toBe(''); // Initial: Idle
|
|
|
|
// Transition to Responding
|
|
rerender(
|
|
<StreamingContext.Provider
|
|
value={{ streamingState: StreamingState.Responding }}
|
|
>
|
|
<LoadingIndicator
|
|
currentLoadingPhrase="Now Responding"
|
|
elapsedTime={2}
|
|
/>
|
|
</StreamingContext.Provider>,
|
|
);
|
|
let output = lastFrame();
|
|
expect(output).toContain('MockSpinner');
|
|
expect(output).toContain('Now Responding');
|
|
expect(output).toContain('(esc to cancel, 2s)');
|
|
|
|
// Transition to WaitingForConfirmation
|
|
rerender(
|
|
<StreamingContext.Provider
|
|
value={{ streamingState: StreamingState.WaitingForConfirmation }}
|
|
>
|
|
<LoadingIndicator
|
|
currentLoadingPhrase="Please Confirm"
|
|
elapsedTime={15}
|
|
/>
|
|
</StreamingContext.Provider>,
|
|
);
|
|
output = lastFrame();
|
|
expect(output).not.toContain('MockSpinner');
|
|
expect(output).toContain('Please Confirm');
|
|
expect(output).not.toContain('(esc to cancel)');
|
|
expect(output).not.toContain(', 15s');
|
|
|
|
// Transition back to Idle
|
|
rerender(
|
|
<StreamingContext.Provider
|
|
value={{ streamingState: StreamingState.Idle }}
|
|
>
|
|
<LoadingIndicator {...defaultProps} />
|
|
</StreamingContext.Provider>,
|
|
);
|
|
expect(lastFrame()).toBe('');
|
|
});
|
|
});
|