Files
habitica/website/server/libs/baseModel.js
Phillip Thelen 8150fef993 Database Access optimisations (#14544)
* Optimize database access during spell casting

* load less data when casting spells

* Begin migrating update calls to updateOne and updateMany

* Only update user objects that don’t have notification yet

* fix test

* fix spy

* Don’t unnecessarily update user when requesting invalid guild

* fix sort order for middlewares to not load user twice every request

* fix tests

* fix integration test

* fix skill usage not always deducting mp

* addtest case for blessing spell

* fix healAll

* fix lint

* Fix error for when some spells are used outside of party

* Add check to not run bulk spells in web client

* fix(tags): change const to let

---------

Co-authored-by: SabreCat <sabe@habitica.com>
2023-05-16 12:21:45 -05:00

91 lines
2.7 KiB
JavaScript

import { v4 as uuid } from 'uuid';
import validator from 'validator';
import _ from 'lodash';
export default function baseModel (schema, options = {}) {
if (schema.options.typeKey !== '$type') {
throw new Error('Every schema must use $type as the typeKey, see https://mongoosejs.com/docs/guide.html#typeKey');
}
if (options._id !== false) {
schema.add({
_id: {
$type: String,
default: uuid,
validate: [v => validator.isUUID(v), 'Invalid uuid in baseModel.'],
},
});
}
if (options.timestamps) {
schema.add({
createdAt: {
$type: Date,
default: Date.now,
},
updatedAt: {
$type: Date,
default: Date.now,
},
});
}
if (options.timestamps) {
schema.pre('save', function updateUpdatedAt (next) {
if (!this.isNew) this.updatedAt = Date.now();
next();
});
schema.pre('update', function preUpdateModel () {
this.set({}, { $set: { updatedAt: new Date() } });
});
schema.pre('updateOne', function preUpdateModel () {
this.set({}, { $set: { updatedAt: new Date() } });
});
schema.pre('updateMany', function preUpdateModel () {
this.set({}, { $set: { updatedAt: new Date() } });
});
}
const noSetFields = ['createdAt', 'updatedAt'];
const privateFields = ['__v'];
if (Array.isArray(options.noSet)) noSetFields.push(...options.noSet);
// This method accepts an additional array of fields to be sanitized that can be passed at runtime
schema.statics.sanitize = function sanitize (objToSanitize = {}, additionalFields = []) {
noSetFields.concat(additionalFields).forEach(fieldPath => {
_.unset(objToSanitize, fieldPath);
});
// Allow a sanitize transform function to be used
return options.sanitizeTransform ? options.sanitizeTransform(objToSanitize) : objToSanitize;
};
if (Array.isArray(options.private)) privateFields.push(...options.private);
if (!schema.options.toJSON) schema.options.toJSON = {};
schema.options.toJSON.transform = function transformToObject (doc, plainObj) {
privateFields.forEach(fieldPath => {
_.unset(plainObj, fieldPath);
});
// Always return `id`
if (!plainObj.id && plainObj._id) plainObj.id = plainObj._id;
// Allow an additional toJSON transform function to be used
return options.toJSONTransform ? options.toJSONTransform(plainObj, doc) : plainObj;
};
schema.statics.getModelPaths = function getModelPaths () {
return _.reduce(this.schema.paths, (result, field, path) => {
if (privateFields.indexOf(path) === -1) {
result[path] = field.instance || 'Boolean';
}
return result;
}, {});
};
}