mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-04-15 07:41:03 -07:00
235 lines
6.5 KiB
TypeScript
235 lines
6.5 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
import { renderWithProviders } from '../../test-utils/render.js';
|
|
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
import { SessionSummaryDisplay } from './SessionSummaryDisplay.js';
|
|
import * as SessionContext from '../contexts/SessionContext.js';
|
|
import { useConfig } from '../contexts/ConfigContext.js';
|
|
import { type SessionMetrics } from '../contexts/SessionContext.js';
|
|
import {
|
|
ToolCallDecision,
|
|
getShellConfiguration,
|
|
type WorktreeSettings,
|
|
} from '@google/gemini-cli-core';
|
|
|
|
vi.mock('@google/gemini-cli-core', async (importOriginal) => {
|
|
const actual =
|
|
await importOriginal<typeof import('@google/gemini-cli-core')>();
|
|
return {
|
|
...actual,
|
|
getShellConfiguration: vi.fn(),
|
|
};
|
|
});
|
|
|
|
vi.mock('../contexts/SessionContext.js', async (importOriginal) => {
|
|
const actual =
|
|
await importOriginal<typeof import('../contexts/SessionContext.js')>();
|
|
return {
|
|
...actual,
|
|
useSessionStats: vi.fn(),
|
|
};
|
|
});
|
|
|
|
vi.mock('../contexts/ConfigContext.js', async (importOriginal) => {
|
|
const actual =
|
|
await importOriginal<typeof import('../contexts/ConfigContext.js')>();
|
|
return {
|
|
...actual,
|
|
useConfig: vi.fn(),
|
|
};
|
|
});
|
|
|
|
const getShellConfigurationMock = vi.mocked(getShellConfiguration);
|
|
const useSessionStatsMock = vi.mocked(SessionContext.useSessionStats);
|
|
|
|
const renderWithMockedStats = async (
|
|
metrics: SessionMetrics,
|
|
sessionId = 'test-session',
|
|
worktreeSettings?: WorktreeSettings,
|
|
) => {
|
|
useSessionStatsMock.mockReturnValue({
|
|
stats: {
|
|
sessionId,
|
|
sessionStartTime: new Date(),
|
|
metrics,
|
|
lastPromptTokenCount: 0,
|
|
promptCount: 5,
|
|
},
|
|
|
|
getPromptCount: () => 5,
|
|
startNewPrompt: vi.fn(),
|
|
} as unknown as ReturnType<typeof SessionContext.useSessionStats>);
|
|
|
|
vi.mocked(useConfig).mockReturnValue({
|
|
getWorktreeSettings: () => worktreeSettings,
|
|
} as never);
|
|
|
|
const result = await renderWithProviders(
|
|
<SessionSummaryDisplay duration="1h 23m 45s" />,
|
|
{
|
|
width: 100,
|
|
},
|
|
);
|
|
await result.waitUntilReady();
|
|
return result;
|
|
};
|
|
|
|
describe('<SessionSummaryDisplay />', () => {
|
|
const emptyMetrics: SessionMetrics = {
|
|
models: {},
|
|
tools: {
|
|
totalCalls: 0,
|
|
totalSuccess: 0,
|
|
totalFail: 0,
|
|
totalDurationMs: 0,
|
|
totalDecisions: {
|
|
accept: 0,
|
|
reject: 0,
|
|
modify: 0,
|
|
[ToolCallDecision.AUTO_ACCEPT]: 0,
|
|
},
|
|
byName: {},
|
|
},
|
|
files: {
|
|
totalLinesAdded: 0,
|
|
totalLinesRemoved: 0,
|
|
},
|
|
};
|
|
|
|
beforeEach(() => {
|
|
getShellConfigurationMock.mockReturnValue({
|
|
executable: 'bash',
|
|
argsPrefix: ['-c'],
|
|
shell: 'bash',
|
|
});
|
|
});
|
|
|
|
it('renders the summary display with a title', async () => {
|
|
const metrics: SessionMetrics = {
|
|
...emptyMetrics,
|
|
models: {
|
|
'gemini-2.5-pro': {
|
|
api: { totalRequests: 10, totalErrors: 1, totalLatencyMs: 50234 },
|
|
tokens: {
|
|
input: 500,
|
|
prompt: 1000,
|
|
candidates: 2000,
|
|
total: 3500,
|
|
cached: 500,
|
|
thoughts: 300,
|
|
tool: 200,
|
|
},
|
|
roles: {},
|
|
},
|
|
},
|
|
files: {
|
|
totalLinesAdded: 42,
|
|
totalLinesRemoved: 15,
|
|
},
|
|
};
|
|
|
|
const { lastFrame, unmount } = await renderWithMockedStats(metrics);
|
|
const output = lastFrame();
|
|
|
|
expect(output).toContain('Agent powering down. Goodbye!');
|
|
expect(output).toMatchSnapshot();
|
|
unmount();
|
|
});
|
|
|
|
describe('Session ID escaping', () => {
|
|
it('renders a standard UUID-formatted session ID in the footer (bash)', async () => {
|
|
const uuidSessionId = '1234-abcd-5678-efgh';
|
|
const { lastFrame, unmount } = await renderWithMockedStats(
|
|
emptyMetrics,
|
|
uuidSessionId,
|
|
);
|
|
const output = lastFrame();
|
|
|
|
// Standard UUID characters should not be escaped/quoted by default for bash.
|
|
expect(output).toContain('gemini --resume 1234-abcd-5678-efgh');
|
|
unmount();
|
|
});
|
|
|
|
it('sanitizes a malicious session ID in the footer (bash)', async () => {
|
|
const maliciousSessionId = "'; rm -rf / #";
|
|
const { lastFrame, unmount } = await renderWithMockedStats(
|
|
emptyMetrics,
|
|
maliciousSessionId,
|
|
);
|
|
const output = lastFrame();
|
|
|
|
// escapeShellArg (using shell-quote for bash) will wrap special characters in double quotes.
|
|
expect(output).toContain('gemini --resume "\'; rm -rf / #"');
|
|
unmount();
|
|
});
|
|
|
|
it('renders a standard UUID-formatted session ID in the footer (powershell)', async () => {
|
|
getShellConfigurationMock.mockReturnValue({
|
|
executable: 'powershell.exe',
|
|
argsPrefix: ['-NoProfile', '-Command'],
|
|
shell: 'powershell',
|
|
});
|
|
|
|
const uuidSessionId = '1234-abcd-5678-efgh';
|
|
const { lastFrame, unmount } = await renderWithMockedStats(
|
|
emptyMetrics,
|
|
uuidSessionId,
|
|
);
|
|
const output = lastFrame();
|
|
|
|
// PowerShell wraps strings in single quotes
|
|
expect(output).toContain("gemini --resume '1234-abcd-5678-efgh'");
|
|
unmount();
|
|
});
|
|
|
|
it('sanitizes a malicious session ID in the footer (powershell)', async () => {
|
|
getShellConfigurationMock.mockReturnValue({
|
|
executable: 'powershell.exe',
|
|
argsPrefix: ['-NoProfile', '-Command'],
|
|
shell: 'powershell',
|
|
});
|
|
|
|
const maliciousSessionId = "'; rm -rf / #";
|
|
const { lastFrame, unmount } = await renderWithMockedStats(
|
|
emptyMetrics,
|
|
maliciousSessionId,
|
|
);
|
|
const output = lastFrame();
|
|
|
|
// PowerShell wraps in single quotes and escapes internal single quotes by doubling them
|
|
expect(output).toContain("gemini --resume '''; rm -rf / #'");
|
|
unmount();
|
|
});
|
|
});
|
|
|
|
describe('Worktree status', () => {
|
|
it('renders worktree instructions when worktreeSettings are present', async () => {
|
|
const worktreeSettings: WorktreeSettings = {
|
|
name: 'foo-bar',
|
|
path: '/path/to/foo-bar',
|
|
baseSha: 'base-sha',
|
|
};
|
|
|
|
const { lastFrame, unmount } = await renderWithMockedStats(
|
|
emptyMetrics,
|
|
'test-session',
|
|
worktreeSettings,
|
|
);
|
|
const output = lastFrame();
|
|
|
|
expect(output).toContain('To resume work in this worktree:');
|
|
expect(output).toContain(
|
|
'cd /path/to/foo-bar && gemini --resume test-session',
|
|
);
|
|
expect(output).toContain(
|
|
'To remove manually: git worktree remove /path/to/foo-bar',
|
|
);
|
|
unmount();
|
|
});
|
|
});
|
|
});
|