LibreChat/server/routes/ask.js

173 lines
5.2 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();
const { titleConvo } = require('../../app/chatgpt');
const { askClient } = require('../../app/chatgpt-client');
const { askBing } = require('../../app/bingai');
2023-02-12 16:38:33 -05:00
const { saveMessage, deleteMessages } = require('../../models/Message');
const { saveConvo } = require('../../models/Conversation');
const handleError = (res, errorMessage) => {
res.status(500).write(`event: error\ndata: ${errorMessage}`);
res.end();
};
const sendMessage = (res, message) => {
res.write(`event: message\ndata: ${JSON.stringify(message)}\n\n`);
};
router.post('/bing', async (req, res) => {
2023-02-20 21:16:40 -05:00
const { model, text, ...convo } = req.body;
if (!text.trim().includes(' ') && text.length < 5) {
return handleError(res, 'Prompt empty or too short');
}
const userMessageId = crypto.randomUUID();
let userMessage = { id: userMessageId, sender: 'User', text };
2023-02-20 21:16:40 -05:00
console.log('ask log', { model, ...userMessage, ...convo });
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'
});
try {
let tokens = '';
const progressCallback = async (partial) => {
2023-02-19 21:06:21 -05:00
tokens += partial;
sendMessage(res, { text: tokens, message: true });
};
let response = await askBing({
text,
2023-02-20 21:16:40 -05:00
progressCallback,
convo
});
2023-02-20 21:52:08 -05:00
// console.log('CLIENT RESPONSE');
// console.dir(response, { depth: null });
2023-02-19 21:06:21 -05:00
userMessage.conversationSignature =
2023-02-20 21:16:40 -05:00
convo.conversationSignature || response.conversationSignature;
userMessage.conversationId = convo.conversationId || response.conversationId;
2023-02-19 21:06:21 -05:00
userMessage.invocationId = response.invocationId;
await saveMessage(userMessage);
2023-02-20 21:16:40 -05:00
if (!convo.conversationSignature) {
response.title = await titleConvo(text, response.response, model);
2023-02-19 21:06:21 -05:00
}
response.text = response.response;
response.id = response.details.messageId;
response.suggestions =
response.details.suggestedResponses &&
response.details.suggestedResponses.map((s) => s.text);
2023-02-19 21:06:21 -05:00
response.sender = model;
response.final = true;
2023-02-19 21:06:21 -05:00
await saveMessage(response);
await saveConvo(response);
sendMessage(res, response);
res.end();
} catch (error) {
console.log(error);
2023-02-19 21:06:21 -05:00
await deleteMessages({ id: userMessageId });
handleError(res, error.message);
}
});
2023-02-12 16:38:33 -05:00
router.post('/', async (req, res) => {
const { model, text, parentMessageId, conversationId } = req.body;
2023-02-12 16:38:33 -05:00
if (!text.trim().includes(' ') && text.length < 5) {
return handleError(res, 'Prompt empty or too short');
2023-02-12 16:38:33 -05:00
}
const userMessageId = crypto.randomUUID();
let userMessage = { id: userMessageId, sender: 'User', text };
console.log('ask log', { model, ...userMessage, parentMessageId, conversationId });
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'
});
try {
let i = 0;
let tokens = '';
2023-02-12 16:38:33 -05:00
const progressCallback = async (partial) => {
if (i === 0 && typeof partial === 'object') {
2023-02-12 16:38:33 -05:00
userMessage.parentMessageId = parentMessageId ? parentMessageId : partial.id;
userMessage.conversationId = conversationId ? conversationId : partial.conversationId;
await saveMessage(userMessage);
sendMessage(res, { ...partial, initial: true });
2023-02-12 16:38:33 -05:00
i++;
}
if (typeof partial === 'object') {
sendMessage(res, { ...partial, message: true });
} else {
tokens += partial;
if (tokens.includes('[DONE]')) {
tokens = tokens.replace('[DONE]', '');
}
sendMessage(res, { text: tokens, message: true });
}
2023-02-12 16:38:33 -05:00
};
let gptResponse = await askClient({
model,
text,
progressCallback,
convo: {
parentMessageId,
conversationId
}
});
console.log('CLIENT RESPONSE', gptResponse);
if (!gptResponse.parentMessageId) {
gptResponse.text = gptResponse.response;
gptResponse.id = gptResponse.messageId;
gptResponse.parentMessageId = gptResponse.messageId;
userMessage.parentMessageId = parentMessageId ? parentMessageId : gptResponse.messageId;
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
) {
return handleError(res, 'Prompt empty or too short');
2023-02-12 16:38:33 -05:00
}
2023-02-19 21:06:21 -05:00
if (!parentMessageId) {
gptResponse.title = await titleConvo(text, gptResponse.text, model);
2023-02-19 21:06:21 -05:00
}
gptResponse.sender = model;
2023-02-12 22:57:59 -05:00
gptResponse.final = true;
2023-02-12 16:38:33 -05:00
await saveMessage(gptResponse);
await saveConvo(gptResponse);
sendMessage(res, gptResponse);
2023-02-12 16:38:33 -05:00
res.end();
} catch (error) {
console.log(error);
await deleteMessages({ id: userMessageId });
handleError(res, error.message);
2023-02-12 16:38:33 -05:00
}
});
module.exports = router;