mirror of
https://github.com/run-llama/ai-chatbot.git
synced 2026-07-01 21:14:02 -04:00
142 lines
3.9 KiB
TypeScript
142 lines
3.9 KiB
TypeScript
'use client';
|
|
|
|
import { useChat } from 'ai/react';
|
|
import { useEffect, useRef, useState } from 'react';
|
|
import { BlockKind } from './block';
|
|
import { Suggestion } from '@/lib/db/schema';
|
|
import { initialBlockData, useBlock } from '@/hooks/use-block';
|
|
import { useUserMessageId } from '@/hooks/use-user-message-id';
|
|
import { cx } from 'class-variance-authority';
|
|
import { useSWRConfig } from 'swr';
|
|
|
|
type DataStreamDelta = {
|
|
type:
|
|
| 'text-delta'
|
|
| 'code-delta'
|
|
| 'title'
|
|
| 'id'
|
|
| 'suggestion'
|
|
| 'clear'
|
|
| 'finish'
|
|
| 'user-message-id'
|
|
| 'kind';
|
|
content: string | Suggestion;
|
|
};
|
|
|
|
export function DataStreamHandler({ id }: { id: string }) {
|
|
const { data: dataStream } = useChat({ id });
|
|
const { setUserMessageIdFromServer } = useUserMessageId();
|
|
const { setBlock } = useBlock();
|
|
const lastProcessedIndex = useRef(-1);
|
|
|
|
const { mutate } = useSWRConfig();
|
|
const [optimisticSuggestions, setOptimisticSuggestions] = useState<
|
|
Array<Suggestion>
|
|
>([]);
|
|
|
|
useEffect(() => {
|
|
if (optimisticSuggestions && optimisticSuggestions.length > 0) {
|
|
const [optimisticSuggestion] = optimisticSuggestions;
|
|
const url = `/api/suggestions?documentId=${optimisticSuggestion.documentId}`;
|
|
mutate(url, optimisticSuggestions, false);
|
|
}
|
|
}, [optimisticSuggestions, mutate]);
|
|
|
|
useEffect(() => {
|
|
if (!dataStream?.length) return;
|
|
|
|
const newDeltas = dataStream.slice(lastProcessedIndex.current + 1);
|
|
lastProcessedIndex.current = dataStream.length - 1;
|
|
|
|
(newDeltas as DataStreamDelta[]).forEach((delta: DataStreamDelta) => {
|
|
if (delta.type === 'user-message-id') {
|
|
setUserMessageIdFromServer(delta.content as string);
|
|
return;
|
|
}
|
|
|
|
setBlock((draftBlock) => {
|
|
if (!draftBlock) {
|
|
return { ...initialBlockData, status: 'streaming' };
|
|
}
|
|
|
|
switch (delta.type) {
|
|
case 'id':
|
|
return {
|
|
...draftBlock,
|
|
documentId: delta.content as string,
|
|
status: 'streaming',
|
|
};
|
|
|
|
case 'title':
|
|
return {
|
|
...draftBlock,
|
|
title: delta.content as string,
|
|
status: 'streaming',
|
|
};
|
|
|
|
case 'kind':
|
|
return {
|
|
...draftBlock,
|
|
kind: delta.content as BlockKind,
|
|
status: 'streaming',
|
|
};
|
|
|
|
case 'text-delta':
|
|
return {
|
|
...draftBlock,
|
|
content: draftBlock.content + (delta.content as string),
|
|
isVisible:
|
|
draftBlock.status === 'streaming' &&
|
|
draftBlock.content.length > 400 &&
|
|
draftBlock.content.length < 450
|
|
? true
|
|
: draftBlock.isVisible,
|
|
status: 'streaming',
|
|
};
|
|
|
|
case 'code-delta':
|
|
return {
|
|
...draftBlock,
|
|
content: delta.content as string,
|
|
isVisible:
|
|
draftBlock.status === 'streaming' &&
|
|
draftBlock.content.length > 300 &&
|
|
draftBlock.content.length < 310
|
|
? true
|
|
: draftBlock.isVisible,
|
|
status: 'streaming',
|
|
};
|
|
|
|
case 'suggestion':
|
|
setTimeout(() => {
|
|
setOptimisticSuggestions((currentSuggestions) => [
|
|
...currentSuggestions,
|
|
delta.content as Suggestion,
|
|
]);
|
|
}, 0);
|
|
|
|
return draftBlock;
|
|
|
|
case 'clear':
|
|
return {
|
|
...draftBlock,
|
|
content: '',
|
|
status: 'streaming',
|
|
};
|
|
|
|
case 'finish':
|
|
return {
|
|
...draftBlock,
|
|
status: 'idle',
|
|
};
|
|
|
|
default:
|
|
return draftBlock;
|
|
}
|
|
});
|
|
});
|
|
}, [dataStream, setBlock, setUserMessageIdFromServer]);
|
|
|
|
return null;
|
|
}
|