mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-03-13 07:30:52 -07:00
372 lines
9.3 KiB
TypeScript
372 lines
9.3 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
import { render } from '../../test-utils/render.js';
|
|
import { describe, it, expect, vi, beforeAll, afterAll } from 'vitest';
|
|
import { ModelStatsDisplay } from './ModelStatsDisplay.js';
|
|
import * as SessionContext from '../contexts/SessionContext.js';
|
|
import type { SessionMetrics } from '../contexts/SessionContext.js';
|
|
import { ToolCallDecision } from '@google/gemini-cli-core';
|
|
|
|
// Mock the context to provide controlled data for testing
|
|
vi.mock('../contexts/SessionContext.js', async (importOriginal) => {
|
|
const actual = await importOriginal<typeof SessionContext>();
|
|
return {
|
|
...actual,
|
|
useSessionStats: vi.fn(),
|
|
};
|
|
});
|
|
|
|
const useSessionStatsMock = vi.mocked(SessionContext.useSessionStats);
|
|
|
|
const renderWithMockedStats = (metrics: SessionMetrics, width?: number) => {
|
|
useSessionStatsMock.mockReturnValue({
|
|
stats: {
|
|
sessionId: 'test-session',
|
|
sessionStartTime: new Date(),
|
|
metrics,
|
|
lastPromptTokenCount: 0,
|
|
promptCount: 5,
|
|
},
|
|
|
|
getPromptCount: () => 5,
|
|
startNewPrompt: vi.fn(),
|
|
});
|
|
|
|
return render(<ModelStatsDisplay />, width);
|
|
};
|
|
|
|
describe('<ModelStatsDisplay />', () => {
|
|
beforeAll(() => {
|
|
vi.spyOn(Number.prototype, 'toLocaleString').mockImplementation(function (
|
|
this: number,
|
|
) {
|
|
// Use a stable 'en-US' format for test consistency.
|
|
return new Intl.NumberFormat('en-US').format(this);
|
|
});
|
|
});
|
|
|
|
afterAll(() => {
|
|
vi.restoreAllMocks();
|
|
});
|
|
|
|
it('should render "no API calls" message when there are no active models', () => {
|
|
const { lastFrame } = renderWithMockedStats({
|
|
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,
|
|
},
|
|
});
|
|
|
|
expect(lastFrame()).toContain(
|
|
'No API calls have been made in this session.',
|
|
);
|
|
expect(lastFrame()).toMatchSnapshot();
|
|
});
|
|
|
|
it('should not display conditional rows if no model has data for them', () => {
|
|
const { lastFrame } = renderWithMockedStats({
|
|
models: {
|
|
'gemini-2.5-pro': {
|
|
api: { totalRequests: 1, totalErrors: 0, totalLatencyMs: 100 },
|
|
tokens: {
|
|
input: 10,
|
|
prompt: 10,
|
|
candidates: 20,
|
|
total: 30,
|
|
cached: 0,
|
|
thoughts: 0,
|
|
tool: 0,
|
|
},
|
|
},
|
|
},
|
|
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,
|
|
},
|
|
});
|
|
|
|
const output = lastFrame();
|
|
expect(output).not.toContain('Cache Reads');
|
|
expect(output).not.toContain('Thoughts');
|
|
expect(output).not.toContain('Tool');
|
|
expect(output).toMatchSnapshot();
|
|
});
|
|
|
|
it('should display conditional rows if at least one model has data', () => {
|
|
const { lastFrame } = renderWithMockedStats({
|
|
models: {
|
|
'gemini-2.5-pro': {
|
|
api: { totalRequests: 1, totalErrors: 0, totalLatencyMs: 100 },
|
|
tokens: {
|
|
input: 5,
|
|
prompt: 10,
|
|
candidates: 20,
|
|
total: 30,
|
|
cached: 5,
|
|
thoughts: 2,
|
|
tool: 0,
|
|
},
|
|
},
|
|
'gemini-2.5-flash': {
|
|
api: { totalRequests: 1, totalErrors: 0, totalLatencyMs: 50 },
|
|
tokens: {
|
|
input: 5,
|
|
prompt: 5,
|
|
candidates: 10,
|
|
total: 15,
|
|
cached: 0,
|
|
thoughts: 0,
|
|
tool: 3,
|
|
},
|
|
},
|
|
},
|
|
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,
|
|
},
|
|
});
|
|
|
|
const output = lastFrame();
|
|
expect(output).toContain('Cache Reads');
|
|
expect(output).toContain('Thoughts');
|
|
expect(output).toContain('Tool');
|
|
expect(output).toMatchSnapshot();
|
|
});
|
|
|
|
it('should display stats for multiple models correctly', () => {
|
|
const { lastFrame } = renderWithMockedStats({
|
|
models: {
|
|
'gemini-2.5-pro': {
|
|
api: { totalRequests: 10, totalErrors: 1, totalLatencyMs: 1000 },
|
|
tokens: {
|
|
input: 50,
|
|
prompt: 100,
|
|
candidates: 200,
|
|
total: 300,
|
|
cached: 50,
|
|
thoughts: 10,
|
|
tool: 5,
|
|
},
|
|
},
|
|
'gemini-2.5-flash': {
|
|
api: { totalRequests: 20, totalErrors: 2, totalLatencyMs: 500 },
|
|
tokens: {
|
|
input: 100,
|
|
prompt: 200,
|
|
candidates: 400,
|
|
total: 600,
|
|
cached: 100,
|
|
thoughts: 20,
|
|
tool: 10,
|
|
},
|
|
},
|
|
},
|
|
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,
|
|
},
|
|
});
|
|
|
|
const output = lastFrame();
|
|
expect(output).toContain('gemini-2.5-pro');
|
|
expect(output).toContain('gemini-2.5-flash');
|
|
expect(output).toMatchSnapshot();
|
|
});
|
|
|
|
it('should handle large values without wrapping or overlapping', () => {
|
|
const { lastFrame } = renderWithMockedStats({
|
|
models: {
|
|
'gemini-2.5-pro': {
|
|
api: {
|
|
totalRequests: 999999999,
|
|
totalErrors: 123456789,
|
|
totalLatencyMs: 9876,
|
|
},
|
|
tokens: {
|
|
input: 987654321 - 123456789,
|
|
prompt: 987654321,
|
|
candidates: 123456789,
|
|
total: 999999999,
|
|
cached: 123456789,
|
|
thoughts: 111111111,
|
|
tool: 222222222,
|
|
},
|
|
},
|
|
},
|
|
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,
|
|
},
|
|
});
|
|
|
|
expect(lastFrame()).toMatchSnapshot();
|
|
});
|
|
|
|
it('should display a single model correctly', () => {
|
|
const { lastFrame } = renderWithMockedStats({
|
|
models: {
|
|
'gemini-2.5-pro': {
|
|
api: { totalRequests: 1, totalErrors: 0, totalLatencyMs: 100 },
|
|
tokens: {
|
|
input: 5,
|
|
prompt: 10,
|
|
candidates: 20,
|
|
total: 30,
|
|
cached: 5,
|
|
thoughts: 2,
|
|
tool: 1,
|
|
},
|
|
},
|
|
},
|
|
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,
|
|
},
|
|
});
|
|
|
|
const output = lastFrame();
|
|
expect(output).toContain('gemini-2.5-pro');
|
|
expect(output).not.toContain('gemini-2.5-flash');
|
|
expect(output).toMatchSnapshot();
|
|
});
|
|
|
|
it('should handle models with long names (gemini-3-*-preview) without layout breaking', () => {
|
|
const { lastFrame } = renderWithMockedStats(
|
|
{
|
|
models: {
|
|
'gemini-3-pro-preview': {
|
|
api: { totalRequests: 10, totalErrors: 0, totalLatencyMs: 2000 },
|
|
tokens: {
|
|
input: 1000,
|
|
prompt: 2000,
|
|
candidates: 4000,
|
|
total: 6000,
|
|
cached: 500,
|
|
thoughts: 100,
|
|
tool: 50,
|
|
},
|
|
},
|
|
'gemini-3-flash-preview': {
|
|
api: { totalRequests: 20, totalErrors: 0, totalLatencyMs: 1000 },
|
|
tokens: {
|
|
input: 2000,
|
|
prompt: 4000,
|
|
candidates: 8000,
|
|
total: 12000,
|
|
cached: 1000,
|
|
thoughts: 200,
|
|
tool: 100,
|
|
},
|
|
},
|
|
},
|
|
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,
|
|
},
|
|
},
|
|
80,
|
|
);
|
|
|
|
const output = lastFrame();
|
|
expect(output).toContain('gemini-3-pro-');
|
|
expect(output).toContain('gemini-3-flash-');
|
|
expect(output).toMatchSnapshot();
|
|
});
|
|
});
|