2025-04-18 17:44:24 -07:00
|
|
|
/**
|
|
|
|
|
* @license
|
|
|
|
|
* Copyright 2025 Google LLC
|
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
*/
|
|
|
|
|
|
2025-04-15 21:41:08 -07:00
|
|
|
import React from 'react';
|
|
|
|
|
import { Box, Text } from 'ink';
|
|
|
|
|
import Spinner from 'ink-spinner';
|
2025-04-19 19:45:42 +01:00
|
|
|
import {
|
|
|
|
|
IndividualToolCallDisplay,
|
|
|
|
|
ToolCallStatus,
|
|
|
|
|
ToolCallConfirmationDetails,
|
|
|
|
|
ToolEditConfirmationDetails,
|
|
|
|
|
ToolExecuteConfirmationDetails,
|
|
|
|
|
} from '../../types.js';
|
2025-04-18 19:09:41 -04:00
|
|
|
import { DiffRenderer } from './DiffRenderer.js';
|
2025-04-19 19:45:42 +01:00
|
|
|
import { FileDiff, ToolResultDisplay } from '../../../tools/tools.js';
|
2025-04-15 21:41:08 -07:00
|
|
|
|
2025-04-19 19:45:42 +01:00
|
|
|
export const ToolMessage: React.FC<IndividualToolCallDisplay> = ({
|
|
|
|
|
callId,
|
2025-04-17 18:06:21 -04:00
|
|
|
name,
|
|
|
|
|
description,
|
|
|
|
|
resultDisplay,
|
|
|
|
|
status,
|
2025-04-19 19:45:42 +01:00
|
|
|
confirmationDetails,
|
2025-04-17 18:06:21 -04:00
|
|
|
}) => {
|
2025-04-19 19:45:42 +01:00
|
|
|
// Explicitly type the props to help the type checker
|
|
|
|
|
const typedConfirmationDetails = confirmationDetails as
|
|
|
|
|
| ToolCallConfirmationDetails
|
|
|
|
|
| undefined;
|
|
|
|
|
const typedResultDisplay = resultDisplay as ToolResultDisplay | undefined;
|
|
|
|
|
|
|
|
|
|
let color = 'gray';
|
|
|
|
|
let prefix = '';
|
|
|
|
|
switch (status) {
|
|
|
|
|
case ToolCallStatus.Pending:
|
|
|
|
|
prefix = 'Pending:';
|
|
|
|
|
break;
|
|
|
|
|
case ToolCallStatus.Invoked:
|
|
|
|
|
prefix = 'Executing:';
|
|
|
|
|
break;
|
|
|
|
|
case ToolCallStatus.Confirming:
|
|
|
|
|
color = 'yellow';
|
|
|
|
|
prefix = 'Confirm:';
|
|
|
|
|
break;
|
|
|
|
|
case ToolCallStatus.Success:
|
|
|
|
|
color = 'green';
|
|
|
|
|
prefix = 'Success:';
|
|
|
|
|
break;
|
|
|
|
|
case ToolCallStatus.Error:
|
|
|
|
|
color = 'red';
|
|
|
|
|
prefix = 'Error:';
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
// Handle unexpected status if necessary, or just break
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const title = `${prefix} ${name}`;
|
2025-04-15 21:41:08 -07:00
|
|
|
|
2025-04-17 18:06:21 -04:00
|
|
|
return (
|
2025-04-19 19:45:42 +01:00
|
|
|
<Box key={callId} borderStyle="round" paddingX={1} flexDirection="column">
|
|
|
|
|
<Box>
|
|
|
|
|
{status === ToolCallStatus.Invoked && (
|
|
|
|
|
<Box marginRight={1}>
|
|
|
|
|
<Text color="blue">
|
|
|
|
|
<Spinner type="dots" />
|
2025-04-17 18:06:21 -04:00
|
|
|
</Text>
|
2025-04-19 19:45:42 +01:00
|
|
|
</Box>
|
|
|
|
|
)}
|
|
|
|
|
<Text bold color={color}>
|
|
|
|
|
{title}
|
|
|
|
|
</Text>
|
|
|
|
|
<Text color={color}>
|
|
|
|
|
{status === ToolCallStatus.Error && typedResultDisplay
|
|
|
|
|
? `: ${typedResultDisplay}`
|
|
|
|
|
: ` - ${description}`}
|
|
|
|
|
</Text>
|
|
|
|
|
</Box>
|
|
|
|
|
{status === ToolCallStatus.Confirming && typedConfirmationDetails && (
|
|
|
|
|
<Box flexDirection="column" marginLeft={2}>
|
|
|
|
|
{/* Display diff for edit/write */}
|
|
|
|
|
{'fileDiff' in typedConfirmationDetails && (
|
|
|
|
|
<DiffRenderer
|
|
|
|
|
diffContent={
|
|
|
|
|
(typedConfirmationDetails as ToolEditConfirmationDetails)
|
|
|
|
|
.fileDiff
|
|
|
|
|
}
|
|
|
|
|
/>
|
2025-04-17 18:06:21 -04:00
|
|
|
)}
|
2025-04-19 19:45:42 +01:00
|
|
|
{/* Display command for execute */}
|
|
|
|
|
{'command' in typedConfirmationDetails && (
|
|
|
|
|
<Text color="yellow">
|
|
|
|
|
Command:{' '}
|
|
|
|
|
{
|
|
|
|
|
(typedConfirmationDetails as ToolExecuteConfirmationDetails)
|
|
|
|
|
.command
|
|
|
|
|
}
|
|
|
|
|
</Text>
|
|
|
|
|
)}
|
|
|
|
|
{/* <ConfirmInput onConfirm={handleConfirm} isFocused={isFocused} /> */}
|
2025-04-17 18:06:21 -04:00
|
|
|
</Box>
|
2025-04-19 19:45:42 +01:00
|
|
|
)}
|
|
|
|
|
{status === ToolCallStatus.Success && typedResultDisplay && (
|
|
|
|
|
<Box flexDirection="column" marginLeft={2}>
|
|
|
|
|
{typeof typedResultDisplay === 'string' ? (
|
|
|
|
|
<Text>{typedResultDisplay}</Text>
|
|
|
|
|
) : (
|
|
|
|
|
<DiffRenderer
|
|
|
|
|
diffContent={(typedResultDisplay as FileDiff).fileDiff}
|
|
|
|
|
/>
|
|
|
|
|
)}
|
2025-04-15 21:41:08 -07:00
|
|
|
</Box>
|
2025-04-17 18:06:21 -04:00
|
|
|
)}
|
|
|
|
|
</Box>
|
|
|
|
|
);
|
2025-04-15 21:41:08 -07:00
|
|
|
};
|