Mongo-Node/node_modules/mongoose/lib/schema/map.js
2025-11-21 09:23:11 +01:00

201 lines
5.4 KiB
JavaScript

'use strict';
/*!
* ignore
*/
const MongooseMap = require('../types/map');
const SchemaMapOptions = require('../options/schemaMapOptions');
const SchemaType = require('../schemaType');
const createJSONSchemaTypeDefinition = require('../helpers/createJSONSchemaTypeDefinition');
const MongooseError = require('../error/mongooseError');
const Schema = require('../schema');
const utils = require('../utils');
class SchemaMap extends SchemaType {
/**
* Map SchemaType constructor.
*
* @param {String} path
* @param {Object} options
* @param {Object} schemaOptions
* @param {Schema} parentSchema
* @inherits SchemaType
* @api public
*/
constructor(key, options, schemaOptions, parentSchema) {
super(key, options, 'Map', parentSchema);
this.$isSchemaMap = true;
// Create the nested schema type for the map values
this._createNestedSchemaType(parentSchema, key, options, schemaOptions);
}
/**
* Sets a default option for all Map instances.
*
* @param {String} option The option you'd like to set the value for
* @param {Any} value value for option
* @return {undefined}
* @function set
* @api public
*/
set(option, value) {
return SchemaType.set(option, value);
}
/**
* Casts to Map
*
* @param {Object} value
* @param {Object} model this value is optional
* @api private
*/
cast(val, doc, init, prev, options) {
if (val instanceof MongooseMap) {
return val;
}
const path = this.path;
if (init) {
const map = new MongooseMap({}, path, doc, this.$__schemaType, options);
// Use the map's path for passing to nested casts.
// If map's parent is a subdocument, use the relative path so nested casts get relative paths.
const mapPath = map.$__pathRelativeToParent != null ? map.$__pathRelativeToParent : map.$__path;
if (val instanceof global.Map) {
for (const key of val.keys()) {
let _val = val.get(key);
if (_val == null) {
_val = map.$__schemaType._castNullish(_val);
} else {
_val = map.$__schemaType.cast(_val, doc, true, null, { ...options, path: mapPath + '.' + key });
}
map.$init(key, _val);
}
} else {
for (const key of Object.keys(val)) {
let _val = val[key];
if (_val == null) {
_val = map.$__schemaType._castNullish(_val);
} else {
_val = map.$__schemaType.cast(_val, doc, true, null, { ...options, path: mapPath + '.' + key });
}
map.$init(key, _val);
}
}
return map;
}
return new MongooseMap(val, path, doc, this.$__schemaType, options);
}
/**
* Creates a copy of this map schema type.
*
* @api private
*/
clone() {
const schematype = super.clone();
if (this.$__schemaType != null) {
schematype.$__schemaType = this.$__schemaType.clone();
}
return schematype;
}
/**
* Returns the embedded schema type (i.e. the `.$*` path)
*
* @api public
*/
getEmbeddedSchemaType() {
return this.$__schemaType;
}
/**
* Returns this schema type's representation in a JSON schema.
*
* @param [options]
* @param [options.useBsonType=false] If true, return a representation with `bsonType` for use with MongoDB's `$jsonSchema`.
* @returns {Object} JSON schema properties
*/
toJSONSchema(options) {
const useBsonType = options?.useBsonType;
const embeddedSchemaType = this.getEmbeddedSchemaType();
const isRequired = this.options.required && typeof this.options.required !== 'function';
const result = createJSONSchemaTypeDefinition('object', 'object', useBsonType, isRequired);
result.additionalProperties = embeddedSchemaType.toJSONSchema(options);
return result;
}
/**
* Returns the auto encryption type for this schema type.
*
* @api public
*/
autoEncryptionType() {
return 'object';
}
}
/**
* This schema type's name, to defend against minifiers that mangle
* function names.
*
* @api public
*/
SchemaMap.schemaName = 'Map';
SchemaMap.prototype.OptionsConstructor = SchemaMapOptions;
SchemaMap.defaultOptions = {};
/*!
* ignore
*/
SchemaMap.prototype._createNestedSchemaType = function _createNestedSchemaType(schema, path, obj, options) {
const mapPath = path + '.$*';
let _mapType = { type: {} };
if (utils.hasUserDefinedProperty(obj, 'of')) {
const isInlineSchema = utils.isPOJO(obj.of) &&
Object.keys(obj.of).length > 0 &&
!utils.hasUserDefinedProperty(obj.of, schema.options.typeKey);
if (isInlineSchema) {
_mapType = { [schema.options.typeKey]: new Schema(obj.of) };
} else if (utils.isPOJO(obj.of)) {
_mapType = Object.assign({}, obj.of);
} else {
_mapType = { [schema.options.typeKey]: obj.of };
}
if (_mapType[schema.options.typeKey] && _mapType[schema.options.typeKey].instanceOfSchema) {
const subdocumentSchema = _mapType[schema.options.typeKey];
subdocumentSchema.eachPath((subpath, type) => {
if (type.options.select === true || type.options.select === false) {
throw new MongooseError('Cannot use schema-level projections (`select: true` or `select: false`) within maps at path "' + path + '.' + subpath + '"');
}
});
}
if (utils.hasUserDefinedProperty(obj, 'ref')) {
_mapType.ref = obj.ref;
}
}
this.$__schemaType = schema.interpretAsType(mapPath, _mapType, options);
};
module.exports = SchemaMap;