Index: test/mjsunit/harmony/async-debug-caught-exception2.js |
diff --git a/test/mjsunit/harmony/async-debug-caught-exception2.js b/test/mjsunit/harmony/async-debug-caught-exception2.js |
new file mode 100644 |
index 0000000000000000000000000000000000000000..0a585a2ae5723beff433a3a5db011276fb7a6c6f |
--- /dev/null |
+++ b/test/mjsunit/harmony/async-debug-caught-exception2.js |
@@ -0,0 +1,182 @@ |
+// Copyright 2016 the V8 project authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+// Flags: --allow-natives-syntax --harmony-async-await --expose-debug-as debug |
+ |
+Debug = debug.Debug |
+ |
+let exception = null; |
+let events = 0; |
+ |
+function listener(event, exec_state, event_data, data) { |
+ if (event != Debug.DebugEvent.Exception) return; |
+ try { |
+ events++; |
+ } catch (e) { |
+ %AbortJS(e + "\n" + e.stack); |
+ } |
+} |
+ |
+async function thrower() { |
+ throw "a"; // Exception a |
+} |
+ |
+var reject = () => Promise.reject("b"); // Exception b |
+ |
+async function awaitReturn() { await 1; return; } |
+ |
+async function scalar() { return 1; } |
+ |
+function nothing() { return 1; } |
+ |
+function rejectConstructor() { |
+ return new Promise((resolve, reject) => reject("c")); // Exception c |
+} |
+ |
+async function argThrower(x = (() => { throw "d"; })()) { } // Exception d |
+ |
+async function awaitThrow() { |
+ await undefined; |
+ throw "e"; // Exception e |
+} |
+ |
+function constructorThrow() { |
+ return new Promise((resolve, reject) => |
+ Promise.resolve().then(() => |
+ reject("f") // Exception f |
+ ) |
+ ); |
+} |
+ |
+function suppressThrow() { |
+ return thrower(); |
+} |
+ |
+async function caught(producer) { |
+ try { |
+ await producer(); |
+ } catch (e) { |
+ } |
+} |
+ |
+async function uncaught(producer) { |
+ await producer(); |
+} |
+ |
+async function caught(producer) { |
+ try { |
+ await producer(); |
+ } catch (e) { |
+ } |
+} |
+ |
+async function indirectUncaught(producer) { |
+ await uncaught(producer); |
+} |
+ |
+async function indirectCaught(producer) { |
+ try { |
+ await uncaught(producer); |
+ } catch (e) { |
+ } |
+} |
+ |
+function dotCatch(producer) { |
+ Promise.resolve(producer()).catch(() => {}); |
+} |
+ |
+function indirectReturnDotCatch(producer) { |
+ (async() => producer())().catch(() => {}); |
+} |
+ |
+function indirectAwaitDotCatch(producer) { |
+ (async() => await producer())().catch(() => {}); |
+} |
+ |
+function nestedDotCatch(producer) { |
+ Promise.resolve(producer()).then().catch(() => {}); |
+} |
+ |
+async function indirectAwaitCatch(producer) { |
+ try { |
+ await (() => producer())(); |
+ } catch (e) { |
+ } |
+} |
+ |
+let catches = [caught, indirectCaught, indirectAwaitCatch]; |
+let noncatches = [uncaught, indirectUncaught]; |
+let lateCatches = [dotCatch, indirectReturnDotCatch, indirectAwaitDotCatch, nestedDotCatch]; |
+ |
+let throws = [thrower, reject, argThrower, suppressThrow]; |
+let nonthrows = [awaitReturn, scalar, nothing]; |
+// TODO(littledan): Build a dependency graph to track the relationship between Promises |
+// linked by async/await, and enable tests so that |
+// let lateThrows = [awaitThrow, constructorThrow, rejectConstructor]; |
+let lateThrows = []; |
+ |
+let cases = []; |
+ |
+for (let producer of throws.concat(lateThrows)) { |
+ for (let consumer of catches) { |
+ cases.push({ producer, consumer, expectedEvents: 1, caught: true }); |
+ cases.push({ producer, consumer, expectedEvents: 0, caught: false }); |
+ } |
+} |
+ |
+for (let producer of throws.concat(lateThrows)) { |
+ for (let consumer of noncatches) { |
+ cases.push({ producer, consumer, expectedEvents: 1, caught: true }); |
+ cases.push({ producer, consumer, expectedEvents: 1, caught: false }); |
+ } |
+} |
+ |
+for (let producer of nonthrows) { |
+ for (let consumer of catches.concat(noncatches, lateCatches)) { |
+ cases.push({ producer, consumer, expectedEvents: 0, caught: true }); |
+ cases.push({ producer, consumer, expectedEvents: 0, caught: false }); |
+ } |
+} |
+ |
+for (let producer of lateThrows) { |
+ for (let consumer of lateCatches) { |
+ cases.push({ producer, consumer, expectedEvents: 1, caught: true }); |
+ cases.push({ producer, consumer, expectedEvents: 0, caught: false }); |
+ } |
+} |
+ |
+for (let producer of throws) { |
+ for (let consumer of lateCatches) { |
+ cases.push({ producer, consumer, expectedEvents: 1, caught: true }); |
+ cases.push({ producer, consumer, expectedEvents: 1, caught: false }); |
+ } |
+} |
+ |
+const tonsOfTasks = 50; |
+ |
+function afterWaiting(callback, tasks = tonsOfTasks) { |
+ if (tasks === 0) callback(); |
+ else %EnqueueMicrotask(() => afterWaiting(callback, tasks - 1)); |
+} |
+ |
+let failureCount = 0; |
+ |
+for (let {producer, consumer, expectedEvents, caught} of cases) { |
+ events = 0; |
+ Debug.setListener(listener); |
+ if (caught) |
+ Debug.setBreakOnException(); |
+ else |
+ Debug.setBreakOnUncaughtException(); |
+ |
+ consumer(producer); |
+ |
+ %RunMicrotasks(); |
+ |
+ Debug.setListener(null); |
+ if (caught) Debug.clearBreakOnException(); |
+ else Debug.clearBreakOnUncaughtException(); |
+ assertEquals(expectedEvents, events); |
+ assertNull(exception); |
+} |