mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-12-17 08:50:15 +01:00
fix: rewrite ask openAI and ask BingAI. now all code cleaned.
This commit is contained in:
parent
f922a1d102
commit
d864da6a21
8 changed files with 165 additions and 341 deletions
|
|
@ -30,23 +30,38 @@ const askBing = async ({
|
||||||
proxy: process.env.PROXY || null
|
proxy: process.env.PROXY || null
|
||||||
});
|
});
|
||||||
|
|
||||||
let options = {
|
let options = {};
|
||||||
jailbreakConversationId: jailbreakConversationId || jailbreak,
|
|
||||||
context,
|
|
||||||
systemMessage,
|
|
||||||
parentMessageId,
|
|
||||||
conversationId: jailbreakConversationId ? jailbreakConversationId : conversationId,
|
|
||||||
toneStyle,
|
|
||||||
onProgress
|
|
||||||
};
|
|
||||||
|
|
||||||
if (conversationSignature) options.conversationSignature = conversationSignature;
|
if (jailbreakConversationId == 'false') {
|
||||||
if (conversationSignature) options.clientId = clientId;
|
jailbreakConversationId = false;
|
||||||
if (conversationSignature) options.invocationId = invocationId;
|
}
|
||||||
if (conversationSignature) options.toneStyle = toneStyle;
|
|
||||||
|
|
||||||
if (options?.jailbreakConversationId == 'false') {
|
if (jailbreak)
|
||||||
options.jailbreakConversationId = false;
|
options = {
|
||||||
|
jailbreakConversationId: jailbreakConversationId || jailbreak,
|
||||||
|
context,
|
||||||
|
systemMessage,
|
||||||
|
parentMessageId,
|
||||||
|
toneStyle,
|
||||||
|
onProgress
|
||||||
|
};
|
||||||
|
else {
|
||||||
|
options = {
|
||||||
|
conversationId,
|
||||||
|
context,
|
||||||
|
systemMessage,
|
||||||
|
parentMessageId,
|
||||||
|
toneStyle,
|
||||||
|
onProgress
|
||||||
|
};
|
||||||
|
|
||||||
|
// don't give those parameters for new conversation
|
||||||
|
// for new conversation, conversationSignature always is null
|
||||||
|
if (conversationSignature) {
|
||||||
|
options.conversationSignature = conversationSignature;
|
||||||
|
options.clientId = clientId;
|
||||||
|
options.invocationId = invocationId;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log('bing options', options);
|
console.log('bing options', options);
|
||||||
|
|
|
||||||
|
|
@ -31,36 +31,6 @@ module.exports = {
|
||||||
return { message: 'Error saving message' };
|
return { message: 'Error saving message' };
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
saveBingMessage: async ({
|
|
||||||
messageId,
|
|
||||||
newMessageId,
|
|
||||||
conversationId,
|
|
||||||
parentMessageId,
|
|
||||||
sender,
|
|
||||||
text,
|
|
||||||
isCreatedByUser = false,
|
|
||||||
error
|
|
||||||
}) => {
|
|
||||||
try {
|
|
||||||
await Message.findOneAndUpdate(
|
|
||||||
{ messageId },
|
|
||||||
{
|
|
||||||
messageId: newMessageId || messageId,
|
|
||||||
conversationId,
|
|
||||||
parentMessageId,
|
|
||||||
sender,
|
|
||||||
text,
|
|
||||||
isCreatedByUser,
|
|
||||||
error
|
|
||||||
},
|
|
||||||
{ upsert: true, new: true }
|
|
||||||
);
|
|
||||||
return { messageId, conversationId, parentMessageId, sender, text, isCreatedByUser };
|
|
||||||
} catch (error) {
|
|
||||||
console.error(error);
|
|
||||||
return { message: 'Error saving message' };
|
|
||||||
}
|
|
||||||
},
|
|
||||||
deleteMessagesSince: async ({ messageId, conversationId }) => {
|
deleteMessagesSince: async ({ messageId, conversationId }) => {
|
||||||
try {
|
try {
|
||||||
const message = await Message.findOne({ messageId }).exec();
|
const message = await Message.findOne({ messageId }).exec();
|
||||||
|
|
|
||||||
|
|
@ -1,10 +1,4 @@
|
||||||
const {
|
const { getMessages, saveMessage, deleteMessagesSince, deleteMessages } = require('./Message');
|
||||||
getMessages,
|
|
||||||
saveMessage,
|
|
||||||
saveBingMessage,
|
|
||||||
deleteMessagesSince,
|
|
||||||
deleteMessages
|
|
||||||
} = require('./Message');
|
|
||||||
const { getCustomGpts, updateCustomGpt, updateByLabel, deleteCustomGpts } = require('./CustomGpt');
|
const { getCustomGpts, updateCustomGpt, updateByLabel, deleteCustomGpts } = require('./CustomGpt');
|
||||||
const { getConvoTitle, getConvo, saveConvo, updateConvo } = require('./Conversation');
|
const { getConvoTitle, getConvo, saveConvo, updateConvo } = require('./Conversation');
|
||||||
const { getPreset, getPresets, savePreset, deletePresets } = require('./Preset');
|
const { getPreset, getPresets, savePreset, deletePresets } = require('./Preset');
|
||||||
|
|
@ -12,7 +6,6 @@ const { getPreset, getPresets, savePreset, deletePresets } = require('./Preset')
|
||||||
module.exports = {
|
module.exports = {
|
||||||
getMessages,
|
getMessages,
|
||||||
saveMessage,
|
saveMessage,
|
||||||
saveBingMessage,
|
|
||||||
deleteMessagesSince,
|
deleteMessagesSince,
|
||||||
deleteMessages,
|
deleteMessages,
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -47,14 +47,6 @@ module.exports = {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: false
|
default: false
|
||||||
},
|
},
|
||||||
jailbreakConversationId: {
|
|
||||||
type: String,
|
|
||||||
default: null
|
|
||||||
},
|
|
||||||
conversationSignature: {
|
|
||||||
type: String,
|
|
||||||
default: null
|
|
||||||
},
|
|
||||||
context: {
|
context: {
|
||||||
type: String,
|
type: String,
|
||||||
default: null
|
default: null
|
||||||
|
|
@ -63,14 +55,6 @@ module.exports = {
|
||||||
type: String,
|
type: String,
|
||||||
default: null
|
default: null
|
||||||
},
|
},
|
||||||
clientId: {
|
|
||||||
type: String,
|
|
||||||
default: null
|
|
||||||
},
|
|
||||||
invocationId: {
|
|
||||||
type: Number,
|
|
||||||
default: 1
|
|
||||||
},
|
|
||||||
toneStyle: {
|
toneStyle: {
|
||||||
type: String,
|
type: String,
|
||||||
default: null
|
default: null
|
||||||
|
|
|
||||||
|
|
@ -20,7 +20,24 @@ const convoSchema = mongoose.Schema(
|
||||||
default: null
|
default: null
|
||||||
},
|
},
|
||||||
messages: [{ type: mongoose.Schema.Types.ObjectId, ref: 'Message' }],
|
messages: [{ type: mongoose.Schema.Types.ObjectId, ref: 'Message' }],
|
||||||
...conversationPreset
|
...conversationPreset,
|
||||||
|
// for bingAI only
|
||||||
|
jailbreakConversationId: {
|
||||||
|
type: String,
|
||||||
|
default: null
|
||||||
|
},
|
||||||
|
conversationSignature: {
|
||||||
|
type: String,
|
||||||
|
default: null
|
||||||
|
},
|
||||||
|
clientId: {
|
||||||
|
type: String,
|
||||||
|
default: null
|
||||||
|
},
|
||||||
|
invocationId: {
|
||||||
|
type: Number,
|
||||||
|
default: 1
|
||||||
|
}
|
||||||
},
|
},
|
||||||
{ timestamps: true }
|
{ timestamps: true }
|
||||||
);
|
);
|
||||||
|
|
|
||||||
|
|
@ -2,7 +2,7 @@ const express = require('express');
|
||||||
const crypto = require('crypto');
|
const crypto = require('crypto');
|
||||||
const router = express.Router();
|
const router = express.Router();
|
||||||
const { titleConvo, askBing } = require('../../app');
|
const { titleConvo, askBing } = require('../../app');
|
||||||
const { saveBingMessage, getConvoTitle, saveConvo, getConvo } = require('../../models');
|
const { saveMessage, getConvoTitle, saveConvo, getConvo } = require('../../models');
|
||||||
const { handleError, sendMessage, createOnProgress, handleText } = require('./handlers');
|
const { handleError, sendMessage, createOnProgress, handleText } = require('./handlers');
|
||||||
|
|
||||||
router.post('/', async (req, res) => {
|
router.post('/', async (req, res) => {
|
||||||
|
|
@ -32,17 +32,25 @@ router.post('/', async (req, res) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
// build endpoint option
|
// build endpoint option
|
||||||
const endpointOption = {
|
let endpointOption = {};
|
||||||
jailbreak: req.body?.jailbreak || false,
|
if (req.body?.jailbreak)
|
||||||
jailbreakConversationId: req.body?.jailbreakConversationId || null,
|
endpointOption = {
|
||||||
systemMessage: req.body?.systemMessage || null,
|
jailbreak: req.body?.jailbreak || false,
|
||||||
context: req.body?.context || null,
|
jailbreakConversationId: req.body?.jailbreakConversationId || null,
|
||||||
conversationSignature: req.body?.conversationSignature || null,
|
systemMessage: req.body?.systemMessage || null,
|
||||||
clientId: req.body?.clientId || null,
|
context: req.body?.context || null,
|
||||||
invocationId: req.body?.invocationId || null,
|
toneStyle: req.body?.toneStyle || 'fast'
|
||||||
toneStyle: req.body?.toneStyle || 'fast',
|
};
|
||||||
suggestions: req.body?.suggestions || []
|
else
|
||||||
};
|
endpointOption = {
|
||||||
|
jailbreak: req.body?.jailbreak || false,
|
||||||
|
systemMessage: req.body?.systemMessage || null,
|
||||||
|
context: req.body?.context || null,
|
||||||
|
conversationSignature: req.body?.conversationSignature || null,
|
||||||
|
clientId: req.body?.clientId || null,
|
||||||
|
invocationId: req.body?.invocationId || null,
|
||||||
|
toneStyle: req.body?.toneStyle || 'fast'
|
||||||
|
};
|
||||||
|
|
||||||
console.log('ask log', {
|
console.log('ask log', {
|
||||||
userMessage,
|
userMessage,
|
||||||
|
|
@ -51,7 +59,7 @@ router.post('/', async (req, res) => {
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!overrideParentMessageId) {
|
if (!overrideParentMessageId) {
|
||||||
await saveBingMessage(userMessage);
|
await saveMessage(userMessage);
|
||||||
await saveConvo(req?.session?.user?.username, {
|
await saveConvo(req?.session?.user?.username, {
|
||||||
...userMessage,
|
...userMessage,
|
||||||
...endpointOption,
|
...endpointOption,
|
||||||
|
|
@ -114,57 +122,81 @@ const ask = async ({
|
||||||
|
|
||||||
console.log('BING RESPONSE', response);
|
console.log('BING RESPONSE', response);
|
||||||
|
|
||||||
// STEP1 update the convosation
|
// STEP1 generate response message
|
||||||
|
response.text = response.response || response.details.spokenText || '**Bing refused to answer.**';
|
||||||
|
|
||||||
|
let responseMessage = {
|
||||||
|
text: await handleText(response, true),
|
||||||
|
suggestions:
|
||||||
|
response.details.suggestedResponses && response.details.suggestedResponses.map(s => s.text),
|
||||||
|
jailbreak: endpointOption?.jailbreak
|
||||||
|
};
|
||||||
|
// // response.text = await handleText(response, true);
|
||||||
|
// response.suggestions =
|
||||||
|
// response.details.suggestedResponses && response.details.suggestedResponses.map(s => s.text);
|
||||||
|
|
||||||
|
if (endpointOption?.jailbreak) {
|
||||||
|
responseMessage.conversationId = response.jailbreakConversationId;
|
||||||
|
responseMessage.messageId = response.messageId || response.details.messageId;
|
||||||
|
responseMessage.parentMessageId = overrideParentMessageId || response.parentMessageId || userMessageId;
|
||||||
|
responseMessage.sender = 'Sydney';
|
||||||
|
} else {
|
||||||
|
responseMessage.conversationId = response.conversationId;
|
||||||
|
responseMessage.messageId = response.messageId || response.details.messageId;
|
||||||
|
response.parentMessageId =
|
||||||
|
overrideParentMessageId || response.parentMessageId || response.details.requestId || userMessageId;
|
||||||
|
responseMessage.sender = 'BingAI';
|
||||||
|
}
|
||||||
|
|
||||||
|
await saveMessage(responseMessage);
|
||||||
|
|
||||||
|
// STEP2 update the convosation.
|
||||||
|
|
||||||
|
// First update conversationId if needed
|
||||||
|
// Note!
|
||||||
// Bing API will not use our conversationId at the first time,
|
// Bing API will not use our conversationId at the first time,
|
||||||
// so change the placeholder conversationId to the real one.
|
// so change the placeholder conversationId to the real one.
|
||||||
// Attition: the api will also create new conversationId while using invalid userMessage.parentMessageId,
|
// 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.
|
// but in this situation, don't change the conversationId, but create new convo.
|
||||||
if (conversationId != response.conversationId && isNewConversation)
|
|
||||||
await saveConvo(req?.session?.user?.username, {
|
|
||||||
conversationId: conversationId,
|
|
||||||
newConversationId: response.conversationId || conversationId
|
|
||||||
});
|
|
||||||
conversationId = response.conversationId || conversationId;
|
|
||||||
|
|
||||||
// STEP2 update the user message
|
let conversationUpdate = { conversationId, endpoint: 'bingAI' };
|
||||||
userMessage.conversationSignature =
|
if (conversationId != responseMessage.conversationId && isNewConversation)
|
||||||
endpointOption.conversationSignature || response.conversationSignature;
|
conversationUpdate = {
|
||||||
|
...conversationUpdate,
|
||||||
|
conversationId: conversationId,
|
||||||
|
newConversationId: responseMessage.conversationId || conversationId
|
||||||
|
};
|
||||||
|
conversationId = responseMessage.conversationId || conversationId;
|
||||||
|
|
||||||
|
if (endpointOption?.jailbreak) {
|
||||||
|
conversationUpdate.jailbreak = true;
|
||||||
|
conversationUpdate.jailbreakConversationId = response.jailbreakConversationId;
|
||||||
|
} else {
|
||||||
|
conversationUpdate.jailbreak = false;
|
||||||
|
conversationUpdate.conversationSignature = response.conversationSignature;
|
||||||
|
conversationUpdate.clientId = response.clientId;
|
||||||
|
conversationUpdate.invocationId = response.invocationId;
|
||||||
|
}
|
||||||
|
|
||||||
|
await saveConvo(req?.session?.user?.username, conversationUpdate);
|
||||||
|
|
||||||
|
// STEP3 update the user message
|
||||||
userMessage.conversationId = conversationId;
|
userMessage.conversationId = conversationId;
|
||||||
userMessage.invocationId = endpointOption.invocationId;
|
userMessage.messageId = responseMessage.parentMessageId;
|
||||||
userMessage.messageId = response.details.requestId || userMessageId;
|
|
||||||
|
|
||||||
// If response has parentMessageId, the fake userMessage.messageId should be updated to the real one.
|
// If response has parentMessageId, the fake userMessage.messageId should be updated to the real one.
|
||||||
if (!overrideParentMessageId) {
|
if (!overrideParentMessageId) {
|
||||||
const oldUserMessageId = userMessageId;
|
const oldUserMessageId = userMessageId;
|
||||||
userMessageId = response.details.requestId;
|
await saveMessage({ ...userMessage, messageId: oldUserMessageId, newMessageId: userMessage.messageId });
|
||||||
await saveBingMessage({ ...userMessage, messageId: oldUserMessageId, newMessageId: userMessageId });
|
|
||||||
}
|
}
|
||||||
|
userMessageId = userMessage.messageId;
|
||||||
response.text = response.response || response.details.spokenText || '**Bing refused to answer.**';
|
|
||||||
// delete response.response;
|
|
||||||
// response.id = response.details.messageId;
|
|
||||||
response.suggestions =
|
|
||||||
response.details.suggestedResponses && response.details.suggestedResponses.map(s => s.text);
|
|
||||||
response.sender = endpointOption?.jailbreak ? 'Sydney' : 'BingAI';
|
|
||||||
// response.final = true;
|
|
||||||
|
|
||||||
response.messageId = response.details.messageId;
|
|
||||||
// override the parentMessageId, for the regeneration.
|
|
||||||
response.parentMessageId = overrideParentMessageId || response.details.requestId || userMessageId;
|
|
||||||
|
|
||||||
response.text = await handleText(response, true);
|
|
||||||
await saveBingMessage(response);
|
|
||||||
await saveConvo(req?.session?.user?.username, {
|
|
||||||
...endpointOption,
|
|
||||||
...response
|
|
||||||
});
|
|
||||||
|
|
||||||
sendMessage(res, {
|
sendMessage(res, {
|
||||||
title: await getConvoTitle(req?.session?.user?.username, conversationId),
|
title: await getConvoTitle(req?.session?.user?.username, conversationId),
|
||||||
final: true,
|
final: true,
|
||||||
conversation: await getConvo(req?.session?.user?.username, conversationId),
|
conversation: await getConvo(req?.session?.user?.username, conversationId),
|
||||||
requestMessage: userMessage,
|
requestMessage: userMessage,
|
||||||
responseMessage: response
|
responseMessage: responseMessage
|
||||||
});
|
});
|
||||||
res.end();
|
res.end();
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -95,47 +95,57 @@ const ask = async ({
|
||||||
const progressCallback = createOnProgress();
|
const progressCallback = createOnProgress();
|
||||||
const abortController = new AbortController();
|
const abortController = new AbortController();
|
||||||
res.on('close', () => abortController.abort());
|
res.on('close', () => abortController.abort());
|
||||||
let gptResponse = await client({
|
let response = await client({
|
||||||
text,
|
text,
|
||||||
parentMessageId: userParentMessageId,
|
parentMessageId: userParentMessageId,
|
||||||
conversationId,
|
conversationId,
|
||||||
...endpointOption,
|
...endpointOption,
|
||||||
onProgress: progressCallback.call(null, { res, text }),
|
onProgress: progressCallback.call(null, {
|
||||||
|
res,
|
||||||
|
text,
|
||||||
|
parentMessageId: overrideParentMessageId || userMessageId
|
||||||
|
}),
|
||||||
abortController
|
abortController
|
||||||
});
|
});
|
||||||
|
|
||||||
gptResponse.text = gptResponse.response;
|
console.log('CLIENT RESPONSE', response);
|
||||||
console.log('CLIENT RESPONSE', gptResponse);
|
|
||||||
|
|
||||||
if (gptResponse.parentMessageId) {
|
// STEP1 generate response message
|
||||||
// If gptResponse has parentMessageId, the fake userMessage.messageId should be updated to the real one.
|
response.text = response.response || '**ChatGPT refused to answer.**';
|
||||||
if (!overrideParentMessageId) {
|
|
||||||
const oldUserMessageId = userMessageId;
|
let responseMessage = {
|
||||||
userMessageId = gptResponse.parentMessageId;
|
conversationId: response.conversationId,
|
||||||
userMessage.messageId = userMessageId;
|
messageId: response.messageId,
|
||||||
await saveMessage({ ...userMessage, messageId: oldUserMessageId, newMessageId: userMessageId });
|
parentMessageId: overrideParentMessageId || userMessageId,
|
||||||
}
|
text: await handleText(response),
|
||||||
} else {
|
sender: endpointOption?.chatGptLabel || 'ChatGPT'
|
||||||
delete gptResponse.response;
|
};
|
||||||
|
|
||||||
|
await saveMessage(responseMessage);
|
||||||
|
|
||||||
|
// STEP2 update the conversation
|
||||||
|
conversationId = responseMessage.conversationId || conversationId;
|
||||||
|
// it seems openAI will not change the conversationId.
|
||||||
|
// let conversationUpdate = { conversationId, endpoint: 'openAI' };
|
||||||
|
// await saveConvo(req?.session?.user?.username, conversationUpdate);
|
||||||
|
|
||||||
|
// STEP3 update the user message
|
||||||
|
userMessage.conversationId = conversationId;
|
||||||
|
userMessage.messageId = responseMessage.parentMessageId;
|
||||||
|
|
||||||
|
// If gptResponse has parentMessageId, the fake userMessage.messageId should be updated to the real one.
|
||||||
|
if (!overrideParentMessageId) {
|
||||||
|
const oldUserMessageId = userMessageId;
|
||||||
|
await saveMessage({ ...userMessage, messageId: oldUserMessageId, newMessageId: userMessage.messageId });
|
||||||
}
|
}
|
||||||
|
userMessageId = userMessage.messageId;
|
||||||
gptResponse.parentMessageId = overrideParentMessageId || userMessageId;
|
|
||||||
gptResponse.sender = endpointOption?.chatGptLabel || 'ChatGPT';
|
|
||||||
// gptResponse.model = model;
|
|
||||||
gptResponse.text = await handleText(gptResponse);
|
|
||||||
|
|
||||||
await saveMessage(gptResponse);
|
|
||||||
await updateConvo(req?.session?.user?.username, {
|
|
||||||
...gptResponse,
|
|
||||||
oldConvoId: conversationId
|
|
||||||
});
|
|
||||||
|
|
||||||
sendMessage(res, {
|
sendMessage(res, {
|
||||||
title: await getConvoTitle(req?.session?.user?.username, conversationId),
|
title: await getConvoTitle(req?.session?.user?.username, conversationId),
|
||||||
final: true,
|
final: true,
|
||||||
conversation: await getConvo(req?.session?.user?.username, conversationId),
|
conversation: await getConvo(req?.session?.user?.username, conversationId),
|
||||||
requestMessage: userMessage,
|
requestMessage: userMessage,
|
||||||
responseMessage: gptResponse
|
responseMessage: responseMessage
|
||||||
});
|
});
|
||||||
res.end();
|
res.end();
|
||||||
|
|
||||||
|
|
@ -147,6 +157,7 @@ const ask = async ({
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
const errorMessage = {
|
const errorMessage = {
|
||||||
messageId: crypto.randomUUID(),
|
messageId: crypto.randomUUID(),
|
||||||
sender: endpointOption?.chatGptLabel || 'ChatGPT',
|
sender: endpointOption?.chatGptLabel || 'ChatGPT',
|
||||||
|
|
|
||||||
|
|
@ -1,198 +0,0 @@
|
||||||
const express = require('express');
|
|
||||||
const crypto = require('crypto');
|
|
||||||
const router = express.Router();
|
|
||||||
const { titleConvo, askSydney } = require('../../app/');
|
|
||||||
const { saveBingMessage, saveConvo, updateConvo, getConvoTitle } = require('../../models');
|
|
||||||
const { handleError, sendMessage, createOnProgress, handleText } = require('./handlers');
|
|
||||||
|
|
||||||
router.post('/', async (req, res) => {
|
|
||||||
const {
|
|
||||||
model,
|
|
||||||
text,
|
|
||||||
overrideParentMessageId=null,
|
|
||||||
parentMessageId,
|
|
||||||
conversationId: oldConversationId,
|
|
||||||
...convo
|
|
||||||
} = req.body;
|
|
||||||
if (text.length === 0) {
|
|
||||||
return handleError(res, { text: 'Prompt empty or too short' });
|
|
||||||
}
|
|
||||||
|
|
||||||
const conversationId = oldConversationId || crypto.randomUUID();
|
|
||||||
const isNewConversation = !oldConversationId;
|
|
||||||
|
|
||||||
const userMessageId = convo.messageId;
|
|
||||||
const userParentMessageId = parentMessageId || '00000000-0000-0000-0000-000000000000';
|
|
||||||
let userMessage = {
|
|
||||||
messageId: userMessageId,
|
|
||||||
sender: 'User',
|
|
||||||
text,
|
|
||||||
parentMessageId: userParentMessageId,
|
|
||||||
conversationId,
|
|
||||||
isCreatedByUser: true
|
|
||||||
};
|
|
||||||
|
|
||||||
console.log('ask log', {
|
|
||||||
model,
|
|
||||||
...convo,
|
|
||||||
...userMessage
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!overrideParentMessageId) {
|
|
||||||
await saveBingMessage(userMessage);
|
|
||||||
await saveConvo(req?.session?.user?.username, { model, ...convo, ...userMessage });
|
|
||||||
}
|
|
||||||
|
|
||||||
return await ask({
|
|
||||||
isNewConversation,
|
|
||||||
userMessage,
|
|
||||||
model,
|
|
||||||
convo,
|
|
||||||
preSendRequest: true,
|
|
||||||
overrideParentMessageId,
|
|
||||||
req,
|
|
||||||
res
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
const ask = async ({
|
|
||||||
isNewConversation,
|
|
||||||
overrideParentMessageId = null,
|
|
||||||
userMessage,
|
|
||||||
model,
|
|
||||||
convo,
|
|
||||||
preSendRequest = true,
|
|
||||||
req,
|
|
||||||
res
|
|
||||||
}) => {
|
|
||||||
let {
|
|
||||||
text,
|
|
||||||
parentMessageId: userParentMessageId,
|
|
||||||
conversationId,
|
|
||||||
messageId: userMessageId
|
|
||||||
} = userMessage;
|
|
||||||
|
|
||||||
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 });
|
|
||||||
|
|
||||||
try {
|
|
||||||
const progressCallback = createOnProgress();
|
|
||||||
|
|
||||||
const abortController = new AbortController();
|
|
||||||
res.on('close', () => {
|
|
||||||
console.log('The client has disconnected.');
|
|
||||||
// 执行其他操作
|
|
||||||
abortController.abort();
|
|
||||||
})
|
|
||||||
|
|
||||||
let response = await askSydney({
|
|
||||||
text,
|
|
||||||
onProgress: progressCallback.call(null, model, {
|
|
||||||
res,
|
|
||||||
text,
|
|
||||||
parentMessageId: overrideParentMessageId || userMessageId
|
|
||||||
}),
|
|
||||||
convo: {
|
|
||||||
...convo,
|
|
||||||
parentMessageId: userParentMessageId,
|
|
||||||
conversationId
|
|
||||||
},
|
|
||||||
abortController
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log('SYDNEY RESPONSE', response);
|
|
||||||
// console.dir(response, { depth: null });
|
|
||||||
|
|
||||||
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.
|
|
||||||
userMessage.messageId = response.parentMessageId || userMessageId;
|
|
||||||
|
|
||||||
// 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();
|
|
||||||
response.text = response.response || response.details.spokenText || '**Bing refused to answer.**';
|
|
||||||
// delete response.response;
|
|
||||||
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;
|
|
||||||
if (!overrideParentMessageId)
|
|
||||||
await saveBingMessage({ oldMessageId: userMessageId, ...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 updateConvo(
|
|
||||||
req?.session?.user?.username,
|
|
||||||
{
|
|
||||||
conversationId: conversationId,
|
|
||||||
newConversationId: userMessage.conversationId
|
|
||||||
}
|
|
||||||
);
|
|
||||||
conversationId = userMessage.conversationId;
|
|
||||||
|
|
||||||
response.text = await handleText(response, true);
|
|
||||||
// Save sydney response & convo, then send
|
|
||||||
await saveBingMessage(response);
|
|
||||||
await updateConvo(req?.session?.user?.username, { model, chatGptLabel: null, promptPrefix: null, ...convo, ...response });
|
|
||||||
|
|
||||||
sendMessage(res, {
|
|
||||||
title: await getConvoTitle(req?.session?.user?.username, conversationId),
|
|
||||||
final: true,
|
|
||||||
requestMessage: userMessage,
|
|
||||||
responseMessage: response
|
|
||||||
});
|
|
||||||
res.end();
|
|
||||||
|
|
||||||
if (userParentMessageId == '00000000-0000-0000-0000-000000000000') {
|
|
||||||
const title = await titleConvo({ model, text, response });
|
|
||||||
|
|
||||||
await updateConvo(
|
|
||||||
req?.session?.user?.username,
|
|
||||||
{
|
|
||||||
conversationId,
|
|
||||||
title
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
} 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 saveBingMessage(errorMessage);
|
|
||||||
handleError(res, errorMessage);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = router;
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue