const express = require('express'); const router = express.Router(); const { getResponseSender } = require('../endpoints/schemas'); const { sendMessage, createOnProgress } = require('../../utils'); const { addTitle, initializeClient } = require('../endpoints/openAI'); const { saveMessage, getConvoTitle, getConvo } = require('../../../models'); const { handleAbort, createAbortController, handleAbortError, setHeaders, validateEndpoint, buildEndpointOption, } = require('../../middleware'); router.post('/abort', handleAbort()); router.post('/', validateEndpoint, buildEndpointOption, setHeaders, async (req, res) => { let { text, endpointOption, conversationId, parentMessageId = null, overrideParentMessageId = null, } = req.body; console.log('ask log'); console.dir({ text, conversationId, endpointOption }, { depth: null }); let metadata; let userMessage; let promptTokens; let userMessageId; let responseMessageId; let lastSavedTimestamp = 0; let saveDelay = 100; const sender = getResponseSender(endpointOption); const newConvo = !conversationId; const user = req.user.id; const addMetadata = (data) => (metadata = data); const getReqData = (data = {}) => { for (let key in data) { if (key === 'userMessage') { userMessage = data[key]; userMessageId = data[key].messageId; } else if (key === 'responseMessageId') { responseMessageId = data[key]; } else if (key === 'promptTokens') { promptTokens = data[key]; } else if (!conversationId && key === 'conversationId') { conversationId = data[key]; } } }; const { onProgress: progressCallback, getPartialText } = createOnProgress({ onProgress: ({ text: partialText }) => { const currentTimestamp = Date.now(); if (currentTimestamp - lastSavedTimestamp > saveDelay) { lastSavedTimestamp = currentTimestamp; saveMessage({ messageId: responseMessageId, sender, conversationId, parentMessageId: overrideParentMessageId ?? userMessageId, text: partialText, model: endpointOption.modelOptions.model, unfinished: true, cancelled: false, error: false, user, }); } if (saveDelay < 500) { saveDelay = 500; } }, }); const getAbortData = () => ({ sender, conversationId, messageId: responseMessageId, parentMessageId: overrideParentMessageId ?? userMessageId, text: getPartialText(), userMessage, promptTokens, }); const { abortController, onStart } = createAbortController(req, res, getAbortData); try { const { client } = await initializeClient({ req, res, endpointOption }); let response = await client.sendMessage(text, { user, parentMessageId, conversationId, overrideParentMessageId, getReqData, onStart, addMetadata, abortController, onProgress: progressCallback.call(null, { res, text, parentMessageId: overrideParentMessageId || userMessageId, }), }); if (overrideParentMessageId) { response.parentMessageId = overrideParentMessageId; } if (metadata) { response = { ...response, ...metadata }; } await saveMessage({ ...response, user }); sendMessage(res, { title: await getConvoTitle(user, conversationId), final: true, conversation: await getConvo(user, conversationId), requestMessage: userMessage, responseMessage: response, }); res.end(); if (parentMessageId === '00000000-0000-0000-0000-000000000000' && newConvo) { addTitle(req, { text, response, client, }); } } catch (error) { const partialText = getPartialText(); handleAbortError(res, req, error, { partialText, conversationId, sender, messageId: responseMessageId, parentMessageId: userMessageId ?? parentMessageId, }); } }); module.exports = router;