Files
habitica/website/server/libs/baseModel.js
Phillip Thelen f8d315ff6e Upgrade to mongoose 7 (#14971)
* remove some unused dependencies

* update mongoose version

* make common tests pass

* Make unit tests pass

* make api v3 integration tests pass

* fix lint issues

* fix issue with package-lock

* fix(lint): we don't need no .js

* fix(lint): update to latest config-habitrpg

* chore(npm): update package locks

* fix(test): replace deprecated fn

* chore(package): update eslint-habitrpg again

* fix(lint): server linting

* fix(lint): client linting

* fix(client): correct mangled common imports

* chore(npm): update package-locks

* fix(lint): punctuation, module

---------

Co-authored-by: SabreCat <sabrecat@gmail.com>
Co-authored-by: SabreCat <sabe@habitica.com>
2024-01-16 15:18:47 -06:00

87 lines
2.5 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('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;
}, {});
};
}