schema.js 62 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147
  1. 'use strict';
  2. /*!
  3. * Module dependencies.
  4. */
  5. const EventEmitter = require('events').EventEmitter;
  6. const Kareem = require('kareem');
  7. const MongooseError = require('./error/mongooseError');
  8. const SchemaType = require('./schematype');
  9. const SchemaTypeOptions = require('./options/SchemaTypeOptions');
  10. const VirtualOptions = require('./options/VirtualOptions');
  11. const VirtualType = require('./virtualtype');
  12. const addAutoId = require('./helpers/schema/addAutoId');
  13. const arrayParentSymbol = require('./helpers/symbols').arrayParentSymbol;
  14. const get = require('./helpers/get');
  15. const getIndexes = require('./helpers/schema/getIndexes');
  16. const merge = require('./helpers/schema/merge');
  17. const mpath = require('mpath');
  18. const readPref = require('./driver').get().ReadPreference;
  19. const setupTimestamps = require('./helpers/timestamps/setupTimestamps');
  20. const util = require('util');
  21. const utils = require('./utils');
  22. const validateRef = require('./helpers/populate/validateRef');
  23. let MongooseTypes;
  24. const queryHooks = require('./helpers/query/applyQueryMiddleware').
  25. middlewareFunctions;
  26. const documentHooks = require('./helpers/model/applyHooks').middlewareFunctions;
  27. const hookNames = queryHooks.concat(documentHooks).
  28. reduce((s, hook) => s.add(hook), new Set());
  29. let id = 0;
  30. /**
  31. * Schema constructor.
  32. *
  33. * ####Example:
  34. *
  35. * const child = new Schema({ name: String });
  36. * const schema = new Schema({ name: String, age: Number, children: [child] });
  37. * const Tree = mongoose.model('Tree', schema);
  38. *
  39. * // setting schema options
  40. * new Schema({ name: String }, { _id: false, autoIndex: false })
  41. *
  42. * ####Options:
  43. *
  44. * - [autoIndex](/docs/guide.html#autoIndex): bool - defaults to null (which means use the connection's autoIndex option)
  45. * - [autoCreate](/docs/guide.html#autoCreate): bool - defaults to null (which means use the connection's autoCreate option)
  46. * - [bufferCommands](/docs/guide.html#bufferCommands): bool - defaults to true
  47. * - [bufferTimeoutMS](/docs/guide.html#bufferTimeoutMS): number - defaults to 10000 (10 seconds). If `bufferCommands` is enabled, the amount of time Mongoose will wait for connectivity to be restablished before erroring out.
  48. * - [capped](/docs/guide.html#capped): bool - defaults to false
  49. * - [collection](/docs/guide.html#collection): string - no default
  50. * - [id](/docs/guide.html#id): bool - defaults to true
  51. * - [_id](/docs/guide.html#_id): bool - defaults to true
  52. * - [minimize](/docs/guide.html#minimize): bool - controls [document#toObject](#document_Document-toObject) behavior when called manually - defaults to true
  53. * - [read](/docs/guide.html#read): string
  54. * - [writeConcern](/docs/guide.html#writeConcern): object - defaults to null, use to override [the MongoDB server's default write concern settings](https://docs.mongodb.com/manual/reference/write-concern/)
  55. * - [shardKey](/docs/guide.html#shardKey): object - defaults to `null`
  56. * - [strict](/docs/guide.html#strict): bool - defaults to true
  57. * - [strictQuery](/docs/guide.html#strictQuery): bool - defaults to false
  58. * - [toJSON](/docs/guide.html#toJSON) - object - no default
  59. * - [toObject](/docs/guide.html#toObject) - object - no default
  60. * - [typeKey](/docs/guide.html#typeKey) - string - defaults to 'type'
  61. * - [typePojoToMixed](/docs/guide.html#typePojoToMixed) - boolean - defaults to true. Determines whether a type set to a POJO becomes a Mixed path or a Subdocument
  62. * - [useNestedStrict](/docs/guide.html#useNestedStrict) - boolean - defaults to false
  63. * - [validateBeforeSave](/docs/guide.html#validateBeforeSave) - bool - defaults to `true`
  64. * - [versionKey](/docs/guide.html#versionKey): string or object - defaults to "__v"
  65. * - [optimisticConcurrency](/docs/guide.html#optimisticConcurrency): bool - defaults to false. Set to true to enable [optimistic concurrency](https://thecodebarbarian.com/whats-new-in-mongoose-5-10-optimistic-concurrency.html).
  66. * - [collation](/docs/guide.html#collation): object - defaults to null (which means use no collation)
  67. * - [selectPopulatedPaths](/docs/guide.html#selectPopulatedPaths): boolean - defaults to `true`
  68. * - [skipVersioning](/docs/guide.html#skipVersioning): object - paths to exclude from versioning
  69. * - [timestamps](/docs/guide.html#timestamps): object or boolean - defaults to `false`. If true, Mongoose adds `createdAt` and `updatedAt` properties to your schema and manages those properties for you.
  70. * - [storeSubdocValidationError](/docs/guide.html#storeSubdocValidationError): boolean - Defaults to true. If false, Mongoose will wrap validation errors in single nested document subpaths into a single validation error on the single nested subdoc's path.
  71. *
  72. * ####Options for Nested Schemas:
  73. * - `excludeIndexes`: bool - defaults to `false`. If `true`, skip building indexes on this schema's paths.
  74. *
  75. * ####Note:
  76. *
  77. * _When nesting schemas, (`children` in the example above), always declare the child schema first before passing it into its parent._
  78. *
  79. * @param {Object|Schema|Array} [definition] Can be one of: object describing schema paths, or schema to copy, or array of objects and schemas
  80. * @param {Object} [options]
  81. * @inherits NodeJS EventEmitter http://nodejs.org/api/events.html#events_class_events_eventemitter
  82. * @event `init`: Emitted after the schema is compiled into a `Model`.
  83. * @api public
  84. */
  85. function Schema(obj, options) {
  86. if (!(this instanceof Schema)) {
  87. return new Schema(obj, options);
  88. }
  89. this.obj = obj;
  90. this.paths = {};
  91. this.aliases = {};
  92. this.subpaths = {};
  93. this.virtuals = {};
  94. this.singleNestedPaths = {};
  95. this.nested = {};
  96. this.inherits = {};
  97. this.callQueue = [];
  98. this._indexes = [];
  99. this.methods = {};
  100. this.methodOptions = {};
  101. this.statics = {};
  102. this.tree = {};
  103. this.query = {};
  104. this.childSchemas = [];
  105. this.plugins = [];
  106. // For internal debugging. Do not use this to try to save a schema in MDB.
  107. this.$id = ++id;
  108. this.s = {
  109. hooks: new Kareem()
  110. };
  111. this.options = this.defaultOptions(options);
  112. // build paths
  113. if (Array.isArray(obj)) {
  114. for (const definition of obj) {
  115. this.add(definition);
  116. }
  117. } else if (obj) {
  118. this.add(obj);
  119. }
  120. // check if _id's value is a subdocument (gh-2276)
  121. const _idSubDoc = obj && obj._id && utils.isObject(obj._id);
  122. // ensure the documents get an auto _id unless disabled
  123. const auto_id = !this.paths['_id'] &&
  124. (!this.options.noId && this.options._id) && !_idSubDoc;
  125. if (auto_id) {
  126. addAutoId(this);
  127. }
  128. this.setupTimestamp(this.options.timestamps);
  129. }
  130. /*!
  131. * Create virtual properties with alias field
  132. */
  133. function aliasFields(schema, paths) {
  134. paths = paths || Object.keys(schema.paths);
  135. for (const path of paths) {
  136. const options = get(schema.paths[path], 'options');
  137. if (options == null) {
  138. continue;
  139. }
  140. const prop = schema.paths[path].path;
  141. const alias = options.alias;
  142. if (!alias) {
  143. continue;
  144. }
  145. if (typeof alias !== 'string') {
  146. throw new Error('Invalid value for alias option on ' + prop + ', got ' + alias);
  147. }
  148. schema.aliases[alias] = prop;
  149. schema.
  150. virtual(alias).
  151. get((function(p) {
  152. return function() {
  153. if (typeof this.get === 'function') {
  154. return this.get(p);
  155. }
  156. return this[p];
  157. };
  158. })(prop)).
  159. set((function(p) {
  160. return function(v) {
  161. return this.$set(p, v);
  162. };
  163. })(prop));
  164. }
  165. }
  166. /*!
  167. * Inherit from EventEmitter.
  168. */
  169. Schema.prototype = Object.create(EventEmitter.prototype);
  170. Schema.prototype.constructor = Schema;
  171. Schema.prototype.instanceOfSchema = true;
  172. /*!
  173. * ignore
  174. */
  175. Object.defineProperty(Schema.prototype, '$schemaType', {
  176. configurable: false,
  177. enumerable: false,
  178. writable: true
  179. });
  180. /**
  181. * Array of child schemas (from document arrays and single nested subdocs)
  182. * and their corresponding compiled models. Each element of the array is
  183. * an object with 2 properties: `schema` and `model`.
  184. *
  185. * This property is typically only useful for plugin authors and advanced users.
  186. * You do not need to interact with this property at all to use mongoose.
  187. *
  188. * @api public
  189. * @property childSchemas
  190. * @memberOf Schema
  191. * @instance
  192. */
  193. Object.defineProperty(Schema.prototype, 'childSchemas', {
  194. configurable: false,
  195. enumerable: true,
  196. writable: true
  197. });
  198. /**
  199. * The original object passed to the schema constructor
  200. *
  201. * ####Example:
  202. *
  203. * const schema = new Schema({ a: String }).add({ b: String });
  204. * schema.obj; // { a: String }
  205. *
  206. * @api public
  207. * @property obj
  208. * @memberOf Schema
  209. * @instance
  210. */
  211. Schema.prototype.obj;
  212. /**
  213. * The paths defined on this schema. The keys are the top-level paths
  214. * in this schema, and the values are instances of the SchemaType class.
  215. *
  216. * ####Example:
  217. * const schema = new Schema({ name: String }, { _id: false });
  218. * schema.paths; // { name: SchemaString { ... } }
  219. *
  220. * schema.add({ age: Number });
  221. * schema.paths; // { name: SchemaString { ... }, age: SchemaNumber { ... } }
  222. *
  223. * @api public
  224. * @property paths
  225. * @memberOf Schema
  226. * @instance
  227. */
  228. Schema.prototype.paths;
  229. /**
  230. * Schema as a tree
  231. *
  232. * ####Example:
  233. * {
  234. * '_id' : ObjectId
  235. * , 'nested' : {
  236. * 'key' : String
  237. * }
  238. * }
  239. *
  240. * @api private
  241. * @property tree
  242. * @memberOf Schema
  243. * @instance
  244. */
  245. Schema.prototype.tree;
  246. /**
  247. * Returns a deep copy of the schema
  248. *
  249. * ####Example:
  250. *
  251. * const schema = new Schema({ name: String });
  252. * const clone = schema.clone();
  253. * clone === schema; // false
  254. * clone.path('name'); // SchemaString { ... }
  255. *
  256. * @return {Schema} the cloned schema
  257. * @api public
  258. * @memberOf Schema
  259. * @instance
  260. */
  261. Schema.prototype.clone = function() {
  262. const Constructor = this.base == null ? Schema : this.base.Schema;
  263. const s = new Constructor({}, this._userProvidedOptions);
  264. s.base = this.base;
  265. s.obj = this.obj;
  266. s.options = utils.clone(this.options);
  267. s.callQueue = this.callQueue.map(function(f) { return f; });
  268. s.methods = utils.clone(this.methods);
  269. s.methodOptions = utils.clone(this.methodOptions);
  270. s.statics = utils.clone(this.statics);
  271. s.query = utils.clone(this.query);
  272. s.plugins = Array.prototype.slice.call(this.plugins);
  273. s._indexes = utils.clone(this._indexes);
  274. s.s.hooks = this.s.hooks.clone();
  275. s.tree = utils.clone(this.tree);
  276. s.paths = utils.clone(this.paths);
  277. s.nested = utils.clone(this.nested);
  278. s.subpaths = utils.clone(this.subpaths);
  279. s.singleNestedPaths = utils.clone(this.singleNestedPaths);
  280. s.childSchemas = gatherChildSchemas(s);
  281. s.virtuals = utils.clone(this.virtuals);
  282. s.$globalPluginsApplied = this.$globalPluginsApplied;
  283. s.$isRootDiscriminator = this.$isRootDiscriminator;
  284. s.$implicitlyCreated = this.$implicitlyCreated;
  285. if (this.discriminatorMapping != null) {
  286. s.discriminatorMapping = Object.assign({}, this.discriminatorMapping);
  287. }
  288. if (this.discriminators != null) {
  289. s.discriminators = Object.assign({}, this.discriminators);
  290. }
  291. s.aliases = Object.assign({}, this.aliases);
  292. // Bubble up `init` for backwards compat
  293. s.on('init', v => this.emit('init', v));
  294. return s;
  295. };
  296. /**
  297. * Returns a new schema that has the picked `paths` from this schema.
  298. *
  299. * This method is analagous to [Lodash's `pick()` function](https://lodash.com/docs/4.17.15#pick) for Mongoose schemas.
  300. *
  301. * ####Example:
  302. *
  303. * const schema = Schema({ name: String, age: Number });
  304. * // Creates a new schema with the same `name` path as `schema`,
  305. * // but no `age` path.
  306. * const newSchema = schema.pick(['name']);
  307. *
  308. * newSchema.path('name'); // SchemaString { ... }
  309. * newSchema.path('age'); // undefined
  310. *
  311. * @param {Array} paths list of paths to pick
  312. * @param {Object} [options] options to pass to the schema constructor. Defaults to `this.options` if not set.
  313. * @return {Schema}
  314. * @api public
  315. */
  316. Schema.prototype.pick = function(paths, options) {
  317. const newSchema = new Schema({}, options || this.options);
  318. if (!Array.isArray(paths)) {
  319. throw new MongooseError('Schema#pick() only accepts an array argument, ' +
  320. 'got "' + typeof paths + '"');
  321. }
  322. for (const path of paths) {
  323. if (this.nested[path]) {
  324. newSchema.add({ [path]: get(this.tree, path) });
  325. } else {
  326. const schematype = this.path(path);
  327. if (schematype == null) {
  328. throw new MongooseError('Path `' + path + '` is not in the schema');
  329. }
  330. newSchema.add({ [path]: schematype });
  331. }
  332. }
  333. return newSchema;
  334. };
  335. /**
  336. * Returns default options for this schema, merged with `options`.
  337. *
  338. * @param {Object} options
  339. * @return {Object}
  340. * @api private
  341. */
  342. Schema.prototype.defaultOptions = function(options) {
  343. if (options && options.safe === false) {
  344. options.safe = { w: 0 };
  345. }
  346. if (options && options.safe && options.safe.w === 0) {
  347. // if you turn off safe writes, then versioning goes off as well
  348. options.versionKey = false;
  349. }
  350. this._userProvidedOptions = options == null ? {} : utils.clone(options);
  351. const baseOptions = get(this, 'base.options', {});
  352. options = utils.options({
  353. strict: 'strict' in baseOptions ? baseOptions.strict : true,
  354. strictQuery: 'strictQuery' in baseOptions ? baseOptions.strictQuery : false,
  355. bufferCommands: true,
  356. capped: false, // { size, max, autoIndexId }
  357. versionKey: '__v',
  358. optimisticConcurrency: false,
  359. discriminatorKey: '__t',
  360. minimize: true,
  361. autoIndex: null,
  362. shardKey: null,
  363. read: null,
  364. validateBeforeSave: true,
  365. // the following are only applied at construction time
  366. noId: false, // deprecated, use { _id: false }
  367. _id: true,
  368. noVirtualId: false, // deprecated, use { id: false }
  369. id: true,
  370. typeKey: 'type',
  371. typePojoToMixed: 'typePojoToMixed' in baseOptions ? baseOptions.typePojoToMixed : true
  372. }, utils.clone(options));
  373. if (options.read) {
  374. options.read = readPref(options.read);
  375. }
  376. if (options.optimisticConcurrency && !options.versionKey) {
  377. throw new MongooseError('Must set `versionKey` if using `optimisticConcurrency`');
  378. }
  379. return options;
  380. };
  381. /**
  382. * Adds key path / schema type pairs to this schema.
  383. *
  384. * ####Example:
  385. *
  386. * const ToySchema = new Schema();
  387. * ToySchema.add({ name: 'string', color: 'string', price: 'number' });
  388. *
  389. * const TurboManSchema = new Schema();
  390. * // You can also `add()` another schema and copy over all paths, virtuals,
  391. * // getters, setters, indexes, methods, and statics.
  392. * TurboManSchema.add(ToySchema).add({ year: Number });
  393. *
  394. * @param {Object|Schema} obj plain object with paths to add, or another schema
  395. * @param {String} [prefix] path to prefix the newly added paths with
  396. * @return {Schema} the Schema instance
  397. * @api public
  398. */
  399. Schema.prototype.add = function add(obj, prefix) {
  400. if (obj instanceof Schema || (obj != null && obj.instanceOfSchema)) {
  401. merge(this, obj);
  402. return this;
  403. }
  404. // Special case: setting top-level `_id` to false should convert to disabling
  405. // the `_id` option. This behavior never worked before 5.4.11 but numerous
  406. // codebases use it (see gh-7516, gh-7512).
  407. if (obj._id === false && prefix == null) {
  408. this.options._id = false;
  409. }
  410. prefix = prefix || '';
  411. // avoid prototype pollution
  412. if (prefix === '__proto__.' || prefix === 'constructor.' || prefix === 'prototype.') {
  413. return this;
  414. }
  415. const keys = Object.keys(obj);
  416. for (const key of keys) {
  417. const fullPath = prefix + key;
  418. if (obj[key] == null) {
  419. throw new TypeError('Invalid value for schema path `' + fullPath +
  420. '`, got value "' + obj[key] + '"');
  421. }
  422. // Retain `_id: false` but don't set it as a path, re: gh-8274.
  423. if (key === '_id' && obj[key] === false) {
  424. continue;
  425. }
  426. if (obj[key] instanceof VirtualType || get(obj[key], 'constructor.name', null) === 'VirtualType') {
  427. this.virtual(obj[key]);
  428. continue;
  429. }
  430. if (Array.isArray(obj[key]) && obj[key].length === 1 && obj[key][0] == null) {
  431. throw new TypeError('Invalid value for schema Array path `' + fullPath +
  432. '`, got value "' + obj[key][0] + '"');
  433. }
  434. if (!(utils.isPOJO(obj[key]) || obj[key] instanceof SchemaTypeOptions)) {
  435. // Special-case: Non-options definitely a path so leaf at this node
  436. // Examples: Schema instances, SchemaType instances
  437. if (prefix) {
  438. this.nested[prefix.substr(0, prefix.length - 1)] = true;
  439. }
  440. this.path(prefix + key, obj[key]);
  441. } else if (Object.keys(obj[key]).length < 1) {
  442. // Special-case: {} always interpreted as Mixed path so leaf at this node
  443. if (prefix) {
  444. this.nested[prefix.substr(0, prefix.length - 1)] = true;
  445. }
  446. this.path(fullPath, obj[key]); // mixed type
  447. } else if (!obj[key][this.options.typeKey] || (this.options.typeKey === 'type' && obj[key].type.type)) {
  448. // Special-case: POJO with no bona-fide type key - interpret as tree of deep paths so recurse
  449. // nested object { last: { name: String }}
  450. this.nested[fullPath] = true;
  451. this.add(obj[key], fullPath + '.');
  452. } else {
  453. // There IS a bona-fide type key that may also be a POJO
  454. if (!this.options.typePojoToMixed && utils.isPOJO(obj[key][this.options.typeKey])) {
  455. // If a POJO is the value of a type key, make it a subdocument
  456. if (prefix) {
  457. this.nested[prefix.substr(0, prefix.length - 1)] = true;
  458. }
  459. // Propage `typePojoToMixed` to implicitly created schemas
  460. const opts = { typePojoToMixed: false };
  461. const _schema = new Schema(obj[key][this.options.typeKey], opts);
  462. const schemaWrappedPath = Object.assign({}, obj[key], { [this.options.typeKey]: _schema });
  463. this.path(prefix + key, schemaWrappedPath);
  464. } else {
  465. // Either the type is non-POJO or we interpret it as Mixed anyway
  466. if (prefix) {
  467. this.nested[prefix.substr(0, prefix.length - 1)] = true;
  468. }
  469. this.path(prefix + key, obj[key]);
  470. }
  471. }
  472. }
  473. const addedKeys = Object.keys(obj).
  474. map(key => prefix ? prefix + key : key);
  475. aliasFields(this, addedKeys);
  476. return this;
  477. };
  478. /**
  479. * Reserved document keys.
  480. *
  481. * Keys in this object are names that are rejected in schema declarations
  482. * because they conflict with Mongoose functionality. If you create a schema
  483. * using `new Schema()` with one of these property names, Mongoose will throw
  484. * an error.
  485. *
  486. * - _posts
  487. * - _pres
  488. * - collection
  489. * - emit
  490. * - errors
  491. * - get
  492. * - init
  493. * - isModified
  494. * - isNew
  495. * - listeners
  496. * - modelName
  497. * - on
  498. * - once
  499. * - populated
  500. * - prototype
  501. * - remove
  502. * - removeListener
  503. * - save
  504. * - schema
  505. * - toObject
  506. * - validate
  507. *
  508. * _NOTE:_ Use of these terms as method names is permitted, but play at your own risk, as they may be existing mongoose document methods you are stomping on.
  509. *
  510. * const schema = new Schema(..);
  511. * schema.methods.init = function () {} // potentially breaking
  512. */
  513. Schema.reserved = Object.create(null);
  514. Schema.prototype.reserved = Schema.reserved;
  515. const reserved = Schema.reserved;
  516. // Core object
  517. reserved['prototype'] =
  518. // EventEmitter
  519. reserved.emit =
  520. reserved.listeners =
  521. reserved.on =
  522. reserved.removeListener =
  523. // document properties and functions
  524. reserved.collection =
  525. reserved.errors =
  526. reserved.get =
  527. reserved.init =
  528. reserved.isModified =
  529. reserved.isNew =
  530. reserved.populated =
  531. reserved.remove =
  532. reserved.save =
  533. reserved.toObject =
  534. reserved.validate = 1;
  535. /**
  536. * Gets/sets schema paths.
  537. *
  538. * Sets a path (if arity 2)
  539. * Gets a path (if arity 1)
  540. *
  541. * ####Example
  542. *
  543. * schema.path('name') // returns a SchemaType
  544. * schema.path('name', Number) // changes the schemaType of `name` to Number
  545. *
  546. * @param {String} path
  547. * @param {Object} constructor
  548. * @api public
  549. */
  550. Schema.prototype.path = function(path, obj) {
  551. // Convert to '.$' to check subpaths re: gh-6405
  552. const cleanPath = _pathToPositionalSyntax(path);
  553. if (obj === undefined) {
  554. let schematype = _getPath(this, path, cleanPath);
  555. if (schematype != null) {
  556. return schematype;
  557. }
  558. // Look for maps
  559. const mapPath = getMapPath(this, path);
  560. if (mapPath != null) {
  561. return mapPath;
  562. }
  563. // Look if a parent of this path is mixed
  564. schematype = this.hasMixedParent(cleanPath);
  565. if (schematype != null) {
  566. return schematype;
  567. }
  568. // subpaths?
  569. return /\.\d+\.?.*$/.test(path)
  570. ? getPositionalPath(this, path)
  571. : undefined;
  572. }
  573. // some path names conflict with document methods
  574. const firstPieceOfPath = path.split('.')[0];
  575. if (reserved[firstPieceOfPath]) {
  576. throw new Error('`' + firstPieceOfPath + '` may not be used as a schema pathname');
  577. }
  578. if (typeof obj === 'object' && utils.hasUserDefinedProperty(obj, 'ref')) {
  579. validateRef(obj.ref, path);
  580. }
  581. // update the tree
  582. const subpaths = path.split(/\./);
  583. const last = subpaths.pop();
  584. let branch = this.tree;
  585. let fullPath = '';
  586. for (const sub of subpaths) {
  587. fullPath = fullPath += (fullPath.length > 0 ? '.' : '') + sub;
  588. if (!branch[sub]) {
  589. this.nested[fullPath] = true;
  590. branch[sub] = {};
  591. }
  592. if (typeof branch[sub] !== 'object') {
  593. const msg = 'Cannot set nested path `' + path + '`. '
  594. + 'Parent path `'
  595. + fullPath
  596. + '` already set to type ' + branch[sub].name
  597. + '.';
  598. throw new Error(msg);
  599. }
  600. branch = branch[sub];
  601. }
  602. branch[last] = utils.clone(obj);
  603. this.paths[path] = this.interpretAsType(path, obj, this.options);
  604. const schemaType = this.paths[path];
  605. if (schemaType.$isSchemaMap) {
  606. // Maps can have arbitrary keys, so `$*` is internal shorthand for "any key"
  607. // The '$' is to imply this path should never be stored in MongoDB so we
  608. // can easily build a regexp out of this path, and '*' to imply "any key."
  609. const mapPath = path + '.$*';
  610. let _mapType = { type: {} };
  611. if (utils.hasUserDefinedProperty(obj, 'of')) {
  612. const isInlineSchema = utils.isPOJO(obj.of) &&
  613. Object.keys(obj.of).length > 0 &&
  614. !utils.hasUserDefinedProperty(obj.of, this.options.typeKey);
  615. _mapType = isInlineSchema ? new Schema(obj.of) : obj.of;
  616. }
  617. this.paths[mapPath] = this.interpretAsType(mapPath,
  618. _mapType, this.options);
  619. schemaType.$__schemaType = this.paths[mapPath];
  620. }
  621. if (schemaType.$isSingleNested) {
  622. for (const key of Object.keys(schemaType.schema.paths)) {
  623. this.singleNestedPaths[path + '.' + key] = schemaType.schema.paths[key];
  624. }
  625. for (const key of Object.keys(schemaType.schema.singleNestedPaths)) {
  626. this.singleNestedPaths[path + '.' + key] =
  627. schemaType.schema.singleNestedPaths[key];
  628. }
  629. for (const key of Object.keys(schemaType.schema.subpaths)) {
  630. this.singleNestedPaths[path + '.' + key] =
  631. schemaType.schema.subpaths[key];
  632. }
  633. for (const key of Object.keys(schemaType.schema.nested)) {
  634. this.singleNestedPaths[path + '.' + key] = 'nested';
  635. }
  636. Object.defineProperty(schemaType.schema, 'base', {
  637. configurable: true,
  638. enumerable: false,
  639. writable: false,
  640. value: this.base
  641. });
  642. schemaType.caster.base = this.base;
  643. this.childSchemas.push({
  644. schema: schemaType.schema,
  645. model: schemaType.caster
  646. });
  647. } else if (schemaType.$isMongooseDocumentArray) {
  648. Object.defineProperty(schemaType.schema, 'base', {
  649. configurable: true,
  650. enumerable: false,
  651. writable: false,
  652. value: this.base
  653. });
  654. schemaType.casterConstructor.base = this.base;
  655. this.childSchemas.push({
  656. schema: schemaType.schema,
  657. model: schemaType.casterConstructor
  658. });
  659. }
  660. if (schemaType.$isMongooseArray && schemaType.caster instanceof SchemaType) {
  661. let arrayPath = path;
  662. let _schemaType = schemaType;
  663. const toAdd = [];
  664. while (_schemaType.$isMongooseArray) {
  665. arrayPath = arrayPath + '.$';
  666. // Skip arrays of document arrays
  667. if (_schemaType.$isMongooseDocumentArray) {
  668. _schemaType.$embeddedSchemaType._arrayPath = arrayPath;
  669. _schemaType.$embeddedSchemaType._arrayParentPath = path;
  670. _schemaType = _schemaType.$embeddedSchemaType.clone();
  671. } else {
  672. _schemaType.caster._arrayPath = arrayPath;
  673. _schemaType.caster._arrayParentPath = path;
  674. _schemaType = _schemaType.caster.clone();
  675. }
  676. _schemaType.path = arrayPath;
  677. toAdd.push(_schemaType);
  678. }
  679. for (const _schemaType of toAdd) {
  680. this.subpaths[_schemaType.path] = _schemaType;
  681. }
  682. }
  683. if (schemaType.$isMongooseDocumentArray) {
  684. for (const key of Object.keys(schemaType.schema.paths)) {
  685. this.subpaths[path + '.' + key] = schemaType.schema.paths[key];
  686. schemaType.schema.paths[key].$isUnderneathDocArray = true;
  687. }
  688. for (const key of Object.keys(schemaType.schema.subpaths)) {
  689. this.subpaths[path + '.' + key] = schemaType.schema.subpaths[key];
  690. schemaType.schema.subpaths[key].$isUnderneathDocArray = true;
  691. }
  692. for (const key of Object.keys(schemaType.schema.singleNestedPaths)) {
  693. const _schemaType = schemaType.schema.singleNestedPaths[key];
  694. this.subpaths[path + '.' + key] = _schemaType;
  695. if (typeof _schemaType === 'object' && _schemaType != null) {
  696. _schemaType.$isUnderneathDocArray = true;
  697. }
  698. }
  699. }
  700. return this;
  701. };
  702. /*!
  703. * ignore
  704. */
  705. function gatherChildSchemas(schema) {
  706. const childSchemas = [];
  707. for (const path of Object.keys(schema.paths)) {
  708. const schematype = schema.paths[path];
  709. if (schematype.$isMongooseDocumentArray || schematype.$isSingleNested) {
  710. childSchemas.push({ schema: schematype.schema, model: schematype.caster });
  711. }
  712. }
  713. return childSchemas;
  714. }
  715. /*!
  716. * ignore
  717. */
  718. function _getPath(schema, path, cleanPath) {
  719. if (schema.paths.hasOwnProperty(path)) {
  720. return schema.paths[path];
  721. }
  722. if (schema.subpaths.hasOwnProperty(cleanPath)) {
  723. return schema.subpaths[cleanPath];
  724. }
  725. if (schema.singleNestedPaths.hasOwnProperty(cleanPath) && typeof schema.singleNestedPaths[cleanPath] === 'object') {
  726. return schema.singleNestedPaths[cleanPath];
  727. }
  728. return null;
  729. }
  730. /*!
  731. * ignore
  732. */
  733. function _pathToPositionalSyntax(path) {
  734. if (!/\.\d+/.test(path)) {
  735. return path;
  736. }
  737. return path.replace(/\.\d+\./g, '.$.').replace(/\.\d+$/, '.$');
  738. }
  739. /*!
  740. * ignore
  741. */
  742. function getMapPath(schema, path) {
  743. for (const _path of Object.keys(schema.paths)) {
  744. if (!_path.includes('.$*')) {
  745. continue;
  746. }
  747. const re = new RegExp('^' + _path.replace(/\.\$\*/g, '\\.[^.]+') + '$');
  748. if (re.test(path)) {
  749. return schema.paths[_path];
  750. }
  751. }
  752. return null;
  753. }
  754. /**
  755. * The Mongoose instance this schema is associated with
  756. *
  757. * @property base
  758. * @api private
  759. */
  760. Object.defineProperty(Schema.prototype, 'base', {
  761. configurable: true,
  762. enumerable: false,
  763. writable: true,
  764. value: null
  765. });
  766. /**
  767. * Converts type arguments into Mongoose Types.
  768. *
  769. * @param {String} path
  770. * @param {Object} obj constructor
  771. * @api private
  772. */
  773. Schema.prototype.interpretAsType = function(path, obj, options) {
  774. if (obj instanceof SchemaType) {
  775. if (obj.path === path) {
  776. return obj;
  777. }
  778. const clone = obj.clone();
  779. clone.path = path;
  780. return clone;
  781. }
  782. // If this schema has an associated Mongoose object, use the Mongoose object's
  783. // copy of SchemaTypes re: gh-7158 gh-6933
  784. const MongooseTypes = this.base != null ? this.base.Schema.Types : Schema.Types;
  785. if (!utils.isPOJO(obj) && !(obj instanceof SchemaTypeOptions)) {
  786. const constructorName = utils.getFunctionName(obj.constructor);
  787. if (constructorName !== 'Object') {
  788. const oldObj = obj;
  789. obj = {};
  790. obj[options.typeKey] = oldObj;
  791. }
  792. }
  793. // Get the type making sure to allow keys named "type"
  794. // and default to mixed if not specified.
  795. // { type: { type: String, default: 'freshcut' } }
  796. let type = obj[options.typeKey] && (options.typeKey !== 'type' || !obj.type.type)
  797. ? obj[options.typeKey]
  798. : {};
  799. let name;
  800. if (utils.isPOJO(type) || type === 'mixed') {
  801. return new MongooseTypes.Mixed(path, obj);
  802. }
  803. if (Array.isArray(type) || type === Array || type === 'array' || type === MongooseTypes.Array) {
  804. // if it was specified through { type } look for `cast`
  805. let cast = (type === Array || type === 'array')
  806. ? obj.cast || obj.of
  807. : type[0];
  808. if (cast && cast.instanceOfSchema) {
  809. return new MongooseTypes.DocumentArray(path, cast, obj);
  810. }
  811. if (cast &&
  812. cast[options.typeKey] &&
  813. cast[options.typeKey].instanceOfSchema) {
  814. return new MongooseTypes.DocumentArray(path, cast[options.typeKey], obj, cast);
  815. }
  816. if (Array.isArray(cast)) {
  817. return new MongooseTypes.Array(path, this.interpretAsType(path, cast, options), obj);
  818. }
  819. if (typeof cast === 'string') {
  820. cast = MongooseTypes[cast.charAt(0).toUpperCase() + cast.substring(1)];
  821. } else if (cast && (!cast[options.typeKey] || (options.typeKey === 'type' && cast.type.type))
  822. && utils.isPOJO(cast)) {
  823. if (Object.keys(cast).length) {
  824. // The `minimize` and `typeKey` options propagate to child schemas
  825. // declared inline, like `{ arr: [{ val: { $type: String } }] }`.
  826. // See gh-3560
  827. const childSchemaOptions = { minimize: options.minimize };
  828. if (options.typeKey) {
  829. childSchemaOptions.typeKey = options.typeKey;
  830. }
  831. // propagate 'strict' option to child schema
  832. if (options.hasOwnProperty('strict')) {
  833. childSchemaOptions.strict = options.strict;
  834. }
  835. if (options.hasOwnProperty('typePojoToMixed')) {
  836. childSchemaOptions.typePojoToMixed = options.typePojoToMixed;
  837. }
  838. if (this._userProvidedOptions.hasOwnProperty('_id')) {
  839. childSchemaOptions._id = this._userProvidedOptions._id;
  840. } else if (Schema.Types.DocumentArray.defaultOptions &&
  841. Schema.Types.DocumentArray.defaultOptions._id != null) {
  842. childSchemaOptions._id = Schema.Types.DocumentArray.defaultOptions._id;
  843. }
  844. const childSchema = new Schema(cast, childSchemaOptions);
  845. childSchema.$implicitlyCreated = true;
  846. return new MongooseTypes.DocumentArray(path, childSchema, obj);
  847. } else {
  848. // Special case: empty object becomes mixed
  849. return new MongooseTypes.Array(path, MongooseTypes.Mixed, obj);
  850. }
  851. }
  852. if (cast) {
  853. type = cast[options.typeKey] && (options.typeKey !== 'type' || !cast.type.type)
  854. ? cast[options.typeKey]
  855. : cast;
  856. name = typeof type === 'string'
  857. ? type
  858. : type.schemaName || utils.getFunctionName(type);
  859. if (!MongooseTypes.hasOwnProperty(name)) {
  860. throw new TypeError('Invalid schema configuration: ' +
  861. `\`${name}\` is not a valid type within the array \`${path}\`.` +
  862. 'See http://bit.ly/mongoose-schematypes for a list of valid schema types.');
  863. }
  864. }
  865. return new MongooseTypes.Array(path, cast || MongooseTypes.Mixed, obj, options);
  866. }
  867. if (type && type.instanceOfSchema) {
  868. return new MongooseTypes.Embedded(type, path, obj);
  869. }
  870. if (Buffer.isBuffer(type)) {
  871. name = 'Buffer';
  872. } else if (typeof type === 'function' || typeof type === 'object') {
  873. name = type.schemaName || utils.getFunctionName(type);
  874. } else {
  875. name = type == null ? '' + type : type.toString();
  876. }
  877. if (name) {
  878. name = name.charAt(0).toUpperCase() + name.substring(1);
  879. }
  880. // Special case re: gh-7049 because the bson `ObjectID` class' capitalization
  881. // doesn't line up with Mongoose's.
  882. if (name === 'ObjectID') {
  883. name = 'ObjectId';
  884. }
  885. if (MongooseTypes[name] == null) {
  886. throw new TypeError(`Invalid schema configuration: \`${name}\` is not ` +
  887. `a valid type at path \`${path}\`. See ` +
  888. 'http://bit.ly/mongoose-schematypes for a list of valid schema types.');
  889. }
  890. return new MongooseTypes[name](path, obj);
  891. };
  892. /**
  893. * Iterates the schemas paths similar to Array#forEach.
  894. *
  895. * The callback is passed the pathname and the schemaType instance.
  896. *
  897. * ####Example:
  898. *
  899. * const userSchema = new Schema({ name: String, registeredAt: Date });
  900. * userSchema.eachPath((pathname, schematype) => {
  901. * // Prints twice:
  902. * // name SchemaString { ... }
  903. * // registeredAt SchemaDate { ... }
  904. * console.log(pathname, schematype);
  905. * });
  906. *
  907. * @param {Function} fn callback function
  908. * @return {Schema} this
  909. * @api public
  910. */
  911. Schema.prototype.eachPath = function(fn) {
  912. const keys = Object.keys(this.paths);
  913. const len = keys.length;
  914. for (let i = 0; i < len; ++i) {
  915. fn(keys[i], this.paths[keys[i]]);
  916. }
  917. return this;
  918. };
  919. /**
  920. * Returns an Array of path strings that are required by this schema.
  921. *
  922. * ####Example:
  923. * const s = new Schema({
  924. * name: { type: String, required: true },
  925. * age: { type: String, required: true },
  926. * notes: String
  927. * });
  928. * s.requiredPaths(); // [ 'age', 'name' ]
  929. *
  930. * @api public
  931. * @param {Boolean} invalidate refresh the cache
  932. * @return {Array}
  933. */
  934. Schema.prototype.requiredPaths = function requiredPaths(invalidate) {
  935. if (this._requiredpaths && !invalidate) {
  936. return this._requiredpaths;
  937. }
  938. const paths = Object.keys(this.paths);
  939. let i = paths.length;
  940. const ret = [];
  941. while (i--) {
  942. const path = paths[i];
  943. if (this.paths[path].isRequired) {
  944. ret.push(path);
  945. }
  946. }
  947. this._requiredpaths = ret;
  948. return this._requiredpaths;
  949. };
  950. /**
  951. * Returns indexes from fields and schema-level indexes (cached).
  952. *
  953. * @api private
  954. * @return {Array}
  955. */
  956. Schema.prototype.indexedPaths = function indexedPaths() {
  957. if (this._indexedpaths) {
  958. return this._indexedpaths;
  959. }
  960. this._indexedpaths = this.indexes();
  961. return this._indexedpaths;
  962. };
  963. /**
  964. * Returns the pathType of `path` for this schema.
  965. *
  966. * Given a path, returns whether it is a real, virtual, nested, or ad-hoc/undefined path.
  967. *
  968. * ####Example:
  969. * const s = new Schema({ name: String, nested: { foo: String } });
  970. * s.virtual('foo').get(() => 42);
  971. * s.pathType('name'); // "real"
  972. * s.pathType('nested'); // "nested"
  973. * s.pathType('foo'); // "virtual"
  974. * s.pathType('fail'); // "adhocOrUndefined"
  975. *
  976. * @param {String} path
  977. * @return {String}
  978. * @api public
  979. */
  980. Schema.prototype.pathType = function(path) {
  981. // Convert to '.$' to check subpaths re: gh-6405
  982. const cleanPath = _pathToPositionalSyntax(path);
  983. if (this.paths.hasOwnProperty(path)) {
  984. return 'real';
  985. }
  986. if (this.virtuals.hasOwnProperty(path)) {
  987. return 'virtual';
  988. }
  989. if (this.nested.hasOwnProperty(path)) {
  990. return 'nested';
  991. }
  992. if (this.subpaths.hasOwnProperty(cleanPath) || this.subpaths.hasOwnProperty(path)) {
  993. return 'real';
  994. }
  995. const singleNestedPath = this.singleNestedPaths.hasOwnProperty(cleanPath) || this.singleNestedPaths.hasOwnProperty(path);
  996. if (singleNestedPath) {
  997. return singleNestedPath === 'nested' ? 'nested' : 'real';
  998. }
  999. // Look for maps
  1000. const mapPath = getMapPath(this, path);
  1001. if (mapPath != null) {
  1002. return 'real';
  1003. }
  1004. if (/\.\d+\.|\.\d+$/.test(path)) {
  1005. return getPositionalPathType(this, path);
  1006. }
  1007. return 'adhocOrUndefined';
  1008. };
  1009. /**
  1010. * Returns true iff this path is a child of a mixed schema.
  1011. *
  1012. * @param {String} path
  1013. * @return {Boolean}
  1014. * @api private
  1015. */
  1016. Schema.prototype.hasMixedParent = function(path) {
  1017. const subpaths = path.split(/\./g);
  1018. path = '';
  1019. for (let i = 0; i < subpaths.length; ++i) {
  1020. path = i > 0 ? path + '.' + subpaths[i] : subpaths[i];
  1021. if (this.paths.hasOwnProperty(path) &&
  1022. this.paths[path] instanceof MongooseTypes.Mixed) {
  1023. return this.paths[path];
  1024. }
  1025. }
  1026. return null;
  1027. };
  1028. /**
  1029. * Setup updatedAt and createdAt timestamps to documents if enabled
  1030. *
  1031. * @param {Boolean|Object} timestamps timestamps options
  1032. * @api private
  1033. */
  1034. Schema.prototype.setupTimestamp = function(timestamps) {
  1035. return setupTimestamps(this, timestamps);
  1036. };
  1037. /*!
  1038. * ignore. Deprecated re: #6405
  1039. */
  1040. function getPositionalPathType(self, path) {
  1041. const subpaths = path.split(/\.(\d+)\.|\.(\d+)$/).filter(Boolean);
  1042. if (subpaths.length < 2) {
  1043. return self.paths.hasOwnProperty(subpaths[0]) ?
  1044. self.paths[subpaths[0]] :
  1045. 'adhocOrUndefined';
  1046. }
  1047. let val = self.path(subpaths[0]);
  1048. let isNested = false;
  1049. if (!val) {
  1050. return 'adhocOrUndefined';
  1051. }
  1052. const last = subpaths.length - 1;
  1053. for (let i = 1; i < subpaths.length; ++i) {
  1054. isNested = false;
  1055. const subpath = subpaths[i];
  1056. if (i === last && val && !/\D/.test(subpath)) {
  1057. if (val.$isMongooseDocumentArray) {
  1058. val = val.$embeddedSchemaType;
  1059. } else if (val instanceof MongooseTypes.Array) {
  1060. // StringSchema, NumberSchema, etc
  1061. val = val.caster;
  1062. } else {
  1063. val = undefined;
  1064. }
  1065. break;
  1066. }
  1067. // ignore if its just a position segment: path.0.subpath
  1068. if (!/\D/.test(subpath)) {
  1069. // Nested array
  1070. if (val instanceof MongooseTypes.Array && i !== last) {
  1071. val = val.caster;
  1072. }
  1073. continue;
  1074. }
  1075. if (!(val && val.schema)) {
  1076. val = undefined;
  1077. break;
  1078. }
  1079. const type = val.schema.pathType(subpath);
  1080. isNested = (type === 'nested');
  1081. val = val.schema.path(subpath);
  1082. }
  1083. self.subpaths[path] = val;
  1084. if (val) {
  1085. return 'real';
  1086. }
  1087. if (isNested) {
  1088. return 'nested';
  1089. }
  1090. return 'adhocOrUndefined';
  1091. }
  1092. /*!
  1093. * ignore
  1094. */
  1095. function getPositionalPath(self, path) {
  1096. getPositionalPathType(self, path);
  1097. return self.subpaths[path];
  1098. }
  1099. /**
  1100. * Adds a method call to the queue.
  1101. *
  1102. * ####Example:
  1103. *
  1104. * schema.methods.print = function() { console.log(this); };
  1105. * schema.queue('print', []); // Print the doc every one is instantiated
  1106. *
  1107. * const Model = mongoose.model('Test', schema);
  1108. * new Model({ name: 'test' }); // Prints '{"_id": ..., "name": "test" }'
  1109. *
  1110. * @param {String} name name of the document method to call later
  1111. * @param {Array} args arguments to pass to the method
  1112. * @api public
  1113. */
  1114. Schema.prototype.queue = function(name, args) {
  1115. this.callQueue.push([name, args]);
  1116. return this;
  1117. };
  1118. /**
  1119. * Defines a pre hook for the model.
  1120. *
  1121. * ####Example
  1122. *
  1123. * const toySchema = new Schema({ name: String, created: Date });
  1124. *
  1125. * toySchema.pre('save', function(next) {
  1126. * if (!this.created) this.created = new Date;
  1127. * next();
  1128. * });
  1129. *
  1130. * toySchema.pre('validate', function(next) {
  1131. * if (this.name !== 'Woody') this.name = 'Woody';
  1132. * next();
  1133. * });
  1134. *
  1135. * // Equivalent to calling `pre()` on `find`, `findOne`, `findOneAndUpdate`.
  1136. * toySchema.pre(/^find/, function(next) {
  1137. * console.log(this.getFilter());
  1138. * });
  1139. *
  1140. * // Equivalent to calling `pre()` on `updateOne`, `findOneAndUpdate`.
  1141. * toySchema.pre(['updateOne', 'findOneAndUpdate'], function(next) {
  1142. * console.log(this.getFilter());
  1143. * });
  1144. *
  1145. * toySchema.pre('deleteOne', function() {
  1146. * // Runs when you call `Toy.deleteOne()`
  1147. * });
  1148. *
  1149. * toySchema.pre('deleteOne', { document: true }, function() {
  1150. * // Runs when you call `doc.deleteOne()`
  1151. * });
  1152. *
  1153. * @param {String|RegExp} The method name or regular expression to match method name
  1154. * @param {Object} [options]
  1155. * @param {Boolean} [options.document] If `name` is a hook for both document and query middleware, set to `true` to run on document middleware. For example, set `options.document` to `true` to apply this hook to `Document#deleteOne()` rather than `Query#deleteOne()`.
  1156. * @param {Boolean} [options.query] If `name` is a hook for both document and query middleware, set to `true` to run on query middleware.
  1157. * @param {Function} callback
  1158. * @api public
  1159. */
  1160. Schema.prototype.pre = function(name) {
  1161. if (name instanceof RegExp) {
  1162. const remainingArgs = Array.prototype.slice.call(arguments, 1);
  1163. for (const fn of hookNames) {
  1164. if (name.test(fn)) {
  1165. this.pre.apply(this, [fn].concat(remainingArgs));
  1166. }
  1167. }
  1168. return this;
  1169. }
  1170. if (Array.isArray(name)) {
  1171. const remainingArgs = Array.prototype.slice.call(arguments, 1);
  1172. for (const el of name) {
  1173. this.pre.apply(this, [el].concat(remainingArgs));
  1174. }
  1175. return this;
  1176. }
  1177. this.s.hooks.pre.apply(this.s.hooks, arguments);
  1178. return this;
  1179. };
  1180. /**
  1181. * Defines a post hook for the document
  1182. *
  1183. * const schema = new Schema(..);
  1184. * schema.post('save', function (doc) {
  1185. * console.log('this fired after a document was saved');
  1186. * });
  1187. *
  1188. * schema.post('find', function(docs) {
  1189. * console.log('this fired after you ran a find query');
  1190. * });
  1191. *
  1192. * schema.post(/Many$/, function(res) {
  1193. * console.log('this fired after you ran `updateMany()` or `deleteMany()`);
  1194. * });
  1195. *
  1196. * const Model = mongoose.model('Model', schema);
  1197. *
  1198. * const m = new Model(..);
  1199. * m.save(function(err) {
  1200. * console.log('this fires after the `post` hook');
  1201. * });
  1202. *
  1203. * m.find(function(err, docs) {
  1204. * console.log('this fires after the post find hook');
  1205. * });
  1206. *
  1207. * @param {String|RegExp} The method name or regular expression to match method name
  1208. * @param {Object} [options]
  1209. * @param {Boolean} [options.document] If `name` is a hook for both document and query middleware, set to `true` to run on document middleware.
  1210. * @param {Boolean} [options.query] If `name` is a hook for both document and query middleware, set to `true` to run on query middleware.
  1211. * @param {Function} fn callback
  1212. * @see middleware http://mongoosejs.com/docs/middleware.html
  1213. * @see kareem http://npmjs.org/package/kareem
  1214. * @api public
  1215. */
  1216. Schema.prototype.post = function(name) {
  1217. if (name instanceof RegExp) {
  1218. const remainingArgs = Array.prototype.slice.call(arguments, 1);
  1219. for (const fn of hookNames) {
  1220. if (name.test(fn)) {
  1221. this.post.apply(this, [fn].concat(remainingArgs));
  1222. }
  1223. }
  1224. return this;
  1225. }
  1226. if (Array.isArray(name)) {
  1227. const remainingArgs = Array.prototype.slice.call(arguments, 1);
  1228. for (const el of name) {
  1229. this.post.apply(this, [el].concat(remainingArgs));
  1230. }
  1231. return this;
  1232. }
  1233. this.s.hooks.post.apply(this.s.hooks, arguments);
  1234. return this;
  1235. };
  1236. /**
  1237. * Registers a plugin for this schema.
  1238. *
  1239. * ####Example:
  1240. *
  1241. * const s = new Schema({ name: String });
  1242. * s.plugin(schema => console.log(schema.path('name').path));
  1243. * mongoose.model('Test', s); // Prints 'name'
  1244. *
  1245. * @param {Function} plugin callback
  1246. * @param {Object} [opts]
  1247. * @see plugins
  1248. * @api public
  1249. */
  1250. Schema.prototype.plugin = function(fn, opts) {
  1251. if (typeof fn !== 'function') {
  1252. throw new Error('First param to `schema.plugin()` must be a function, ' +
  1253. 'got "' + (typeof fn) + '"');
  1254. }
  1255. if (opts && opts.deduplicate) {
  1256. for (const plugin of this.plugins) {
  1257. if (plugin.fn === fn) {
  1258. return this;
  1259. }
  1260. }
  1261. }
  1262. this.plugins.push({ fn: fn, opts: opts });
  1263. fn(this, opts);
  1264. return this;
  1265. };
  1266. /**
  1267. * Adds an instance method to documents constructed from Models compiled from this schema.
  1268. *
  1269. * ####Example
  1270. *
  1271. * const schema = kittySchema = new Schema(..);
  1272. *
  1273. * schema.method('meow', function () {
  1274. * console.log('meeeeeoooooooooooow');
  1275. * })
  1276. *
  1277. * const Kitty = mongoose.model('Kitty', schema);
  1278. *
  1279. * const fizz = new Kitty;
  1280. * fizz.meow(); // meeeeeooooooooooooow
  1281. *
  1282. * If a hash of name/fn pairs is passed as the only argument, each name/fn pair will be added as methods.
  1283. *
  1284. * schema.method({
  1285. * purr: function () {}
  1286. * , scratch: function () {}
  1287. * });
  1288. *
  1289. * // later
  1290. * fizz.purr();
  1291. * fizz.scratch();
  1292. *
  1293. * NOTE: `Schema.method()` adds instance methods to the `Schema.methods` object. You can also add instance methods directly to the `Schema.methods` object as seen in the [guide](/docs/guide.html#methods)
  1294. *
  1295. * @param {String|Object} method name
  1296. * @param {Function} [fn]
  1297. * @api public
  1298. */
  1299. Schema.prototype.method = function(name, fn, options) {
  1300. if (typeof name !== 'string') {
  1301. for (const i in name) {
  1302. this.methods[i] = name[i];
  1303. this.methodOptions[i] = utils.clone(options);
  1304. }
  1305. } else {
  1306. this.methods[name] = fn;
  1307. this.methodOptions[name] = utils.clone(options);
  1308. }
  1309. return this;
  1310. };
  1311. /**
  1312. * Adds static "class" methods to Models compiled from this schema.
  1313. *
  1314. * ####Example
  1315. *
  1316. * const schema = new Schema(..);
  1317. * // Equivalent to `schema.statics.findByName = function(name) {}`;
  1318. * schema.static('findByName', function(name) {
  1319. * return this.find({ name: name });
  1320. * });
  1321. *
  1322. * const Drink = mongoose.model('Drink', schema);
  1323. * await Drink.findByName('LaCroix');
  1324. *
  1325. * If a hash of name/fn pairs is passed as the only argument, each name/fn pair will be added as statics.
  1326. *
  1327. * @param {String|Object} name
  1328. * @param {Function} [fn]
  1329. * @api public
  1330. * @see Statics /docs/guide.html#statics
  1331. */
  1332. Schema.prototype.static = function(name, fn) {
  1333. if (typeof name !== 'string') {
  1334. for (const i in name) {
  1335. this.statics[i] = name[i];
  1336. }
  1337. } else {
  1338. this.statics[name] = fn;
  1339. }
  1340. return this;
  1341. };
  1342. /**
  1343. * Defines an index (most likely compound) for this schema.
  1344. *
  1345. * ####Example
  1346. *
  1347. * schema.index({ first: 1, last: -1 })
  1348. *
  1349. * @param {Object} fields
  1350. * @param {Object} [options] Options to pass to [MongoDB driver's `createIndex()` function](http://mongodb.github.io/node-mongodb-native/2.0/api/Collection.html#createIndex)
  1351. * @param {String} [options.expires=null] Mongoose-specific syntactic sugar, uses [ms](https://www.npmjs.com/package/ms) to convert `expires` option into seconds for the `expireAfterSeconds` in the above link.
  1352. * @api public
  1353. */
  1354. Schema.prototype.index = function(fields, options) {
  1355. fields || (fields = {});
  1356. options || (options = {});
  1357. if (options.expires) {
  1358. utils.expires(options);
  1359. }
  1360. this._indexes.push([fields, options]);
  1361. return this;
  1362. };
  1363. /**
  1364. * Sets a schema option.
  1365. *
  1366. * ####Example
  1367. *
  1368. * schema.set('strict'); // 'true' by default
  1369. * schema.set('strict', false); // Sets 'strict' to false
  1370. * schema.set('strict'); // 'false'
  1371. *
  1372. * @param {String} key option name
  1373. * @param {Object} [value] if not passed, the current option value is returned
  1374. * @see Schema ./
  1375. * @api public
  1376. */
  1377. Schema.prototype.set = function(key, value, _tags) {
  1378. if (arguments.length === 1) {
  1379. return this.options[key];
  1380. }
  1381. switch (key) {
  1382. case 'read':
  1383. this.options[key] = readPref(value, _tags);
  1384. this._userProvidedOptions[key] = this.options[key];
  1385. break;
  1386. case 'safe':
  1387. setSafe(this.options, value);
  1388. this._userProvidedOptions[key] = this.options[key];
  1389. break;
  1390. case 'timestamps':
  1391. this.setupTimestamp(value);
  1392. this.options[key] = value;
  1393. this._userProvidedOptions[key] = this.options[key];
  1394. break;
  1395. case '_id':
  1396. this.options[key] = value;
  1397. this._userProvidedOptions[key] = this.options[key];
  1398. if (value && !this.paths['_id']) {
  1399. addAutoId(this);
  1400. } else if (!value && this.paths['_id'] != null && this.paths['_id'].auto) {
  1401. this.remove('_id');
  1402. }
  1403. break;
  1404. default:
  1405. this.options[key] = value;
  1406. this._userProvidedOptions[key] = this.options[key];
  1407. break;
  1408. }
  1409. return this;
  1410. };
  1411. /*!
  1412. * ignore
  1413. */
  1414. const safeDeprecationWarning = 'Mongoose: The `safe` option for schemas is ' +
  1415. 'deprecated. Use the `writeConcern` option instead: ' +
  1416. 'http://bit.ly/mongoose-write-concern';
  1417. const setSafe = util.deprecate(function setSafe(options, value) {
  1418. options.safe = value === false ?
  1419. { w: 0 } :
  1420. value;
  1421. }, safeDeprecationWarning);
  1422. /**
  1423. * Gets a schema option.
  1424. *
  1425. * ####Example:
  1426. *
  1427. * schema.get('strict'); // true
  1428. * schema.set('strict', false);
  1429. * schema.get('strict'); // false
  1430. *
  1431. * @param {String} key option name
  1432. * @api public
  1433. * @return {Any} the option's value
  1434. */
  1435. Schema.prototype.get = function(key) {
  1436. return this.options[key];
  1437. };
  1438. /**
  1439. * The allowed index types
  1440. *
  1441. * @receiver Schema
  1442. * @static indexTypes
  1443. * @api public
  1444. */
  1445. const indexTypes = '2d 2dsphere hashed text'.split(' ');
  1446. Object.defineProperty(Schema, 'indexTypes', {
  1447. get: function() {
  1448. return indexTypes;
  1449. },
  1450. set: function() {
  1451. throw new Error('Cannot overwrite Schema.indexTypes');
  1452. }
  1453. });
  1454. /**
  1455. * Returns a list of indexes that this schema declares, via `schema.index()`
  1456. * or by `index: true` in a path's options.
  1457. *
  1458. * ####Example:
  1459. *
  1460. * const userSchema = new Schema({
  1461. * email: { type: String, required: true, unique: true },
  1462. * registeredAt: { type: Date, index: true }
  1463. * });
  1464. *
  1465. * // [ [ { email: 1 }, { unique: true, background: true } ],
  1466. * // [ { registeredAt: 1 }, { background: true } ] ]
  1467. * userSchema.indexes();
  1468. *
  1469. * @api public
  1470. * @return {Array} list of indexes defined in the schema
  1471. */
  1472. Schema.prototype.indexes = function() {
  1473. return getIndexes(this);
  1474. };
  1475. /**
  1476. * Creates a virtual type with the given name.
  1477. *
  1478. * @param {String} name
  1479. * @param {Object} [options]
  1480. * @param {String|Model} [options.ref] model name or model instance. Marks this as a [populate virtual](populate.html#populate-virtuals).
  1481. * @param {String|Function} [options.localField] Required for populate virtuals. See [populate virtual docs](populate.html#populate-virtuals) for more information.
  1482. * @param {String|Function} [options.foreignField] Required for populate virtuals. See [populate virtual docs](populate.html#populate-virtuals) for more information.
  1483. * @param {Boolean|Function} [options.justOne=false] Only works with populate virtuals. If [truthy](https://masteringjs.io/tutorials/fundamentals/truthy), will be a single doc or `null`. Otherwise, the populate virtual will be an array.
  1484. * @param {Boolean} [options.count=false] Only works with populate virtuals. If [truthy](https://masteringjs.io/tutorials/fundamentals/truthy), this populate virtual will contain the number of documents rather than the documents themselves when you `populate()`.
  1485. * @param {Function|null} [options.get=null] Adds a [getter](/docs/tutorials/getters-setters.html) to this virtual to transform the populated doc.
  1486. * @return {VirtualType}
  1487. */
  1488. Schema.prototype.virtual = function(name, options) {
  1489. if (name instanceof VirtualType || (name != null && name.constructor.name === 'VirtualType')) {
  1490. return this.virtual(name.path, name.options);
  1491. }
  1492. options = new VirtualOptions(options);
  1493. if (utils.hasUserDefinedProperty(options, ['ref', 'refPath'])) {
  1494. if (options.localField == null) {
  1495. throw new Error('Reference virtuals require `localField` option');
  1496. }
  1497. if (options.foreignField == null) {
  1498. throw new Error('Reference virtuals require `foreignField` option');
  1499. }
  1500. this.pre('init', function(obj) {
  1501. if (mpath.has(name, obj)) {
  1502. const _v = mpath.get(name, obj);
  1503. if (!this.$$populatedVirtuals) {
  1504. this.$$populatedVirtuals = {};
  1505. }
  1506. if (options.justOne || options.count) {
  1507. this.$$populatedVirtuals[name] = Array.isArray(_v) ?
  1508. _v[0] :
  1509. _v;
  1510. } else {
  1511. this.$$populatedVirtuals[name] = Array.isArray(_v) ?
  1512. _v :
  1513. _v == null ? [] : [_v];
  1514. }
  1515. mpath.unset(name, obj);
  1516. }
  1517. });
  1518. const virtual = this.virtual(name);
  1519. virtual.options = options;
  1520. virtual.
  1521. set(function(_v) {
  1522. if (!this.$$populatedVirtuals) {
  1523. this.$$populatedVirtuals = {};
  1524. }
  1525. if (options.justOne || options.count) {
  1526. this.$$populatedVirtuals[name] = Array.isArray(_v) ?
  1527. _v[0] :
  1528. _v;
  1529. if (typeof this.$$populatedVirtuals[name] !== 'object') {
  1530. this.$$populatedVirtuals[name] = options.count ? _v : null;
  1531. }
  1532. } else {
  1533. this.$$populatedVirtuals[name] = Array.isArray(_v) ?
  1534. _v :
  1535. _v == null ? [] : [_v];
  1536. this.$$populatedVirtuals[name] = this.$$populatedVirtuals[name].filter(function(doc) {
  1537. return doc && typeof doc === 'object';
  1538. });
  1539. }
  1540. });
  1541. if (typeof options.get === 'function') {
  1542. virtual.get(options.get);
  1543. }
  1544. return virtual;
  1545. }
  1546. const virtuals = this.virtuals;
  1547. const parts = name.split('.');
  1548. if (this.pathType(name) === 'real') {
  1549. throw new Error('Virtual path "' + name + '"' +
  1550. ' conflicts with a real path in the schema');
  1551. }
  1552. virtuals[name] = parts.reduce(function(mem, part, i) {
  1553. mem[part] || (mem[part] = (i === parts.length - 1)
  1554. ? new VirtualType(options, name)
  1555. : {});
  1556. return mem[part];
  1557. }, this.tree);
  1558. // Workaround for gh-8198: if virtual is under document array, make a fake
  1559. // virtual. See gh-8210
  1560. let cur = parts[0];
  1561. for (let i = 0; i < parts.length - 1; ++i) {
  1562. if (this.paths[cur] != null && this.paths[cur].$isMongooseDocumentArray) {
  1563. const remnant = parts.slice(i + 1).join('.');
  1564. const v = this.paths[cur].schema.virtual(remnant);
  1565. v.get((v, virtual, doc) => {
  1566. const parent = doc.__parentArray[arrayParentSymbol];
  1567. const path = cur + '.' + doc.__index + '.' + remnant;
  1568. return parent.get(path);
  1569. });
  1570. break;
  1571. }
  1572. cur += '.' + parts[i + 1];
  1573. }
  1574. return virtuals[name];
  1575. };
  1576. /**
  1577. * Returns the virtual type with the given `name`.
  1578. *
  1579. * @param {String} name
  1580. * @return {VirtualType}
  1581. */
  1582. Schema.prototype.virtualpath = function(name) {
  1583. return this.virtuals.hasOwnProperty(name) ? this.virtuals[name] : null;
  1584. };
  1585. /**
  1586. * Removes the given `path` (or [`paths`]).
  1587. *
  1588. * ####Example:
  1589. *
  1590. * const schema = new Schema({ name: String, age: Number });
  1591. * schema.remove('name');
  1592. * schema.path('name'); // Undefined
  1593. * schema.path('age'); // SchemaNumber { ... }
  1594. *
  1595. * @param {String|Array} path
  1596. * @return {Schema} the Schema instance
  1597. * @api public
  1598. */
  1599. Schema.prototype.remove = function(path) {
  1600. if (typeof path === 'string') {
  1601. path = [path];
  1602. }
  1603. if (Array.isArray(path)) {
  1604. path.forEach(function(name) {
  1605. if (this.path(name) == null && !this.nested[name]) {
  1606. return;
  1607. }
  1608. if (this.nested[name]) {
  1609. const allKeys = Object.keys(this.paths).
  1610. concat(Object.keys(this.nested));
  1611. for (const path of allKeys) {
  1612. if (path.startsWith(name + '.')) {
  1613. delete this.paths[path];
  1614. delete this.nested[path];
  1615. _deletePath(this, path);
  1616. }
  1617. }
  1618. delete this.nested[name];
  1619. _deletePath(this, name);
  1620. return;
  1621. }
  1622. delete this.paths[name];
  1623. _deletePath(this, name);
  1624. }, this);
  1625. }
  1626. return this;
  1627. };
  1628. /*!
  1629. * ignore
  1630. */
  1631. function _deletePath(schema, name) {
  1632. const pieces = name.split('.');
  1633. const last = pieces.pop();
  1634. let branch = schema.tree;
  1635. for (const piece of pieces) {
  1636. branch = branch[piece];
  1637. }
  1638. delete branch[last];
  1639. }
  1640. /**
  1641. * Loads an ES6 class into a schema. Maps [setters](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Functions/set) + [getters](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Functions/get), [static methods](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes/static),
  1642. * and [instance methods](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes#Class_body_and_method_definitions)
  1643. * to schema [virtuals](/docs/guide.html#virtuals),
  1644. * [statics](/docs/guide.html#statics), and
  1645. * [methods](/docs/guide.html#methods).
  1646. *
  1647. * ####Example:
  1648. *
  1649. * ```javascript
  1650. * const md5 = require('md5');
  1651. * const userSchema = new Schema({ email: String });
  1652. * class UserClass {
  1653. * // `gravatarImage` becomes a virtual
  1654. * get gravatarImage() {
  1655. * const hash = md5(this.email.toLowerCase());
  1656. * return `https://www.gravatar.com/avatar/${hash}`;
  1657. * }
  1658. *
  1659. * // `getProfileUrl()` becomes a document method
  1660. * getProfileUrl() {
  1661. * return `https://mysite.com/${this.email}`;
  1662. * }
  1663. *
  1664. * // `findByEmail()` becomes a static
  1665. * static findByEmail(email) {
  1666. * return this.findOne({ email });
  1667. * }
  1668. * }
  1669. *
  1670. * // `schema` will now have a `gravatarImage` virtual, a `getProfileUrl()` method,
  1671. * // and a `findByEmail()` static
  1672. * userSchema.loadClass(UserClass);
  1673. * ```
  1674. *
  1675. * @param {Function} model
  1676. * @param {Boolean} [virtualsOnly] if truthy, only pulls virtuals from the class, not methods or statics
  1677. */
  1678. Schema.prototype.loadClass = function(model, virtualsOnly) {
  1679. if (model === Object.prototype ||
  1680. model === Function.prototype ||
  1681. model.prototype.hasOwnProperty('$isMongooseModelPrototype')) {
  1682. return this;
  1683. }
  1684. this.loadClass(Object.getPrototypeOf(model), virtualsOnly);
  1685. // Add static methods
  1686. if (!virtualsOnly) {
  1687. Object.getOwnPropertyNames(model).forEach(function(name) {
  1688. if (name.match(/^(length|name|prototype|constructor|__proto__)$/)) {
  1689. return;
  1690. }
  1691. const prop = Object.getOwnPropertyDescriptor(model, name);
  1692. this.static(name, prop.value);
  1693. }, this);
  1694. }
  1695. // Add methods and virtuals
  1696. Object.getOwnPropertyNames(model.prototype).forEach(function(name) {
  1697. if (name.match(/^(constructor)$/)) {
  1698. return;
  1699. }
  1700. const method = Object.getOwnPropertyDescriptor(model.prototype, name);
  1701. if (!virtualsOnly) {
  1702. if (typeof method.value === 'function') {
  1703. this.method(name, method.value);
  1704. }
  1705. }
  1706. if (typeof method.get === 'function') {
  1707. if (this.virtuals[name]) {
  1708. this.virtuals[name].getters = [];
  1709. }
  1710. this.virtual(name).get(method.get);
  1711. }
  1712. if (typeof method.set === 'function') {
  1713. if (this.virtuals[name]) {
  1714. this.virtuals[name].setters = [];
  1715. }
  1716. this.virtual(name).set(method.set);
  1717. }
  1718. }, this);
  1719. return this;
  1720. };
  1721. /*!
  1722. * ignore
  1723. */
  1724. Schema.prototype._getSchema = function(path) {
  1725. const _this = this;
  1726. const pathschema = _this.path(path);
  1727. const resultPath = [];
  1728. if (pathschema) {
  1729. pathschema.$fullPath = path;
  1730. return pathschema;
  1731. }
  1732. function search(parts, schema) {
  1733. let p = parts.length + 1;
  1734. let foundschema;
  1735. let trypath;
  1736. while (p--) {
  1737. trypath = parts.slice(0, p).join('.');
  1738. foundschema = schema.path(trypath);
  1739. if (foundschema) {
  1740. resultPath.push(trypath);
  1741. if (foundschema.caster) {
  1742. // array of Mixed?
  1743. if (foundschema.caster instanceof MongooseTypes.Mixed) {
  1744. foundschema.caster.$fullPath = resultPath.join('.');
  1745. return foundschema.caster;
  1746. }
  1747. // Now that we found the array, we need to check if there
  1748. // are remaining document paths to look up for casting.
  1749. // Also we need to handle array.$.path since schema.path
  1750. // doesn't work for that.
  1751. // If there is no foundschema.schema we are dealing with
  1752. // a path like array.$
  1753. if (p !== parts.length) {
  1754. if (foundschema.schema) {
  1755. let ret;
  1756. if (parts[p] === '$' || isArrayFilter(parts[p])) {
  1757. if (p + 1 === parts.length) {
  1758. // comments.$
  1759. return foundschema;
  1760. }
  1761. // comments.$.comments.$.title
  1762. ret = search(parts.slice(p + 1), foundschema.schema);
  1763. if (ret) {
  1764. ret.$isUnderneathDocArray = ret.$isUnderneathDocArray ||
  1765. !foundschema.schema.$isSingleNested;
  1766. }
  1767. return ret;
  1768. }
  1769. // this is the last path of the selector
  1770. ret = search(parts.slice(p), foundschema.schema);
  1771. if (ret) {
  1772. ret.$isUnderneathDocArray = ret.$isUnderneathDocArray ||
  1773. !foundschema.schema.$isSingleNested;
  1774. }
  1775. return ret;
  1776. }
  1777. }
  1778. } else if (foundschema.$isSchemaMap) {
  1779. if (p + 1 >= parts.length) {
  1780. return foundschema.$__schemaType;
  1781. }
  1782. const ret = search(parts.slice(p + 1), foundschema.$__schemaType.schema);
  1783. return ret;
  1784. }
  1785. foundschema.$fullPath = resultPath.join('.');
  1786. return foundschema;
  1787. }
  1788. }
  1789. }
  1790. // look for arrays
  1791. const parts = path.split('.');
  1792. for (let i = 0; i < parts.length; ++i) {
  1793. if (parts[i] === '$' || isArrayFilter(parts[i])) {
  1794. // Re: gh-5628, because `schema.path()` doesn't take $ into account.
  1795. parts[i] = '0';
  1796. }
  1797. }
  1798. return search(parts, _this);
  1799. };
  1800. /*!
  1801. * ignore
  1802. */
  1803. Schema.prototype._getPathType = function(path) {
  1804. const _this = this;
  1805. const pathschema = _this.path(path);
  1806. if (pathschema) {
  1807. return 'real';
  1808. }
  1809. function search(parts, schema) {
  1810. let p = parts.length + 1,
  1811. foundschema,
  1812. trypath;
  1813. while (p--) {
  1814. trypath = parts.slice(0, p).join('.');
  1815. foundschema = schema.path(trypath);
  1816. if (foundschema) {
  1817. if (foundschema.caster) {
  1818. // array of Mixed?
  1819. if (foundschema.caster instanceof MongooseTypes.Mixed) {
  1820. return { schema: foundschema, pathType: 'mixed' };
  1821. }
  1822. // Now that we found the array, we need to check if there
  1823. // are remaining document paths to look up for casting.
  1824. // Also we need to handle array.$.path since schema.path
  1825. // doesn't work for that.
  1826. // If there is no foundschema.schema we are dealing with
  1827. // a path like array.$
  1828. if (p !== parts.length && foundschema.schema) {
  1829. if (parts[p] === '$' || isArrayFilter(parts[p])) {
  1830. if (p === parts.length - 1) {
  1831. return { schema: foundschema, pathType: 'nested' };
  1832. }
  1833. // comments.$.comments.$.title
  1834. return search(parts.slice(p + 1), foundschema.schema);
  1835. }
  1836. // this is the last path of the selector
  1837. return search(parts.slice(p), foundschema.schema);
  1838. }
  1839. return {
  1840. schema: foundschema,
  1841. pathType: foundschema.$isSingleNested ? 'nested' : 'array'
  1842. };
  1843. }
  1844. return { schema: foundschema, pathType: 'real' };
  1845. } else if (p === parts.length && schema.nested[trypath]) {
  1846. return { schema: schema, pathType: 'nested' };
  1847. }
  1848. }
  1849. return { schema: foundschema || schema, pathType: 'undefined' };
  1850. }
  1851. // look for arrays
  1852. return search(path.split('.'), _this);
  1853. };
  1854. /*!
  1855. * ignore
  1856. */
  1857. function isArrayFilter(piece) {
  1858. return piece.startsWith('$[') && piece.endsWith(']');
  1859. }
  1860. /*!
  1861. * Module exports.
  1862. */
  1863. module.exports = exports = Schema;
  1864. // require down here because of reference issues
  1865. /**
  1866. * The various built-in Mongoose Schema Types.
  1867. *
  1868. * ####Example:
  1869. *
  1870. * const mongoose = require('mongoose');
  1871. * const ObjectId = mongoose.Schema.Types.ObjectId;
  1872. *
  1873. * ####Types:
  1874. *
  1875. * - [String](#schema-string-js)
  1876. * - [Number](#schema-number-js)
  1877. * - [Boolean](#schema-boolean-js) | Bool
  1878. * - [Array](#schema-array-js)
  1879. * - [Buffer](#schema-buffer-js)
  1880. * - [Date](#schema-date-js)
  1881. * - [ObjectId](#schema-objectid-js) | Oid
  1882. * - [Mixed](#schema-mixed-js)
  1883. *
  1884. * Using this exposed access to the `Mixed` SchemaType, we can use them in our schema.
  1885. *
  1886. * const Mixed = mongoose.Schema.Types.Mixed;
  1887. * new mongoose.Schema({ _user: Mixed })
  1888. *
  1889. * @api public
  1890. */
  1891. Schema.Types = MongooseTypes = require('./schema/index');
  1892. /*!
  1893. * ignore
  1894. */
  1895. exports.ObjectId = MongooseTypes.ObjectId;