
- Fixed TypeScript errors in ClaudeCodeSession.tsx: - Removed unused imports (Square, PreviewPromptDialog, etc.) - Removed unused handleOpenPreview function - Fixed unused detectedUrl state variable - Fixed TypeScript error in StreamMessage.tsx: - Removed unused useMemo import - Fixed TypeScript errors in ToolWidgets.tsx: - Prefixed unused result props with underscore in multiple widgets - Fixed Rust warnings: - Removed unused imports in commands modules - Prefixed unused variables with underscore - Added #[allow(dead_code)] for API methods intended for future use Closes #31 Closes #23 Closes #21 Closes #22
681 lines
30 KiB
TypeScript
681 lines
30 KiB
TypeScript
import React, { useState, useEffect } from "react";
|
|
import {
|
|
Terminal,
|
|
User,
|
|
Bot,
|
|
AlertCircle,
|
|
CheckCircle2
|
|
} from "lucide-react";
|
|
import { Card, CardContent } from "@/components/ui/card";
|
|
import { cn } from "@/lib/utils";
|
|
import ReactMarkdown from "react-markdown";
|
|
import remarkGfm from "remark-gfm";
|
|
import { Prism as SyntaxHighlighter } from "react-syntax-highlighter";
|
|
import { claudeSyntaxTheme } from "@/lib/claudeSyntaxTheme";
|
|
import type { ClaudeStreamMessage } from "./AgentExecution";
|
|
import {
|
|
TodoWidget,
|
|
LSWidget,
|
|
ReadWidget,
|
|
ReadResultWidget,
|
|
GlobWidget,
|
|
BashWidget,
|
|
WriteWidget,
|
|
GrepWidget,
|
|
EditWidget,
|
|
EditResultWidget,
|
|
MCPWidget,
|
|
CommandWidget,
|
|
CommandOutputWidget,
|
|
SummaryWidget,
|
|
MultiEditWidget,
|
|
MultiEditResultWidget,
|
|
SystemReminderWidget,
|
|
SystemInitializedWidget,
|
|
TaskWidget,
|
|
LSResultWidget
|
|
} from "./ToolWidgets";
|
|
|
|
interface StreamMessageProps {
|
|
message: ClaudeStreamMessage;
|
|
className?: string;
|
|
streamMessages: ClaudeStreamMessage[];
|
|
onLinkDetected?: (url: string) => void;
|
|
}
|
|
|
|
/**
|
|
* Component to render a single Claude Code stream message
|
|
*/
|
|
const StreamMessageComponent: React.FC<StreamMessageProps> = ({ message, className, streamMessages, onLinkDetected }) => {
|
|
// State to track tool results mapped by tool call ID
|
|
const [toolResults, setToolResults] = useState<Map<string, any>>(new Map());
|
|
|
|
// Extract all tool results from stream messages
|
|
useEffect(() => {
|
|
const results = new Map<string, any>();
|
|
|
|
// Iterate through all messages to find tool results
|
|
streamMessages.forEach(msg => {
|
|
if (msg.type === "user" && msg.message?.content && Array.isArray(msg.message.content)) {
|
|
msg.message.content.forEach((content: any) => {
|
|
if (content.type === "tool_result" && content.tool_use_id) {
|
|
results.set(content.tool_use_id, content);
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
setToolResults(results);
|
|
}, [streamMessages]);
|
|
|
|
// Helper to get tool result for a specific tool call ID
|
|
const getToolResult = (toolId: string | undefined): any => {
|
|
if (!toolId) return null;
|
|
return toolResults.get(toolId) || null;
|
|
};
|
|
try {
|
|
// Skip rendering for meta messages that don't have meaningful content
|
|
if (message.isMeta && !message.leafUuid && !message.summary) {
|
|
return null;
|
|
}
|
|
|
|
// Handle summary messages
|
|
if (message.leafUuid && message.summary && (message as any).type === "summary") {
|
|
return <SummaryWidget summary={message.summary} leafUuid={message.leafUuid} />;
|
|
}
|
|
|
|
// System initialization message
|
|
if (message.type === "system" && message.subtype === "init") {
|
|
return (
|
|
<SystemInitializedWidget
|
|
sessionId={message.session_id}
|
|
model={message.model}
|
|
cwd={message.cwd}
|
|
tools={message.tools}
|
|
/>
|
|
);
|
|
}
|
|
|
|
// Assistant message
|
|
if (message.type === "assistant" && message.message) {
|
|
const msg = message.message;
|
|
|
|
return (
|
|
<Card className={cn("border-primary/20 bg-primary/5", className)}>
|
|
<CardContent className="p-4">
|
|
<div className="flex items-start gap-3">
|
|
<Bot className="h-5 w-5 text-primary mt-0.5" />
|
|
<div className="flex-1 space-y-2 min-w-0">
|
|
{msg.content && Array.isArray(msg.content) && msg.content.map((content: any, idx: number) => {
|
|
// Text content - render as markdown
|
|
if (content.type === "text") {
|
|
// Ensure we have a string to render
|
|
const textContent = typeof content.text === 'string'
|
|
? content.text
|
|
: (content.text?.text || JSON.stringify(content.text || content));
|
|
|
|
return (
|
|
<div key={idx} className="prose prose-sm dark:prose-invert max-w-none">
|
|
<ReactMarkdown
|
|
remarkPlugins={[remarkGfm]}
|
|
components={{
|
|
code({ node, inline, className, children, ...props }: any) {
|
|
const match = /language-(\w+)/.exec(className || '');
|
|
return !inline && match ? (
|
|
<SyntaxHighlighter
|
|
style={claudeSyntaxTheme}
|
|
language={match[1]}
|
|
PreTag="div"
|
|
{...props}
|
|
>
|
|
{String(children).replace(/\n$/, '')}
|
|
</SyntaxHighlighter>
|
|
) : (
|
|
<code className={className} {...props}>
|
|
{children}
|
|
</code>
|
|
);
|
|
}
|
|
}}
|
|
>
|
|
{textContent}
|
|
</ReactMarkdown>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
// Tool use - render custom widgets based on tool name
|
|
if (content.type === "tool_use") {
|
|
const toolName = content.name?.toLowerCase();
|
|
const input = content.input;
|
|
const toolId = content.id;
|
|
|
|
// Get the tool result if available
|
|
const toolResult = getToolResult(toolId);
|
|
|
|
// Function to render the appropriate tool widget
|
|
const renderToolWidget = () => {
|
|
// Task tool - for sub-agent tasks
|
|
if (toolName === "task" && input) {
|
|
return <TaskWidget description={input.description} prompt={input.prompt} result={toolResult} />;
|
|
}
|
|
|
|
// Edit tool
|
|
if (toolName === "edit" && input?.file_path) {
|
|
return <EditWidget {...input} result={toolResult} />;
|
|
}
|
|
|
|
// MultiEdit tool
|
|
if (toolName === "multiedit" && input?.file_path && input?.edits) {
|
|
return <MultiEditWidget {...input} result={toolResult} />;
|
|
}
|
|
|
|
// MCP tools (starting with mcp__)
|
|
if (content.name?.startsWith("mcp__")) {
|
|
return <MCPWidget toolName={content.name} input={input} result={toolResult} />;
|
|
}
|
|
|
|
// TodoWrite tool
|
|
if (toolName === "todowrite" && input?.todos) {
|
|
return <TodoWidget todos={input.todos} result={toolResult} />;
|
|
}
|
|
|
|
// LS tool
|
|
if (toolName === "ls" && input?.path) {
|
|
return <LSWidget path={input.path} result={toolResult} />;
|
|
}
|
|
|
|
// Read tool
|
|
if (toolName === "read" && input?.file_path) {
|
|
return <ReadWidget filePath={input.file_path} result={toolResult} />;
|
|
}
|
|
|
|
// Glob tool
|
|
if (toolName === "glob" && input?.pattern) {
|
|
return <GlobWidget pattern={input.pattern} result={toolResult} />;
|
|
}
|
|
|
|
// Bash tool
|
|
if (toolName === "bash" && input?.command) {
|
|
return <BashWidget command={input.command} description={input.description} result={toolResult} />;
|
|
}
|
|
|
|
// Write tool
|
|
if (toolName === "write" && input?.file_path && input?.content) {
|
|
return <WriteWidget filePath={input.file_path} content={input.content} result={toolResult} />;
|
|
}
|
|
|
|
// Grep tool
|
|
if (toolName === "grep" && input?.pattern) {
|
|
return <GrepWidget pattern={input.pattern} include={input.include} path={input.path} exclude={input.exclude} result={toolResult} />;
|
|
}
|
|
|
|
// Default - return null
|
|
return null;
|
|
};
|
|
|
|
// Render the tool widget
|
|
const widget = renderToolWidget();
|
|
if (widget) {
|
|
return <div key={idx}>{widget}</div>;
|
|
}
|
|
|
|
// Fallback to basic tool display
|
|
return (
|
|
<div key={idx} className="space-y-2">
|
|
<div className="flex items-center gap-2">
|
|
<Terminal className="h-4 w-4 text-muted-foreground" />
|
|
<span className="text-sm font-medium">
|
|
Using tool: <code className="font-mono">{content.name}</code>
|
|
</span>
|
|
</div>
|
|
{content.input && (
|
|
<div className="ml-6 p-2 bg-background rounded-md border">
|
|
<pre className="text-xs font-mono overflow-x-auto">
|
|
{JSON.stringify(content.input, null, 2)}
|
|
</pre>
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return null;
|
|
})}
|
|
{msg.usage && (
|
|
<div className="text-xs text-muted-foreground mt-2">
|
|
Tokens: {msg.usage.input_tokens} in, {msg.usage.output_tokens} out
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
);
|
|
}
|
|
|
|
// User message
|
|
if (message.type === "user" && message.message) {
|
|
// Don't render meta messages, which are for system use
|
|
if (message.isMeta) return null;
|
|
|
|
const msg = message.message;
|
|
|
|
// Skip empty user messages
|
|
if (!msg.content || (Array.isArray(msg.content) && msg.content.length === 0)) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<Card className={cn("border-muted-foreground/20 bg-muted/20", className)}>
|
|
<CardContent className="p-4">
|
|
<div className="flex items-start gap-3">
|
|
<User className="h-5 w-5 text-muted-foreground mt-0.5" />
|
|
<div className="flex-1 space-y-2 min-w-0">
|
|
{/* Handle content that is a simple string (e.g. from user commands) */}
|
|
{typeof msg.content === 'string' && (
|
|
(() => {
|
|
const contentStr = msg.content as string;
|
|
|
|
// Check if it's a command message
|
|
const commandMatch = contentStr.match(/<command-name>(.+?)<\/command-name>[\s\S]*?<command-message>(.+?)<\/command-message>[\s\S]*?<command-args>(.*?)<\/command-args>/);
|
|
if (commandMatch) {
|
|
const [, commandName, commandMessage, commandArgs] = commandMatch;
|
|
return (
|
|
<CommandWidget
|
|
commandName={commandName.trim()}
|
|
commandMessage={commandMessage.trim()}
|
|
commandArgs={commandArgs?.trim()}
|
|
/>
|
|
);
|
|
}
|
|
|
|
// Check if it's command output
|
|
const stdoutMatch = contentStr.match(/<local-command-stdout>([\s\S]*?)<\/local-command-stdout>/);
|
|
if (stdoutMatch) {
|
|
const [, output] = stdoutMatch;
|
|
return <CommandOutputWidget output={output} onLinkDetected={onLinkDetected} />;
|
|
}
|
|
|
|
// Otherwise render as plain text
|
|
return (
|
|
<pre className="text-sm font-mono whitespace-pre-wrap text-muted-foreground">
|
|
{contentStr}
|
|
</pre>
|
|
);
|
|
})()
|
|
)}
|
|
|
|
{/* Handle content that is an array of parts */}
|
|
{Array.isArray(msg.content) && msg.content.map((content: any, idx: number) => {
|
|
// Tool result
|
|
if (content.type === "tool_result") {
|
|
// Skip rendering tool results that are already displayed by tool widgets
|
|
// We need to check if this result corresponds to a tool that has its own widget
|
|
|
|
// Find the corresponding tool use for this result
|
|
let hasCorrespondingWidget = false;
|
|
if (content.tool_use_id && streamMessages) {
|
|
// Look for the matching tool_use in previous assistant messages
|
|
for (let i = streamMessages.length - 1; i >= 0; i--) {
|
|
const prevMsg = streamMessages[i];
|
|
if (prevMsg.type === 'assistant' && prevMsg.message?.content && Array.isArray(prevMsg.message.content)) {
|
|
const toolUse = prevMsg.message.content.find((c: any) =>
|
|
c.type === 'tool_use' && c.id === content.tool_use_id
|
|
);
|
|
if (toolUse) {
|
|
const toolName = toolUse.name?.toLowerCase();
|
|
// List of tools that display their own results
|
|
const toolsWithWidgets = [
|
|
'task', 'edit', 'multiedit', 'todowrite', 'ls', 'read',
|
|
'glob', 'bash', 'write', 'grep'
|
|
];
|
|
// Also check for MCP tools
|
|
if (toolsWithWidgets.includes(toolName) || toolUse.name?.startsWith('mcp__')) {
|
|
hasCorrespondingWidget = true;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// If the tool has its own widget that displays results, skip rendering the duplicate
|
|
if (hasCorrespondingWidget && !content.is_error) {
|
|
return null;
|
|
}
|
|
// Extract the actual content string
|
|
let contentText = '';
|
|
if (typeof content.content === 'string') {
|
|
contentText = content.content;
|
|
} else if (content.content && typeof content.content === 'object') {
|
|
// Handle object with text property
|
|
if (content.content.text) {
|
|
contentText = content.content.text;
|
|
} else if (Array.isArray(content.content)) {
|
|
// Handle array of content blocks
|
|
contentText = content.content
|
|
.map((c: any) => (typeof c === 'string' ? c : c.text || JSON.stringify(c)))
|
|
.join('\n');
|
|
} else {
|
|
// Fallback to JSON stringify
|
|
contentText = JSON.stringify(content.content, null, 2);
|
|
}
|
|
}
|
|
|
|
// Always show system reminders regardless of widget status
|
|
const reminderMatch = contentText.match(/<system-reminder>(.*?)<\/system-reminder>/s);
|
|
if (reminderMatch) {
|
|
const reminderMessage = reminderMatch[1].trim();
|
|
const beforeReminder = contentText.substring(0, reminderMatch.index || 0).trim();
|
|
const afterReminder = contentText.substring((reminderMatch.index || 0) + reminderMatch[0].length).trim();
|
|
|
|
return (
|
|
<div key={idx} className="space-y-2">
|
|
<div className="flex items-center gap-2">
|
|
<CheckCircle2 className="h-4 w-4 text-green-500" />
|
|
<span className="text-sm font-medium">Tool Result</span>
|
|
</div>
|
|
|
|
{beforeReminder && (
|
|
<div className="ml-6 p-2 bg-background rounded-md border">
|
|
<pre className="text-xs font-mono overflow-x-auto whitespace-pre-wrap">
|
|
{beforeReminder}
|
|
</pre>
|
|
</div>
|
|
)}
|
|
|
|
<div className="ml-6">
|
|
<SystemReminderWidget message={reminderMessage} />
|
|
</div>
|
|
|
|
{afterReminder && (
|
|
<div className="ml-6 p-2 bg-background rounded-md border">
|
|
<pre className="text-xs font-mono overflow-x-auto whitespace-pre-wrap">
|
|
{afterReminder}
|
|
</pre>
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
// Check if this is an Edit tool result
|
|
const isEditResult = contentText.includes("has been updated. Here's the result of running `cat -n`");
|
|
|
|
if (isEditResult) {
|
|
return (
|
|
<div key={idx} className="space-y-2">
|
|
<div className="flex items-center gap-2">
|
|
<CheckCircle2 className="h-4 w-4 text-green-500" />
|
|
<span className="text-sm font-medium">Edit Result</span>
|
|
</div>
|
|
<EditResultWidget content={contentText} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
// Check if this is a MultiEdit tool result
|
|
const isMultiEditResult = contentText.includes("has been updated with multiple edits") ||
|
|
contentText.includes("MultiEdit completed successfully") ||
|
|
contentText.includes("Applied multiple edits to");
|
|
|
|
if (isMultiEditResult) {
|
|
return (
|
|
<div key={idx} className="space-y-2">
|
|
<div className="flex items-center gap-2">
|
|
<CheckCircle2 className="h-4 w-4 text-green-500" />
|
|
<span className="text-sm font-medium">MultiEdit Result</span>
|
|
</div>
|
|
<MultiEditResultWidget content={contentText} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
// Check if this is an LS tool result (directory tree structure)
|
|
const isLSResult = (() => {
|
|
if (!content.tool_use_id || typeof contentText !== 'string') return false;
|
|
|
|
// Check if this result came from an LS tool by looking for the tool call
|
|
let isFromLSTool = false;
|
|
|
|
// Search in previous assistant messages for the matching tool_use
|
|
if (streamMessages) {
|
|
for (let i = streamMessages.length - 1; i >= 0; i--) {
|
|
const prevMsg = streamMessages[i];
|
|
// Only check assistant messages
|
|
if (prevMsg.type === 'assistant' && prevMsg.message?.content && Array.isArray(prevMsg.message.content)) {
|
|
const toolUse = prevMsg.message.content.find((c: any) =>
|
|
c.type === 'tool_use' &&
|
|
c.id === content.tool_use_id &&
|
|
c.name?.toLowerCase() === 'ls'
|
|
);
|
|
if (toolUse) {
|
|
isFromLSTool = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Only proceed if this is from an LS tool
|
|
if (!isFromLSTool) return false;
|
|
|
|
// Additional validation: check for tree structure pattern
|
|
const lines = contentText.split('\n');
|
|
const hasTreeStructure = lines.some(line => /^\s*-\s+/.test(line));
|
|
const hasNoteAtEnd = lines.some(line => line.trim().startsWith('NOTE: do any of the files'));
|
|
|
|
return hasTreeStructure || hasNoteAtEnd;
|
|
})();
|
|
|
|
if (isLSResult) {
|
|
return (
|
|
<div key={idx} className="space-y-2">
|
|
<div className="flex items-center gap-2">
|
|
<CheckCircle2 className="h-4 w-4 text-green-500" />
|
|
<span className="text-sm font-medium">Directory Contents</span>
|
|
</div>
|
|
<LSResultWidget content={contentText} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
// Check if this is a Read tool result (contains line numbers with arrow separator)
|
|
const isReadResult = content.tool_use_id && typeof contentText === 'string' &&
|
|
/^\s*\d+→/.test(contentText);
|
|
|
|
if (isReadResult) {
|
|
// Try to find the corresponding Read tool call to get the file path
|
|
let filePath: string | undefined;
|
|
|
|
// Search in previous assistant messages for the matching tool_use
|
|
if (streamMessages) {
|
|
for (let i = streamMessages.length - 1; i >= 0; i--) {
|
|
const prevMsg = streamMessages[i];
|
|
// Only check assistant messages
|
|
if (prevMsg.type === 'assistant' && prevMsg.message?.content && Array.isArray(prevMsg.message.content)) {
|
|
const toolUse = prevMsg.message.content.find((c: any) =>
|
|
c.type === 'tool_use' &&
|
|
c.id === content.tool_use_id &&
|
|
c.name?.toLowerCase() === 'read'
|
|
);
|
|
if (toolUse?.input?.file_path) {
|
|
filePath = toolUse.input.file_path;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return (
|
|
<div key={idx} className="space-y-2">
|
|
<div className="flex items-center gap-2">
|
|
<CheckCircle2 className="h-4 w-4 text-green-500" />
|
|
<span className="text-sm font-medium">Read Result</span>
|
|
</div>
|
|
<ReadResultWidget content={contentText} filePath={filePath} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
// Handle empty tool results
|
|
if (!contentText || contentText.trim() === '') {
|
|
return (
|
|
<div key={idx} className="space-y-2">
|
|
<div className="flex items-center gap-2">
|
|
<CheckCircle2 className="h-4 w-4 text-green-500" />
|
|
<span className="text-sm font-medium">Tool Result</span>
|
|
</div>
|
|
<div className="ml-6 p-3 bg-muted/50 rounded-md border text-sm text-muted-foreground italic">
|
|
Tool did not return any output
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div key={idx} className="space-y-2">
|
|
<div className="flex items-center gap-2">
|
|
{content.is_error ? (
|
|
<AlertCircle className="h-4 w-4 text-destructive" />
|
|
) : (
|
|
<CheckCircle2 className="h-4 w-4 text-green-500" />
|
|
)}
|
|
<span className="text-sm font-medium">Tool Result</span>
|
|
</div>
|
|
<div className="ml-6 p-2 bg-background rounded-md border">
|
|
<pre className="text-xs font-mono overflow-x-auto whitespace-pre-wrap">
|
|
{contentText}
|
|
</pre>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
// Text content
|
|
if (content.type === "text") {
|
|
// Handle both string and object formats
|
|
const textContent = typeof content.text === 'string'
|
|
? content.text
|
|
: (content.text?.text || JSON.stringify(content.text));
|
|
|
|
return (
|
|
<div key={idx} className="text-sm">
|
|
{textContent}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return null;
|
|
})}
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
);
|
|
}
|
|
|
|
// Result message - render with markdown
|
|
if (message.type === "result") {
|
|
const isError = message.is_error || message.subtype?.includes("error");
|
|
|
|
return (
|
|
<Card className={cn(
|
|
isError ? "border-destructive/20 bg-destructive/5" : "border-green-500/20 bg-green-500/5",
|
|
className
|
|
)}>
|
|
<CardContent className="p-4">
|
|
<div className="flex items-start gap-3">
|
|
{isError ? (
|
|
<AlertCircle className="h-5 w-5 text-destructive mt-0.5" />
|
|
) : (
|
|
<CheckCircle2 className="h-5 w-5 text-green-500 mt-0.5" />
|
|
)}
|
|
<div className="flex-1 space-y-2">
|
|
<h4 className="font-semibold text-sm">
|
|
{isError ? "Execution Failed" : "Execution Complete"}
|
|
</h4>
|
|
|
|
{message.result && (
|
|
<div className="prose prose-sm dark:prose-invert max-w-none">
|
|
<ReactMarkdown
|
|
remarkPlugins={[remarkGfm]}
|
|
components={{
|
|
code({ node, inline, className, children, ...props }: any) {
|
|
const match = /language-(\w+)/.exec(className || '');
|
|
return !inline && match ? (
|
|
<SyntaxHighlighter
|
|
style={claudeSyntaxTheme}
|
|
language={match[1]}
|
|
PreTag="div"
|
|
{...props}
|
|
>
|
|
{String(children).replace(/\n$/, '')}
|
|
</SyntaxHighlighter>
|
|
) : (
|
|
<code className={className} {...props}>
|
|
{children}
|
|
</code>
|
|
);
|
|
}
|
|
}}
|
|
>
|
|
{message.result}
|
|
</ReactMarkdown>
|
|
</div>
|
|
)}
|
|
|
|
{message.error && (
|
|
<div className="text-sm text-destructive">{message.error}</div>
|
|
)}
|
|
|
|
<div className="text-xs text-muted-foreground space-y-1 mt-2">
|
|
{message.cost_usd !== undefined && (
|
|
<div>Cost: ${message.cost_usd.toFixed(4)} USD</div>
|
|
)}
|
|
{message.duration_ms !== undefined && (
|
|
<div>Duration: {(message.duration_ms / 1000).toFixed(2)}s</div>
|
|
)}
|
|
{message.num_turns !== undefined && (
|
|
<div>Turns: {message.num_turns}</div>
|
|
)}
|
|
{message.usage && (
|
|
<div>
|
|
Total tokens: {message.usage.input_tokens + message.usage.output_tokens}
|
|
({message.usage.input_tokens} in, {message.usage.output_tokens} out)
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
);
|
|
}
|
|
|
|
// Skip rendering if no meaningful content
|
|
return null;
|
|
} catch (error) {
|
|
// If any error occurs during rendering, show a safe error message
|
|
console.error("Error rendering stream message:", error, message);
|
|
return (
|
|
<Card className={cn("border-destructive/20 bg-destructive/5", className)}>
|
|
<CardContent className="p-4">
|
|
<div className="flex items-start gap-3">
|
|
<AlertCircle className="h-5 w-5 text-destructive mt-0.5" />
|
|
<div className="flex-1">
|
|
<p className="text-sm font-medium">Error rendering message</p>
|
|
<p className="text-xs text-muted-foreground mt-1">
|
|
{error instanceof Error ? error.message : 'Unknown error'}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
);
|
|
}
|
|
};
|
|
|
|
export const StreamMessage = React.memo(StreamMessageComponent);
|