184 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			184 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| "use strict";
 | |
| module.exports = function(Promise,
 | |
|                           PromiseArray,
 | |
|                           apiRejection,
 | |
|                           tryConvertToPromise,
 | |
|                           INTERNAL,
 | |
|                           debug) {
 | |
| var util = require("./util");
 | |
| var tryCatch = util.tryCatch;
 | |
| 
 | |
| function ReductionPromiseArray(promises, fn, initialValue, _each) {
 | |
|     this.constructor$(promises);
 | |
|     var context = Promise._getContext();
 | |
|     this._fn = util.contextBind(context, fn);
 | |
|     if (initialValue !== undefined) {
 | |
|         initialValue = Promise.resolve(initialValue);
 | |
|         initialValue._attachCancellationCallback(this);
 | |
|     }
 | |
|     this._initialValue = initialValue;
 | |
|     this._currentCancellable = null;
 | |
|     if(_each === INTERNAL) {
 | |
|         this._eachValues = Array(this._length);
 | |
|     } else if (_each === 0) {
 | |
|         this._eachValues = null;
 | |
|     } else {
 | |
|         this._eachValues = undefined;
 | |
|     }
 | |
|     this._promise._captureStackTrace();
 | |
|     this._init$(undefined, -5);
 | |
| }
 | |
| util.inherits(ReductionPromiseArray, PromiseArray);
 | |
| 
 | |
| ReductionPromiseArray.prototype._gotAccum = function(accum) {
 | |
|     if (this._eachValues !== undefined &&
 | |
|         this._eachValues !== null &&
 | |
|         accum !== INTERNAL) {
 | |
|         this._eachValues.push(accum);
 | |
|     }
 | |
| };
 | |
| 
 | |
| ReductionPromiseArray.prototype._eachComplete = function(value) {
 | |
|     if (this._eachValues !== null) {
 | |
|         this._eachValues.push(value);
 | |
|     }
 | |
|     return this._eachValues;
 | |
| };
 | |
| 
 | |
| ReductionPromiseArray.prototype._init = function() {};
 | |
| 
 | |
| ReductionPromiseArray.prototype._resolveEmptyArray = function() {
 | |
|     this._resolve(this._eachValues !== undefined ? this._eachValues
 | |
|                                                  : this._initialValue);
 | |
| };
 | |
| 
 | |
| ReductionPromiseArray.prototype.shouldCopyValues = function () {
 | |
|     return false;
 | |
| };
 | |
| 
 | |
| ReductionPromiseArray.prototype._resolve = function(value) {
 | |
|     this._promise._resolveCallback(value);
 | |
|     this._values = null;
 | |
| };
 | |
| 
 | |
| ReductionPromiseArray.prototype._resultCancelled = function(sender) {
 | |
|     if (sender === this._initialValue) return this._cancel();
 | |
|     if (this._isResolved()) return;
 | |
|     this._resultCancelled$();
 | |
|     if (this._currentCancellable instanceof Promise) {
 | |
|         this._currentCancellable.cancel();
 | |
|     }
 | |
|     if (this._initialValue instanceof Promise) {
 | |
|         this._initialValue.cancel();
 | |
|     }
 | |
| };
 | |
| 
 | |
| ReductionPromiseArray.prototype._iterate = function (values) {
 | |
|     this._values = values;
 | |
|     var value;
 | |
|     var i;
 | |
|     var length = values.length;
 | |
|     if (this._initialValue !== undefined) {
 | |
|         value = this._initialValue;
 | |
|         i = 0;
 | |
|     } else {
 | |
|         value = Promise.resolve(values[0]);
 | |
|         i = 1;
 | |
|     }
 | |
| 
 | |
|     this._currentCancellable = value;
 | |
| 
 | |
|     for (var j = i; j < length; ++j) {
 | |
|         var maybePromise = values[j];
 | |
|         if (maybePromise instanceof Promise) {
 | |
|             maybePromise.suppressUnhandledRejections();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (!value.isRejected()) {
 | |
|         for (; i < length; ++i) {
 | |
|             var ctx = {
 | |
|                 accum: null,
 | |
|                 value: values[i],
 | |
|                 index: i,
 | |
|                 length: length,
 | |
|                 array: this
 | |
|             };
 | |
| 
 | |
|             value = value._then(gotAccum, undefined, undefined, ctx, undefined);
 | |
| 
 | |
|             if ((i & 127) === 0) {
 | |
|                 value._setNoAsyncGuarantee();
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (this._eachValues !== undefined) {
 | |
|         value = value
 | |
|             ._then(this._eachComplete, undefined, undefined, this, undefined);
 | |
|     }
 | |
|     value._then(completed, completed, undefined, value, this);
 | |
| };
 | |
| 
 | |
| Promise.prototype.reduce = function (fn, initialValue) {
 | |
|     return reduce(this, fn, initialValue, null);
 | |
| };
 | |
| 
 | |
| Promise.reduce = function (promises, fn, initialValue, _each) {
 | |
|     return reduce(promises, fn, initialValue, _each);
 | |
| };
 | |
| 
 | |
| function completed(valueOrReason, array) {
 | |
|     if (this.isFulfilled()) {
 | |
|         array._resolve(valueOrReason);
 | |
|     } else {
 | |
|         array._reject(valueOrReason);
 | |
|     }
 | |
| }
 | |
| 
 | |
| function reduce(promises, fn, initialValue, _each) {
 | |
|     if (typeof fn !== "function") {
 | |
|         return apiRejection("expecting a function but got " + util.classString(fn));
 | |
|     }
 | |
|     var array = new ReductionPromiseArray(promises, fn, initialValue, _each);
 | |
|     return array.promise();
 | |
| }
 | |
| 
 | |
| function gotAccum(accum) {
 | |
|     this.accum = accum;
 | |
|     this.array._gotAccum(accum);
 | |
|     var value = tryConvertToPromise(this.value, this.array._promise);
 | |
|     if (value instanceof Promise) {
 | |
|         this.array._currentCancellable = value;
 | |
|         return value._then(gotValue, undefined, undefined, this, undefined);
 | |
|     } else {
 | |
|         return gotValue.call(this, value);
 | |
|     }
 | |
| }
 | |
| 
 | |
| function gotValue(value) {
 | |
|     var array = this.array;
 | |
|     var promise = array._promise;
 | |
|     var fn = tryCatch(array._fn);
 | |
|     promise._pushContext();
 | |
|     var ret;
 | |
|     if (array._eachValues !== undefined) {
 | |
|         ret = fn.call(promise._boundValue(), value, this.index, this.length);
 | |
|     } else {
 | |
|         ret = fn.call(promise._boundValue(),
 | |
|                               this.accum, value, this.index, this.length);
 | |
|     }
 | |
|     if (ret instanceof Promise) {
 | |
|         array._currentCancellable = ret;
 | |
|     }
 | |
|     var promiseCreated = promise._popContext();
 | |
|     debug.checkForgottenReturns(
 | |
|         ret,
 | |
|         promiseCreated,
 | |
|         array._eachValues !== undefined ? "Promise.each" : "Promise.reduce",
 | |
|         promise
 | |
|     );
 | |
|     return ret;
 | |
| }
 | |
| };
 |