LibreChat/api/server/routes/ask/askChatGPTBrowser.js

190 lines
5.8 KiB
JavaScript
Raw Normal View History

2023-03-31 03:22:57 +08:00
const express = require('express');
const crypto = require('crypto');
const router = express.Router();
const { getChatGPTBrowserModels } = require('../endpoints');
2023-04-05 16:15:46 +08:00
const { titleConvo, browserClient } = require('../../../app/');
const { saveMessage, getConvoTitle, saveConvo, updateConvo, getConvo } = require('../../../models');
2023-03-31 03:22:57 +08:00
const { handleError, sendMessage, createOnProgress, handleText } = require('./handlers');
router.post('/', async (req, res) => {
const {
endpoint,
text,
overrideParentMessageId = null,
parentMessageId,
conversationId: oldConversationId
} = req.body;
if (text.length === 0) return handleError(res, { text: 'Prompt empty or too short' });
if (endpoint !== 'chatGPTBrowser') return handleError(res, { text: 'Illegal request' });
// build user message
const conversationId = oldConversationId || crypto.randomUUID();
const isNewConversation = !oldConversationId;
2023-03-31 03:22:57 +08:00
const userMessageId = crypto.randomUUID();
const userParentMessageId = parentMessageId || '00000000-0000-0000-0000-000000000000';
const userMessage = {
messageId: userMessageId,
sender: 'User',
text,
parentMessageId: userParentMessageId,
conversationId,
isCreatedByUser: true
};
// build endpoint option
const endpointOption = {
2023-04-06 23:14:42 +08:00
model: req.body?.model ?? 'text-davinci-002-render-sha'
2023-03-31 03:22:57 +08:00
};
const availableModels = getChatGPTBrowserModels();
if (availableModels.find(model => model === endpointOption.model) === undefined)
return handleError(res, { text: 'Illegal request: model' });
2023-03-31 03:22:57 +08:00
console.log('ask log', {
userMessage,
endpointOption,
conversationId
});
if (!overrideParentMessageId) {
await saveMessage(userMessage);
await saveConvo(req?.session?.user?.username, {
...userMessage,
...endpointOption,
conversationId,
endpoint
});
2023-03-31 03:22:57 +08:00
}
// eslint-disable-next-line no-use-before-define
2023-03-31 03:22:57 +08:00
return await ask({
isNewConversation,
2023-03-31 03:22:57 +08:00
userMessage,
endpointOption,
conversationId,
preSendRequest: true,
overrideParentMessageId,
req,
res
});
});
const ask = async ({
isNewConversation,
2023-03-31 03:22:57 +08:00
userMessage,
endpointOption,
conversationId,
preSendRequest = true,
overrideParentMessageId = null,
req,
res
}) => {
let { text, parentMessageId: userParentMessageId, messageId: userMessageId } = userMessage;
2023-03-31 03:22:57 +08:00
res.writeHead(200, {
Connection: 'keep-alive',
'Content-Type': 'text/event-stream',
'Cache-Control': 'no-cache, no-transform',
'Access-Control-Allow-Origin': '*',
'X-Accel-Buffering': 'no'
});
if (preSendRequest) sendMessage(res, { message: userMessage, created: true });
try {
const progressCallback = createOnProgress();
const abortController = new AbortController();
res.on('close', () => abortController.abort());
let response = await browserClient({
2023-03-31 03:22:57 +08:00
text,
parentMessageId: userParentMessageId,
conversationId,
...endpointOption,
onProgress: progressCallback.call(null, { res, text }),
abortController
});
console.log('CLIENT RESPONSE', response);
2023-03-31 03:22:57 +08:00
2023-04-06 23:14:42 +08:00
const newConversationId = response.conversationId || conversationId;
const newUserMassageId = response.parentMessageId || userMessageId;
const newResponseMessageId = response.messageId;
// STEP1 generate response message
response.text = response.response || '**ChatGPT refused to answer.**';
2023-03-31 03:22:57 +08:00
let responseMessage = {
2023-04-06 23:14:42 +08:00
conversationId: newConversationId,
messageId: newResponseMessageId,
parentMessageId: overrideParentMessageId || newUserMassageId,
text: await handleText(response),
sender: endpointOption?.chatGptLabel || 'ChatGPT'
};
2023-03-31 03:22:57 +08:00
await saveMessage(responseMessage);
2023-03-31 03:22:57 +08:00
// STEP2 update the conversation
2023-03-31 03:22:57 +08:00
// First update conversationId if needed
2023-04-06 23:14:42 +08:00
let conversationUpdate = { conversationId: newConversationId, endpoint: 'chatGPTBrowser' };
if (conversationId != newConversationId)
if (isNewConversation) {
// change the conversationId to new one
conversationUpdate = {
...conversationUpdate,
conversationId: conversationId,
newConversationId: newConversationId
};
} else {
// create new conversation
conversationUpdate = {
...conversationUpdate,
...endpointOption
};
}
2023-03-31 03:22:57 +08:00
await saveConvo(req?.session?.user?.username, conversationUpdate);
2023-04-06 23:14:42 +08:00
conversationId = newConversationId;
// STEP3 update the user message
2023-04-06 23:14:42 +08:00
userMessage.conversationId = newConversationId;
userMessage.messageId = newUserMassageId;
// If response has parentMessageId, the fake userMessage.messageId should be updated to the real one.
2023-04-06 23:14:42 +08:00
if (!overrideParentMessageId)
await saveMessage({ ...userMessage, messageId: userMessageId, newMessageId: newUserMassageId });
userMessageId = newUserMassageId;
2023-03-31 03:22:57 +08:00
sendMessage(res, {
title: await getConvoTitle(req?.session?.user?.username, conversationId),
final: true,
conversation: await getConvo(req?.session?.user?.username, conversationId),
requestMessage: userMessage,
responseMessage: responseMessage
2023-03-31 03:22:57 +08:00
});
res.end();
if (userParentMessageId == '00000000-0000-0000-0000-000000000000') {
// const title = await titleConvo({ endpoint: endpointOption?.endpoint, text, response: responseMessage });
const title = await response.details.title;
2023-03-31 03:22:57 +08:00
await updateConvo(req?.session?.user?.username, {
conversationId: conversationId,
2023-03-31 03:22:57 +08:00
title
});
}
} catch (error) {
const errorMessage = {
messageId: crypto.randomUUID(),
sender: 'ChatGPT',
conversationId,
parentMessageId: overrideParentMessageId || userMessageId,
error: true,
text: error.message
};
await saveMessage(errorMessage);
handleError(res, errorMessage);
}
};
module.exports = router;