mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-15 05:37:22 +01:00
* 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>
91 lines
2.7 KiB
JavaScript
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;
|
|
}, {});
|
|
};
|
|
}
|