You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					1163 lines
				
				34 KiB
			
		
		
			
		
	
	
					1163 lines
				
				34 KiB
			| 
											3 years ago
										 | 'use strict'; | ||
|  | 
 | ||
|  | /*! | ||
|  |  * Module dependencies | ||
|  |  */ | ||
|  | 
 | ||
|  | const AggregationCursor = require('./cursor/AggregationCursor'); | ||
|  | const Query = require('./query'); | ||
|  | const applyGlobalMaxTimeMS = require('./helpers/query/applyGlobalMaxTimeMS'); | ||
|  | const getConstructorName = require('./helpers/getConstructorName'); | ||
|  | const promiseOrCallback = require('./helpers/promiseOrCallback'); | ||
|  | const stringifyFunctionOperators = require('./helpers/aggregate/stringifyFunctionOperators'); | ||
|  | const util = require('util'); | ||
|  | const utils = require('./utils'); | ||
|  | const read = Query.prototype.read; | ||
|  | const readConcern = Query.prototype.readConcern; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Aggregate constructor used for building aggregation pipelines. Do not | ||
|  |  * instantiate this class directly, use [Model.aggregate()](/docs/api.html#model_Model.aggregate) instead. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     const aggregate = Model.aggregate([ | ||
|  |  *       { $project: { a: 1, b: 1 } }, | ||
|  |  *       { $skip: 5 } | ||
|  |  *     ]); | ||
|  |  * | ||
|  |  *     Model. | ||
|  |  *       aggregate([{ $match: { age: { $gte: 21 }}}]). | ||
|  |  *       unwind('tags'). | ||
|  |  *       exec(callback); | ||
|  |  * | ||
|  |  * ####Note: | ||
|  |  * | ||
|  |  * - The documents returned are plain javascript objects, not mongoose documents (since any shape of document can be returned). | ||
|  |  * - Mongoose does **not** cast pipeline stages. The below will **not** work unless `_id` is a string in the database | ||
|  |  * | ||
|  |  * ```javascript
 | ||
|  |  *   new Aggregate([{ $match: { _id: '00000000000000000000000a' } }]); | ||
|  |  *   // Do this instead to cast to an ObjectId
 | ||
|  |  *   new Aggregate([{ $match: { _id: mongoose.Types.ObjectId('00000000000000000000000a') } }]); | ||
|  |  * ```
 | ||
|  |  * | ||
|  |  * @see MongoDB http://docs.mongodb.org/manual/applications/aggregation/
 | ||
|  |  * @see driver http://mongodb.github.com/node-mongodb-native/api-generated/collection.html#aggregate
 | ||
|  |  * @param {Array} [pipeline] aggregation pipeline as an array of objects | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | function Aggregate(pipeline) { | ||
|  |   this._pipeline = []; | ||
|  |   this._model = undefined; | ||
|  |   this.options = {}; | ||
|  | 
 | ||
|  |   if (arguments.length === 1 && util.isArray(pipeline)) { | ||
|  |     this.append.apply(this, pipeline); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | /** | ||
|  |  * Contains options passed down to the [aggregate command](https://docs.mongodb.com/manual/reference/command/aggregate/).
 | ||
|  |  * Supported options are: | ||
|  |  * | ||
|  |  * - `readPreference` | ||
|  |  * - [`cursor`](./api.html#aggregate_Aggregate-cursor) | ||
|  |  * - [`explain`](./api.html#aggregate_Aggregate-explain) | ||
|  |  * - [`allowDiskUse`](./api.html#aggregate_Aggregate-allowDiskUse) | ||
|  |  * - `maxTimeMS` | ||
|  |  * - `bypassDocumentValidation` | ||
|  |  * - `raw` | ||
|  |  * - `promoteLongs` | ||
|  |  * - `promoteValues` | ||
|  |  * - `promoteBuffers` | ||
|  |  * - [`collation`](./api.html#aggregate_Aggregate-collation) | ||
|  |  * - `comment` | ||
|  |  * - [`session`](./api.html#aggregate_Aggregate-session) | ||
|  |  * | ||
|  |  * @property options | ||
|  |  * @memberOf Aggregate | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.options; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Get/set the model that this aggregation will execute on. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  *     const aggregate = MyModel.aggregate([{ $match: { answer: 42 } }]); | ||
|  |  *     aggregate.model() === MyModel; // true
 | ||
|  |  * | ||
|  |  *     // Change the model. There's rarely any reason to do this.
 | ||
|  |  *     aggregate.model(SomeOtherModel); | ||
|  |  *     aggregate.model() === SomeOtherModel; // true
 | ||
|  |  * | ||
|  |  * @param {Model} [model] the model to which the aggregate is to be bound | ||
|  |  * @return {Aggregate|Model} if model is passed, will return `this`, otherwise will return the model | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.model = function(model) { | ||
|  |   if (arguments.length === 0) { | ||
|  |     return this._model; | ||
|  |   } | ||
|  | 
 | ||
|  |   this._model = model; | ||
|  |   if (model.schema != null) { | ||
|  |     if (this.options.readPreference == null && | ||
|  |         model.schema.options.read != null) { | ||
|  |       this.options.readPreference = model.schema.options.read; | ||
|  |     } | ||
|  |     if (this.options.collation == null && | ||
|  |         model.schema.options.collation != null) { | ||
|  |       this.options.collation = model.schema.options.collation; | ||
|  |     } | ||
|  |   } | ||
|  |   return this; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends new operators to this aggregate pipeline | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.append({ $project: { field: 1 }}, { $limit: 2 }); | ||
|  |  * | ||
|  |  *     // or pass an array
 | ||
|  |  *     const pipeline = [{ $match: { daw: 'Logic Audio X' }} ]; | ||
|  |  *     aggregate.append(pipeline); | ||
|  |  * | ||
|  |  * @param {Object} ops operator(s) to append | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.append = function() { | ||
|  |   const args = (arguments.length === 1 && util.isArray(arguments[0])) | ||
|  |     ? arguments[0] | ||
|  |     : utils.args(arguments); | ||
|  | 
 | ||
|  |   if (!args.every(isOperator)) { | ||
|  |     throw new Error('Arguments must be aggregate pipeline operators'); | ||
|  |   } | ||
|  | 
 | ||
|  |   this._pipeline = this._pipeline.concat(args); | ||
|  | 
 | ||
|  |   return this; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new $addFields operator to this aggregate pipeline. | ||
|  |  * Requires MongoDB v3.4+ to work | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |   * | ||
|  |  *     // adding new fields based on existing fields
 | ||
|  |  *     aggregate.addFields({ | ||
|  |  *         newField: '$b.nested' | ||
|  |  *       , plusTen: { $add: ['$val', 10]} | ||
|  |  *       , sub: { | ||
|  |  *            name: '$a' | ||
|  |  *         } | ||
|  |  *     }) | ||
|  |  * | ||
|  |  *     // etc
 | ||
|  |  *     aggregate.addFields({ salary_k: { $divide: [ "$salary", 1000 ] } }); | ||
|  |  * | ||
|  |  * @param {Object} arg field specification | ||
|  |  * @see $addFields https://docs.mongodb.com/manual/reference/operator/aggregation/addFields/
 | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | Aggregate.prototype.addFields = function(arg) { | ||
|  |   const fields = {}; | ||
|  |   if (typeof arg === 'object' && !util.isArray(arg)) { | ||
|  |     Object.keys(arg).forEach(function(field) { | ||
|  |       fields[field] = arg[field]; | ||
|  |     }); | ||
|  |   } else { | ||
|  |     throw new Error('Invalid addFields() argument. Must be an object'); | ||
|  |   } | ||
|  |   return this.append({ $addFields: fields }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new $project operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * Mongoose query [selection syntax](#query_Query-select) is also supported. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     // include a, include b, exclude _id
 | ||
|  |  *     aggregate.project("a b -_id"); | ||
|  |  * | ||
|  |  *     // or you may use object notation, useful when
 | ||
|  |  *     // you have keys already prefixed with a "-"
 | ||
|  |  *     aggregate.project({a: 1, b: 1, _id: 0}); | ||
|  |  * | ||
|  |  *     // reshaping documents
 | ||
|  |  *     aggregate.project({ | ||
|  |  *         newField: '$b.nested' | ||
|  |  *       , plusTen: { $add: ['$val', 10]} | ||
|  |  *       , sub: { | ||
|  |  *            name: '$a' | ||
|  |  *         } | ||
|  |  *     }) | ||
|  |  * | ||
|  |  *     // etc
 | ||
|  |  *     aggregate.project({ salary_k: { $divide: [ "$salary", 1000 ] } }); | ||
|  |  * | ||
|  |  * @param {Object|String} arg field specification | ||
|  |  * @see projection http://docs.mongodb.org/manual/reference/aggregation/project/
 | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.project = function(arg) { | ||
|  |   const fields = {}; | ||
|  | 
 | ||
|  |   if (typeof arg === 'object' && !util.isArray(arg)) { | ||
|  |     Object.keys(arg).forEach(function(field) { | ||
|  |       fields[field] = arg[field]; | ||
|  |     }); | ||
|  |   } else if (arguments.length === 1 && typeof arg === 'string') { | ||
|  |     arg.split(/\s+/).forEach(function(field) { | ||
|  |       if (!field) { | ||
|  |         return; | ||
|  |       } | ||
|  |       const include = field[0] === '-' ? 0 : 1; | ||
|  |       if (include === 0) { | ||
|  |         field = field.substring(1); | ||
|  |       } | ||
|  |       fields[field] = include; | ||
|  |     }); | ||
|  |   } else { | ||
|  |     throw new Error('Invalid project() argument. Must be string or object'); | ||
|  |   } | ||
|  | 
 | ||
|  |   return this.append({ $project: fields }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new custom $group operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.group({ _id: "$department" }); | ||
|  |  * | ||
|  |  * @see $group http://docs.mongodb.org/manual/reference/aggregation/group/
 | ||
|  |  * @method group | ||
|  |  * @memberOf Aggregate | ||
|  |  * @instance | ||
|  |  * @param {Object} arg $group operator contents | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new custom $match operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.match({ department: { $in: [ "sales", "engineering" ] } }); | ||
|  |  * | ||
|  |  * @see $match http://docs.mongodb.org/manual/reference/aggregation/match/
 | ||
|  |  * @method match | ||
|  |  * @memberOf Aggregate | ||
|  |  * @instance | ||
|  |  * @param {Object} arg $match operator contents | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new $skip operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.skip(10); | ||
|  |  * | ||
|  |  * @see $skip http://docs.mongodb.org/manual/reference/aggregation/skip/
 | ||
|  |  * @method skip | ||
|  |  * @memberOf Aggregate | ||
|  |  * @instance | ||
|  |  * @param {Number} num number of records to skip before next stage | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new $limit operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.limit(10); | ||
|  |  * | ||
|  |  * @see $limit http://docs.mongodb.org/manual/reference/aggregation/limit/
 | ||
|  |  * @method limit | ||
|  |  * @memberOf Aggregate | ||
|  |  * @instance | ||
|  |  * @param {Number} num maximum number of records to pass to the next stage | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new $geoNear operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * ####NOTE: | ||
|  |  * | ||
|  |  * **MUST** be used as the first operator in the pipeline. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.near({ | ||
|  |  *       near: [40.724, -73.997], | ||
|  |  *       distanceField: "dist.calculated", // required
 | ||
|  |  *       maxDistance: 0.008, | ||
|  |  *       query: { type: "public" }, | ||
|  |  *       includeLocs: "dist.location", | ||
|  |  *       uniqueDocs: true, | ||
|  |  *       num: 5 | ||
|  |  *     }); | ||
|  |  * | ||
|  |  * @see $geoNear http://docs.mongodb.org/manual/reference/aggregation/geoNear/
 | ||
|  |  * @method near | ||
|  |  * @memberOf Aggregate | ||
|  |  * @instance | ||
|  |  * @param {Object} arg | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.near = function(arg) { | ||
|  |   const op = {}; | ||
|  |   op.$geoNear = arg; | ||
|  |   return this.append(op); | ||
|  | }; | ||
|  | 
 | ||
|  | /*! | ||
|  |  * define methods | ||
|  |  */ | ||
|  | 
 | ||
|  | 'group match skip limit out'.split(' ').forEach(function($operator) { | ||
|  |   Aggregate.prototype[$operator] = function(arg) { | ||
|  |     const op = {}; | ||
|  |     op['$' + $operator] = arg; | ||
|  |     return this.append(op); | ||
|  |   }; | ||
|  | }); | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends new custom $unwind operator(s) to this aggregate pipeline. | ||
|  |  * | ||
|  |  * Note that the `$unwind` operator requires the path name to start with '$'. | ||
|  |  * Mongoose will prepend '$' if the specified field doesn't start '$'. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.unwind("tags"); | ||
|  |  *     aggregate.unwind("a", "b", "c"); | ||
|  |  *     aggregate.unwind({ path: '$tags', preserveNullAndEmptyArrays: true }); | ||
|  |  * | ||
|  |  * @see $unwind http://docs.mongodb.org/manual/reference/aggregation/unwind/
 | ||
|  |  * @param {String|Object} fields the field(s) to unwind, either as field names or as [objects with options](https://docs.mongodb.com/manual/reference/operator/aggregation/unwind/#document-operand-with-options). If passing a string, prefixing the field name with '$' is optional. If passing an object, `path` must start with '$'.
 | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.unwind = function() { | ||
|  |   const args = utils.args(arguments); | ||
|  | 
 | ||
|  |   const res = []; | ||
|  |   for (const arg of args) { | ||
|  |     if (arg && typeof arg === 'object') { | ||
|  |       res.push({ $unwind: arg }); | ||
|  |     } else if (typeof arg === 'string') { | ||
|  |       res.push({ | ||
|  |         $unwind: (arg && arg.startsWith('$')) ? arg : '$' + arg | ||
|  |       }); | ||
|  |     } else { | ||
|  |       throw new Error('Invalid arg "' + arg + '" to unwind(), ' + | ||
|  |         'must be string or object'); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return this.append.apply(this, res); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new $replaceRoot operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * Note that the `$replaceRoot` operator requires field strings to start with '$'. | ||
|  |  * If you are passing in a string Mongoose will prepend '$' if the specified field doesn't start '$'. | ||
|  |  * If you are passing in an object the strings in your expression will not be altered. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.replaceRoot("user"); | ||
|  |  * | ||
|  |  *     aggregate.replaceRoot({ x: { $concat: ['$this', '$that'] } }); | ||
|  |  * | ||
|  |  * @see $replaceRoot https://docs.mongodb.org/manual/reference/operator/aggregation/replaceRoot
 | ||
|  |  * @param {String|Object} the field or document which will become the new root document | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.replaceRoot = function(newRoot) { | ||
|  |   let ret; | ||
|  | 
 | ||
|  |   if (typeof newRoot === 'string') { | ||
|  |     ret = newRoot.startsWith('$') ? newRoot : '$' + newRoot; | ||
|  |   } else { | ||
|  |     ret = newRoot; | ||
|  |   } | ||
|  | 
 | ||
|  |   return this.append({ | ||
|  |     $replaceRoot: { | ||
|  |       newRoot: ret | ||
|  |     } | ||
|  |   }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new $count operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.count("userCount"); | ||
|  |  * | ||
|  |  * @see $count https://docs.mongodb.org/manual/reference/operator/aggregation/count
 | ||
|  |  * @param {String} the name of the count field | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.count = function(countName) { | ||
|  |   return this.append({ $count: countName }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new $sortByCount operator to this aggregate pipeline. Accepts either a string field name | ||
|  |  * or a pipeline object. | ||
|  |  * | ||
|  |  * Note that the `$sortByCount` operator requires the new root to start with '$'. | ||
|  |  * Mongoose will prepend '$' if the specified field name doesn't start with '$'. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.sortByCount('users'); | ||
|  |  *     aggregate.sortByCount({ $mergeObjects: [ "$employee", "$business" ] }) | ||
|  |  * | ||
|  |  * @see $sortByCount https://docs.mongodb.com/manual/reference/operator/aggregation/sortByCount/
 | ||
|  |  * @param {Object|String} arg | ||
|  |  * @return {Aggregate} this | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.sortByCount = function(arg) { | ||
|  |   if (arg && typeof arg === 'object') { | ||
|  |     return this.append({ $sortByCount: arg }); | ||
|  |   } else if (typeof arg === 'string') { | ||
|  |     return this.append({ | ||
|  |       $sortByCount: (arg && arg.startsWith('$')) ? arg : '$' + arg | ||
|  |     }); | ||
|  |   } else { | ||
|  |     throw new TypeError('Invalid arg "' + arg + '" to sortByCount(), ' + | ||
|  |       'must be string or object'); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends new custom $lookup operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.lookup({ from: 'users', localField: 'userId', foreignField: '_id', as: 'users' }); | ||
|  |  * | ||
|  |  * @see $lookup https://docs.mongodb.org/manual/reference/operator/aggregation/lookup/#pipe._S_lookup
 | ||
|  |  * @param {Object} options to $lookup as described in the above link | ||
|  |  * @return {Aggregate}* @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.lookup = function(options) { | ||
|  |   return this.append({ $lookup: options }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends new custom $graphLookup operator(s) to this aggregate pipeline, performing a recursive search on a collection. | ||
|  |  * | ||
|  |  * Note that graphLookup can only consume at most 100MB of memory, and does not allow disk use even if `{ allowDiskUse: true }` is specified. | ||
|  |  * | ||
|  |  * #### Examples: | ||
|  |  *      // Suppose we have a collection of courses, where a document might look like `{ _id: 0, name: 'Calculus', prerequisite: 'Trigonometry'}` and `{ _id: 0, name: 'Trigonometry', prerequisite: 'Algebra' }`
 | ||
|  |  *      aggregate.graphLookup({ from: 'courses', startWith: '$prerequisite', connectFromField: 'prerequisite', connectToField: 'name', as: 'prerequisites', maxDepth: 3 }) // this will recursively search the 'courses' collection up to 3 prerequisites
 | ||
|  |  * | ||
|  |  * @see $graphLookup https://docs.mongodb.com/manual/reference/operator/aggregation/graphLookup/#pipe._S_graphLookup
 | ||
|  |  * @param {Object} options to $graphLookup as described in the above link | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.graphLookup = function(options) { | ||
|  |   const cloneOptions = {}; | ||
|  |   if (options) { | ||
|  |     if (!utils.isObject(options)) { | ||
|  |       throw new TypeError('Invalid graphLookup() argument. Must be an object.'); | ||
|  |     } | ||
|  | 
 | ||
|  |     utils.mergeClone(cloneOptions, options); | ||
|  |     const startWith = cloneOptions.startWith; | ||
|  | 
 | ||
|  |     if (startWith && typeof startWith === 'string') { | ||
|  |       cloneOptions.startWith = cloneOptions.startWith.startsWith('$') ? | ||
|  |         cloneOptions.startWith : | ||
|  |         '$' + cloneOptions.startWith; | ||
|  |     } | ||
|  | 
 | ||
|  |   } | ||
|  |   return this.append({ $graphLookup: cloneOptions }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends new custom $sample operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     aggregate.sample(3); // Add a pipeline that picks 3 random documents
 | ||
|  |  * | ||
|  |  * @see $sample https://docs.mongodb.org/manual/reference/operator/aggregation/sample/#pipe._S_sample
 | ||
|  |  * @param {Number} size number of random documents to pick | ||
|  |  * @return {Aggregate} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.sample = function(size) { | ||
|  |   return this.append({ $sample: { size: size } }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new $sort operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * If an object is passed, values allowed are `asc`, `desc`, `ascending`, `descending`, `1`, and `-1`. | ||
|  |  * | ||
|  |  * If a string is passed, it must be a space delimited list of path names. The sort order of each path is ascending unless the path name is prefixed with `-` which will be treated as descending. | ||
|  |  * | ||
|  |  * ####Examples: | ||
|  |  * | ||
|  |  *     // these are equivalent
 | ||
|  |  *     aggregate.sort({ field: 'asc', test: -1 }); | ||
|  |  *     aggregate.sort('field -test'); | ||
|  |  * | ||
|  |  * @see $sort http://docs.mongodb.org/manual/reference/aggregation/sort/
 | ||
|  |  * @param {Object|String} arg | ||
|  |  * @return {Aggregate} this | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.sort = function(arg) { | ||
|  |   // TODO refactor to reuse the query builder logic
 | ||
|  | 
 | ||
|  |   const sort = {}; | ||
|  | 
 | ||
|  |   if (getConstructorName(arg) === 'Object') { | ||
|  |     const desc = ['desc', 'descending', -1]; | ||
|  |     Object.keys(arg).forEach(function(field) { | ||
|  |       // If sorting by text score, skip coercing into 1/-1
 | ||
|  |       if (arg[field] instanceof Object && arg[field].$meta) { | ||
|  |         sort[field] = arg[field]; | ||
|  |         return; | ||
|  |       } | ||
|  |       sort[field] = desc.indexOf(arg[field]) === -1 ? 1 : -1; | ||
|  |     }); | ||
|  |   } else if (arguments.length === 1 && typeof arg === 'string') { | ||
|  |     arg.split(/\s+/).forEach(function(field) { | ||
|  |       if (!field) { | ||
|  |         return; | ||
|  |       } | ||
|  |       const ascend = field[0] === '-' ? -1 : 1; | ||
|  |       if (ascend === -1) { | ||
|  |         field = field.substring(1); | ||
|  |       } | ||
|  |       sort[field] = ascend; | ||
|  |     }); | ||
|  |   } else { | ||
|  |     throw new TypeError('Invalid sort() argument. Must be a string or object.'); | ||
|  |   } | ||
|  | 
 | ||
|  |   return this.append({ $sort: sort }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Sets the readPreference option for the aggregation query. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     Model.aggregate(..).read('primaryPreferred').exec(callback) | ||
|  |  * | ||
|  |  * @param {String} pref one of the listed preference options or their aliases | ||
|  |  * @param {Array} [tags] optional tags for this query | ||
|  |  * @return {Aggregate} this | ||
|  |  * @api public | ||
|  |  * @see mongodb http://docs.mongodb.org/manual/applications/replication/#read-preference
 | ||
|  |  * @see driver http://mongodb.github.com/node-mongodb-native/driver-articles/anintroductionto1_1and2_2.html#read-preferences
 | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.read = function(pref, tags) { | ||
|  |   if (!this.options) { | ||
|  |     this.options = {}; | ||
|  |   } | ||
|  |   read.call(this, pref, tags); | ||
|  |   return this; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Sets the readConcern level for the aggregation query. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     Model.aggregate(..).readConcern('majority').exec(callback) | ||
|  |  * | ||
|  |  * @param {String} level one of the listed read concern level or their aliases | ||
|  |  * @see mongodb https://docs.mongodb.com/manual/reference/read-concern/
 | ||
|  |  * @return {Aggregate} this | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.readConcern = function(level) { | ||
|  |   if (!this.options) { | ||
|  |     this.options = {}; | ||
|  |   } | ||
|  |   readConcern.call(this, level); | ||
|  |   return this; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Appends a new $redact operator to this aggregate pipeline. | ||
|  |  * | ||
|  |  * If 3 arguments are supplied, Mongoose will wrap them with if-then-else of $cond operator respectively | ||
|  |  * If `thenExpr` or `elseExpr` is string, make sure it starts with $$, like `$$DESCEND`, `$$PRUNE` or `$$KEEP`. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     Model.aggregate(...) | ||
|  |  *      .redact({ | ||
|  |  *        $cond: { | ||
|  |  *          if: { $eq: [ '$level', 5 ] }, | ||
|  |  *          then: '$$PRUNE', | ||
|  |  *          else: '$$DESCEND' | ||
|  |  *        } | ||
|  |  *      }) | ||
|  |  *      .exec(); | ||
|  |  * | ||
|  |  *     // $redact often comes with $cond operator, you can also use the following syntax provided by mongoose
 | ||
|  |  *     Model.aggregate(...) | ||
|  |  *      .redact({ $eq: [ '$level', 5 ] }, '$$PRUNE', '$$DESCEND') | ||
|  |  *      .exec(); | ||
|  |  * | ||
|  |  * @param {Object} expression redact options or conditional expression | ||
|  |  * @param {String|Object} [thenExpr] true case for the condition | ||
|  |  * @param {String|Object} [elseExpr] false case for the condition | ||
|  |  * @return {Aggregate} this | ||
|  |  * @see $redact https://docs.mongodb.com/manual/reference/operator/aggregation/redact/
 | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.redact = function(expression, thenExpr, elseExpr) { | ||
|  |   if (arguments.length === 3) { | ||
|  |     if ((typeof thenExpr === 'string' && !thenExpr.startsWith('$$')) || | ||
|  |         (typeof elseExpr === 'string' && !elseExpr.startsWith('$$'))) { | ||
|  |       throw new Error('If thenExpr or elseExpr is string, it must start with $$. e.g. $$DESCEND, $$PRUNE, $$KEEP'); | ||
|  |     } | ||
|  | 
 | ||
|  |     expression = { | ||
|  |       $cond: { | ||
|  |         if: expression, | ||
|  |         then: thenExpr, | ||
|  |         else: elseExpr | ||
|  |       } | ||
|  |     }; | ||
|  |   } else if (arguments.length !== 1) { | ||
|  |     throw new TypeError('Invalid arguments'); | ||
|  |   } | ||
|  | 
 | ||
|  |   return this.append({ $redact: expression }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Execute the aggregation with explain | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     Model.aggregate(..).explain(callback) | ||
|  |  * | ||
|  |  * @param {Function} callback | ||
|  |  * @return {Promise} | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.explain = function(callback) { | ||
|  |   const model = this._model; | ||
|  | 
 | ||
|  |   return promiseOrCallback(callback, cb => { | ||
|  |     if (!this._pipeline.length) { | ||
|  |       const err = new Error('Aggregate has empty pipeline'); | ||
|  |       return cb(err); | ||
|  |     } | ||
|  | 
 | ||
|  |     prepareDiscriminatorPipeline(this); | ||
|  | 
 | ||
|  |     model.hooks.execPre('aggregate', this, error => { | ||
|  |       if (error) { | ||
|  |         const _opts = { error: error }; | ||
|  |         return model.hooks.execPost('aggregate', this, [null], _opts, error => { | ||
|  |           cb(error); | ||
|  |         }); | ||
|  |       } | ||
|  | 
 | ||
|  |       this.options.explain = true; | ||
|  | 
 | ||
|  |       model.collection. | ||
|  |         aggregate(this._pipeline, this.options || {}). | ||
|  |         explain((error, result) => { | ||
|  |           const _opts = { error: error }; | ||
|  |           return model.hooks.execPost('aggregate', this, [result], _opts, error => { | ||
|  |             if (error) { | ||
|  |               return cb(error); | ||
|  |             } | ||
|  |             return cb(null, result); | ||
|  |           }); | ||
|  |         }); | ||
|  |     }); | ||
|  |   }, model.events); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Sets the allowDiskUse option for the aggregation query (ignored for < 2.6.0) | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     await Model.aggregate([{ $match: { foo: 'bar' } }]).allowDiskUse(true); | ||
|  |  * | ||
|  |  * @param {Boolean} value Should tell server it can use hard drive to store data during aggregation. | ||
|  |  * @param {Array} [tags] optional tags for this query | ||
|  |  * @see mongodb http://docs.mongodb.org/manual/reference/command/aggregate/
 | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.allowDiskUse = function(value) { | ||
|  |   this.options.allowDiskUse = value; | ||
|  |   return this; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Sets the hint option for the aggregation query (ignored for < 3.6.0) | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     Model.aggregate(..).hint({ qty: 1, category: 1 }).exec(callback) | ||
|  |  * | ||
|  |  * @param {Object|String} value a hint object or the index name | ||
|  |  * @see mongodb http://docs.mongodb.org/manual/reference/command/aggregate/
 | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.hint = function(value) { | ||
|  |   this.options.hint = value; | ||
|  |   return this; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Sets the session for this aggregation. Useful for [transactions](/docs/transactions.html). | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     const session = await Model.startSession(); | ||
|  |  *     await Model.aggregate(..).session(session); | ||
|  |  * | ||
|  |  * @param {ClientSession} session | ||
|  |  * @see mongodb http://docs.mongodb.org/manual/reference/command/aggregate/
 | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.session = function(session) { | ||
|  |   if (session == null) { | ||
|  |     delete this.options.session; | ||
|  |   } else { | ||
|  |     this.options.session = session; | ||
|  |   } | ||
|  |   return this; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Lets you set arbitrary options, for middleware or plugins. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     const agg = Model.aggregate(..).option({ allowDiskUse: true }); // Set the `allowDiskUse` option
 | ||
|  |  *     agg.options; // `{ allowDiskUse: true }`
 | ||
|  |  * | ||
|  |  * @param {Object} options keys to merge into current options | ||
|  |  * @param [options.maxTimeMS] number limits the time this aggregation will run, see [MongoDB docs on `maxTimeMS`](https://docs.mongodb.com/manual/reference/operator/meta/maxTimeMS/)
 | ||
|  |  * @param [options.allowDiskUse] boolean if true, the MongoDB server will use the hard drive to store data during this aggregation | ||
|  |  * @param [options.collation] object see [`Aggregate.prototype.collation()`](./docs/api.html#aggregate_Aggregate-collation) | ||
|  |  * @param [options.session] ClientSession see [`Aggregate.prototype.session()`](./docs/api.html#aggregate_Aggregate-session) | ||
|  |  * @see mongodb http://docs.mongodb.org/manual/reference/command/aggregate/
 | ||
|  |  * @return {Aggregate} this | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.option = function(value) { | ||
|  |   for (const key in value) { | ||
|  |     this.options[key] = value[key]; | ||
|  |   } | ||
|  |   return this; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Sets the cursor option option for the aggregation query (ignored for < 2.6.0). | ||
|  |  * Note the different syntax below: .exec() returns a cursor object, and no callback | ||
|  |  * is necessary. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     const cursor = Model.aggregate(..).cursor({ batchSize: 1000 }).exec(); | ||
|  |  *     cursor.eachAsync(function(doc, i) { | ||
|  |  *       // use doc
 | ||
|  |  *     }); | ||
|  |  * | ||
|  |  * @param {Object} options | ||
|  |  * @param {Number} options.batchSize set the cursor batch size | ||
|  |  * @param {Boolean} [options.useMongooseAggCursor] use experimental mongoose-specific aggregation cursor (for `eachAsync()` and other query cursor semantics) | ||
|  |  * @return {Aggregate} this | ||
|  |  * @api public | ||
|  |  * @see mongodb http://mongodb.github.io/node-mongodb-native/2.0/api/AggregationCursor.html
 | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.cursor = function(options) { | ||
|  |   if (!this.options) { | ||
|  |     this.options = {}; | ||
|  |   } | ||
|  |   this.options.cursor = options || {}; | ||
|  |   return this; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Sets an option on this aggregation. This function will be deprecated in a | ||
|  |  * future release. Use the [`cursor()`](./api.html#aggregate_Aggregate-cursor), | ||
|  |  * [`collation()`](./api.html#aggregate_Aggregate-collation), etc. helpers to | ||
|  |  * set individual options, or access `agg.options` directly. | ||
|  |  * | ||
|  |  * Note that MongoDB aggregations [do **not** support the `noCursorTimeout` flag](https://jira.mongodb.org/browse/SERVER-6036),
 | ||
|  |  * if you try setting that flag with this function you will get a "unrecognized field 'noCursorTimeout'" error. | ||
|  |  * | ||
|  |  * @param {String} flag | ||
|  |  * @param {Boolean} value | ||
|  |  * @return {Aggregate} this | ||
|  |  * @api public | ||
|  |  * @deprecated Use [`.option()`](api.html#aggregate_Aggregate-option) instead. Note that MongoDB aggregations do **not** support a `noCursorTimeout` option. | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.addCursorFlag = util.deprecate(function(flag, value) { | ||
|  |   if (!this.options) { | ||
|  |     this.options = {}; | ||
|  |   } | ||
|  |   this.options[flag] = value; | ||
|  |   return this; | ||
|  | }, 'Mongoose: `Aggregate#addCursorFlag()` is deprecated, use `option()` instead'); | ||
|  | 
 | ||
|  | /** | ||
|  |  * Adds a collation | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     Model.aggregate(..).collation({ locale: 'en_US', strength: 1 }).exec(); | ||
|  |  * | ||
|  |  * @param {Object} collation options | ||
|  |  * @return {Aggregate} this | ||
|  |  * @api public | ||
|  |  * @see mongodb http://mongodb.github.io/node-mongodb-native/2.2/api/Collection.html#aggregate
 | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.collation = function(collation) { | ||
|  |   if (!this.options) { | ||
|  |     this.options = {}; | ||
|  |   } | ||
|  |   this.options.collation = collation; | ||
|  |   return this; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Combines multiple aggregation pipelines. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     Model.aggregate(...) | ||
|  |  *      .facet({ | ||
|  |  *        books: [{ groupBy: '$author' }], | ||
|  |  *        price: [{ $bucketAuto: { groupBy: '$price', buckets: 2 } }] | ||
|  |  *      }) | ||
|  |  *      .exec(); | ||
|  |  * | ||
|  |  *     // Output: { books: [...], price: [{...}, {...}] }
 | ||
|  |  * | ||
|  |  * @param {Object} facet options | ||
|  |  * @return {Aggregate} this | ||
|  |  * @see $facet https://docs.mongodb.com/v3.4/reference/operator/aggregation/facet/
 | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.facet = function(options) { | ||
|  |   return this.append({ $facet: options }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Helper for [Atlas Text Search](https://docs.atlas.mongodb.com/reference/atlas-search/tutorial/)'s
 | ||
|  |  * `$search` stage. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     Model.aggregate(). | ||
|  |  *      search({ | ||
|  |  *        text: { | ||
|  |  *          query: 'baseball', | ||
|  |  *          path: 'plot' | ||
|  |  *        } | ||
|  |  *      }); | ||
|  |  * | ||
|  |  *     // Output: [{ plot: '...', title: '...' }]
 | ||
|  |  * | ||
|  |  * @param {Object} $search options | ||
|  |  * @return {Aggregate} this | ||
|  |  * @see $search https://docs.atlas.mongodb.com/reference/atlas-search/tutorial/
 | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.search = function(options) { | ||
|  |   return this.append({ $search: options }); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Returns the current pipeline | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     MyModel.aggregate().match({ test: 1 }).pipeline(); // [{ $match: { test: 1 } }]
 | ||
|  |  * | ||
|  |  * @return {Array} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | 
 | ||
|  | Aggregate.prototype.pipeline = function() { | ||
|  |   return this._pipeline; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Executes the aggregate pipeline on the currently bound Model. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     aggregate.exec(callback); | ||
|  |  * | ||
|  |  *     // Because a promise is returned, the `callback` is optional.
 | ||
|  |  *     const promise = aggregate.exec(); | ||
|  |  *     promise.then(..); | ||
|  |  * | ||
|  |  * @see Promise #promise_Promise | ||
|  |  * @param {Function} [callback] | ||
|  |  * @return {Promise} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.exec = function(callback) { | ||
|  |   if (!this._model) { | ||
|  |     throw new Error('Aggregate not bound to any Model'); | ||
|  |   } | ||
|  |   const model = this._model; | ||
|  |   const collection = this._model.collection; | ||
|  | 
 | ||
|  |   applyGlobalMaxTimeMS(this.options, model); | ||
|  | 
 | ||
|  |   if (this.options && this.options.cursor) { | ||
|  |     return new AggregationCursor(this); | ||
|  |   } | ||
|  | 
 | ||
|  |   return promiseOrCallback(callback, cb => { | ||
|  |     prepareDiscriminatorPipeline(this); | ||
|  |     stringifyFunctionOperators(this._pipeline); | ||
|  | 
 | ||
|  |     model.hooks.execPre('aggregate', this, error => { | ||
|  |       if (error) { | ||
|  |         const _opts = { error: error }; | ||
|  |         return model.hooks.execPost('aggregate', this, [null], _opts, error => { | ||
|  |           cb(error); | ||
|  |         }); | ||
|  |       } | ||
|  |       if (!this._pipeline.length) { | ||
|  |         return cb(new Error('Aggregate has empty pipeline')); | ||
|  |       } | ||
|  | 
 | ||
|  |       const options = utils.clone(this.options || {}); | ||
|  |       collection.aggregate(this._pipeline, options, (error, cursor) => { | ||
|  |         if (error) { | ||
|  |           const _opts = { error: error }; | ||
|  |           return model.hooks.execPost('aggregate', this, [null], _opts, error => { | ||
|  |             if (error) { | ||
|  |               return cb(error); | ||
|  |             } | ||
|  |             return cb(null); | ||
|  |           }); | ||
|  |         } | ||
|  |         cursor.toArray((error, result) => { | ||
|  |           const _opts = { error: error }; | ||
|  |           model.hooks.execPost('aggregate', this, [result], _opts, (error, result) => { | ||
|  |             if (error) { | ||
|  |               return cb(error); | ||
|  |             } | ||
|  | 
 | ||
|  |             cb(null, result); | ||
|  |           }); | ||
|  |         }); | ||
|  |       }); | ||
|  |     }); | ||
|  |   }, model.events); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Provides promise for aggregate. | ||
|  |  * | ||
|  |  * ####Example: | ||
|  |  * | ||
|  |  *     Model.aggregate(..).then(successCallback, errorCallback); | ||
|  |  * | ||
|  |  * @see Promise #promise_Promise | ||
|  |  * @param {Function} [resolve] successCallback | ||
|  |  * @param {Function} [reject]  errorCallback | ||
|  |  * @return {Promise} | ||
|  |  */ | ||
|  | Aggregate.prototype.then = function(resolve, reject) { | ||
|  |   return this.exec().then(resolve, reject); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Executes the query returning a `Promise` which will be | ||
|  |  * resolved with either the doc(s) or rejected with the error. | ||
|  |  * Like [`.then()`](#query_Query-then), but only takes a rejection handler. | ||
|  |  * | ||
|  |  * @param {Function} [reject] | ||
|  |  * @return {Promise} | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate.prototype.catch = function(reject) { | ||
|  |   return this.exec().then(null, reject); | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Returns an asyncIterator for use with [`for/await/of` loops](https://thecodebarbarian.com/getting-started-with-async-iterators-in-node-js
 | ||
|  |  * You do not need to call this function explicitly, the JavaScript runtime | ||
|  |  * will call it for you. | ||
|  |  * | ||
|  |  * ####Example | ||
|  |  * | ||
|  |  *     const agg = Model.aggregate([{ $match: { age: { $gte: 25 } } }]); | ||
|  |  *     for await (const doc of agg) { | ||
|  |  *       console.log(doc.name); | ||
|  |  *     } | ||
|  |  * | ||
|  |  * Node.js 10.x supports async iterators natively without any flags. You can | ||
|  |  * enable async iterators in Node.js 8.x using the [`--harmony_async_iteration` flag](https://github.com/tc39/proposal-async-iteration/issues/117#issuecomment-346695187).
 | ||
|  |  * | ||
|  |  * **Note:** This function is not set if `Symbol.asyncIterator` is undefined. If | ||
|  |  * `Symbol.asyncIterator` is undefined, that means your Node.js version does not | ||
|  |  * support async iterators. | ||
|  |  * | ||
|  |  * @method Symbol.asyncIterator | ||
|  |  * @memberOf Aggregate | ||
|  |  * @instance | ||
|  |  * @api public | ||
|  |  */ | ||
|  | 
 | ||
|  | if (Symbol.asyncIterator != null) { | ||
|  |   Aggregate.prototype[Symbol.asyncIterator] = function() { | ||
|  |     return this.cursor({ useMongooseAggCursor: true }). | ||
|  |       exec(). | ||
|  |       transformNull(). | ||
|  |       _transformForAsyncIterator(); | ||
|  |   }; | ||
|  | } | ||
|  | 
 | ||
|  | /*! | ||
|  |  * Helpers | ||
|  |  */ | ||
|  | 
 | ||
|  | /** | ||
|  |  * Checks whether an object is likely a pipeline operator | ||
|  |  * | ||
|  |  * @param {Object} obj object to check | ||
|  |  * @return {Boolean} | ||
|  |  * @api private | ||
|  |  */ | ||
|  | 
 | ||
|  | function isOperator(obj) { | ||
|  |   if (typeof obj !== 'object') { | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   const k = Object.keys(obj); | ||
|  | 
 | ||
|  |   return k.length === 1 && k.some(key => { return key[0] === '$'; }); | ||
|  | } | ||
|  | 
 | ||
|  | /*! | ||
|  |  * Adds the appropriate `$match` pipeline step to the top of an aggregate's | ||
|  |  * pipeline, should it's model is a non-root discriminator type. This is | ||
|  |  * analogous to the `prepareDiscriminatorCriteria` function in `lib/query.js`. | ||
|  |  * | ||
|  |  * @param {Aggregate} aggregate Aggregate to prepare | ||
|  |  */ | ||
|  | 
 | ||
|  | Aggregate._prepareDiscriminatorPipeline = prepareDiscriminatorPipeline; | ||
|  | 
 | ||
|  | function prepareDiscriminatorPipeline(aggregate) { | ||
|  |   const schema = aggregate._model.schema; | ||
|  |   const discriminatorMapping = schema && schema.discriminatorMapping; | ||
|  | 
 | ||
|  |   if (discriminatorMapping && !discriminatorMapping.isRoot) { | ||
|  |     const originalPipeline = aggregate._pipeline; | ||
|  |     const discriminatorKey = discriminatorMapping.key; | ||
|  |     const discriminatorValue = discriminatorMapping.value; | ||
|  | 
 | ||
|  |     // If the first pipeline stage is a match and it doesn't specify a `__t`
 | ||
|  |     // key, add the discriminator key to it. This allows for potential
 | ||
|  |     // aggregation query optimizations not to be disturbed by this feature.
 | ||
|  |     if (originalPipeline[0] && originalPipeline[0].$match && !originalPipeline[0].$match[discriminatorKey]) { | ||
|  |       originalPipeline[0].$match[discriminatorKey] = discriminatorValue; | ||
|  |       // `originalPipeline` is a ref, so there's no need for
 | ||
|  |       // aggregate._pipeline = originalPipeline
 | ||
|  |     } else if (originalPipeline[0] && originalPipeline[0].$geoNear) { | ||
|  |       originalPipeline[0].$geoNear.query = | ||
|  |           originalPipeline[0].$geoNear.query || {}; | ||
|  |       originalPipeline[0].$geoNear.query[discriminatorKey] = discriminatorValue; | ||
|  |     } else if (originalPipeline[0] && originalPipeline[0].$search) { | ||
|  |       if (originalPipeline[1] && originalPipeline[1].$match != null) { | ||
|  |         originalPipeline[1].$match[discriminatorKey] = originalPipeline[1].$match[discriminatorKey] || discriminatorValue; | ||
|  |       } else { | ||
|  |         const match = {}; | ||
|  |         match[discriminatorKey] = discriminatorValue; | ||
|  |         originalPipeline.splice(1, 0, { $match: match }); | ||
|  |       } | ||
|  |     } else { | ||
|  |       const match = {}; | ||
|  |       match[discriminatorKey] = discriminatorValue; | ||
|  |       aggregate._pipeline.unshift({ $match: match }); | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | /*! | ||
|  |  * Exports | ||
|  |  */ | ||
|  | 
 | ||
|  | module.exports = Aggregate; |