feat: utitlize lean queries, remove migration script, index createdAt timestamps (#698)

* feat(mongoDb): utitlize lean queries and index createdAt timestamps for cosmosDB support

* fix: remove unnecessary lean() method from deleteMany calls

* fix: remove unnecessary lean() method from deleteMany calls

* fix: remove lean() from queries that need hydration

* chore(migrateDb.js): remove unused migration script
fix(Preset.js): return lean documents when retrieving presets
refactor(index.js): remove migration script from server initialization
refactor(convos.js): remove toObject() when sending conversation object
refactor(presets.js): remove toObject() when sending presets object
This commit is contained in:
Danny Avila 2023-07-25 19:27:55 -04:00 committed by GitHub
parent 2f7658e39f
commit 19af2b06ce
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
13 changed files with 34 additions and 160 deletions

View file

@ -1,124 +0,0 @@
const mongoose = require('mongoose');
const { Conversation } = require('../../models/Conversation');
const { getMessages } = require('../../models/');
const migrateToStrictFollowParentMessageIdChain = async () => {
try {
const conversations = await Conversation.find({ endpoint: null, model: null }).exec();
if (!conversations || conversations.length === 0) {
return { noNeed: true };
}
console.log('Migration: To strict follow the parentMessageId chain.');
for (let convo of conversations) {
const messages = await getMessages({
conversationId: convo.conversationId,
messageId: { $exists: false },
});
let model;
let oldId;
const promises = [];
messages.forEach((message, i) => {
const msgObj = message.toObject();
const newId = msgObj.id;
if (i === 0) {
message.parentMessageId = '00000000-0000-0000-0000-000000000000';
} else {
message.parentMessageId = oldId;
}
oldId = newId;
message.messageId = newId;
if (message.sender.toLowerCase() !== 'user' && !model) {
model = message.sender.toLowerCase();
}
if (message.sender.toLowerCase() === 'user') {
message.isCreatedByUser = true;
}
promises.push(message.save());
});
await Promise.all(promises);
await Conversation.findOneAndUpdate(
{ conversationId: convo.conversationId },
{ model },
{ new: true },
).exec();
}
try {
await mongoose.connection.db.collection('messages').dropIndex('id_1');
} catch (error) {
console.log('[Migrate] Index doesn\'t exist or already dropped');
}
} catch (error) {
console.log(error);
return { message: '[Migrate] Error migrating conversations' };
}
};
const migrateToSupportBetterCustomization = async () => {
try {
const conversations = await Conversation.find({ endpoint: null }).exec();
if (!conversations || conversations.length === 0) {
return { noNeed: true };
}
console.log('Migration: To support better customization.');
const promises = [];
for (let convo of conversations) {
const originalModel = convo?.model;
if (originalModel === 'chatgpt') {
convo.endpoint = 'openAI';
convo.model = 'gpt-3.5-turbo';
} else if (originalModel === 'chatgptCustom') {
convo.endpoint = 'openAI';
convo.model = 'gpt-3.5-turbo';
} else if (originalModel === 'bingai') {
convo.endpoint = 'bingAI';
convo.model = null;
convo.jailbreak = false;
} else if (originalModel === 'sydney') {
convo.endpoint = 'bingAI';
convo.model = null;
convo.jailbreak = true;
} else if (originalModel === 'chatgptBrowser') {
convo.endpoint = 'chatGPTBrowser';
convo.model = 'text-davinci-002-render-sha';
convo.jailbreak = true;
} else {
convo.endpoint = 'openAI';
convo.model = 'gpt-3.5-turbo';
}
promises.push(convo.save());
}
await Promise.all(promises);
} catch (error) {
console.log(error);
return { message: '[Migrate] Error migrating conversations' };
}
};
async function migrateDb() {
let ret = [];
ret[0] = await migrateToStrictFollowParentMessageIdChain();
ret[1] = await migrateToSupportBetterCustomization();
const isMigrated = !!ret.find((element) => !element?.noNeed);
if (!isMigrated) {
console.log('[Migrate] Nothing to migrate');
}
}
module.exports = migrateDb;

View file

@ -55,7 +55,7 @@ configSchema.methods.incrementCount = function () {
// Static methods // Static methods
configSchema.statics.findByTag = async function (tag) { configSchema.statics.findByTag = async function (tag) {
return await this.findOne({ tag }); return await this.findOne({ tag }).lean();
}; };
configSchema.statics.updateByTag = async function (tag, update) { configSchema.statics.updateByTag = async function (tag, update) {
@ -67,7 +67,7 @@ const Config = mongoose.models.Config || mongoose.model('Config', configSchema);
module.exports = { module.exports = {
getConfigs: async (filter) => { getConfigs: async (filter) => {
try { try {
return await Config.find(filter).exec(); return await Config.find(filter).lean();
} catch (error) { } catch (error) {
console.error(error); console.error(error);
return { config: 'Error getting configs' }; return { config: 'Error getting configs' };
@ -75,7 +75,7 @@ module.exports = {
}, },
deleteConfigs: async (filter) => { deleteConfigs: async (filter) => {
try { try {
return await Config.deleteMany(filter).exec(); return await Config.deleteMany(filter);
} catch (error) { } catch (error) {
console.error(error); console.error(error);
return { config: 'Error deleting configs' }; return { config: 'Error deleting configs' };

View file

@ -4,7 +4,7 @@ const { getMessages, deleteMessages } = require('./Message');
const getConvo = async (user, conversationId) => { const getConvo = async (user, conversationId) => {
try { try {
return await Conversation.findOne({ user, conversationId }).exec(); return await Conversation.findOne({ user, conversationId }).lean();
} catch (error) { } catch (error) {
console.log(error); console.log(error);
return { message: 'Error getting single conversation' }; return { message: 'Error getting single conversation' };
@ -24,7 +24,7 @@ module.exports = {
return await Conversation.findOneAndUpdate({ conversationId: conversationId, user }, update, { return await Conversation.findOneAndUpdate({ conversationId: conversationId, user }, update, {
new: true, new: true,
upsert: true, upsert: true,
}).exec(); });
} catch (error) { } catch (error) {
console.log(error); console.log(error);
return { message: 'Error saving conversation' }; return { message: 'Error saving conversation' };
@ -35,10 +35,10 @@ module.exports = {
const totalConvos = (await Conversation.countDocuments({ user })) || 1; const totalConvos = (await Conversation.countDocuments({ user })) || 1;
const totalPages = Math.ceil(totalConvos / pageSize); const totalPages = Math.ceil(totalConvos / pageSize);
const convos = await Conversation.find({ user }) const convos = await Conversation.find({ user })
.sort({ createdAt: -1, created: -1 }) .sort({ createdAt: -1 })
.skip((pageNumber - 1) * pageSize) .skip((pageNumber - 1) * pageSize)
.limit(pageSize) .limit(pageSize)
.exec(); .lean();
return { conversations: convos, pages: totalPages, pageNumber, pageSize }; return { conversations: convos, pages: totalPages, pageNumber, pageSize };
} catch (error) { } catch (error) {
console.log(error); console.log(error);
@ -62,7 +62,7 @@ module.exports = {
Conversation.findOne({ Conversation.findOne({
user, user,
conversationId: convo.conversationId, conversationId: convo.conversationId,
}).exec(), }).lean(),
), ),
); );
@ -121,7 +121,7 @@ module.exports = {
deleteConvos: async (user, filter) => { deleteConvos: async (user, filter) => {
let toRemove = await Conversation.find({ ...filter, user }).select('conversationId'); let toRemove = await Conversation.find({ ...filter, user }).select('conversationId');
const ids = toRemove.map((instance) => instance.conversationId); const ids = toRemove.map((instance) => instance.conversationId);
let deleteCount = await Conversation.deleteMany({ ...filter, user }).exec(); let deleteCount = await Conversation.deleteMany({ ...filter, user });
deleteCount.messages = await deleteMessages({ conversationId: { $in: ids } }); deleteCount.messages = await deleteMessages({ conversationId: { $in: ids } });
return deleteCount; return deleteCount;
}, },

View file

@ -78,12 +78,12 @@ module.exports = {
}, },
async deleteMessagesSince({ messageId, conversationId }) { async deleteMessagesSince({ messageId, conversationId }) {
try { try {
const message = await Message.findOne({ messageId }).exec(); const message = await Message.findOne({ messageId }).lean();
if (message) { if (message) {
return await Message.find({ conversationId }) return await Message.find({ conversationId }).deleteMany({
.deleteMany({ createdAt: { $gt: message.createdAt } }) createdAt: { $gt: message.createdAt },
.exec(); });
} }
} catch (err) { } catch (err) {
console.error(`Error deleting messages: ${err}`); console.error(`Error deleting messages: ${err}`);
@ -93,7 +93,7 @@ module.exports = {
async getMessages(filter) { async getMessages(filter) {
try { try {
return await Message.find(filter).sort({ createdAt: 1 }).exec(); return await Message.find(filter).sort({ createdAt: 1 }).lean();
} catch (err) { } catch (err) {
console.error(`Error getting messages: ${err}`); console.error(`Error getting messages: ${err}`);
throw new Error('Failed to get messages.'); throw new Error('Failed to get messages.');
@ -102,7 +102,7 @@ module.exports = {
async deleteMessages(filter) { async deleteMessages(filter) {
try { try {
return await Message.deleteMany(filter).exec(); return await Message.deleteMany(filter);
} catch (err) { } catch (err) {
console.error(`Error deleting messages: ${err}`); console.error(`Error deleting messages: ${err}`);
throw new Error('Failed to delete messages.'); throw new Error('Failed to delete messages.');

View file

@ -2,7 +2,7 @@ const Preset = require('./schema/presetSchema');
const getPreset = async (user, presetId) => { const getPreset = async (user, presetId) => {
try { try {
return await Preset.findOne({ user, presetId }).exec(); return await Preset.findOne({ user, presetId }).lean();
} catch (error) { } catch (error) {
console.log(error); console.log(error);
return { message: 'Error getting single preset' }; return { message: 'Error getting single preset' };
@ -14,10 +14,10 @@ module.exports = {
getPreset, getPreset,
getPresets: async (user, filter) => { getPresets: async (user, filter) => {
try { try {
return await Preset.find({ ...filter, user }).exec(); return await Preset.find({ ...filter, user }).lean();
} catch (error) { } catch (error) {
console.log(error); console.log(error);
return { message: 'Error retriving presets' }; return { message: 'Error retrieving presets' };
} }
}, },
savePreset: async (user, { presetId, newPresetId, ...preset }) => { savePreset: async (user, { presetId, newPresetId, ...preset }) => {
@ -31,7 +31,7 @@ module.exports = {
{ presetId, user }, { presetId, user },
{ $set: update }, { $set: update },
{ new: true, upsert: true }, { new: true, upsert: true },
).exec(); );
} catch (error) { } catch (error) {
console.log(error); console.log(error);
return { message: 'Error saving preset' }; return { message: 'Error saving preset' };
@ -40,7 +40,7 @@ module.exports = {
deletePresets: async (user, filter) => { deletePresets: async (user, filter) => {
// let toRemove = await Preset.find({ ...filter, user }).select('presetId'); // let toRemove = await Preset.find({ ...filter, user }).select('presetId');
// const ids = toRemove.map((instance) => instance.presetId); // const ids = toRemove.map((instance) => instance.presetId);
let deleteCount = await Preset.deleteMany({ ...filter, user }).exec(); let deleteCount = await Preset.deleteMany({ ...filter, user });
return deleteCount; return deleteCount;
}, },
}; };

View file

@ -34,7 +34,7 @@ module.exports = {
}, },
getPrompts: async (filter) => { getPrompts: async (filter) => {
try { try {
return await Prompt.find(filter).exec(); return await Prompt.find(filter).lean();
} catch (error) { } catch (error) {
console.error(error); console.error(error);
return { prompt: 'Error getting prompts' }; return { prompt: 'Error getting prompts' };
@ -42,7 +42,7 @@ module.exports = {
}, },
deletePrompts: async (filter) => { deletePrompts: async (filter) => {
try { try {
return await Prompt.deleteMany(filter).exec(); return await Prompt.deleteMany(filter);
} catch (error) { } catch (error) {
console.error(error); console.error(error);
return { prompt: 'Error deleting prompts' }; return { prompt: 'Error deleting prompts' };

View file

@ -61,6 +61,8 @@ if (process.env.MEILI_HOST && process.env.MEILI_MASTER_KEY) {
}); });
} }
convoSchema.index({ createdAt: 1 });
const Conversation = mongoose.models.Conversation || mongoose.model('Conversation', convoSchema); const Conversation = mongoose.models.Conversation || mongoose.model('Conversation', convoSchema);
module.exports = Conversation; module.exports = Conversation;

View file

@ -100,6 +100,8 @@ if (process.env.MEILI_HOST && process.env.MEILI_MASTER_KEY) {
}); });
} }
messageSchema.index({ createdAt: 1 });
const Message = mongoose.models.Message || mongoose.model('Message', messageSchema); const Message = mongoose.models.Message || mongoose.model('Message', messageSchema);
module.exports = Message; module.exports = Message;

View file

@ -1,7 +1,6 @@
const express = require('express'); const express = require('express');
const session = require('express-session'); const session = require('express-session');
const connectDb = require('../lib/db/connectDb'); const connectDb = require('../lib/db/connectDb');
const migrateDb = require('../lib/db/migrateDb');
const indexSync = require('../lib/db/indexSync'); const indexSync = require('../lib/db/indexSync');
const path = require('path'); const path = require('path');
const cors = require('cors'); const cors = require('cors');
@ -28,7 +27,6 @@ config.validate(); // Validate the config
(async () => { (async () => {
await connectDb(); await connectDb();
console.log('Connected to MongoDB'); console.log('Connected to MongoDB');
await migrateDb();
await indexSync(); await indexSync();
const app = express(); const app = express();

View file

@ -14,7 +14,7 @@ router.get('/:conversationId', requireJwtAuth, async (req, res) => {
const convo = await getConvo(req.user.id, conversationId); const convo = await getConvo(req.user.id, conversationId);
if (convo) { if (convo) {
res.status(200).send(convo.toObject()); res.status(200).send(convo);
} else { } else {
res.status(404).end(); res.status(404).end();
} }

View file

@ -6,7 +6,7 @@ const requireJwtAuth = require('../../middleware/requireJwtAuth');
router.get('/', requireJwtAuth, async (req, res) => { router.get('/', requireJwtAuth, async (req, res) => {
const presets = (await getPresets(req.user.id)).map((preset) => { const presets = (await getPresets(req.user.id)).map((preset) => {
return preset.toObject(); return preset;
}); });
res.status(200).send(presets); res.status(200).send(presets);
}); });
@ -20,7 +20,7 @@ router.post('/', requireJwtAuth, async (req, res) => {
await savePreset(req.user.id, update); await savePreset(req.user.id, update);
const presets = (await getPresets(req.user.id)).map((preset) => { const presets = (await getPresets(req.user.id)).map((preset) => {
return preset.toObject(); return preset;
}); });
res.status(201).send(presets); res.status(201).send(presets);
} catch (error) { } catch (error) {
@ -41,12 +41,8 @@ router.post('/delete', requireJwtAuth, async (req, res) => {
try { try {
await deletePresets(req.user.id, filter); await deletePresets(req.user.id, filter);
const presets = await getPresets(req.user.id);
const presets = (await getPresets(req.user.id)).map((preset) => preset.toObject());
// console.log('delete preset response', presets);
res.status(201).send(presets); res.status(201).send(presets);
// res.status(201).send(dbResponse);
} catch (error) { } catch (error) {
console.error(error); console.error(error);
res.status(500).send(error); res.status(500).send(error);

View file

@ -3,7 +3,7 @@ const { encrypt, decrypt } = require('../../utils/');
const getUserPluginAuthValue = async (user, authField) => { const getUserPluginAuthValue = async (user, authField) => {
try { try {
const pluginAuth = await PluginAuth.findOne({ user, authField }); const pluginAuth = await PluginAuth.findOne({ user, authField }).lean();
if (!pluginAuth) { if (!pluginAuth) {
return null; return null;
} }
@ -43,7 +43,7 @@ const getUserPluginAuthValue = async (user, authField) => {
const updateUserPluginAuth = async (userId, authField, pluginKey, value) => { const updateUserPluginAuth = async (userId, authField, pluginKey, value) => {
try { try {
const encryptedValue = encrypt(value); const encryptedValue = encrypt(value);
const pluginAuth = await PluginAuth.findOne({ userId, authField }); const pluginAuth = await PluginAuth.findOne({ userId, authField }).lean();
if (pluginAuth) { if (pluginAuth) {
const pluginAuth = await PluginAuth.updateOne( const pluginAuth = await PluginAuth.updateOne(
{ userId, authField }, { userId, authField },

View file

@ -54,7 +54,7 @@ const registerUser = async (user) => {
const { email, password, name, username } = user; const { email, password, name, username } = user;
try { try {
const existingUser = await User.findOne({ email }); const existingUser = await User.findOne({ email }).lean();
if (existingUser) { if (existingUser) {
console.info( console.info(
@ -104,7 +104,7 @@ const registerUser = async (user) => {
* @returns * @returns
*/ */
const requestPasswordReset = async (email) => { const requestPasswordReset = async (email) => {
const user = await User.findOne({ email }); const user = await User.findOne({ email }).lean();
if (!user) { if (!user) {
return new Error('Email does not exist'); return new Error('Email does not exist');
} }