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.
129 lines
3.6 KiB
129 lines
3.6 KiB
"use strict"; |
|
module.exports = function(Promise, PromiseArray, apiRejection, debug) { |
|
var util = require("./util"); |
|
var tryCatch = util.tryCatch; |
|
var errorObj = util.errorObj; |
|
var async = Promise._async; |
|
|
|
Promise.prototype["break"] = Promise.prototype.cancel = function() { |
|
if (!debug.cancellation()) return this._warn("cancellation is disabled"); |
|
|
|
var promise = this; |
|
var child = promise; |
|
while (promise._isCancellable()) { |
|
if (!promise._cancelBy(child)) { |
|
if (child._isFollowing()) { |
|
child._followee().cancel(); |
|
} else { |
|
child._cancelBranched(); |
|
} |
|
break; |
|
} |
|
|
|
var parent = promise._cancellationParent; |
|
if (parent == null || !parent._isCancellable()) { |
|
if (promise._isFollowing()) { |
|
promise._followee().cancel(); |
|
} else { |
|
promise._cancelBranched(); |
|
} |
|
break; |
|
} else { |
|
if (promise._isFollowing()) promise._followee().cancel(); |
|
promise._setWillBeCancelled(); |
|
child = promise; |
|
promise = parent; |
|
} |
|
} |
|
}; |
|
|
|
Promise.prototype._branchHasCancelled = function() { |
|
this._branchesRemainingToCancel--; |
|
}; |
|
|
|
Promise.prototype._enoughBranchesHaveCancelled = function() { |
|
return this._branchesRemainingToCancel === undefined || |
|
this._branchesRemainingToCancel <= 0; |
|
}; |
|
|
|
Promise.prototype._cancelBy = function(canceller) { |
|
if (canceller === this) { |
|
this._branchesRemainingToCancel = 0; |
|
this._invokeOnCancel(); |
|
return true; |
|
} else { |
|
this._branchHasCancelled(); |
|
if (this._enoughBranchesHaveCancelled()) { |
|
this._invokeOnCancel(); |
|
return true; |
|
} |
|
} |
|
return false; |
|
}; |
|
|
|
Promise.prototype._cancelBranched = function() { |
|
if (this._enoughBranchesHaveCancelled()) { |
|
this._cancel(); |
|
} |
|
}; |
|
|
|
Promise.prototype._cancel = function() { |
|
if (!this._isCancellable()) return; |
|
this._setCancelled(); |
|
async.invoke(this._cancelPromises, this, undefined); |
|
}; |
|
|
|
Promise.prototype._cancelPromises = function() { |
|
if (this._length() > 0) this._settlePromises(); |
|
}; |
|
|
|
Promise.prototype._unsetOnCancel = function() { |
|
this._onCancelField = undefined; |
|
}; |
|
|
|
Promise.prototype._isCancellable = function() { |
|
return this.isPending() && !this._isCancelled(); |
|
}; |
|
|
|
Promise.prototype.isCancellable = function() { |
|
return this.isPending() && !this.isCancelled(); |
|
}; |
|
|
|
Promise.prototype._doInvokeOnCancel = function(onCancelCallback, internalOnly) { |
|
if (util.isArray(onCancelCallback)) { |
|
for (var i = 0; i < onCancelCallback.length; ++i) { |
|
this._doInvokeOnCancel(onCancelCallback[i], internalOnly); |
|
} |
|
} else if (onCancelCallback !== undefined) { |
|
if (typeof onCancelCallback === "function") { |
|
if (!internalOnly) { |
|
var e = tryCatch(onCancelCallback).call(this._boundValue()); |
|
if (e === errorObj) { |
|
this._attachExtraTrace(e.e); |
|
async.throwLater(e.e); |
|
} |
|
} |
|
} else { |
|
onCancelCallback._resultCancelled(this); |
|
} |
|
} |
|
}; |
|
|
|
Promise.prototype._invokeOnCancel = function() { |
|
var onCancelCallback = this._onCancel(); |
|
this._unsetOnCancel(); |
|
async.invoke(this._doInvokeOnCancel, this, onCancelCallback); |
|
}; |
|
|
|
Promise.prototype._invokeInternalOnCancel = function() { |
|
if (this._isCancellable()) { |
|
this._doInvokeOnCancel(this._onCancel(), true); |
|
this._unsetOnCancel(); |
|
} |
|
}; |
|
|
|
Promise.prototype._resultCancelled = function() { |
|
this.cancel(); |
|
}; |
|
|
|
};
|
|
|