LibreChat/api/server/routes/ask.js

213 lines
5.6 KiB
JavaScript
Raw Normal View History

2023-02-12 16:38:33 -05:00
const express = require('express');
const crypto = require('crypto');
const router = express.Router();
2023-02-25 09:04:32 -05:00
const askBing = require('./askBing');
2023-03-08 19:47:23 -05:00
const askSydney = require('./askSydney');
2023-03-15 15:21:04 -04:00
const { titleConvo, askClient, browserClient, customClient } = require('../../app/');
const { getConvo, saveMessage, getConvoTitle, saveConvo } = require('../../models');
2023-03-15 15:21:04 -04:00
const { handleError, sendMessage, createOnProgress, handleText } = require('./handlers');
const { getMessages } = require('../../models/Message');
2023-02-12 16:38:33 -05:00
router.use('/bing', askBing);
router.use('/sydney', askSydney);
2023-02-12 16:38:33 -05:00
router.post('/', async (req, res) => {
let { model, text, parentMessageId, conversationId: oldConversationId, ...convo } = req.body;
2023-03-11 12:10:00 -05:00
if (text.length === 0) {
return handleError(res, { text: 'Prompt empty or too short' });
2023-02-12 16:38:33 -05:00
}
const conversationId = oldConversationId || crypto.randomUUID();
const userMessageId = crypto.randomUUID();
const userParentMessageId = parentMessageId || '00000000-0000-0000-0000-000000000000';
let userMessage = {
messageId: userMessageId,
sender: 'User',
text,
parentMessageId: userParentMessageId,
conversationId,
isCreatedByUser: true
};
2023-02-12 16:38:33 -05:00
2023-03-03 15:52:06 -05:00
console.log('ask log', {
model,
...userMessage,
...convo
2023-03-03 15:52:06 -05:00
});
await saveMessage(userMessage);
await saveConvo({ ...userMessage, model, ...convo });
return await ask({
userMessage,
model,
convo,
preSendRequest: true,
req,
res
});
});
router.post('/regenerate', async (req, res) => {
const { model } = req.body;
const oldUserMessage = await getMessages({ messageId: req.body });
if (oldUserMessage) {
const convo = await getConvo(userMessage?.conversationId);
const userMessageId = crypto.randomUUID();
let userMessage = {
...userMessage,
messageId: userMessageId
};
console.log('ask log for regeneration', {
model,
...userMessage,
...convo
});
return await ask({
userMessage,
model,
convo,
preSendRequest: false,
req,
res
});
} else return handleError(res, { text: 'Parent message not found' });
});
const ask = async ({
userMessage,
overrideParentMessageId = null,
model,
convo,
preSendRequest = true,
req,
res
}) => {
let {
text,
parentMessageId: userParentMessageId,
conversationId,
messageId: userMessageId
} = userMessage;
2023-03-03 15:52:06 -05:00
let client;
2023-02-12 16:38:33 -05:00
2023-03-03 15:52:06 -05:00
if (model === 'chatgpt') {
client = askClient;
} else if (model === 'chatgptCustom') {
client = customClient;
} else {
client = browserClient;
}
2023-02-12 16:38:33 -05: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 });
2023-02-12 16:38:33 -05:00
try {
const progressCallback = createOnProgress();
let gptResponse = await client({
text,
onProgress: progressCallback.call(null, model, { res, text }),
convo: {
parentMessageId: userParentMessageId,
conversationId,
...convo
2023-03-03 15:52:06 -05:00
},
...convo
});
console.log('CLIENT RESPONSE', gptResponse);
if (!gptResponse.parentMessageId) {
gptResponse.text = gptResponse.response;
// gptResponse.id = gptResponse.messageId;
gptResponse.parentMessageId = overrideParentMessageId || userMessageId;
userMessage.conversationId = conversationId
? conversationId
: gptResponse.conversationId;
await saveMessage(userMessage);
delete gptResponse.response;
}
2023-02-12 16:38:33 -05:00
if (
(gptResponse.text.includes('2023') && !gptResponse.text.trim().includes(' ')) ||
gptResponse.text.toLowerCase().includes('no response') ||
gptResponse.text.toLowerCase().includes('no answer')
2023-02-12 16:38:33 -05:00
) {
await saveMessage({
messageId: crypto.randomUUID(),
sender: model,
conversationId,
parentMessageId: overrideParentMessageId || userMessageId,
error: true,
text: 'Prompt empty or too short'
});
return handleError(res, { text: 'Prompt empty or too short' });
2023-02-12 16:38:33 -05:00
}
gptResponse.sender = model === 'chatgptCustom' ? convo.chatGptLabel : model;
gptResponse.model = model;
// gptResponse.final = true;
gptResponse.text = await handleText(gptResponse);
if (convo.chatGptLabel?.length > 0 && model === 'chatgptCustom') {
gptResponse.chatGptLabel = convo.chatGptLabel;
}
if (convo.promptPrefix?.length > 0 && model === 'chatgptCustom') {
gptResponse.promptPrefix = convo.promptPrefix;
}
// override the parentMessageId, for the regeneration.
gptResponse.parentMessageId = overrideParentMessageId || userMessageId;
2023-02-12 16:38:33 -05:00
await saveMessage(gptResponse);
await saveConvo(gptResponse);
sendMessage(res, {
title: await getConvoTitle(conversationId),
final: true,
requestMessage: userMessage,
responseMessage: gptResponse
});
2023-02-12 16:38:33 -05:00
res.end();
2023-03-14 11:42:35 +08:00
if (userParentMessageId == '00000000-0000-0000-0000-000000000000') {
2023-03-15 15:21:04 -04:00
const title = await titleConvo({ model, text, response: gptResponse });
2023-03-15 00:53:15 +08:00
2023-03-14 11:42:35 +08:00
await saveConvo({
conversationId,
title
});
2023-03-14 11:42:35 +08:00
}
2023-02-12 16:38:33 -05:00
} catch (error) {
console.log(error);
// await deleteMessages({ messageId: userMessageId });
const errorMessage = {
messageId: crypto.randomUUID(),
sender: model,
conversationId,
parentMessageId: overrideParentMessageId || userMessageId,
error: true,
text: error.message
};
await saveMessage(errorMessage);
handleError(res, errorMessage);
2023-02-12 16:38:33 -05:00
}
};
2023-02-12 16:38:33 -05:00
module.exports = router;