|  | /* | 
|  | * Copyright (C) 2015 Yusuke Suzuki <utatane.tea@gmail.com>. | 
|  | * Copyright (C) 2016 Apple Inc. All rights reserved. | 
|  | * | 
|  | * Redistribution and use in source and binary forms, with or without | 
|  | * modification, are permitted provided that the following conditions | 
|  | * are met: | 
|  | * 1. Redistributions of source code must retain the above copyright | 
|  | *    notice, this list of conditions and the following disclaimer. | 
|  | * 2. Redistributions in binary form must reproduce the above copyright | 
|  | *    notice, this list of conditions and the following disclaimer in the | 
|  | *    documentation and/or other materials provided with the distribution. | 
|  | * | 
|  | * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY | 
|  | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
|  | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | 
|  | * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR | 
|  | * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | 
|  | * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | 
|  | * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | 
|  | * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY | 
|  | * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
|  | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
|  | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  | */ | 
|  |  | 
|  | // @internal | 
|  |  | 
|  | @globalPrivate | 
|  | function isPromise(promise) | 
|  | { | 
|  | "use strict"; | 
|  |  | 
|  | return @isObject(promise) && !!promise.@promiseState; | 
|  | } | 
|  |  | 
|  | @globalPrivate | 
|  | function newPromiseReaction(capability, handler) | 
|  | { | 
|  | "use strict"; | 
|  |  | 
|  | return { | 
|  | @capabilities: capability, | 
|  | @handler: handler | 
|  | }; | 
|  | } | 
|  |  | 
|  | @globalPrivate | 
|  | function newPromiseCapability(constructor) | 
|  | { | 
|  | "use strict"; | 
|  |  | 
|  | // FIXME: Check isConstructor(constructor). | 
|  | if (typeof constructor !== "function") | 
|  | throw new @TypeError("promise capability requires a constructor function"); | 
|  |  | 
|  | var promiseCapability = { | 
|  | @promise: @undefined, | 
|  | @resolve: @undefined, | 
|  | @reject: @undefined | 
|  | }; | 
|  |  | 
|  | function executor(resolve, reject) | 
|  | { | 
|  | if (promiseCapability.@resolve !== @undefined) | 
|  | throw new @TypeError("resolve function is already set"); | 
|  | if (promiseCapability.@reject !== @undefined) | 
|  | throw new @TypeError("reject function is already set"); | 
|  |  | 
|  | promiseCapability.@resolve = resolve; | 
|  | promiseCapability.@reject = reject; | 
|  | } | 
|  |  | 
|  | var promise = new constructor(executor); | 
|  |  | 
|  | if (typeof promiseCapability.@resolve !== "function") | 
|  | throw new @TypeError("executor did not take a resolve function"); | 
|  |  | 
|  | if (typeof promiseCapability.@reject !== "function") | 
|  | throw new @TypeError("executor did not take a reject function"); | 
|  |  | 
|  | promiseCapability.@promise = promise; | 
|  |  | 
|  | return promiseCapability; | 
|  | } | 
|  |  | 
|  | @globalPrivate | 
|  | function triggerPromiseReactions(reactions, argument) | 
|  | { | 
|  | "use strict"; | 
|  |  | 
|  | for (var index = 0, length = reactions.length; index < length; ++index) | 
|  | @enqueueJob(@promiseReactionJob, [reactions[index], argument]); | 
|  | } | 
|  |  | 
|  | @globalPrivate | 
|  | function rejectPromise(promise, reason) | 
|  | { | 
|  | "use strict"; | 
|  |  | 
|  | var reactions = promise.@promiseRejectReactions; | 
|  | promise.@promiseResult = reason; | 
|  | promise.@promiseFulfillReactions = @undefined; | 
|  | promise.@promiseRejectReactions = @undefined; | 
|  | promise.@promiseState = @promiseStateRejected; | 
|  |  | 
|  | @InspectorInstrumentation.promiseRejected(promise, reason, reactions); | 
|  |  | 
|  | @triggerPromiseReactions(reactions, reason); | 
|  | } | 
|  |  | 
|  | @globalPrivate | 
|  | function fulfillPromise(promise, value) | 
|  | { | 
|  | "use strict"; | 
|  |  | 
|  | var reactions = promise.@promiseFulfillReactions; | 
|  | promise.@promiseResult = value; | 
|  | promise.@promiseFulfillReactions = @undefined; | 
|  | promise.@promiseRejectReactions = @undefined; | 
|  | promise.@promiseState = @promiseStateFulfilled; | 
|  |  | 
|  | @InspectorInstrumentation.promiseFulfilled(promise, value, reactions); | 
|  |  | 
|  | @triggerPromiseReactions(reactions, value); | 
|  | } | 
|  |  | 
|  | @globalPrivate | 
|  | function createResolvingFunctions(promise) | 
|  | { | 
|  | "use strict"; | 
|  |  | 
|  | var alreadyResolved = false; | 
|  |  | 
|  | var resolve = function (resolution) { | 
|  | if (alreadyResolved) | 
|  | return @undefined; | 
|  | alreadyResolved = true; | 
|  |  | 
|  | if (resolution === promise) | 
|  | return @rejectPromise(promise, new @TypeError("Resolve a promise with itself")); | 
|  |  | 
|  | if (!@isObject(resolution)) | 
|  | return @fulfillPromise(promise, resolution); | 
|  |  | 
|  | var then; | 
|  | try { | 
|  | then = resolution.then; | 
|  | } catch (error) { | 
|  | return @rejectPromise(promise, error); | 
|  | } | 
|  |  | 
|  | if (typeof then !== 'function') | 
|  | return @fulfillPromise(promise, resolution); | 
|  |  | 
|  | @enqueueJob(@promiseResolveThenableJob, [promise, resolution, then]); | 
|  |  | 
|  | return @undefined; | 
|  | }; | 
|  |  | 
|  | var reject = function (reason) { | 
|  | if (alreadyResolved) | 
|  | return @undefined; | 
|  | alreadyResolved = true; | 
|  |  | 
|  | return @rejectPromise(promise, reason); | 
|  | }; | 
|  |  | 
|  | return { | 
|  | @resolve: resolve, | 
|  | @reject: reject | 
|  | }; | 
|  | } | 
|  |  | 
|  | @globalPrivate | 
|  | function promiseReactionJob(reaction, argument) | 
|  | { | 
|  | "use strict"; | 
|  |  | 
|  | var promiseCapability = reaction.@capabilities; | 
|  |  | 
|  | var result; | 
|  | try { | 
|  | result = reaction.@handler.@call(@undefined, argument); | 
|  | } catch (error) { | 
|  | return promiseCapability.@reject.@call(@undefined, error); | 
|  | } | 
|  |  | 
|  | return promiseCapability.@resolve.@call(@undefined, result); | 
|  | } | 
|  |  | 
|  | @globalPrivate | 
|  | function promiseResolveThenableJob(promiseToResolve, thenable, then) | 
|  | { | 
|  | "use strict"; | 
|  |  | 
|  | var resolvingFunctions = @createResolvingFunctions(promiseToResolve); | 
|  |  | 
|  | try { | 
|  | return then.@call(thenable, resolvingFunctions.@resolve, resolvingFunctions.@reject); | 
|  | } catch (error) { | 
|  | return resolvingFunctions.@reject.@call(@undefined, error); | 
|  | } | 
|  | } | 
|  |  | 
|  | @globalPrivate | 
|  | function initializePromise(executor) | 
|  | { | 
|  | "use strict"; | 
|  |  | 
|  | if (typeof executor !== 'function') | 
|  | throw new @TypeError("Promise constructor takes a function argument"); | 
|  |  | 
|  | this.@promiseState = @promiseStatePending; | 
|  | this.@promiseFulfillReactions = []; | 
|  | this.@promiseRejectReactions = []; | 
|  |  | 
|  | var resolvingFunctions = @createResolvingFunctions(this); | 
|  | try { | 
|  | executor(resolvingFunctions.@resolve, resolvingFunctions.@reject); | 
|  | } catch (error) { | 
|  | return resolvingFunctions.@reject.@call(@undefined, error); | 
|  | } | 
|  |  | 
|  | return this; | 
|  | } |