ai-sdk-next-openai/app/test-openai-web-search/page.tsx
2025-09-26 15:46:29 +00:00

104 lines
3.2 KiB
TypeScript

'use client';
import { useChat } from '@ai-sdk/react';
import { DefaultChatTransport } from 'ai';
import ChatInput from '@/component/chat-input';
import { OpenAIWebSearchMessage } from '@/app/api/chat-openai-web-search/route';
export default function TestOpenAIWebSearch() {
const { error, status, sendMessage, messages, regenerate, stop } =
useChat<OpenAIWebSearchMessage>({
transport: new DefaultChatTransport({
api: '/api/chat-openai-web-search',
}),
});
return (
<div className="flex flex-col py-24 mx-auto w-full max-w-md stretch">
<h1 className="mb-4 text-xl font-bold">OpenAI Web Search Test</h1>
{messages.map(message => (
<div key={message.id} className="whitespace-pre-wrap">
{message.role === 'user' ? 'User: ' : 'AI: '}
{message.parts.map((part, index) => {
if (part.type === 'text') {
return <div key={index}>{part.text}</div>;
}
if (part.type === 'tool-web_search') {
if (part.state === 'input-available') {
return (
<pre
key={index}
className="overflow-auto p-2 text-sm bg-gray-100 rounded"
>
{JSON.stringify(part.input, null, 2)}
</pre>
);
}
if (part.state === 'output-available') {
return (
<pre
key={index}
className="overflow-auto p-2 text-sm bg-gray-100 rounded"
>
{JSON.stringify(part.input, null, 2)}
{`\n\nDONE - Web search completed`}
</pre>
);
}
}
if (part.type === 'source-url') {
return (
<span key={index}>
[
<a
href={part.url}
target="_blank"
rel="noopener noreferrer"
className="text-sm font-bold text-blue-500 hover:underline"
>
{part.title ?? new URL(part.url).hostname}
</a>
]
</span>
);
}
return null;
})}
</div>
))}
{(status === 'submitted' || status === 'streaming') && (
<div className="mt-4 text-gray-500">
{status === 'submitted' && <div>Loading...</div>}
<button
type="button"
className="px-4 py-2 mt-4 text-blue-500 rounded-md border border-blue-500"
onClick={stop}
>
Stop
</button>
</div>
)}
{error && (
<div className="mt-4">
<div className="text-red-500">An error occurred.</div>
<button
type="button"
className="px-4 py-2 mt-4 text-blue-500 rounded-md border border-blue-500"
onClick={() => regenerate()}
>
Retry
</button>
</div>
)}
<ChatInput status={status} onSubmit={text => sendMessage({ text })} />
</div>
);
}