LibreChat/client/src/components/MessageHandler/index.jsx

255 lines
6.9 KiB
React
Raw Normal View History

import { useEffect, useState } from 'react';
import { useRecoilValue, useRecoilState, useResetRecoilState, useSetRecoilState } from 'recoil';
import { SSE } from '~/data-provider/sse.mjs';
import createPayload from '~/data-provider/createPayload';
import store from '~/store';
2023-03-31 03:22:57 +08:00
export default function MessageHandler() {
const submission = useRecoilValue(store.submission);
const setIsSubmitting = useSetRecoilState(store.isSubmitting);
const setMessages = useSetRecoilState(store.messages);
const setConversation = useSetRecoilState(store.conversation);
const resetLatestMessage = useResetRecoilState(store.latestMessage);
const [lastResponse, setLastResponse] = useRecoilState(store.lastResponse);
const setSubmission = useSetRecoilState(store.submission);
const [source, setSource] = useState(null);
const [abortKey, setAbortKey] = useState(null);
const { refreshConversations } = store.useConversations();
const messageHandler = (data, submission) => {
const { messages, message, initialResponse, isRegenerate = false } = submission;
if (isRegenerate)
setMessages([
...messages,
{
...initialResponse,
text: data,
parentMessageId: message?.overrideParentMessageId,
messageId: message?.overrideParentMessageId + '_',
submitting: true
}
]);
else
setMessages([
...messages,
message,
{
...initialResponse,
text: data,
parentMessageId: message?.messageId,
messageId: message?.messageId + '_',
submitting: true
}
]);
};
const cancelHandler = (data, submission) => {
const { messages, message, initialResponse, isRegenerate = false } = submission;
if (isRegenerate) {
setMessages([
...messages,
{
...initialResponse,
text: data,
parentMessageId: message?.overrideParentMessageId,
messageId: message?.overrideParentMessageId + '_',
cancelled: true
}
]);
} else {
setMessages([
...messages,
message,
{
...initialResponse,
text: data,
parentMessageId: message?.messageId,
messageId: message?.messageId + '_',
// cancelled: true
}
]);
setLastResponse('');
setSource(null);
}
};
const createdHandler = (data, submission) => {
const { messages, message, initialResponse, isRegenerate = false } = submission;
if (isRegenerate)
setMessages([
...messages,
{
...initialResponse,
parentMessageId: message?.overrideParentMessageId,
messageId: message?.overrideParentMessageId + '_',
submitting: true
}
]);
else
setMessages([
...messages,
message,
{
...initialResponse,
parentMessageId: message?.messageId,
messageId: message?.messageId + '_',
submitting: true
}
]);
const { conversationId } = message;
setConversation(prevState => ({
...prevState,
conversationId
}));
resetLatestMessage();
};
const finalHandler = (data, submission) => {
2023-03-31 03:22:57 +08:00
const { messages, isRegenerate = false } = submission;
2023-03-31 03:22:57 +08:00
const { requestMessage, responseMessage, conversation } = data;
// update the messages
if (isRegenerate) setMessages([...messages, responseMessage]);
else setMessages([...messages, requestMessage, responseMessage]);
setIsSubmitting(false);
// refresh title
if (requestMessage.parentMessageId == '00000000-0000-0000-0000-000000000000') {
setTimeout(() => {
refreshConversations();
}, 2000);
// in case it takes too long.
setTimeout(() => {
refreshConversations();
}, 5000);
}
2023-03-31 03:22:57 +08:00
setConversation(prevState => ({
...prevState,
...conversation
}));
};
const errorHandler = (data, submission) => {
2023-03-31 03:22:57 +08:00
const { messages, message } = submission;
console.log('Error:', data);
const errorResponse = {
...data,
error: true,
parentMessageId: message?.messageId
};
setIsSubmitting(false);
setMessages([...messages, message, errorResponse]);
return;
};
useEffect(() => {
if (submission === null) return;
if (Object.keys(submission).length === 0) return;
let { message, cancel } = submission;
if (cancel && source) {
console.log('message aborted', submission);
source.close();
const { endpoint } = submission.conversation;
const latestMessage = lastResponse.replaceAll('█', '');
fetch(`/api/ask/${endpoint}/abort?requestId=${abortKey}`)
.then(response => {
if (response.ok) {
console.log('Request aborted');
} else {
console.error('Error aborting request');
}
})
.catch(error => {
console.error(error);
});
console.log('source closed, got this far');
cancelHandler(latestMessage, { ...submission, message });
setIsSubmitting(false);
setSubmission(null);
return;
}
// events.oncancel = () => cancelHandler(latestResponseText, { ...submission, message });
const { server, payload } = createPayload(submission);
const events = new SSE(server, {
payload: JSON.stringify(payload),
headers: { 'Content-Type': 'application/json' }
});
setSource(events);
// let latestResponseText = '';
events.onmessage = e => {
const data = JSON.parse(e.data);
if (data.final) {
finalHandler(data, { ...submission, message });
console.log('final', data);
}
if (data.created) {
message = {
...data.message,
overrideParentMessageId: message?.overrideParentMessageId
};
createdHandler(data, { ...submission, message });
console.log('created', message);
setAbortKey(message?.messageId);
} else {
let text = data.text || data.response;
if (data.initial) console.log(data);
if (data.message) {
// latestResponseText = text;
setLastResponse(text);
messageHandler(text, { ...submission, message });
}
// console.log('dataStream', data);
}
};
events.onopen = () => console.log('connection is opened');
// events.oncancel = () => cancelHandler(latestResponseText, { ...submission, message });
events.onerror = function (e) {
console.log('error in opening conn.');
events.close();
const data = JSON.parse(e.data);
errorHandler(data, { ...submission, message });
};
setIsSubmitting(true);
events.stream();
return () => {
const isCancelled = events.readyState <= 1;
events.close();
setSource(null);
if (isCancelled) {
const e = new Event('cancel');
events.dispatchEvent(e);
}
setIsSubmitting(false);
};
}, [submission]);
return null;
}