mirror of
				https://github.com/tc39/test262.git
				synced 2025-10-25 17:53:53 +02:00 
			
		
		
		
	sourceRevisionAtLastExport: 33f2fb0e53d135f0ee17cfccd9d993eb2a6f47de targetRevisionAtLastExport: 31340cbd9add103f586d501b0c3354b7b182abc0
		
			
				
	
	
		
			1026 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			1026 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| // Copyright 2013 the V8 project authors. All rights reserved.
 | |
| // Redistribution and use in source and binary forms, with or without
 | |
| // modification, are permitted provided that the following conditions are
 | |
| // met:
 | |
| //
 | |
| //     * Redistributions of source code must retain the above copyright
 | |
| //       notice, this list of conditions and the following disclaimer.
 | |
| //     * 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.
 | |
| //     * Neither the name of Google Inc. nor the names of its
 | |
| //       contributors may be used to endorse or promote products derived
 | |
| //       from this software without specific prior written permission.
 | |
| //
 | |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 | |
| // "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 THE COPYRIGHT
 | |
| // OWNER 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.
 | |
| 
 | |
| // Flags: --allow-natives-syntax
 | |
| 
 | |
| // Make sure we don't rely on functions patchable by monkeys.
 | |
| var call = Function.prototype.call.call.bind(Function.prototype.call)
 | |
| var getOwnPropertyNames = Object.getOwnPropertyNames;
 | |
| var defineProperty = Object.defineProperty;
 | |
| var numberPrototype = Number.prototype;
 | |
| var symbolIterator = Symbol.iterator;
 | |
| 
 | |
| function assertUnreachable() {
 | |
|   %AbortJS("Failure: unreachable");
 | |
| }
 | |
| 
 | |
| (function() {
 | |
|   // Test before clearing global (fails otherwise)
 | |
|   assertEquals("[object Promise]",
 | |
|       Object.prototype.toString.call(new Promise(function() {})));
 | |
| })();
 | |
| 
 | |
| 
 | |
| function clear(o) {
 | |
|   if (o === null || (typeof o !== 'object' && typeof o !== 'function')) return
 | |
|   clear(o.__proto__)
 | |
|   var properties = getOwnPropertyNames(o)
 | |
|   for (var i in properties) {
 | |
|     // Do not clobber Object.prototype.toString, which is used by tests.
 | |
|     if (properties[i] === "toString") continue;
 | |
|     clearProp(o, properties[i])
 | |
|   }
 | |
| }
 | |
| 
 | |
| function clearProp(o, name) {
 | |
|   var poisoned = {caller: 0, callee: 0, arguments: 0}
 | |
|   try {
 | |
|     var x = o[name]
 | |
|     o[name] = undefined
 | |
|     clear(x)
 | |
|   } catch(e) {} // assertTrue(name in poisoned) }
 | |
| }
 | |
| 
 | |
| // Find intrinsics and null them out.
 | |
| var globals = Object.getOwnPropertyNames(this)
 | |
| var whitelist = {
 | |
|   Promise: true,
 | |
|   TypeError: true,
 | |
|   String: true,
 | |
|   JSON: true,
 | |
|   Error: true,
 | |
|   MjsUnitAssertionError: true
 | |
| };
 | |
| 
 | |
| for (var i in globals) {
 | |
|   var name = globals[i]
 | |
|   if (name in whitelist || name[0] === name[0].toLowerCase()) delete globals[i]
 | |
| }
 | |
| for (var i in globals) {
 | |
|   if (globals[i]) clearProp(this, globals[i])
 | |
| }
 | |
| 
 | |
| 
 | |
| function defer(constructor) {
 | |
|   var resolve, reject;
 | |
|   var promise = new constructor((res, rej) => { resolve = res; reject = rej });
 | |
|   return { promise, resolve, reject };
 | |
| }
 | |
| 
 | |
| var asyncAssertsExpected = 0;
 | |
| 
 | |
| function assertAsyncRan() { ++asyncAssertsExpected }
 | |
| 
 | |
| function assertAsync(b, s) {
 | |
|   if (b) {
 | |
|     print(s, "succeeded")
 | |
|   } else {
 | |
|     %AbortJS(s + " FAILED!")  // Simply throwing here will have no effect.
 | |
|   }
 | |
|   --asyncAssertsExpected
 | |
| }
 | |
| 
 | |
| function assertLater(f, name) {
 | |
|   assertFalse(f()); // should not be true synchronously
 | |
|   ++asyncAssertsExpected;
 | |
|   var iterations = 0;
 | |
|   function runAssertion() {
 | |
|     if (f()) {
 | |
|       print(name, "succeeded");
 | |
|       --asyncAssertsExpected;
 | |
|     } else if (iterations++ < 10) {
 | |
|       %EnqueueMicrotask(runAssertion);
 | |
|     } else {
 | |
|       %AbortJS(name + " FAILED!");
 | |
|     }
 | |
|   }
 | |
|   %EnqueueMicrotask(runAssertion);
 | |
| }
 | |
| 
 | |
| function assertAsyncDone(iteration) {
 | |
|   var iteration = iteration || 0;
 | |
|   %EnqueueMicrotask(function() {
 | |
|     if (asyncAssertsExpected === 0)
 | |
|       assertAsync(true, "all")
 | |
|     else if (iteration > 10)  // Shouldn't take more.
 | |
|       assertAsync(false, "all... " + asyncAssertsExpected)
 | |
|     else
 | |
|       assertAsyncDone(iteration + 1)
 | |
|   });
 | |
| }
 | |
| 
 | |
| (function() {
 | |
|   assertThrows(function() { Promise(function() {}) }, TypeError)
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   assertTrue(new Promise(function() {}) instanceof Promise)
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   assertThrows(function() { new Promise(5) }, TypeError)
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   assertDoesNotThrow(function() { new Promise(function() { throw 5 }) })
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   (new Promise(function() { throw 5 })).then(
 | |
|     assertUnreachable,
 | |
|     function(r) { assertAsync(r === 5, "new-throw") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   Promise.resolve(5);
 | |
|   Promise.resolve(5).then(undefined, assertUnreachable).then(
 | |
|     function(x) { assertAsync(x === 5, "resolved/then-nohandler") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   Promise.resolve(5).then(undefined, assertUnreachable).then(
 | |
|     function(x) { assertAsync(x === 5, "resolved/then-nohandler-undefined") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
|   Promise.resolve(6).then(null, assertUnreachable).then(
 | |
|     function(x) { assertAsync(x === 6, "resolved/then-nohandler-null") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "resolved/then") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.reject(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 5, "rejected/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(function(x) { return x }, assertUnreachable).then(
 | |
|     function(x) { assertAsync(x === 5, "resolved/then/then") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(function(x){ return Promise.resolve(x+1) }, assertUnreachable).then(
 | |
|     function(x) { assertAsync(x === 6, "resolved/then/then2") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(function(x) { throw 6 }, assertUnreachable).then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 6, "resolved/then-throw/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(function(x) { throw 6 }, assertUnreachable).then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 6, "resolved/then-throw/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(function(x) { throw 6 }, assertUnreachable).then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 6, "resolved/then-throw/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(function(x) { throw 6 }, assertUnreachable).then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 6, "resolved/then-throw/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = {then: function(onResolve, onReject) { onResolve(p1) }}
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "resolved/thenable/then") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = {then: function(onResolve, onReject) { onResolve(p1) }}
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "resolved/thenable/then") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.reject(5)
 | |
|   var p2 = {then: function(onResolve, onReject) { onResolve(p1) }}
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 5, "rejected/thenable/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.reject(5)
 | |
|   var p2 = {then: function(onResolve, onReject) { onResolve(p1) }}
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 5, "rejected/thenable/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "then/resolve") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "then/resolve") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 5, "then/reject") }
 | |
|   )
 | |
|   deferred.reject(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 5, "then/reject") }
 | |
|   )
 | |
|   deferred.reject(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = p1.then(1, 2)
 | |
|   p2.then(
 | |
|     function(x) { assertAsync(x === 5, "then/resolve-non-function") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = p1.then(1, 2)
 | |
|   p2.then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 5, "then/reject-non-function") }
 | |
|   )
 | |
|   deferred.reject(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = {then: function(onResolve, onReject) { onResolve(p1) }}
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "then/resolve/thenable") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = {then: function(onResolve, onReject) { onResolve(p1) }}
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "then/resolve/thenable") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = {then: function(onResolve, onReject) { onResolve(p1) }}
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 5, "then/reject/thenable") }
 | |
|   )
 | |
|   deferred.reject(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = {then: function(onResolve, onReject) { onResolve(p1) }}
 | |
|   var p3 = Promise.resolve(p2)
 | |
|   p3.then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 5, "then/reject/thenable") }
 | |
|   )
 | |
|   deferred.reject(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var deferred = defer(Promise)
 | |
|   var p3 = deferred.promise
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "then/resolve2") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(p2)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var deferred = defer(Promise)
 | |
|   var p3 = deferred.promise
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "then/resolve2") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(p2)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var deferred = defer(Promise)
 | |
|   var p3 = deferred.promise
 | |
|   p3.then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 5, "then/reject2") }
 | |
|   )
 | |
|   deferred.reject(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = Promise.resolve(p1)
 | |
|   var deferred = defer(Promise)
 | |
|   var p3 = deferred.promise
 | |
|   p3.then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 5, "then/reject2") }
 | |
|   )
 | |
|   deferred.reject(5)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = {then: function(onResolve, onReject) { onResolve(p1) }}
 | |
|   var deferred = defer(Promise)
 | |
|   var p3 = deferred.promise
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "then/resolve/thenable2") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(p2)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(5)
 | |
|   var p2 = {then: function(onResolve, onReject) { onResolve(p1) }}
 | |
|   var deferred = defer(Promise)
 | |
|   var p3 = deferred.promise
 | |
|   p3.then(
 | |
|     function(x) { assertAsync(x === 5, "then/resolve/thenable2") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(p2)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(0)
 | |
|   var p2 = p1.then(function(x) { return p2 }, assertUnreachable)
 | |
|   p2.then(
 | |
|     assertUnreachable,
 | |
|     function(r) { assertAsync(r instanceof TypeError, "cyclic/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(0)
 | |
|   var p2 = p1.then(function(x) { return p2 }, assertUnreachable)
 | |
|   p2.then(
 | |
|     assertUnreachable,
 | |
|     function(r) { assertAsync(r instanceof TypeError, "cyclic/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p = deferred.promise
 | |
|   deferred.resolve(p)
 | |
|   p.then(
 | |
|     assertUnreachable,
 | |
|     function(r) { assertAsync(r instanceof TypeError, "cyclic/deferred/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p = deferred.promise
 | |
|   deferred.resolve(p)
 | |
|   p.then(
 | |
|     assertUnreachable,
 | |
|     function(r) { assertAsync(r instanceof TypeError, "cyclic/deferred/then") }
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   Promise.all([]).then(
 | |
|     function(x) { assertAsync(x.length === 0, "all/resolve/empty") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   function testPromiseAllNonIterable(value) {
 | |
|     Promise.all(value).then(
 | |
|         assertUnreachable,
 | |
|         function(r) {
 | |
|           assertAsync(r instanceof TypeError, 'all/non iterable');
 | |
|         });
 | |
|     assertAsyncRan();
 | |
|   }
 | |
|   testPromiseAllNonIterable(null);
 | |
|   testPromiseAllNonIterable(undefined);
 | |
|   testPromiseAllNonIterable({});
 | |
|   testPromiseAllNonIterable(42);
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   Promise.all({[symbolIterator](){ return null; }}).then(
 | |
|     assertUnreachable,
 | |
|     function(r) {
 | |
|       assertAsync(r instanceof TypeError, 'all/non iterable');
 | |
|     });
 | |
|   assertAsyncRan();
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise);
 | |
|   var p = deferred.promise;
 | |
|   function* f() {
 | |
|     yield 1;
 | |
|     yield p;
 | |
|     yield 3;
 | |
|   }
 | |
|   Promise.all(f()).then(
 | |
|       function(x) {
 | |
|         assertAsync(x.length === 3, "all/resolve/iterable");
 | |
|         assertAsync(x[0] === 1, "all/resolve/iterable/0");
 | |
|         assertAsync(x[1] === 2, "all/resolve/iterable/1");
 | |
|         assertAsync(x[2] === 3, "all/resolve/iterable/2");
 | |
|       },
 | |
|       assertUnreachable);
 | |
|   deferred.resolve(2);
 | |
|   assertAsyncRan();
 | |
|   assertAsyncRan();
 | |
|   assertAsyncRan();
 | |
|   assertAsyncRan();
 | |
| })();
 | |
| 
 | |
| 
 | |
| (function() {
 | |
|   var deferred1 = defer(Promise)
 | |
|   var p1 = deferred1.promise
 | |
|   var deferred2 = defer(Promise)
 | |
|   var p2 = deferred2.promise
 | |
|   var deferred3 = defer(Promise)
 | |
|   var p3 = deferred3.promise
 | |
|   Promise.all([p1, p2, p3]).then(
 | |
|     function(x) {
 | |
|       assertAsync(x.length === 3, "all/resolve")
 | |
|       assertAsync(x[0] === 1, "all/resolve/0")
 | |
|       assertAsync(x[1] === 2, "all/resolve/1")
 | |
|       assertAsync(x[2] === 3, "all/resolve/2")
 | |
|     },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred1.resolve(1)
 | |
|   deferred3.resolve(3)
 | |
|   deferred2.resolve(2)
 | |
|   assertAsyncRan()
 | |
|   assertAsyncRan()
 | |
|   assertAsyncRan()
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = Promise.resolve(2)
 | |
|   var p3 = defer(Promise).promise
 | |
|   Promise.all([p1, p2, p3]).then(
 | |
|     assertUnreachable,
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(1)
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred1 = defer(Promise)
 | |
|   var p1 = deferred1.promise
 | |
|   var deferred2 = defer(Promise)
 | |
|   var p2 = deferred2.promise
 | |
|   var deferred3 = defer(Promise)
 | |
|   var p3 = deferred3.promise
 | |
|   Promise.all([p1, p2, p3]).then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 2, "all/reject") }
 | |
|   )
 | |
|   deferred1.resolve(1)
 | |
|   deferred3.resolve(3)
 | |
|   deferred2.reject(2)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   'use strict';
 | |
|   var getCalls = 0;
 | |
|   var funcCalls = 0;
 | |
|   var nextCalls = 0;
 | |
|   defineProperty(numberPrototype, symbolIterator, {
 | |
|     get: function() {
 | |
|       assertEquals('number', typeof this);
 | |
|       getCalls++;
 | |
|       return function() {
 | |
|         assertEquals('number', typeof this);
 | |
|         funcCalls++;
 | |
|         var n = this;
 | |
|         var i = 0
 | |
|         return {
 | |
|           next() {
 | |
|             nextCalls++;
 | |
|             return {value: i++, done: i > n};
 | |
|           }
 | |
|         };
 | |
|       };
 | |
|     },
 | |
|     configurable: true
 | |
|   });
 | |
| 
 | |
|   Promise.all(3).then(
 | |
|       function(x) {
 | |
|         assertAsync(x.length === 3, "all/iterable/number/length");
 | |
|         assertAsync(x[0] === 0, "all/iterable/number/0");
 | |
|         assertAsync(x[1] === 1, "all/iterable/number/1");
 | |
|         assertAsync(x[2] === 2, "all/iterable/number/2");
 | |
|       },
 | |
|       assertUnreachable);
 | |
|   delete numberPrototype[symbolIterator];
 | |
| 
 | |
|   assertEquals(getCalls, 1);
 | |
|   assertEquals(funcCalls, 1);
 | |
|   assertEquals(nextCalls, 3 + 1);  // + 1 for {done: true}
 | |
|   assertAsyncRan();
 | |
|   assertAsyncRan();
 | |
|   assertAsyncRan();
 | |
|   assertAsyncRan();
 | |
| })();
 | |
| 
 | |
| 
 | |
| (function() {
 | |
|   Promise.race([]).then(
 | |
|     assertUnreachable,
 | |
|     assertUnreachable
 | |
|   )
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(1)
 | |
|   var p2 = Promise.resolve(2)
 | |
|   var p3 = Promise.resolve(3)
 | |
|   Promise.race([p1, p2, p3]).then(
 | |
|     function(x) { assertAsync(x === 1, "resolved/one") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p1 = Promise.resolve(1)
 | |
|   var p2 = Promise.resolve(2)
 | |
|   var p3 = Promise.resolve(3)
 | |
|   Promise.race([0, p1, p2, p3]).then(
 | |
|     function(x) { assertAsync(x === 0, "resolved-const/one") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred1 = defer(Promise)
 | |
|   var p1 = deferred1.promise
 | |
|   var deferred2 = defer(Promise)
 | |
|   var p2 = deferred2.promise
 | |
|   var deferred3 = defer(Promise)
 | |
|   var p3 = deferred3.promise
 | |
|   Promise.race([p1, p2, p3]).then(
 | |
|     function(x) { assertAsync(x === 3, "one/resolve") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred3.resolve(3)
 | |
|   deferred1.resolve(1)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred = defer(Promise)
 | |
|   var p1 = deferred.promise
 | |
|   var p2 = Promise.resolve(2)
 | |
|   var p3 = defer(Promise).promise
 | |
|   Promise.race([p1, p2, p3]).then(
 | |
|     function(x) { assertAsync(x === 2, "resolved/one") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred.resolve(1)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred1 = defer(Promise)
 | |
|   var p1 = deferred1.promise
 | |
|   var deferred2 = defer(Promise)
 | |
|   var p2 = deferred2.promise
 | |
|   var deferred3 = defer(Promise)
 | |
|   var p3 = deferred3.promise
 | |
|   Promise.race([p1, p2, p3]).then(
 | |
|     function(x) { assertAsync(x === 3, "one/resolve/reject") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred3.resolve(3)
 | |
|   deferred1.reject(1)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred1 = defer(Promise)
 | |
|   var p1 = deferred1.promise
 | |
|   var deferred2 = defer(Promise)
 | |
|   var p2 = deferred2.promise
 | |
|   var deferred3 = defer(Promise)
 | |
|   var p3 = deferred3.promise
 | |
|   Promise.race([p1, p2, p3]).then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 3, "one/reject/resolve") }
 | |
|   )
 | |
|   deferred3.reject(3)
 | |
|   deferred1.resolve(1)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| 
 | |
| (function() {
 | |
|   function testPromiseRaceNonIterable(value) {
 | |
|     Promise.race(value).then(
 | |
|         assertUnreachable,
 | |
|         function(r) {
 | |
|           assertAsync(r instanceof TypeError, 'race/non iterable');
 | |
|         });
 | |
|     assertAsyncRan();
 | |
|   }
 | |
|   testPromiseRaceNonIterable(null);
 | |
|   testPromiseRaceNonIterable(undefined);
 | |
|   testPromiseRaceNonIterable({});
 | |
|   testPromiseRaceNonIterable(42);
 | |
| })();
 | |
| 
 | |
| 
 | |
| (function() {
 | |
|   var deferred1 = defer(Promise)
 | |
|   var p1 = deferred1.promise
 | |
|   var deferred2 = defer(Promise)
 | |
|   var p2 = deferred2.promise
 | |
|   var deferred3 = defer(Promise)
 | |
|   var p3 = deferred3.promise
 | |
|   function* f() {
 | |
|     yield p1;
 | |
|     yield p2;
 | |
|     yield p3;
 | |
|   }
 | |
|   Promise.race(f()).then(
 | |
|     function(x) { assertAsync(x === 3, "race/iterable/resolve/reject") },
 | |
|     assertUnreachable
 | |
|   )
 | |
|   deferred3.resolve(3)
 | |
|   deferred1.reject(1)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var deferred1 = defer(Promise)
 | |
|   var p1 = deferred1.promise
 | |
|   var deferred2 = defer(Promise)
 | |
|   var p2 = deferred2.promise
 | |
|   var deferred3 = defer(Promise)
 | |
|   var p3 = deferred3.promise
 | |
|   function* f() {
 | |
|     yield p1;
 | |
|     yield p2;
 | |
|     yield p3;
 | |
|   }
 | |
|   Promise.race(f()).then(
 | |
|     assertUnreachable,
 | |
|     function(x) { assertAsync(x === 3, "race/iterable/reject/resolve") }
 | |
|   )
 | |
|   deferred3.reject(3)
 | |
|   deferred1.resolve(1)
 | |
|   assertAsyncRan()
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   'use strict';
 | |
|   var getCalls = 0;
 | |
|   var funcCalls = 0;
 | |
|   var nextCalls = 0;
 | |
|   defineProperty(numberPrototype, symbolIterator, {
 | |
|     get: function() {
 | |
|       assertEquals('number', typeof this);
 | |
|       getCalls++;
 | |
|       return function() {
 | |
|         assertEquals('number', typeof this);
 | |
|         funcCalls++;
 | |
|         var n = this;
 | |
|         var i = 0
 | |
|         return {
 | |
|           next() {
 | |
|             nextCalls++;
 | |
|             return {value: i++, done: i > n};
 | |
|           }
 | |
|         };
 | |
|       };
 | |
|     },
 | |
|     configurable: true
 | |
|   });
 | |
| 
 | |
|   Promise.race(3).then(
 | |
|       function(x) {
 | |
|         assertAsync(x === 0, "race/iterable/number");
 | |
|       },
 | |
|       assertUnreachable);
 | |
|   delete numberPrototype[symbolIterator];
 | |
| 
 | |
|   assertEquals(getCalls, 1);
 | |
|   assertEquals(funcCalls, 1);
 | |
|   assertEquals(nextCalls, 3 + 1);  // + 1 for {done: true}
 | |
|   assertAsyncRan();
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var log
 | |
|   function MyPromise(resolver) {
 | |
|     log += "n"
 | |
|     var promise = new Promise(function(resolve, reject) {
 | |
|       resolver(
 | |
|         function(x) { log += "x" + x; resolve(x) },
 | |
|         function(r) { log += "r" + r; reject(r) }
 | |
|       )
 | |
|     })
 | |
|     promise.__proto__ = MyPromise.prototype
 | |
|     return promise
 | |
|   }
 | |
| 
 | |
|   MyPromise.__proto__ = Promise
 | |
|   MyPromise.defer = function() {
 | |
|     log += "d"
 | |
|     return call(this.__proto__.defer, this)
 | |
|   }
 | |
| 
 | |
|   MyPromise.prototype.__proto__ = Promise.prototype
 | |
|   MyPromise.prototype.then = function(resolve, reject) {
 | |
|     log += "c"
 | |
|     return call(this.__proto__.__proto__.then, this, resolve, reject)
 | |
|   }
 | |
| 
 | |
|   log = ""
 | |
|   var p1 = new MyPromise(function(resolve, reject) { resolve(1) })
 | |
|   var p2 = new MyPromise(function(resolve, reject) { reject(2) })
 | |
|   var d3 = defer(MyPromise)
 | |
|   assertTrue(d3.promise instanceof Promise, "subclass/instance")
 | |
|   assertTrue(d3.promise instanceof MyPromise, "subclass/instance-my3")
 | |
|   assertTrue(log === "nx1nr2n", "subclass/create")
 | |
| 
 | |
|   log = ""
 | |
|   var p4 = MyPromise.resolve(4)
 | |
|   var p5 = MyPromise.reject(5)
 | |
|   assertTrue(p4 instanceof MyPromise, "subclass/instance4")
 | |
|   assertTrue(p4 instanceof MyPromise, "subclass/instance-my4")
 | |
|   assertTrue(p5 instanceof MyPromise, "subclass/instance5")
 | |
|   assertTrue(p5 instanceof MyPromise, "subclass/instance-my5")
 | |
|   d3.resolve(3)
 | |
|   assertTrue(log === "nx4nr5x3", "subclass/resolve")
 | |
| 
 | |
|   log = ""
 | |
|   var d6 = defer(MyPromise)
 | |
|   d6.promise.then(function(x) {
 | |
|     return new Promise(function(resolve) { resolve(x) })
 | |
|   }).then(function() {})
 | |
|   d6.resolve(6)
 | |
|   assertTrue(log === "ncncnx6", "subclass/then")
 | |
| 
 | |
|   log = ""
 | |
|   Promise.all([11, Promise.resolve(12), 13, MyPromise.resolve(14), 15, 16])
 | |
| 
 | |
|   assertTrue(log === "nx14", "subclass/all/arg")
 | |
| 
 | |
|   log = ""
 | |
|   MyPromise.all([21, Promise.resolve(22), 23, MyPromise.resolve(24), 25, 26])
 | |
|   assertTrue(log === "nx24nnx21cnnx[object Promise]cnnx23cncnnx25cnnx26cn",
 | |
|              "subclass/all/self")
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   'use strict';
 | |
| 
 | |
|   class Pact extends Promise { }
 | |
|   class Vow  extends Pact    { }
 | |
|   class Oath extends Vow     { }
 | |
| 
 | |
|   Oath.constructor = Vow;
 | |
| 
 | |
|   assertTrue(Pact.resolve(Pact.resolve()).constructor === Pact,
 | |
|              "subclass/resolve/own");
 | |
| 
 | |
|   assertTrue(Pact.resolve(Promise.resolve()).constructor === Pact,
 | |
|              "subclass/resolve/ancestor");
 | |
| 
 | |
|   assertTrue(Pact.resolve(Vow.resolve()).constructor === Pact,
 | |
|              "subclass/resolve/descendant"); var vow = Vow.resolve();
 | |
| 
 | |
|   vow.constructor = Oath;
 | |
|   assertTrue(Oath.resolve(vow) === vow,
 | |
|              "subclass/resolve/descendant with transplanted own constructor");
 | |
| }());
 | |
| 
 | |
| (function() {
 | |
|   var thenCalled = false;
 | |
| 
 | |
|   var resolve;
 | |
|   var promise = new Promise(function(res) { resolve = res; });
 | |
|   resolve({ then() { thenCalled = true; throw new Error(); } });
 | |
|   assertLater(function() { return thenCalled; }, "resolve-with-thenable");
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var calledWith;
 | |
| 
 | |
|   var resolve;
 | |
|   var p1 = (new Promise(function(res) { resolve = res; }));
 | |
|   var p2 = p1.then(function(v) {
 | |
|     return {
 | |
|       then(resolve, reject) { resolve({ then() { calledWith = v }}); }
 | |
|     };
 | |
|   });
 | |
| 
 | |
|   resolve({ then(resolve) { resolve(2); } });
 | |
|   assertLater(function() { return calledWith === 2; },
 | |
|               "resolve-with-thenable2");
 | |
| })();
 | |
| 
 | |
| (function() {
 | |
|   var p = Promise.resolve();
 | |
|   var callCount = 0;
 | |
|   defineProperty(p, "constructor", {
 | |
|     get: function() { ++callCount; return Promise; }
 | |
|   });
 | |
|   p.then();
 | |
|   assertEquals(1, callCount);
 | |
| })();
 | |
| 
 | |
| assertAsyncDone()
 |