159 lines
5.6 KiB
TypeScript
159 lines
5.6 KiB
TypeScript
'use client';
|
|
|
|
import ChatInput from '@/component/chat-input';
|
|
import { useChat } from '@ai-sdk/react';
|
|
import {
|
|
DefaultChatTransport,
|
|
lastAssistantMessageIsCompleteWithToolCalls,
|
|
} from 'ai';
|
|
import { UseChatToolsMessage } from '../api/use-chat-tools/route';
|
|
|
|
export default function Chat() {
|
|
const { messages, sendMessage, addToolResult, status } =
|
|
useChat<UseChatToolsMessage>({
|
|
transport: new DefaultChatTransport({ api: '/api/use-chat-tools' }),
|
|
sendAutomaticallyWhen: lastAssistantMessageIsCompleteWithToolCalls,
|
|
|
|
// run client-side tools that are automatically executed:
|
|
async onToolCall({ toolCall }) {
|
|
// artificial 2 second delay
|
|
await new Promise(resolve => setTimeout(resolve, 2000));
|
|
|
|
if (toolCall.toolName === 'getLocation') {
|
|
const cities = [
|
|
'New York',
|
|
'Los Angeles',
|
|
'Chicago',
|
|
'San Francisco',
|
|
];
|
|
|
|
addToolResult({
|
|
tool: 'getLocation',
|
|
toolCallId: toolCall.toolCallId,
|
|
output: cities[Math.floor(Math.random() * cities.length)],
|
|
});
|
|
}
|
|
},
|
|
});
|
|
|
|
return (
|
|
<div className="flex flex-col py-24 mx-auto w-full max-w-md stretch">
|
|
{messages?.map(message => (
|
|
<div key={message.id} className="whitespace-pre-wrap">
|
|
<strong>{`${message.role}: `}</strong>
|
|
{message.parts.map((part, index) => {
|
|
switch (part.type) {
|
|
case 'text':
|
|
return <div key={index}>{part.text}</div>;
|
|
|
|
case 'step-start':
|
|
return index > 0 ? (
|
|
<div key={index} className="text-gray-500">
|
|
<hr className="my-2 border-gray-300" />
|
|
</div>
|
|
) : null;
|
|
|
|
case 'tool-askForConfirmation': {
|
|
switch (part.state) {
|
|
case 'input-available':
|
|
return (
|
|
<div key={index} className="text-gray-500">
|
|
{part.input.message}
|
|
<div className="flex gap-2">
|
|
<button
|
|
className="px-4 py-2 font-bold text-white bg-blue-500 rounded hover:bg-blue-700"
|
|
onClick={async () => {
|
|
addToolResult({
|
|
tool: 'askForConfirmation',
|
|
toolCallId: part.toolCallId,
|
|
output: 'Yes, confirmed.',
|
|
});
|
|
}}
|
|
>
|
|
Yes
|
|
</button>
|
|
<button
|
|
className="px-4 py-2 font-bold text-white bg-red-500 rounded hover:bg-red-700"
|
|
onClick={async () => {
|
|
addToolResult({
|
|
tool: 'askForConfirmation',
|
|
toolCallId: part.toolCallId,
|
|
output: 'No, denied',
|
|
});
|
|
}}
|
|
>
|
|
No
|
|
</button>
|
|
</div>
|
|
</div>
|
|
);
|
|
case 'output-available':
|
|
return (
|
|
<div key={index} className="text-gray-500">
|
|
Location access allowed: {part.output}
|
|
</div>
|
|
);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 'tool-getLocation': {
|
|
switch (part.state) {
|
|
case 'input-available':
|
|
return (
|
|
<div key={index} className="text-gray-500">
|
|
Getting location...
|
|
</div>
|
|
);
|
|
case 'output-available':
|
|
return (
|
|
<div key={index} className="text-gray-500">
|
|
Location: {part.output}
|
|
</div>
|
|
);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 'tool-getWeatherInformation': {
|
|
switch (part.state) {
|
|
// example of pre-rendering streaming tool calls:
|
|
case 'input-streaming':
|
|
return (
|
|
<pre key={index}>
|
|
{JSON.stringify(part.input, null, 2)}
|
|
</pre>
|
|
);
|
|
case 'input-available':
|
|
return (
|
|
<div key={index} className="text-gray-500">
|
|
Getting weather information for {part.input.city}...
|
|
</div>
|
|
);
|
|
case 'output-available':
|
|
return (
|
|
<div key={index} className="text-gray-500">
|
|
{part.output.state === 'loading'
|
|
? 'Fetching weather information...'
|
|
: `Weather in ${part.input.city}: ${part.output.weather}`}
|
|
</div>
|
|
);
|
|
case 'output-error':
|
|
return (
|
|
<div key={index} className="text-red-500">
|
|
Error: {part.errorText}
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
}
|
|
})}
|
|
<br />
|
|
</div>
|
|
))}
|
|
|
|
<ChatInput status={status} onSubmit={text => sendMessage({ text })} />
|
|
</div>
|
|
);
|
|
}
|