Skip to content

Commit 509363b

Browse files
authored
Add test on handling broken promises in AsyncGenerator.prototype.return (#3472)
* Add test on handling broken promises in AsyncGenerator.prototype.return * fixup! * fixup!
1 parent 2e7cdfb commit 509363b

3 files changed

Lines changed: 180 additions & 0 deletions

File tree

Lines changed: 62 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
// Copyright (C) 2022 Chengzhong Wu. All rights reserved.
2+
// This code is governed by the BSD license found in the LICENSE file.
3+
4+
/*---
5+
esid: sec-asyncgenerator-prototype-return
6+
description: >
7+
A broken promise should reject the returned promise of
8+
AsyncGenerator.prototype.return when the generator is completed.
9+
info: |
10+
AsyncGenerator.prototype.return ( value )
11+
...
12+
8. If state is either suspendedStart or completed, then
13+
a. Set generator.[[AsyncGeneratorState]] to awaiting-return.
14+
b. Perform ! AsyncGeneratorAwaitReturn(generator).
15+
...
16+
17+
AsyncGeneratorAwaitReturn ( generator )
18+
...
19+
6. Let promise be Completion(PromiseResolve(%Promise%, completion.[[Value]])).
20+
7. If promiseCompletion is an abrupt completion, then
21+
a. Set generator.[[AsyncGeneratorState]] to completed.
22+
b. Perform AsyncGeneratorCompleteStep(generator, promiseCompletion, true).
23+
c. Perform AsyncGeneratorDrainQueue(generator).
24+
d. Return unused.
25+
8. Assert: promiseCompletion.[[Type]] is normal.
26+
9. Let promise be promiseCompletion.[[Value]].
27+
...
28+
29+
flags: [async]
30+
features: [async-iteration]
31+
---*/
32+
33+
let unblock;
34+
let blocking = new Promise(resolve => { unblock = resolve; });
35+
let unblocked = false;
36+
var g = async function*() {
37+
await blocking;
38+
unblocked = true;
39+
};
40+
41+
var it = g();
42+
let brokenPromise = Promise.resolve(42);
43+
Object.defineProperty(brokenPromise, 'constructor', {
44+
get: function () {
45+
throw new Error('broken promise');
46+
}
47+
});
48+
49+
it.next();
50+
it.return(brokenPromise)
51+
.then(
52+
() => {
53+
throw new Test262Error("Expected rejection");
54+
},
55+
err => {
56+
assert(unblocked, false, 'return should be rejected before generator is resumed');
57+
assert.sameValue(err.message, 'broken promise');
58+
}
59+
)
60+
.then($DONE, $DONE);
61+
62+
unblock();
Lines changed: 54 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
// Copyright (C) 2022 Chengzhong Wu. All rights reserved.
2+
// This code is governed by the BSD license found in the LICENSE file.
3+
4+
/*---
5+
esid: sec-asyncgenerator-prototype-return
6+
description: >
7+
A broken promise should reject the returned promise of
8+
AsyncGenerator.prototype.return when the generator's state is suspendedStart.
9+
info: |
10+
AsyncGenerator.prototype.return ( value )
11+
...
12+
8. If state is either suspendedStart or completed, then
13+
a. Set generator.[[AsyncGeneratorState]] to awaiting-return.
14+
b. Perform AsyncGeneratorAwaitReturn(_generator_).
15+
...
16+
17+
AsyncGeneratorAwaitReturn ( generator )
18+
...
19+
6. Let promise be Completion(PromiseResolve(%Promise%, completion.[[Value]])).
20+
7. If promiseCompletion is an abrupt completion, then
21+
a. Set generator.[[AsyncGeneratorState]] to completed.
22+
b. Perform AsyncGeneratorCompleteStep(generator, promiseCompletion, true).
23+
c. Perform AsyncGeneratorDrainQueue(generator).
24+
d. Return unused.
25+
8. Assert: promiseCompletion.[[Type]] is normal.
26+
9. Let promise be promiseCompletion.[[Value]].
27+
...
28+
29+
flags: [async]
30+
features: [async-iteration]
31+
---*/
32+
33+
var g = async function*() {
34+
throw new Test262Error('Generator must not be resumed.');
35+
};
36+
37+
var it = g();
38+
let brokenPromise = Promise.resolve(42);
39+
Object.defineProperty(brokenPromise, 'constructor', {
40+
get: function () {
41+
throw new Error('broken promise');
42+
}
43+
});
44+
45+
it.return(brokenPromise)
46+
.then(
47+
() => {
48+
throw new Test262Error("Expected rejection");
49+
},
50+
err => {
51+
assert.sameValue(err.message, 'broken promise');
52+
}
53+
)
54+
.then($DONE, $DONE);
Lines changed: 64 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
// Copyright (C) 2022 Chengzhong Wu. All rights reserved.
2+
// This code is governed by the BSD license found in the LICENSE file.
3+
4+
/*---
5+
esid: sec-asyncgenerator-prototype-return
6+
description: >
7+
A broken promise should resume the generator and reject with
8+
the exception when the generator's state is suspendedYield.
9+
info: |
10+
AsyncGenerator.prototype.return ( value )
11+
...
12+
9. Else if state is suspendedYield, then
13+
a. Perform AsyncGeneratorResume(generator, completion).
14+
...
15+
16+
AsyncGeneratorCompleteStep ( generator, completion, done [ , realm ] )
17+
Resumes the steps defined at AsyncGeneratorStart ( generator, generatorBody )
18+
...
19+
4. Set the code evaluation state of genContext such that when evaluation is resumed for that execution context the following steps will be performed:
20+
...
21+
i. Perform AsyncGeneratorDrainQueue(generator).
22+
j. Return undefined.
23+
24+
AsyncGeneratorDrainQueue ( generator )
25+
...
26+
5. Repeat, while done is false,
27+
a. Let next be the first element of queue.
28+
b. Let completion be Completion(next.[[Completion]]).
29+
c. If completion.[[Type]] is return, then
30+
i. Set generator.[[AsyncGeneratorState]] to awaiting-return.
31+
ii. Perform AsyncGeneratorAwaitReturn(generator).
32+
iii. Set done to true.
33+
...
34+
35+
flags: [async]
36+
features: [async-iteration]
37+
---*/
38+
39+
let caughtErr;
40+
var g = async function*() {
41+
try {
42+
yield;
43+
return 'this is never returned';
44+
} catch (err) {
45+
caughtErr = err;
46+
return 1;
47+
}
48+
};
49+
50+
let brokenPromise = Promise.resolve(42);
51+
Object.defineProperty(brokenPromise, 'constructor', {
52+
get: function () {
53+
throw new Error('broken promise');
54+
}
55+
});
56+
57+
var it = g();
58+
it.next().then(() => {
59+
return it.return(brokenPromise);
60+
}).then(ret => {
61+
assert.sameValue(caughtErr.message, 'broken promise');
62+
assert.sameValue(ret.value, 1, 'returned value');
63+
assert.sameValue(ret.done, true, 'iterator is closed');
64+
}).then($DONE, $DONE);

0 commit comments

Comments
 (0)