LibreChat/api/server/routes/askSydney.js

200 lines
5.8 KiB
JavaScript
Raw Normal View History

2023-03-08 19:47:23 -05:00
const express = require('express');
const crypto = require('crypto');
const router = express.Router();
const { titleConvo, askSydney } = require('../../app/');
const { saveMessage, saveConvo, getConvoTitle } = require('../../models');
2023-03-15 15:21:04 -04:00
const { handleError, sendMessage, createOnProgress, handleText } = require('./handlers');
2023-03-08 19:47:23 -05:00
router.post('/', async (req, res) => {
const {
model,
text,
2023-03-17 02:08:03 +08:00
overrideParentMessageId=null,
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-03-08 19:47:23 -05:00
}
const conversationId = oldConversationId || crypto.randomUUID();
const isNewConversation = !oldConversationId;
2023-03-08 19:47:23 -05:00
2023-03-13 21:59:25 -04:00
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-03-08 19:47:23 -05:00
console.log('ask log', {
model,
...userMessage,
...convo
});
2023-03-17 02:08:03 +08:00
if (!overrideParentMessageId) {
await saveMessage(userMessage);
await saveConvo(req?.session?.user?.username, { ...userMessage, model, ...convo });
}
return await ask({
isNewConversation,
userMessage,
model,
convo,
preSendRequest: true,
2023-03-17 02:08:03 +08:00
overrideParentMessageId,
req,
res
});
});
const ask = async ({
isNewConversation,
overrideParentMessageId = null,
userMessage,
model,
convo,
preSendRequest = true,
req,
res
}) => {
let {
text,
parentMessageId: userParentMessageId,
conversationId,
messageId: userMessageId
} = userMessage;
2023-03-08 19:47:23 -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-03-08 19:47:23 -05:00
try {
const progressCallback = createOnProgress();
2023-03-17 03:13:42 +08:00
const abortController = new AbortController();
res.on('close', () => {
console.log('The client has disconnected.');
// 执行其他操作
abortController.abort();
})
2023-03-08 19:47:23 -05:00
let response = await askSydney({
text,
onProgress: progressCallback.call(null, model, {
res,
text,
parentMessageId: overrideParentMessageId || userMessageId
}),
convo: {
parentMessageId: userParentMessageId,
conversationId,
...convo
2023-03-17 03:13:42 +08:00
},
abortController
2023-03-08 19:47:23 -05:00
});
console.log('SYDNEY RESPONSE', response);
2023-03-09 16:09:53 -05:00
// console.dir(response, { depth: null });
2023-03-09 18:42:36 -05:00
userMessage.conversationSignature =
convo.conversationSignature || response.conversationSignature;
userMessage.conversationId = response.conversationId || conversationId;
userMessage.invocationId = response.invocationId;
// Unlike gpt and bing, Sydney will never accept our given userMessage.messageId, it will generate its own one.
2023-03-17 02:08:03 +08:00
if (!overrideParentMessageId)
await saveMessage(userMessage);
// Save sydney response
// response.id = response.messageId;
response.invocationId = convo.invocationId ? convo.invocationId + 1 : 1;
response.conversationId = conversationId ? conversationId : crypto.randomUUID();
response.conversationSignature = convo.conversationSignature
? convo.conversationSignature
: crypto.randomUUID();
2023-03-08 19:47:23 -05:00
response.text = response.response;
2023-03-09 18:07:36 -05:00
delete response.response;
2023-03-08 19:47:23 -05:00
response.suggestions =
response.details.suggestedResponses &&
response.details.suggestedResponses.map((s) => s.text);
response.sender = model;
// response.final = true;
// override the parentMessageId, for the regeneration.
response.parentMessageId =
overrideParentMessageId || response.parentMessageId || userMessageId;
// Save user message
userMessage.conversationId = response.conversationId || conversationId;
2023-03-17 02:08:03 +08:00
if (!overrideParentMessageId)
await saveMessage(userMessage);
// Bing API will not use our conversationId at the first time,
// so change the placeholder conversationId to the real one.
// Attition: the api will also create new conversationId while using invalid userMessage.parentMessageId,
// but in this situation, don't change the conversationId, but create new convo.
if (conversationId != userMessage.conversationId && isNewConversation)
await saveConvo(
req?.session?.user?.username,
{
conversationId: conversationId,
newConversationId: userMessage.conversationId
}
);
conversationId = userMessage.conversationId;
response.text = await handleText(response, true);
// Save sydney response & convo, then send
2023-03-08 19:47:23 -05:00
await saveMessage(response);
await saveConvo(req?.session?.user?.username, { ...response, model, chatGptLabel: null, promptPrefix: null, ...convo });
sendMessage(res, {
title: await getConvoTitle(req?.session?.user?.username, conversationId),
final: true,
requestMessage: userMessage,
responseMessage: response
});
2023-03-08 19:47:23 -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 });
await saveConvo(
req?.session?.user?.username,
{
conversationId,
title
}
);
2023-03-14 11:42:35 +08:00
}
2023-03-08 19:47:23 -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-03-08 19:47:23 -05:00
}
};
2023-03-08 19:47:23 -05:00
module.exports = router;