123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314 |
- /*!
- * Module dependencies.
- */
- var SchemaType = require('../schematype');
- var CastError = SchemaType.CastError;
- var Types = {
- Boolean: require('./boolean'),
- Date: require('./date'),
- Number: require('./number'),
- String: require('./string'),
- ObjectId: require('./objectid'),
- Buffer: require('./buffer')
- };
- var MongooseArray = require('../types').Array;
- var EmbeddedDoc = require('../types').Embedded;
- var Mixed = require('./mixed');
- var cast = require('../cast');
- var util = require('util');
- var utils = require('../utils');
- var castToNumber = require('./operators/helpers').castToNumber;
- var geospatial = require('./operators/geospatial');
- /**
- * Array SchemaType constructor
- *
- * @param {String} key
- * @param {SchemaType} cast
- * @param {Object} options
- * @inherits SchemaType
- * @api public
- */
- function SchemaArray(key, cast, options, schemaOptions) {
- var typeKey = 'type';
- if (schemaOptions && schemaOptions.typeKey) {
- typeKey = schemaOptions.typeKey;
- }
- if (cast) {
- var castOptions = {};
- if (utils.getFunctionName(cast.constructor) === 'Object') {
- if (cast[typeKey]) {
- // support { type: Woot }
- castOptions = utils.clone(cast); // do not alter user arguments
- delete castOptions[typeKey];
- cast = cast[typeKey];
- } else {
- cast = Mixed;
- }
- }
- // support { type: 'String' }
- var name = typeof cast === 'string'
- ? cast
- : utils.getFunctionName(cast);
- var caster = name in Types
- ? Types[name]
- : cast;
- this.casterConstructor = caster;
- if (typeof caster === 'function') {
- this.caster = new caster(null, castOptions);
- } else {
- this.caster = caster;
- }
- if (!(this.caster instanceof EmbeddedDoc)) {
- this.caster.path = key;
- }
- }
- SchemaType.call(this, key, options, 'Array');
- var defaultArr;
- var fn;
- if (this.defaultValue != null) {
- defaultArr = this.defaultValue;
- fn = typeof defaultArr === 'function';
- }
- if (!('defaultValue' in this) || this.defaultValue !== void 0) {
- this.default(function() {
- var arr = [];
- if (fn) {
- arr = defaultArr();
- } else if (defaultArr != null) {
- arr = defaultArr;
- }
- // Leave it up to `cast()` to convert the array
- return arr;
- });
- }
- }
- /**
- * This schema type's name, to defend against minifiers that mangle
- * function names.
- *
- * @api public
- */
- SchemaArray.schemaName = 'Array';
- /*!
- * Inherits from SchemaType.
- */
- SchemaArray.prototype = Object.create(SchemaType.prototype);
- SchemaArray.prototype.constructor = SchemaArray;
- /**
- * Check if the given value satisfies a required validator. The given value
- * must be not null nor undefined, and have a non-zero length.
- *
- * @param {Any} value
- * @return {Boolean}
- * @api public
- */
- SchemaArray.prototype.checkRequired = function(value) {
- return !!(value && value.length);
- };
- /**
- * Overrides the getters application for the population special-case
- *
- * @param {Object} value
- * @param {Object} scope
- * @api private
- */
- SchemaArray.prototype.applyGetters = function(value, scope) {
- if (this.caster.options && this.caster.options.ref) {
- // means the object id was populated
- return value;
- }
- return SchemaType.prototype.applyGetters.call(this, value, scope);
- };
- /**
- * Casts values for set().
- *
- * @param {Object} value
- * @param {Document} doc document that triggers the casting
- * @param {Boolean} init whether this is an initialization cast
- * @api private
- */
- SchemaArray.prototype.cast = function(value, doc, init) {
- if (Array.isArray(value)) {
- if (!value.length && doc) {
- var indexes = doc.schema.indexedPaths();
- for (var i = 0, l = indexes.length; i < l; ++i) {
- var pathIndex = indexes[i][0][this.path];
- if (pathIndex === '2dsphere' || pathIndex === '2d') {
- return;
- }
- }
- }
- if (!(value && value.isMongooseArray)) {
- value = new MongooseArray(value, this.path, doc);
- } else if (value && value.isMongooseArray) {
- // We need to create a new array, otherwise change tracking will
- // update the old doc (gh-4449)
- value = new MongooseArray(value, this.path, doc);
- }
- if (this.caster) {
- try {
- for (i = 0, l = value.length; i < l; i++) {
- value[i] = this.caster.cast(value[i], doc, init);
- }
- } catch (e) {
- // rethrow
- throw new CastError('[' + e.kind + ']', util.inspect(value), this.path, e);
- }
- }
- return value;
- }
- // gh-2442: if we're loading this from the db and its not an array, mark
- // the whole array as modified.
- if (!!doc && !!init) {
- doc.markModified(this.path);
- }
- return this.cast([value], doc, init);
- };
- /**
- * Casts values for queries.
- *
- * @param {String} $conditional
- * @param {any} [value]
- * @api private
- */
- SchemaArray.prototype.castForQuery = function($conditional, value) {
- var handler,
- val;
- if (arguments.length === 2) {
- handler = this.$conditionalHandlers[$conditional];
- if (!handler) {
- throw new Error('Can\'t use ' + $conditional + ' with Array.');
- }
- val = handler.call(this, value);
- } else {
- val = $conditional;
- var proto = this.casterConstructor.prototype;
- var method = proto && (proto.castForQuery || proto.cast);
- var caster = this.caster;
- if (Array.isArray(val)) {
- val = val.map(function(v) {
- if (utils.isObject(v) && v.$elemMatch) {
- return v;
- }
- if (method) {
- v = method.call(caster, v);
- }
- return v;
- });
- } else if (method) {
- val = method.call(caster, val);
- }
- }
- return val;
- };
- function cast$all(val) {
- if (!Array.isArray(val)) {
- val = [val];
- }
- val = val.map(function(v) {
- if (utils.isObject(v)) {
- var o = {};
- o[this.path] = v;
- return cast(this.casterConstructor.schema, o)[this.path];
- }
- return v;
- }, this);
- return this.castForQuery(val);
- }
- function cast$elemMatch(val) {
- var keys = Object.keys(val);
- var numKeys = keys.length;
- var key;
- var value;
- for (var i = 0; i < numKeys; ++i) {
- key = keys[i];
- value = val[key];
- if (key.indexOf('$') === 0 && value) {
- val[key] = this.castForQuery(key, value);
- }
- }
- return cast(this.casterConstructor.schema, val);
- }
- var handle = SchemaArray.prototype.$conditionalHandlers = {};
- handle.$all = cast$all;
- handle.$options = String;
- handle.$elemMatch = cast$elemMatch;
- handle.$geoIntersects = geospatial.cast$geoIntersects;
- handle.$or = handle.$and = function(val) {
- if (!Array.isArray(val)) {
- throw new TypeError('conditional $or/$and require array');
- }
- var ret = [];
- for (var i = 0; i < val.length; ++i) {
- ret.push(cast(this.casterConstructor.schema, val[i]));
- }
- return ret;
- };
- handle.$near =
- handle.$nearSphere = geospatial.cast$near;
- handle.$within =
- handle.$geoWithin = geospatial.cast$within;
- handle.$size =
- handle.$minDistance =
- handle.$maxDistance = castToNumber;
- handle.$eq =
- handle.$gt =
- handle.$gte =
- handle.$in =
- handle.$lt =
- handle.$lte =
- handle.$ne =
- handle.$nin =
- handle.$regex = SchemaArray.prototype.castForQuery;
- /*!
- * Module exports.
- */
- module.exports = SchemaArray;
|