mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-04-20 18:14:29 -07:00
388 lines
11 KiB
TypeScript
388 lines
11 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
import { renderWithProviders } from '../../../test-utils/render.js';
|
|
import { createMockSettings } from '../../../test-utils/settings.js';
|
|
import { ToolResultDisplay } from './ToolResultDisplay.js';
|
|
import { describe, it, expect, vi } from 'vitest';
|
|
import { makeFakeConfig, type AnsiOutput } from '@google/gemini-cli-core';
|
|
|
|
describe('ToolResultDisplay', () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks();
|
|
});
|
|
|
|
it('uses ScrollableList for ANSI output in alternate buffer mode', async () => {
|
|
const content = 'ansi content';
|
|
const ansiResult: AnsiOutput = [
|
|
[
|
|
{
|
|
text: content,
|
|
fg: 'red',
|
|
bg: 'black',
|
|
bold: false,
|
|
italic: false,
|
|
underline: false,
|
|
dim: false,
|
|
inverse: false,
|
|
},
|
|
],
|
|
];
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay={ansiResult}
|
|
terminalWidth={80}
|
|
maxLines={10}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: true }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: true } }),
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
|
|
expect(output).toContain(content);
|
|
unmount();
|
|
});
|
|
|
|
it('uses Scrollable for non-ANSI output in alternate buffer mode', async () => {
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay="**Markdown content**"
|
|
terminalWidth={80}
|
|
maxLines={10}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: true }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: true } }),
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
|
|
// With real components, we check for the content itself
|
|
expect(output).toContain('Markdown content');
|
|
unmount();
|
|
});
|
|
|
|
it('passes hasFocus prop to scrollable components', async () => {
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay="Some result"
|
|
terminalWidth={80}
|
|
hasFocus={true}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: true }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: true } }),
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
|
|
expect(lastFrame()).toContain('Some result');
|
|
unmount();
|
|
});
|
|
|
|
it('renders string result as markdown by default', async () => {
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay resultDisplay="**Some result**" terminalWidth={80} />,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: false }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: false } }),
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
|
|
expect(output).toMatchSnapshot();
|
|
unmount();
|
|
});
|
|
|
|
it('renders string result as plain text when renderOutputAsMarkdown is false', async () => {
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay="**Some result**"
|
|
terminalWidth={80}
|
|
availableTerminalHeight={20}
|
|
renderOutputAsMarkdown={false}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: false }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: false } }),
|
|
uiState: { constrainHeight: true },
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
|
|
expect(output).toMatchSnapshot();
|
|
unmount();
|
|
});
|
|
|
|
it('truncates very long string results', { timeout: 20000 }, async () => {
|
|
const longString = 'a'.repeat(1000005);
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay={longString}
|
|
terminalWidth={80}
|
|
availableTerminalHeight={20}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: false }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: false } }),
|
|
uiState: { constrainHeight: true },
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
|
|
expect(output).toMatchSnapshot();
|
|
unmount();
|
|
});
|
|
|
|
it('renders file diff result', async () => {
|
|
const diffResult = {
|
|
fileDiff: 'diff content',
|
|
fileName: 'test.ts',
|
|
};
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay={diffResult}
|
|
terminalWidth={80}
|
|
availableTerminalHeight={20}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: false }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: false } }),
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
|
|
expect(output).toMatchSnapshot();
|
|
unmount();
|
|
});
|
|
|
|
it('renders ANSI output result', async () => {
|
|
const ansiResult: AnsiOutput = [
|
|
[
|
|
{
|
|
text: 'ansi content',
|
|
fg: 'red',
|
|
bg: 'black',
|
|
bold: false,
|
|
italic: false,
|
|
underline: false,
|
|
dim: false,
|
|
inverse: false,
|
|
},
|
|
],
|
|
];
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay={ansiResult as unknown as AnsiOutput}
|
|
terminalWidth={80}
|
|
availableTerminalHeight={20}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: false }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: false } }),
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
|
|
expect(output).toMatchSnapshot();
|
|
unmount();
|
|
});
|
|
|
|
it('renders nothing for todos result', async () => {
|
|
const todoResult = {
|
|
todos: [],
|
|
};
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay={todoResult}
|
|
terminalWidth={80}
|
|
availableTerminalHeight={20}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: false }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: false } }),
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame({ allowEmpty: true });
|
|
|
|
expect(output).toMatchSnapshot();
|
|
unmount();
|
|
});
|
|
|
|
it('does not fall back to plain text if availableHeight is set and not in alternate buffer', async () => {
|
|
// availableHeight calculation: 20 - 1 - 5 = 14 > 3
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay="**Some result**"
|
|
terminalWidth={80}
|
|
availableTerminalHeight={20}
|
|
renderOutputAsMarkdown={true}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: false }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: false } }),
|
|
uiState: { constrainHeight: true },
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
expect(output).toMatchSnapshot();
|
|
unmount();
|
|
});
|
|
|
|
it('keeps markdown if in alternate buffer even with availableHeight', async () => {
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay="**Some result**"
|
|
terminalWidth={80}
|
|
availableTerminalHeight={20}
|
|
renderOutputAsMarkdown={true}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: true }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: true } }),
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
|
|
expect(output).toMatchSnapshot();
|
|
unmount();
|
|
});
|
|
|
|
it('truncates ANSI output when maxLines is provided', async () => {
|
|
const ansiResult: AnsiOutput = [
|
|
[
|
|
{
|
|
text: 'Line 1',
|
|
fg: '',
|
|
bg: '',
|
|
bold: false,
|
|
italic: false,
|
|
underline: false,
|
|
dim: false,
|
|
inverse: false,
|
|
},
|
|
],
|
|
[
|
|
{
|
|
text: 'Line 2',
|
|
fg: '',
|
|
bg: '',
|
|
bold: false,
|
|
italic: false,
|
|
underline: false,
|
|
dim: false,
|
|
inverse: false,
|
|
},
|
|
],
|
|
[
|
|
{
|
|
text: 'Line 3',
|
|
fg: '',
|
|
bg: '',
|
|
bold: false,
|
|
italic: false,
|
|
underline: false,
|
|
dim: false,
|
|
inverse: false,
|
|
},
|
|
],
|
|
[
|
|
{
|
|
text: 'Line 4',
|
|
fg: '',
|
|
bg: '',
|
|
bold: false,
|
|
italic: false,
|
|
underline: false,
|
|
dim: false,
|
|
inverse: false,
|
|
},
|
|
],
|
|
[
|
|
{
|
|
text: 'Line 5',
|
|
fg: '',
|
|
bg: '',
|
|
bold: false,
|
|
italic: false,
|
|
underline: false,
|
|
dim: false,
|
|
inverse: false,
|
|
},
|
|
],
|
|
];
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay={ansiResult}
|
|
terminalWidth={80}
|
|
availableTerminalHeight={20}
|
|
maxLines={3}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: false }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: false } }),
|
|
uiState: { constrainHeight: true },
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
|
|
expect(output).not.toContain('Line 1');
|
|
expect(output).not.toContain('Line 2');
|
|
expect(output).not.toContain('Line 3');
|
|
expect(output).toContain('Line 4');
|
|
expect(output).toContain('Line 5');
|
|
unmount();
|
|
});
|
|
|
|
it('truncates ANSI output when maxLines is provided, even if availableTerminalHeight is undefined', async () => {
|
|
const ansiResult: AnsiOutput = Array.from({ length: 50 }, (_, i) => [
|
|
{
|
|
text: `Line ${i + 1}`,
|
|
fg: '',
|
|
bg: '',
|
|
bold: false,
|
|
italic: false,
|
|
underline: false,
|
|
dim: false,
|
|
inverse: false,
|
|
},
|
|
]);
|
|
const { lastFrame, waitUntilReady, unmount } = await renderWithProviders(
|
|
<ToolResultDisplay
|
|
resultDisplay={ansiResult}
|
|
terminalWidth={80}
|
|
maxLines={25}
|
|
availableTerminalHeight={undefined}
|
|
/>,
|
|
{
|
|
config: makeFakeConfig({ useAlternateBuffer: false }),
|
|
settings: createMockSettings({ ui: { useAlternateBuffer: false } }),
|
|
uiState: { constrainHeight: true },
|
|
},
|
|
);
|
|
await waitUntilReady();
|
|
const output = lastFrame();
|
|
|
|
// It SHOULD truncate to 25 lines because maxLines is provided
|
|
expect(output).not.toContain('Line 1');
|
|
expect(output).toContain('Line 50');
|
|
unmount();
|
|
});
|
|
});
|