1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
|
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at <http://mozilla.org/MPL/2.0/>. */
// Tests tracing all function returns
"use strict";
add_task(async function testTracingFunctionReturn() {
await pushPref("devtools.debugger.features.javascript-tracing", true);
const jsCode = `async function foo() { nullReturn(); falseReturn(); await new Promise(r => setTimeout(r, 0)); return bar(); }; function nullReturn() { return null; } function falseReturn() { return false; } function bar() { return 42; }; function throwingFunction() { throw new Error("the exception") }`;
const dbg = await initDebuggerWithAbsoluteURL(
"data:text/html," +
encodeURIComponent(`<script>${jsCode}</script><body></body>`)
);
await openContextMenuInDebugger(dbg, "trace");
let toggled = waitForDispatch(
dbg.store,
"TOGGLE_JAVASCRIPT_TRACING_FUNCTION_RETURN"
);
selectContextMenuItem(dbg, `#debugger-trace-menu-item-function-return`);
await toggled;
ok(true, "Toggled the trace of function returns");
await clickElement(dbg, "trace");
const topLevelThreadActorID =
dbg.toolbox.commands.targetCommand.targetFront.threadFront.actorID;
info("Wait for tracing to be enabled");
await waitForState(dbg, state => {
return dbg.selectors.getIsThreadCurrentlyTracing(topLevelThreadActorID);
});
invokeInTab("foo");
await hasConsoleMessage(dbg, "⟶ interpreter λ foo");
await hasConsoleMessage(dbg, "⟶ interpreter λ bar");
await hasConsoleMessage(dbg, "⟵ λ bar");
await hasConsoleMessage(dbg, "⟵ λ foo");
await clickElement(dbg, "trace");
info("Wait for tracing to be disabled");
await waitForState(dbg, state => {
return !dbg.selectors.getIsThreadCurrentlyTracing(topLevelThreadActorID);
});
await openContextMenuInDebugger(dbg, "trace");
toggled = waitForDispatch(dbg.store, "TOGGLE_JAVASCRIPT_TRACING_VALUES");
selectContextMenuItem(dbg, `#debugger-trace-menu-item-log-values`);
await toggled;
ok(true, "Toggled the log values setting");
await clickElement(dbg, "trace");
info("Wait for tracing to be re-enabled with logging of returned values");
await waitForState(dbg, state => {
return dbg.selectors.getIsThreadCurrentlyTracing(topLevelThreadActorID);
});
invokeInTab("foo");
await hasConsoleMessage(dbg, "⟶ interpreter λ foo");
await hasConsoleMessage(dbg, "⟶ interpreter λ bar");
await hasConsoleMessage(dbg, "⟵ λ bar return 42");
await hasConsoleMessage(dbg, "⟶ interpreter λ nullReturn");
await hasConsoleMessage(dbg, "⟵ λ nullReturn return null");
await hasConsoleMessage(dbg, "⟶ interpreter λ falseReturn");
await hasConsoleMessage(dbg, "⟵ λ falseReturn return false");
await hasConsoleMessage(
dbg,
`⟵ λ foo return \nPromise { <state>: "fulfilled", <value>: 42 }`
);
invokeInTab("throwingFunction").catch(() => {});
await hasConsoleMessage(
dbg,
`⟵ λ throwingFunction throw \nError: the exception`
);
info("Stop tracing");
await clickElement(dbg, "trace");
await waitForState(dbg, state => {
return !dbg.selectors.getIsThreadCurrentlyTracing(topLevelThreadActorID);
});
info("Toggle the two settings to the default value");
await openContextMenuInDebugger(dbg, "trace");
toggled = waitForDispatch(dbg.store, "TOGGLE_JAVASCRIPT_TRACING_VALUES");
selectContextMenuItem(dbg, `#debugger-trace-menu-item-log-values`);
await toggled;
await openContextMenuInDebugger(dbg, "trace");
toggled = waitForDispatch(
dbg.store,
"TOGGLE_JAVASCRIPT_TRACING_FUNCTION_RETURN"
);
selectContextMenuItem(dbg, `#debugger-trace-menu-item-function-return`);
await toggled;
// Reset the trace on next interaction setting
Services.prefs.clearUserPref(
"devtools.debugger.javascript-tracing-on-next-interaction"
);
});
|