LibreChat/server/routes/ask.js

106 lines
3.3 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 { ask, titleConvo } = require('../../app/chatgpt');
const { askClient } = require('../../app/chatgpt-client');
2023-02-12 16:38:33 -05:00
const { saveMessage, deleteMessages } = require('../../models/Message');
const { saveConvo } = require('../../models/Conversation');
router.post('/', async (req, res) => {
const { text, parentMessageId, conversationId } = req.body;
if (!text.trim().includes(' ') && text.length < 5) {
res.status(500).write('Prompt empty or too short');
res.end();
return;
}
const userMessageId = crypto.randomUUID();
let userMessage = { id: userMessageId, sender: 'User', text };
console.log('ask log', userMessage);
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);
res.write(
`event: message\ndata: ${JSON.stringify({ ...partial, initial: true })}\n\n`
);
i++;
}
if (typeof partial === 'object') {
const data = JSON.stringify({ ...partial, message: true });
res.write(`event: message\ndata: ${data}\n\n`);
} else {
tokens += partial;
res.write(`event: message\ndata: ${JSON.stringify({ text: tokens, message: true })}\n\n`);
}
2023-02-12 16:38:33 -05:00
};
let gptResponse = await askClient(text, progressCallback, { parentMessageId, conversationId });
console.log('CLIENT RESPONSE', gptResponse);
2023-02-12 16:38:33 -05:00
if (!!parentMessageId) {
gptResponse = { ...gptResponse, parentMessageId };
} else {
gptResponse.title = await titleConvo(text, gptResponse.text);
2023-02-12 16:38:33 -05:00
}
if (!gptResponse.parentMessageId && !parentMessageId) {
userMessage.parentMessageId = gptResponse.messageId;
gptResponse.parentMessageId = gptResponse.messageId;
userMessage.conversationId = gptResponse.conversationId;
}
const response = gptResponse.text || gptResponse.response;
if (gptResponse.response) {
await saveMessage(userMessage);
gptResponse.text = gptResponse.response;
gptResponse.id = gptResponse.messageId;
delete gptResponse.response;
}
2023-02-12 16:38:33 -05:00
if (
(response.includes('2023') && !response.trim().includes(' ')) ||
response.toLowerCase().includes('no response') ||
response.toLowerCase().includes('no answer')
2023-02-12 16:38:33 -05:00
) {
res.status(500).write('event: error\ndata: Prompt empty or too short');
res.end();
return;
}
gptResponse.sender = 'GPT';
console.log('gptResponse', gptResponse);
2023-02-12 16:38:33 -05:00
await saveMessage(gptResponse);
await saveConvo(gptResponse);
res.write(`event: message\ndata: ${JSON.stringify(gptResponse)}\n\n`);
res.end();
} catch (error) {
console.log(error);
await deleteMessages({ id: userMessageId });
res.status(500).write('event: error\ndata: ' + error.message);
res.end();
}
});
module.exports = router;