mirror of https://github.com/tc39/test262.git
Add test on handling broken promises in AsyncGenerator.prototype.return (#3472)
* Add test on handling broken promises in AsyncGenerator.prototype.return * fixup! * fixup!
This commit is contained in:
parent
2e7cdfbe18
commit
509363bcfd
|
@ -0,0 +1,62 @@
|
|||
// Copyright (C) 2022 Chengzhong Wu. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
esid: sec-asyncgenerator-prototype-return
|
||||
description: >
|
||||
A broken promise should reject the returned promise of
|
||||
AsyncGenerator.prototype.return when the generator is completed.
|
||||
info: |
|
||||
AsyncGenerator.prototype.return ( value )
|
||||
...
|
||||
8. If state is either suspendedStart or completed, then
|
||||
a. Set generator.[[AsyncGeneratorState]] to awaiting-return.
|
||||
b. Perform ! AsyncGeneratorAwaitReturn(generator).
|
||||
...
|
||||
|
||||
AsyncGeneratorAwaitReturn ( generator )
|
||||
...
|
||||
6. Let promise be Completion(PromiseResolve(%Promise%, completion.[[Value]])).
|
||||
7. If promiseCompletion is an abrupt completion, then
|
||||
a. Set generator.[[AsyncGeneratorState]] to completed.
|
||||
b. Perform AsyncGeneratorCompleteStep(generator, promiseCompletion, true).
|
||||
c. Perform AsyncGeneratorDrainQueue(generator).
|
||||
d. Return unused.
|
||||
8. Assert: promiseCompletion.[[Type]] is normal.
|
||||
9. Let promise be promiseCompletion.[[Value]].
|
||||
...
|
||||
|
||||
flags: [async]
|
||||
features: [async-iteration]
|
||||
---*/
|
||||
|
||||
let unblock;
|
||||
let blocking = new Promise(resolve => { unblock = resolve; });
|
||||
let unblocked = false;
|
||||
var g = async function*() {
|
||||
await blocking;
|
||||
unblocked = true;
|
||||
};
|
||||
|
||||
var it = g();
|
||||
let brokenPromise = Promise.resolve(42);
|
||||
Object.defineProperty(brokenPromise, 'constructor', {
|
||||
get: function () {
|
||||
throw new Error('broken promise');
|
||||
}
|
||||
});
|
||||
|
||||
it.next();
|
||||
it.return(brokenPromise)
|
||||
.then(
|
||||
() => {
|
||||
throw new Test262Error("Expected rejection");
|
||||
},
|
||||
err => {
|
||||
assert(unblocked, false, 'return should be rejected before generator is resumed');
|
||||
assert.sameValue(err.message, 'broken promise');
|
||||
}
|
||||
)
|
||||
.then($DONE, $DONE);
|
||||
|
||||
unblock();
|
|
@ -0,0 +1,54 @@
|
|||
// Copyright (C) 2022 Chengzhong Wu. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
esid: sec-asyncgenerator-prototype-return
|
||||
description: >
|
||||
A broken promise should reject the returned promise of
|
||||
AsyncGenerator.prototype.return when the generator's state is suspendedStart.
|
||||
info: |
|
||||
AsyncGenerator.prototype.return ( value )
|
||||
...
|
||||
8. If state is either suspendedStart or completed, then
|
||||
a. Set generator.[[AsyncGeneratorState]] to awaiting-return.
|
||||
b. Perform AsyncGeneratorAwaitReturn(_generator_).
|
||||
...
|
||||
|
||||
AsyncGeneratorAwaitReturn ( generator )
|
||||
...
|
||||
6. Let promise be Completion(PromiseResolve(%Promise%, completion.[[Value]])).
|
||||
7. If promiseCompletion is an abrupt completion, then
|
||||
a. Set generator.[[AsyncGeneratorState]] to completed.
|
||||
b. Perform AsyncGeneratorCompleteStep(generator, promiseCompletion, true).
|
||||
c. Perform AsyncGeneratorDrainQueue(generator).
|
||||
d. Return unused.
|
||||
8. Assert: promiseCompletion.[[Type]] is normal.
|
||||
9. Let promise be promiseCompletion.[[Value]].
|
||||
...
|
||||
|
||||
flags: [async]
|
||||
features: [async-iteration]
|
||||
---*/
|
||||
|
||||
var g = async function*() {
|
||||
throw new Test262Error('Generator must not be resumed.');
|
||||
};
|
||||
|
||||
var it = g();
|
||||
let brokenPromise = Promise.resolve(42);
|
||||
Object.defineProperty(brokenPromise, 'constructor', {
|
||||
get: function () {
|
||||
throw new Error('broken promise');
|
||||
}
|
||||
});
|
||||
|
||||
it.return(brokenPromise)
|
||||
.then(
|
||||
() => {
|
||||
throw new Test262Error("Expected rejection");
|
||||
},
|
||||
err => {
|
||||
assert.sameValue(err.message, 'broken promise');
|
||||
}
|
||||
)
|
||||
.then($DONE, $DONE);
|
|
@ -0,0 +1,64 @@
|
|||
// Copyright (C) 2022 Chengzhong Wu. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
esid: sec-asyncgenerator-prototype-return
|
||||
description: >
|
||||
A broken promise should resume the generator and reject with
|
||||
the exception when the generator's state is suspendedYield.
|
||||
info: |
|
||||
AsyncGenerator.prototype.return ( value )
|
||||
...
|
||||
9. Else if state is suspendedYield, then
|
||||
a. Perform AsyncGeneratorResume(generator, completion).
|
||||
...
|
||||
|
||||
AsyncGeneratorCompleteStep ( generator, completion, done [ , realm ] )
|
||||
Resumes the steps defined at AsyncGeneratorStart ( generator, generatorBody )
|
||||
...
|
||||
4. Set the code evaluation state of genContext such that when evaluation is resumed for that execution context the following steps will be performed:
|
||||
...
|
||||
i. Perform AsyncGeneratorDrainQueue(generator).
|
||||
j. Return undefined.
|
||||
|
||||
AsyncGeneratorDrainQueue ( generator )
|
||||
...
|
||||
5. Repeat, while done is false,
|
||||
a. Let next be the first element of queue.
|
||||
b. Let completion be Completion(next.[[Completion]]).
|
||||
c. If completion.[[Type]] is return, then
|
||||
i. Set generator.[[AsyncGeneratorState]] to awaiting-return.
|
||||
ii. Perform AsyncGeneratorAwaitReturn(generator).
|
||||
iii. Set done to true.
|
||||
...
|
||||
|
||||
flags: [async]
|
||||
features: [async-iteration]
|
||||
---*/
|
||||
|
||||
let caughtErr;
|
||||
var g = async function*() {
|
||||
try {
|
||||
yield;
|
||||
return 'this is never returned';
|
||||
} catch (err) {
|
||||
caughtErr = err;
|
||||
return 1;
|
||||
}
|
||||
};
|
||||
|
||||
let brokenPromise = Promise.resolve(42);
|
||||
Object.defineProperty(brokenPromise, 'constructor', {
|
||||
get: function () {
|
||||
throw new Error('broken promise');
|
||||
}
|
||||
});
|
||||
|
||||
var it = g();
|
||||
it.next().then(() => {
|
||||
return it.return(brokenPromise);
|
||||
}).then(ret => {
|
||||
assert.sameValue(caughtErr.message, 'broken promise');
|
||||
assert.sameValue(ret.value, 1, 'returned value');
|
||||
assert.sameValue(ret.done, true, 'iterator is closed');
|
||||
}).then($DONE, $DONE);
|
Loading…
Reference in New Issue