Promise.any: updates, corrections and new tests.

This commit is contained in:
Rick Waldron 2020-03-18 14:21:32 -04:00
parent e0f0c7860b
commit 6edaba378e
38 changed files with 959 additions and 182 deletions

View File

@ -12,6 +12,7 @@ info: |
...
7. Let promise be ? Construct(C, « executor »).
features: [Promise.any]
---*/

View File

@ -12,9 +12,34 @@ info: |
NewPromiseCapability ( C )
1. If IsConstructor(C) is false, throw a TypeError exception.
features: [Promise.any]
---*/
assert.throws(TypeError, function() {
Promise.any.call(eval);
});
assert.throws(TypeError, function() {
Promise.any.call(undefined, []);
});
assert.throws(TypeError, function() {
Promise.any.call(null, []);
});
assert.throws(TypeError, function() {
Promise.any.call(86, []);
});
assert.throws(TypeError, function() {
Promise.any.call('string', []);
});
assert.throws(TypeError, function() {
Promise.any.call(true, []);
});
assert.throws(TypeError, function() {
Promise.any.call(Symbol(), []);
});

View File

@ -1,36 +0,0 @@
// Copyright (C) 2019 Sergey Rubanov. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: >
Promise.any invoked on a non-object value
esid: sec-promise.any
info: |
1. Let C be the this value.
2. If Type(C) is not Object, throw a TypeError exception.
features: [Promise.any, Symbol]
---*/
assert.throws(TypeError, function() {
Promise.any.call(undefined, []);
});
assert.throws(TypeError, function() {
Promise.any.call(null, []);
});
assert.throws(TypeError, function() {
Promise.any.call(86, []);
});
assert.throws(TypeError, function() {
Promise.any.call('string', []);
});
assert.throws(TypeError, function() {
Promise.any.call(true, []);
});
assert.throws(TypeError, function() {
Promise.any.call(Symbol(), []);
});

View File

@ -0,0 +1,32 @@
// Copyright (C) 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any
description: >
Promise.any() does not retrieve `Symbol.species` property of the "`this` value"
info: |
1. Let C be the this value.
2. Let promiseCapability be ? NewPromiseCapability(C).
...
NewPromiseCapability ( C )
1. If IsConstructor(C) is false, throw a TypeError exception.
2. NOTE: C is assumed to be a constructor function that supports the parameter conventions of the Promise constructor (see 25.6.3.1).
...
flags: [async]
features: [Promise.any, Symbol.species]
---*/
class C extends Promise {
static get [Symbol.species]() {
throw new Test262Error('Getter for Symbol.species called');
}
static resolve() {
throw new Test262Error('C.resolve was reached');
}
}
Promise.any.call(C, [1]).then(() => $DONE(), $DONE);

View File

@ -0,0 +1,29 @@
// Copyright (C) 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any
description: >
Promise.any() does not retrieve `Symbol.species` property of the "`this` value".
info: |
1. Let C be the this value.
2. Let promiseCapability be ? NewPromiseCapability(C).
...
NewPromiseCapability ( C )
1. If IsConstructor(C) is false, throw a TypeError exception.
2. NOTE: C is assumed to be a constructor function that supports the parameter conventions of the Promise constructor (see 25.6.3.1).
...
flags: [async]
features: [Promise.any, Symbol.species]
---*/
Object.defineProperty(Promise, Symbol.species, {
get() {
throw new Test262Error('Getter for Symbol.species called');
}
});
Promise.any([1]).then(() => $DONE(), $DONE);

View File

@ -1,20 +0,0 @@
// Copyright (C) 2019 Sergey Rubanov. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: Promise.any([]) rejects immediately
esid: sec-promise.any
flags: [async]
includes: [promiseHelper.js]
features: [AggregateError, Promise.any]
---*/
Promise.any([])
.then(
() => $DONE('The promise should be rejected, but was resolved'),
error => {
assert(error instanceof AggregateError);
assert.sameValue(error.errors.length, 0);
$DONE()
}
);

View File

@ -0,0 +1,28 @@
// Copyright (C) 2019 Sergey Rubanov. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: >
If the constructor's `resolve` method is not callable, reject with a TypeError.
esid: sec-promise.any
info: |
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
Runtime Semantics: PerformPromiseAny
6. Let promiseResolve be ? Get(constructor, "resolve").
7. If ! IsCallable(promiseResolve) is false, throw a TypeError exception.
flags: [async]
features: [Promise.any]
---*/
Promise.resolve = null;
Promise.any([1])
.then(
() => $DONE('The promise should not be resolved.'),
error => {
assert(error instanceof TypeError);
}
).then($DONE, $DONE);

View File

@ -0,0 +1,50 @@
// Copyright (C) 2019 Sergey Rubanov. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: >
Invocation of the constructor's `resolve` method for iterable with promise values
esid: sec-promise.any
info: |
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
Runtime Semantics: PerformPromiseAny
8. Repeat
...
i. Let nextPromise be ? Call(promiseResolve, constructor, « nextValue »).
flags: [async]
features: [Promise.any]
---*/
class Custom extends Promise {}
let customs = [
new Custom(resolve => resolve()),
new Custom(resolve => resolve()),
new Custom(resolve => resolve()),
];
let cresolveCallCount = 0;
let presolveCallCount = 0;
let boundCustomResolve = Custom.resolve.bind(Custom);
let boundPromiseResolve = Promise.resolve.bind(Promise);
Custom.resolve = function(...args) {
cresolveCallCount += 1;
return boundCustomResolve(...args);
};
Promise.resolve = function(...args) {
presolveCallCount += 1;
return boundPromiseResolve(...args);
};
Promise.any.call(Custom, customs)
.then(() => {
assert.sameValue(presolveCallCount, 0, '`Promise.resolve` is never invoked');
assert.sameValue(cresolveCallCount, 3, '`Custom.resolve` invoked once for every iterated promise');
}, (error) => {
$DONE(error);
}
).then($DONE, $DONE);

View File

@ -0,0 +1,41 @@
// Copyright (C) 2019 Sergey Rubanov. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: >
Invocation of the constructor's `resolve` method for iterable with promise values
esid: sec-promise.any
info: |
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
Runtime Semantics: PerformPromiseAny
8. Repeat
...
i. Let nextPromise be ? Call(promiseResolve, constructor, « nextValue »).
flags: [async]
features: [Promise.any]
---*/
let promises = [
new Promise(resolve => resolve()),
new Promise(resolve => resolve()),
new Promise(resolve => resolve()),
];
let callCount = 0;
let boundPromiseResolve = Promise.resolve.bind(Promise);
Promise.resolve = function(...args) {
callCount += 1;
return boundPromiseResolve(...args);
};
Promise.any(promises)
.then(() => {
assert.sameValue(callCount, 3, '`then` invoked once for every iterated promise');
}, (error) => {
$DONE(error);
}
).then($DONE, $DONE);

View File

@ -0,0 +1,46 @@
// Copyright (C) 2019 Sergey Rubanov. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: >
Invocation of the constructor's `resolve` method for iterable with non-promise values
esid: sec-promise.any
info: |
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
Runtime Semantics: PerformPromiseAny
8. Repeat
...
i. Let nextPromise be ? Call(promiseResolve, constructor, « nextValue »).
flags: [async]
features: [Promise.any]
---*/
class Custom extends Promise {}
let values = [1, 2, 3];
let cresolveCallCount = 0;
let presolveCallCount = 0;
let boundCustomResolve = Custom.resolve.bind(Custom);
let boundPromiseResolve = Promise.resolve.bind(Promise);
Custom.resolve = function(...args) {
cresolveCallCount += 1;
return boundCustomResolve(...args);
};
Promise.resolve = function(...args) {
presolveCallCount += 1;
return boundPromiseResolve(...args);
};
Promise.any.call(Custom, values)
.then(() => {
assert.sameValue(presolveCallCount, 0, '`Promise.resolve` is never invoked');
assert.sameValue(cresolveCallCount, 3, '`Custom.resolve` invoked once for every iterated promise');
}, (error) => {
$DONE(error);
}
).then($DONE, $DONE);

View File

@ -0,0 +1,37 @@
// Copyright (C) 2019 Sergey Rubanov. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: >
Invocation of the constructor's `resolve` method for iterable with non-promise values
esid: sec-promise.any
info: |
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
Runtime Semantics: PerformPromiseAny
8. Repeat
...
i. Let nextPromise be ? Call(promiseResolve, constructor, « nextValue »).
flags: [async]
features: [Promise.any]
---*/
let values = [1, 2, 3];
let callCount = 0;
let boundPromiseResolve = Promise.resolve.bind(Promise);
Promise.resolve = function(...args) {
callCount += 1;
return boundPromiseResolve(...args);
};
Promise.any(values)
.then(() => {
assert.sameValue(callCount, 3, '`Promise.resolve` invoked once for every iterated value');
}, (error) => {
$DONE(error);
}
).then($DONE, $DONE);

View File

@ -10,46 +10,22 @@ info: |
Runtime Semantics: PerformPromiseAny
6. Repeat
8. Repeat
...
i. Let nextPromise be ? Call(promiseResolve, constructor, « nextValue »).
...
z. Perform ? Invoke(nextPromise, "then", « resolveElement, rejectElement »).
r. Perform ? Invoke(nextPromise, "then", « resultCapability.[[Resolve]], rejectElement »).
flags: [async]
features: [Promise.any]
---*/
var p1 = new Promise(function() {});
var p2 = new Promise(function() {});
var p3 = new Promise(function() {});
var resolve = Promise.resolve;
var callCount = 0;
var current = p1;
var next = p2;
var afterNext = p3;
let boundPromiseResolve = Promise.resolve.bind(Promise);
Promise.resolve = function(nextValue) {
assert.sameValue(
nextValue, current, '`resolve` invoked with next iterated value'
);
assert.sameValue(
arguments.length, 1, '`resolve` invoked with a single argument'
);
Promise.resolve = function(...args) {
assert.sameValue(args.length, 1, '`resolve` invoked with a single argument');
assert.sameValue(this, Promise, '`this` value is the constructor');
current = next;
next = afterNext;
afterNext = null;
callCount += 1;
return resolve.apply(Promise, arguments);
return boundPromiseResolve(...args);
};
Promise.any([p1, p2, p3])
.then(function() {
assert.sameValue(
callCount, 3, '`resolve` invoked once for each iterated value'
);
$DONE();
}, $DONE);
Promise.any([1]).then(() => $DONE(), $DONE);

View File

@ -0,0 +1,45 @@
// Copyright (C) 2019 Sergey Rubanov. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: >
Invocation of the instance's `then` method
esid: sec-promise.any
info: |
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
6. If result is an abrupt completion, then
a. If iteratorRecord.[[Done]] is false, set result to IteratorClose(iteratorRecord, result).
b. IfAbruptRejectPromise(result, promiseCapability).
Runtime Semantics: PerformPromiseAny
r. Perform ? Invoke(nextPromise, "then", « resultCapability.[[Resolve]], rejectElement »).
flags: [async]
features: [Promise.any]
---*/
let promises = [
new Promise(resolve => resolve()),
new Promise(resolve => resolve()),
new Promise(resolve => resolve()),
];
let callCount = 0;
promises.forEach(promise => {
let boundThen = promise.then.bind(promise);
promise.then = function(...args) {
assert.sameValue(this, promises[callCount]);
callCount += 1;
return boundThen(...args);
};
});
Promise.any(promises)
.then(() => {
assert.sameValue(callCount, 3, '`then` invoked once for every iterated value');
}, (error) => {
$DONE(error);
// $DONE('The promise should not be rejected');
}
).then($DONE, $DONE);

View File

@ -19,39 +19,16 @@ flags: [async]
features: [Promise.any]
---*/
var p1 = new Promise(function() {});
var p2 = new Promise(function() {});
var p3 = new Promise(function() {});
var callCount = 0;
var currentThis = p1;
var nextThis = p2;
var afterNextThis = p3;
let promise = new Promise(() => {});
let boundThen = promise.then.bind(promise);
p1.then = p2.then = p3.then = function(a, b) {
assert.sameValue(typeof a, 'function', 'type of first argument');
assert.sameValue(
a.length,
1,
'The length property of a promise resolve function is 1.'
);
assert.sameValue(typeof b, 'function', 'type of second argument');
assert.sameValue(
b.length,
1,
'The length property of a promise reject function is 1.'
);
assert.sameValue(arguments.length, 2, '`then` invoked with two arguments');
assert.sameValue(this, currentThis, '`this` value');
currentThis = nextThis;
nextThis = afterNextThis;
afterNextThis = null;
callCount += 1;
promise.then = function(resolver, rejectElement) {
assert.sameValue(this, promise);
assert.sameValue(typeof resolver, 'function');
assert.sameValue(resolver.length, 1, 'resolver.length is 1');
assert.sameValue(typeof rejectElement, 'function');
assert.sameValue(rejectElement.length, 1, 'rejectElement.length is 0');
return boundThen(resolver, rejectElement);
};
Promise.any([p1, p2, p3])
.then(function() {
assert.sameValue(callCount, 3, '`then` invoked once for every iterated value');
$DONE();
}, $DONE);
Promise.any([promise]).then(() => $DONE(), $DONE);

View File

@ -0,0 +1,39 @@
// Copyright (C) 2019 Sergey Rubanov. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any
description: >
Promise.any([]) rejects with AggregateError, empty errors array.
info: |
Runtime Semantics: PerformPromiseAny ( iteratorRecord, constructor, resultCapability )
...
3. Let errors be a new empty List.
...
8. Repeat,
a. Let next be IteratorStep(iteratorRecord).
b. If next is an abrupt completion, set iteratorRecord.[[Done]] to true.
c. ReturnIfAbrupt(next).
d. If next is false, then
i. Set iteratorRecord.[[Done]] to true.
ii. Set remainingElementsCount.[[Value]] to remainingElementsCount.[[Value]] - 1.
iii. If remainingElementsCount.[[Value]] is 0, then
1. Let error be a newly created AggregateError object.
2. Set error.[[AggregateErrors]] to errors.
3. Return ThrowCompletion(error).
...
flags: [async]
features: [AggregateError, Promise.any]
---*/
Promise.any([])
.then(
() => $DONE('The promise should be rejected, but was resolved'),
error => {
assert.sameValue(Object.getPrototypeOf(error), AggregateError.prototype);
assert(error instanceof AggregateError);
assert.sameValue(error.errors.length, 0);
}
).then($DONE, $DONE);

View File

@ -0,0 +1,39 @@
// Copyright (C) 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any
description: >
Promise.any('') rejects with AggregateError, empty errors array.
info: |
Runtime Semantics: PerformPromiseAny ( iteratorRecord, constructor, resultCapability )
...
3. Let errors be a new empty List.
...
8. Repeat,
a. Let next be IteratorStep(iteratorRecord).
b. If next is an abrupt completion, set iteratorRecord.[[Done]] to true.
c. ReturnIfAbrupt(next).
d. If next is false, then
i. Set iteratorRecord.[[Done]] to true.
ii. Set remainingElementsCount.[[Value]] to remainingElementsCount.[[Value]] - 1.
iii. If remainingElementsCount.[[Value]] is 0, then
1. Let error be a newly created AggregateError object.
2. Set error.[[AggregateErrors]] to errors.
3. Return ThrowCompletion(error).
...
features: [AggregateError, Promise.any]
flags: [async]
---*/
Promise.any('')
.then(
() => $DONE('The promise should be rejected, but was resolved'),
error => {
assert.sameValue(Object.getPrototypeOf(error), AggregateError.prototype);
assert(error instanceof AggregateError);
assert.sameValue(error.errors.length, 0);
}
).then($DONE, $DONE);

View File

@ -4,13 +4,13 @@
/*---
esid: sec-promise.any
description: >
Reject when argument is `false`
Promise.any(false) rejects with TypeError.
info: |
Promise.any ( iterable )
...
4. Let iteratorRecord be GetIterator(iterable).
5. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
3. Let iteratorRecord be GetIterator(iterable).
4. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
...
#sec-getiterator

View File

@ -4,13 +4,13 @@
/*---
esid: sec-promise.any
description: >
Reject when argument is `null`
Promise.any(null) rejects with TypeError.
info: |
Promise.any ( iterable )
...
4. Let iteratorRecord be GetIterator(iterable).
5. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
3. Let iteratorRecord be GetIterator(iterable).
4. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
...
#sec-getiterator

View File

@ -4,13 +4,13 @@
/*---
esid: sec-promise.any
description: >
Reject when argument is a number
Promise.any(number) rejects with TypeError.
info: |
Promise.any ( iterable )
...
4. Let iteratorRecord be GetIterator(iterable).
5. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
3. Let iteratorRecord be GetIterator(iterable).
4. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
...
#sec-getiterator

View File

@ -4,13 +4,13 @@
/*---
esid: sec-promise.any
description: >
Reject with abrupt completion from GetIterator
Promise.any(poisoned iterable) rejects with whatever error is thrown.
info: |
Promise.any ( iterable )
...
4. Let iteratorRecord be GetIterator(iterable).
5. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
3. Let iteratorRecord be GetIterator(iterable).
4. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
...
#sec-getiterator
@ -24,18 +24,18 @@ flags: [async]
---*/
var poison = [];
var error = new Test262Error();
Object.defineProperty(poison, Symbol.iterator, {
get() {
throw error;
throw new Test262Error();
}
});
try {
Promise.any(poison).then(function() {
Promise.any(poison).then(() => {
$DONE('The promise should be rejected, but was resolved');
}, function(err) {
assert.sameValue(err, error);
}, (error) => {
assert.sameValue(Object.getPrototypeOf(error), Test262Error.prototype);
assert(error instanceof Test262Error);
}).then($DONE, $DONE);
} catch (error) {
$DONE(`The promise should be rejected, but threw an exception: ${error.message}`);

View File

@ -4,13 +4,13 @@
/*---
esid: sec-promise.any
description: >
Resolve when argument is a string
Promise.any('non-empty-string') resolves with the first character in the non-empty string
info: |
Promise.any ( iterable )
...
4. Let iteratorRecord be GetIterator(iterable).
5. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
3. Let iteratorRecord be GetIterator(iterable).
4. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
...
#sec-getiterator
@ -25,9 +25,10 @@ flags: [async]
---*/
try {
Promise.any('').then(function(v) {
assert.sameValue(v.length, 0);
}, function(error) {
Promise.any('xyz').then(v => {
assert.sameValue(v, 'x');
assert.sameValue(v.length, 1);
}, error => {
$DONE(`The promise should be resolved, but was rejected with error: ${error.message}`);
}).then($DONE, $DONE);
} catch (error) {

View File

@ -4,13 +4,13 @@
/*---
esid: sec-promise.any
description: >
Reject when argument is a symbol
Promise.any(Symbol()) rejects with TypeError.
info: |
Promise.any ( iterable )
...
4. Let iteratorRecord be GetIterator(iterable).
5. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
3. Let iteratorRecord be GetIterator(iterable).
4. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
...
GetIterator ( obj [ , hint [ , method ] ] )

View File

@ -4,13 +4,13 @@
/*---
esid: sec-promise.any
description: >
Reject when argument is `true`
Promise.any(true) rejects with TypeError.
info: |
Promise.any ( iterable )
...
4. Let iteratorRecord be GetIterator(iterable).
5. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
3. Let iteratorRecord be GetIterator(iterable).
4. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
...
GetIterator ( obj [ , hint [ , method ] ] )

View File

@ -4,13 +4,13 @@
/*---
esid: sec-promise.any
description: >
Reject when argument is `undefined`
Promise.any(undefined) rejects with TypeError.
info: |
Promise.any ( iterable )
...
4. Let iteratorRecord be GetIterator(iterable).
5. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
3. Let iteratorRecord be GetIterator(iterable).
4. IfAbruptRejectPromise(iteratorRecord, promiseCapability).
...
GetIterator ( obj [ , hint [ , method ] ] )

View File

@ -0,0 +1,60 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any
description: >
Error when advancing the provided iterable (not closing iterator)
info: |
Promise.any ( iterable )
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
6. If result is an abrupt completion, then
a. If iteratorRecord.[[Done]] is false, set result to IteratorClose(iteratorRecord, result).
b. IfAbruptRejectPromise(result, promiseCapability).
Runtime Semantics: PerformPromiseAny
8. Repeat
a. Let next be IteratorStep(iteratorRecord).
b. If next is an abrupt completion, set iteratorRecord.[[done]] to true.
c. ReturnIfAbrupt(next).
flags: [async]
features: [Promise.any, Symbol.iterator]
---*/
let returnCount = 0;
let poisonedDone = {};
let error = new Test262Error();
Object.defineProperties(poisonedDone, {
done: {
get() {
throw error;
}
},
value: {
get() {}
}
});
let iterStepThrows = {
[Symbol.iterator]() {
return {
next() {
return poisonedDone;
},
return() {
returnCount += 1;
return {};
}
};
}
};
Promise.any(iterStepThrows).then(
() => {
$DONE('The promise should be rejected.');
}, (reason) => {
assert.sameValue(reason, error);
assert.sameValue(returnCount, 0);
}).then($DONE, $DONE);

View File

@ -0,0 +1,56 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any
description: >
Error when advancing the provided iterable (rejecting promise)
info: |
Promise.any ( iterable )
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
6. If result is an abrupt completion, then
a. If iteratorRecord.[[Done]] is false, set result to IteratorClose(iteratorRecord, result).
b. IfAbruptRejectPromise(result, promiseCapability).
Runtime Semantics: PerformPromiseAny
8. Repeat
a. Let next be IteratorStep(iteratorRecord).
b. If next is an abrupt completion, set iteratorRecord.[[done]] to true.
c. ReturnIfAbrupt(next).
flags: [async]
features: [Promise.any, Symbol.iterator]
---*/
let poisonedDone = {};
let error = new Test262Error();
Object.defineProperties(poisonedDone, {
done: {
get() {
throw error;
}
},
value: {
get() {
$DONE('The `value` property should not be accessed.');
}
}
});
let iterStepThrows = {
[Symbol.iterator]() {
return {
next() {
return poisonedDone;
}
};
}
};
Promise.any(iterStepThrows).then(
() => {
$DONE('The promise should be rejected.');
}, (reason) => {
assert.sameValue(reason, error);
}).then($DONE, $DONE);

View File

@ -0,0 +1,51 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-performpromiseany
description: >
Each Promise.any element is called with a new Promise.any Reject Element function.
info: |
Runtime Semantics: PerformPromiseAny ( iteratorRecord, constructor, resultCapability )
...
k. Let rejectElement be ! CreateBuiltinFunction(steps, « [[AlreadyCalled]], [[Index]], [[Errors]], [[Capability]], [[RemainingElements]] »).
...
r. Perform ? Invoke(nextPromise, "then", « resultCapability.[[Resolve]], rejectElement »).
...
features: [Promise.any]
---*/
function rejectFunction() {}
function Constructor(executor) {
executor(rejectFunction, $ERROR);
}
Constructor.resolve = function(v) {
return v;
};
var callCount1 = 0;
var callCount2 = 0;
var p1OnRejected;
var p1 = {
then(_, onRejected) {
callCount1 += 1;
p1OnRejected = onRejected;
assert.notSameValue(onRejected, rejectFunction, 'p1.then');
}
};
var p2 = {
then(_, onRejected) {
callCount2 += 1;
assert.notSameValue(onRejected, rejectFunction, 'p2.then');
assert.notSameValue(onRejected, p1OnRejected, 'p1.onRejected != p2.onRejected');
}
};
Promise.any.call(Constructor, [p1, p2]);
assert.sameValue(callCount1, 1, 'p1.then call count');
assert.sameValue(callCount2, 1, 'p2.then call count');

View File

@ -0,0 +1,26 @@
// Copyright (C) 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any
description: >
Promise.any rejection reasons from various rejections are all present
flags: [async]
features: [Promise.any]
---*/
let rejections = [
Promise.reject('a'),
new Promise((_, reject) => reject('b')),
Promise.all([Promise.reject('c')]),
Promise.resolve(Promise.reject('d')),
];
Promise.any(rejections)
.then(
() => $DONE('The promise should be rejected, but was resolved'),
error => {
assert.sameValue(error.errors.length, rejections.length);
assert.sameValue(error.errors.join(''), 'abcd');
}
).then($DONE, $DONE);

View File

@ -0,0 +1,31 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any
description: Rejecting through deferred invocation of the provided resolving function
info: |
...
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
...
flags: [async]
features: [AggregateError, Promise.any]
---*/
var rejection = {};
var thenable = {
then(_, reject) {
new Promise((resolve) => resolve())
.then(() => reject(rejection));
}
};
Promise.any([thenable])
.then(() => {
$DONE('The promise should be rejected.');
}, (aggregate) => {
assert(aggregate instanceof AggregateError);
assert.sameValue(aggregate.errors.length, 1);
assert.sameValue(aggregate.errors[0], rejection);
}).then($DONE, $DONE);

View File

@ -0,0 +1,29 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any-reject-element-functions
description: The [[Extensible]] slot of Promise.any Reject Element functions
info: |
17 ECMAScript Standard Built-in Objects:
Unless specified otherwise, the [[Extensible]] internal slot
of a built-in object initially has the value true.
features: [Promise.any]
---*/
var rejectElementFunction;
var thenable = {
then(_, reject) {
rejectElementFunction = reject;
}
};
function NotPromise(executor) {
executor(function() {}, function() {});
}
NotPromise.resolve = function(v) {
return v;
};
Promise.any.call(NotPromise, [thenable]);
assert(Object.isExtensible(rejectElementFunction));

View File

@ -0,0 +1,40 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any-reject-element-functions
description: The `length` property of Promise.any Reject Element functions
info: |
The length property of a Promise.any Reject Element function is 1.
17 ECMAScript Standard Built-in Objects:
Unless otherwise specified, the length property of a built-in Function
object has the attributes { [[Writable]]: false, [[Enumerable]]: false,
[[Configurable]]: true }.
includes: [propertyHelper.js]
features: [Promise.any]
---*/
var rejectElementFunction;
var thenable = {
then(_, reject) {
rejectElementFunction = reject;
}
};
function NotPromise(executor) {
executor(function() {}, function() {});
}
NotPromise.resolve = function(v) {
return v;
};
Promise.any.call(NotPromise, [thenable]);
assert.sameValue(rejectElementFunction.length, 1);
verifyProperty(rejectElementFunction, 'length', {
value: 1,
enumerable: false,
writable: false,
configurable: true,
});

View File

@ -0,0 +1,39 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any-reject-element-functions
description: The `name` property of Promise.any Reject Element functions
info: |
A promise resolve function is an anonymous built-in function.
17 ECMAScript Standard Built-in Objects:
Every built-in function object, including constructors, has a `name`
property whose value is a String. Functions that are identified as
anonymous functions use the empty string as the value of the `name`
property.
Unless otherwise specified, the `name` property of a built-in function
object has the attributes { [[Writable]]: *false*, [[Enumerable]]: *false*,
[[Configurable]]: *true* }.
includes: [propertyHelper.js]
features: [Promise.any]
---*/
var rejectElementFunction;
var thenable = {
then(_, reject) {
rejectElementFunction = reject;
}
};
function NotPromise(executor) {
executor(function() {}, function() {});
}
NotPromise.resolve = function(v) {
return v;
};
Promise.any.call(NotPromise, [thenable]);
verifyProperty(rejectElementFunction, "name", {
value: "", writable: false, enumerable: false, configurable: true
});

View File

@ -0,0 +1,33 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any-reject-element-functions
description: Promise.any Reject Element functions are not constructors
info: |
17 ECMAScript Standard Built-in Objects:
Built-in function objects that are not identified as constructors do not
implement the [[Construct]] internal method unless otherwise specified
in the description of a particular function.
features: [Promise.any]
---*/
var rejectElementFunction;
var thenable = {
then(_, reject) {
rejectElementFunction = reject;
}
};
function NotPromise(executor) {
executor(function() {}, function() {});
}
NotPromise.resolve = function(v) {
return v;
};
Promise.any.call(NotPromise, [thenable]);
assert.sameValue(Object.prototype.hasOwnProperty.call(rejectElementFunction, 'prototype'), false);
assert.throws(TypeError, function() {
new rejectElementFunction();
});

View File

@ -0,0 +1,31 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any-reject-element-functions
description: The [[Prototype]] of Promise.any Reject Element functions
info: |
17 ECMAScript Standard Built-in Objects:
Unless otherwise specified every built-in function and every built-in
constructor has the Function prototype object, which is the initial
value of the expression Function.prototype (19.2.3), as the value of
its [[Prototype]] internal slot.
features: [Promise.any]
---*/
var rejectElementFunction;
var thenable = {
then(_, reject) {
rejectElementFunction = reject;
}
};
function NotPromise(executor) {
executor(function() {}, function() {});
}
NotPromise.resolve = function(v) {
return v;
};
Promise.any.call(NotPromise, [thenable]);
assert.sameValue(Object.getPrototypeOf(rejectElementFunction), Function.prototype);

View File

@ -0,0 +1,41 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: >
Resolved promises ignore rejections through deferred invocation of the
provided resolving function
esid: sec-promise.any
info: |
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
Runtime Semantics: PerformPromiseAny
8. Repeat
...
r. Perform ? Invoke(nextPromise, "then", « resultCapability.[[Resolve]], rejectElement »).
flags: [async]
features: [Promise.any]
---*/
var resolver = {
then(resolve) {
new Promise((resolve) => resolve())
.then(() => resolve(42));
}
};
var lateRejector = {
then(resolve, reject) {
new Promise((resolve) => resolve())
.then(() => {
resolve(9);
reject();
});
}
};
Promise.any([resolver, lateRejector])
.then(resolution => {
assert.sameValue(resolution, 42);
}).then($DONE, $DONE);

View File

@ -0,0 +1,37 @@
// Copyright (C) 2019 Leo Balter, 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: >
Resolved promises ignore rejections through immediate invocation of the
provided resolving function
esid: sec-promise.any
info: |
5. Let result be PerformPromiseAny(iteratorRecord, C, promiseCapability).
Runtime Semantics: PerformPromiseAny
8. Repeat
...
r. Perform ? Invoke(nextPromise, "then", « resultCapability.[[Resolve]], rejectElement »).
flags: [async]
features: [Promise.any]
---*/
var resolver = {
then(resolve) {
resolve(42);
}
};
var lateRejector = {
then(resolve, reject) {
resolve(33);
reject();
}
};
Promise.any([resolver, lateRejector])
.then(resolution => {
assert.sameValue(resolution, 42);
}).then($DONE, $DONE);

View File

@ -0,0 +1,22 @@
// Copyright (C) 2020 Rick Waldron. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-promise.any
description: >
Promise.any resolves with the first item that does not reject.
flags: [async]
features: [Promise.any]
---*/
let fulfillables = [
Promise.reject('a'),
new Promise((resolve, reject) => reject('b')),
Promise.all([Promise.reject('c')]),
Promise.resolve(Promise.reject('d').catch(v => v)),
];
Promise.any(fulfillables)
.then((resolution) => {
assert.sameValue(resolution, 'd');
}).then($DONE, $DONE);

View File

@ -1,29 +0,0 @@
// Copyright (C) 2019 Sergey Rubanov. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
description: >
Promise.any() does not retrieve `Symbol.species` property of the `this` value
esid: sec-promise.any
info: |
1. Let C be the this value.
2. If Type(C) is not Object, throw a TypeError exception.
3. Let promiseCapability be ? NewPromiseCapability(C).
...
features: [Promise.any, Symbol.species]
---*/
function C(executor) {
executor(function() {}, function() {});
}
Object.defineProperty(C, Symbol.species, {
get() {
throw new Test262Error('Getter for Symbol.species called');
}
});
C.resolve = function() {
throw new Test262Error();
};
Promise.any.call(C, []);