mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-03-20 11:00:40 -07:00
237 lines
7.0 KiB
TypeScript
237 lines
7.0 KiB
TypeScript
|
|
/**
|
||
|
|
* @license
|
||
|
|
* Copyright 2026 Google LLC
|
||
|
|
* SPDX-License-Identifier: Apache-2.0
|
||
|
|
*/
|
||
|
|
|
||
|
|
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
||
|
|
import { mapCoreStatusToDisplayStatus, mapToDisplay } from './toolMapping.js';
|
||
|
|
import {
|
||
|
|
debugLogger,
|
||
|
|
type AnyDeclarativeTool,
|
||
|
|
type AnyToolInvocation,
|
||
|
|
type ToolCallRequestInfo,
|
||
|
|
type ToolCallResponseInfo,
|
||
|
|
type Status,
|
||
|
|
type ToolCall,
|
||
|
|
type ScheduledToolCall,
|
||
|
|
type SuccessfulToolCall,
|
||
|
|
type ExecutingToolCall,
|
||
|
|
type WaitingToolCall,
|
||
|
|
type CancelledToolCall,
|
||
|
|
} from '@google/gemini-cli-core';
|
||
|
|
import { ToolCallStatus } from '../types.js';
|
||
|
|
|
||
|
|
vi.mock('@google/gemini-cli-core', async (importOriginal) => {
|
||
|
|
const actual =
|
||
|
|
await importOriginal<typeof import('@google/gemini-cli-core')>();
|
||
|
|
return {
|
||
|
|
...actual,
|
||
|
|
debugLogger: {
|
||
|
|
warn: vi.fn(),
|
||
|
|
},
|
||
|
|
};
|
||
|
|
});
|
||
|
|
|
||
|
|
describe('toolMapping', () => {
|
||
|
|
beforeEach(() => {
|
||
|
|
vi.clearAllMocks();
|
||
|
|
});
|
||
|
|
|
||
|
|
describe('mapCoreStatusToDisplayStatus', () => {
|
||
|
|
it.each([
|
||
|
|
['validating', ToolCallStatus.Executing],
|
||
|
|
['awaiting_approval', ToolCallStatus.Confirming],
|
||
|
|
['executing', ToolCallStatus.Executing],
|
||
|
|
['success', ToolCallStatus.Success],
|
||
|
|
['cancelled', ToolCallStatus.Canceled],
|
||
|
|
['error', ToolCallStatus.Error],
|
||
|
|
['scheduled', ToolCallStatus.Pending],
|
||
|
|
] as const)('maps %s to %s', (coreStatus, expectedDisplayStatus) => {
|
||
|
|
expect(mapCoreStatusToDisplayStatus(coreStatus)).toBe(
|
||
|
|
expectedDisplayStatus,
|
||
|
|
);
|
||
|
|
});
|
||
|
|
|
||
|
|
it('logs warning and defaults to Error for unknown status', () => {
|
||
|
|
const result = mapCoreStatusToDisplayStatus('unknown_status' as Status);
|
||
|
|
expect(result).toBe(ToolCallStatus.Error);
|
||
|
|
expect(debugLogger.warn).toHaveBeenCalledWith(
|
||
|
|
'Unknown core status encountered: unknown_status',
|
||
|
|
);
|
||
|
|
});
|
||
|
|
});
|
||
|
|
|
||
|
|
describe('mapToDisplay', () => {
|
||
|
|
const mockRequest: ToolCallRequestInfo = {
|
||
|
|
callId: 'call-1',
|
||
|
|
name: 'test_tool',
|
||
|
|
args: { arg1: 'val1' },
|
||
|
|
isClientInitiated: false,
|
||
|
|
prompt_id: 'p1',
|
||
|
|
};
|
||
|
|
|
||
|
|
const mockTool = {
|
||
|
|
name: 'test_tool',
|
||
|
|
displayName: 'Test Tool',
|
||
|
|
isOutputMarkdown: true,
|
||
|
|
} as unknown as AnyDeclarativeTool;
|
||
|
|
|
||
|
|
const mockInvocation = {
|
||
|
|
getDescription: () => 'Calling test_tool with args...',
|
||
|
|
} as unknown as AnyToolInvocation;
|
||
|
|
|
||
|
|
const mockResponse: ToolCallResponseInfo = {
|
||
|
|
callId: 'call-1',
|
||
|
|
responseParts: [],
|
||
|
|
resultDisplay: 'Success output',
|
||
|
|
error: undefined,
|
||
|
|
errorType: undefined,
|
||
|
|
};
|
||
|
|
|
||
|
|
it('handles a single tool call input', () => {
|
||
|
|
const toolCall: ScheduledToolCall = {
|
||
|
|
status: 'scheduled',
|
||
|
|
request: mockRequest,
|
||
|
|
tool: mockTool,
|
||
|
|
invocation: mockInvocation,
|
||
|
|
};
|
||
|
|
|
||
|
|
const result = mapToDisplay(toolCall);
|
||
|
|
expect(result.type).toBe('tool_group');
|
||
|
|
expect(result.tools).toHaveLength(1);
|
||
|
|
expect(result.tools[0]?.callId).toBe('call-1');
|
||
|
|
});
|
||
|
|
|
||
|
|
it('handles an array of tool calls', () => {
|
||
|
|
const toolCall1: ScheduledToolCall = {
|
||
|
|
status: 'scheduled',
|
||
|
|
request: mockRequest,
|
||
|
|
tool: mockTool,
|
||
|
|
invocation: mockInvocation,
|
||
|
|
};
|
||
|
|
const toolCall2: ScheduledToolCall = {
|
||
|
|
status: 'scheduled',
|
||
|
|
request: { ...mockRequest, callId: 'call-2' },
|
||
|
|
tool: mockTool,
|
||
|
|
invocation: mockInvocation,
|
||
|
|
};
|
||
|
|
|
||
|
|
const result = mapToDisplay([toolCall1, toolCall2]);
|
||
|
|
expect(result.tools).toHaveLength(2);
|
||
|
|
expect(result.tools[0]?.callId).toBe('call-1');
|
||
|
|
expect(result.tools[1]?.callId).toBe('call-2');
|
||
|
|
});
|
||
|
|
|
||
|
|
it('maps successful tool call properties correctly', () => {
|
||
|
|
const toolCall: SuccessfulToolCall = {
|
||
|
|
status: 'success',
|
||
|
|
request: mockRequest,
|
||
|
|
tool: mockTool,
|
||
|
|
invocation: mockInvocation,
|
||
|
|
response: {
|
||
|
|
...mockResponse,
|
||
|
|
outputFile: '/tmp/output.txt',
|
||
|
|
},
|
||
|
|
};
|
||
|
|
|
||
|
|
const result = mapToDisplay(toolCall);
|
||
|
|
const displayTool = result.tools[0];
|
||
|
|
|
||
|
|
expect(displayTool).toEqual(
|
||
|
|
expect.objectContaining({
|
||
|
|
callId: 'call-1',
|
||
|
|
name: 'Test Tool',
|
||
|
|
description: 'Calling test_tool with args...',
|
||
|
|
renderOutputAsMarkdown: true,
|
||
|
|
status: ToolCallStatus.Success,
|
||
|
|
resultDisplay: 'Success output',
|
||
|
|
outputFile: '/tmp/output.txt',
|
||
|
|
}),
|
||
|
|
);
|
||
|
|
});
|
||
|
|
|
||
|
|
it('maps executing tool call properties correctly with live output and ptyId', () => {
|
||
|
|
const toolCall: ExecutingToolCall = {
|
||
|
|
status: 'executing',
|
||
|
|
request: mockRequest,
|
||
|
|
tool: mockTool,
|
||
|
|
invocation: mockInvocation,
|
||
|
|
liveOutput: 'Loading...',
|
||
|
|
pid: 12345,
|
||
|
|
};
|
||
|
|
|
||
|
|
const result = mapToDisplay(toolCall);
|
||
|
|
const displayTool = result.tools[0];
|
||
|
|
|
||
|
|
expect(displayTool.status).toBe(ToolCallStatus.Executing);
|
||
|
|
expect(displayTool.resultDisplay).toBe('Loading...');
|
||
|
|
expect(displayTool.ptyId).toBe(12345);
|
||
|
|
});
|
||
|
|
|
||
|
|
it('maps awaiting_approval tool call properties with correlationId', () => {
|
||
|
|
const confirmationDetails = {
|
||
|
|
type: 'exec' as const,
|
||
|
|
title: 'Confirm Exec',
|
||
|
|
command: 'ls',
|
||
|
|
rootCommand: 'ls',
|
||
|
|
rootCommands: ['ls'],
|
||
|
|
onConfirm: vi.fn(),
|
||
|
|
};
|
||
|
|
|
||
|
|
const toolCall: WaitingToolCall = {
|
||
|
|
status: 'awaiting_approval',
|
||
|
|
request: mockRequest,
|
||
|
|
tool: mockTool,
|
||
|
|
invocation: mockInvocation,
|
||
|
|
confirmationDetails,
|
||
|
|
correlationId: 'corr-id-123',
|
||
|
|
};
|
||
|
|
|
||
|
|
const result = mapToDisplay(toolCall);
|
||
|
|
const displayTool = result.tools[0];
|
||
|
|
|
||
|
|
expect(displayTool.status).toBe(ToolCallStatus.Confirming);
|
||
|
|
expect(displayTool.confirmationDetails).toEqual(confirmationDetails);
|
||
|
|
});
|
||
|
|
|
||
|
|
it('maps error tool call missing tool definition', () => {
|
||
|
|
// e.g. "TOOL_NOT_REGISTERED" errors
|
||
|
|
const toolCall: ToolCall = {
|
||
|
|
status: 'error',
|
||
|
|
request: mockRequest, // name: 'test_tool'
|
||
|
|
response: { ...mockResponse, resultDisplay: 'Tool not found' },
|
||
|
|
// notice: no `tool` or `invocation` defined here
|
||
|
|
};
|
||
|
|
|
||
|
|
const result = mapToDisplay(toolCall);
|
||
|
|
const displayTool = result.tools[0];
|
||
|
|
|
||
|
|
expect(displayTool.status).toBe(ToolCallStatus.Error);
|
||
|
|
expect(displayTool.name).toBe('test_tool'); // falls back to request.name
|
||
|
|
expect(displayTool.description).toBe('{"arg1":"val1"}'); // falls back to stringified args
|
||
|
|
expect(displayTool.resultDisplay).toBe('Tool not found');
|
||
|
|
expect(displayTool.renderOutputAsMarkdown).toBe(false);
|
||
|
|
});
|
||
|
|
|
||
|
|
it('maps cancelled tool call properties correctly', () => {
|
||
|
|
const toolCall: CancelledToolCall = {
|
||
|
|
status: 'cancelled',
|
||
|
|
request: mockRequest,
|
||
|
|
tool: mockTool,
|
||
|
|
invocation: mockInvocation,
|
||
|
|
response: {
|
||
|
|
...mockResponse,
|
||
|
|
resultDisplay: 'User cancelled', // Could be diff output for edits
|
||
|
|
},
|
||
|
|
};
|
||
|
|
|
||
|
|
const result = mapToDisplay(toolCall);
|
||
|
|
const displayTool = result.tools[0];
|
||
|
|
|
||
|
|
expect(displayTool.status).toBe(ToolCallStatus.Canceled);
|
||
|
|
expect(displayTool.resultDisplay).toBe('User cancelled');
|
||
|
|
});
|
||
|
|
});
|
||
|
|
});
|