summaryrefslogtreecommitdiffstats
path: root/dom/tests/browser/browser_ConsoleAPITests.js
diff options
context:
space:
mode:
Diffstat (limited to 'dom/tests/browser/browser_ConsoleAPITests.js')
-rw-r--r--dom/tests/browser/browser_ConsoleAPITests.js638
1 files changed, 638 insertions, 0 deletions
diff --git a/dom/tests/browser/browser_ConsoleAPITests.js b/dom/tests/browser/browser_ConsoleAPITests.js
new file mode 100644
index 0000000000..8287ecfb21
--- /dev/null
+++ b/dom/tests/browser/browser_ConsoleAPITests.js
@@ -0,0 +1,638 @@
+/* vim:set ts=2 sw=2 sts=2 et: */
+/* 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/. */
+
+const TEST_URI =
+ "http://example.com/browser/dom/tests/browser/test-console-api.html";
+
+add_task(async function() {
+ let tab = await BrowserTestUtils.openNewForegroundTab(gBrowser, TEST_URI);
+ registerCleanupFunction(() => gBrowser.removeTab(tab));
+ let browser = gBrowser.selectedBrowser;
+
+ await consoleAPISanityTest(browser);
+ await observeConsoleTest(browser);
+ await startTraceTest(browser);
+ await startLocationTest(browser);
+ await startNativeCallbackTest(browser);
+ await startGroupTest(browser);
+ await startTimeTest(browser);
+ await startTimeEndTest(browser);
+ await startTimeStampTest(browser);
+ await startEmptyTimeStampTest(browser);
+ await startEmptyTimerTest(browser);
+});
+
+function spawnWithObserver(browser, observerFunc, func) {
+ // Build the observer generating function.
+ let source = [
+ "const TEST_URI = 'http://example.com/browser/dom/tests/browser/test-console-api.html';",
+ // Create a promise to be resolved when the text is complete. It is stored
+ // on content, such that it can be waited on by calling waitForResolve. This
+ // is done rather than returning it from this function such that the
+ // initialization can be yeilded on before yeilding on the conclusion of the
+ // test.
+ "content._promise = new Promise(_resolve => {",
+ // These are variables which are used by the test runner to communicate
+ // state to the observer.
+ " let gLevel, gArgs, gStyle;",
+ " let expect = function(level) {",
+ " gLevel = level;",
+ " gArgs = Array.prototype.slice.call(arguments, 1);",
+ " }",
+ // To ease the transition to the new format, content.window is avaliable as gWindow
+ // in the content.
+ " let gWindow = content.window;",
+ // This method is called rather than _resolve such that the observer is removed
+ // before exiting the test
+ " let resolve = () => {",
+ " Services.obs.removeObserver(ConsoleObserver, 'console-api-log-event');",
+ " _resolve();",
+ " };",
+ // This is the observer itself, it calls the passed-in function whenever
+ // it encounters an event
+ " let ConsoleObserver = {",
+ " QueryInterface: ChromeUtils.generateQI([Ci.nsIObserver]),",
+ " observe: function(aSubject, aTopic, aData) {",
+ " try {",
+ " (" + observerFunc.toString() + ")(aSubject.wrappedJSObject);",
+ " } catch (ex) {",
+ " ok(false, 'Exception thrown in observe: ' + ex);",
+ " }",
+ " }",
+ " };",
+ " Services.obs.addObserver(ConsoleObserver, 'console-api-log-event', false);",
+ // Call the initialization function (if present)
+ func ? "(" + func.toString() + ")();" : "",
+ "});",
+ ].join("\n");
+
+ return SpecialPowers.spawn(browser, [], new Function(source));
+}
+
+function waitForResolve(browser) {
+ return SpecialPowers.spawn(browser, [], function() {
+ return content._promise;
+ });
+}
+
+async function consoleAPISanityTest(browser) {
+ await SpecialPowers.spawn(browser, [], function() {
+ let win = XPCNativeWrapper.unwrap(content.window);
+
+ ok(win.console, "we have a console attached");
+ ok(win.console, "we have a console attached, 2nd attempt");
+
+ ok(win.console.log, "console.log is here");
+ ok(win.console.info, "console.info is here");
+ ok(win.console.warn, "console.warn is here");
+ ok(win.console.error, "console.error is here");
+ ok(win.console.exception, "console.exception is here");
+ ok(win.console.trace, "console.trace is here");
+ ok(win.console.dir, "console.dir is here");
+ ok(win.console.group, "console.group is here");
+ ok(win.console.groupCollapsed, "console.groupCollapsed is here");
+ ok(win.console.groupEnd, "console.groupEnd is here");
+ ok(win.console.time, "console.time is here");
+ ok(win.console.timeEnd, "console.timeEnd is here");
+ ok(win.console.timeStamp, "console.timeStamp is here");
+ ok(win.console.assert, "console.assert is here");
+ ok(win.console.count, "console.count is here");
+ });
+}
+
+// These globals are all defined in spawnWithObserver in a sub-process.
+/* global gWindow, gArgs:true, gLevel:true, gStyle:true, expect, resolve */
+function testConsoleData(aMessageObject) {
+ let messageWindow = Services.wm.getOuterWindowWithId(aMessageObject.ID);
+ is(messageWindow, gWindow, "found correct window by window ID");
+
+ is(aMessageObject.level, gLevel, "expected level received");
+ ok(aMessageObject.arguments, "we have arguments");
+
+ switch (gLevel) {
+ case "trace": {
+ is(aMessageObject.arguments.length, 0, "arguments.length matches");
+ is(
+ aMessageObject.stacktrace.toSource(),
+ gArgs.toSource(),
+ "stack trace is correct"
+ );
+ break;
+ }
+ case "count": {
+ is(aMessageObject.counter.label, gArgs[0].label, "label matches");
+ is(aMessageObject.counter.count, gArgs[0].count, "count matches");
+ break;
+ }
+ default: {
+ is(
+ aMessageObject.arguments.length,
+ gArgs.length,
+ "arguments.length matches"
+ );
+ gArgs.forEach(function(a, i) {
+ // Waive Xray so that we don't get messed up by Xray ToString.
+ //
+ // It'd be nice to just use XPCNativeWrapper.unwrap here, but there are
+ // a number of dumb reasons we can't. See bug 868675.
+ var arg = aMessageObject.arguments[i];
+ if (Cu.isXrayWrapper(arg)) {
+ arg = arg.wrappedJSObject;
+ }
+ is(arg, a, "correct arg " + i);
+ });
+
+ if (gStyle) {
+ is(
+ aMessageObject.styles.length,
+ gStyle.length,
+ "styles.length matches"
+ );
+ is(aMessageObject.styles + "", gStyle + "", "styles match");
+ } else {
+ ok(
+ !aMessageObject.styles || aMessageObject.styles.length === 0,
+ "styles match"
+ );
+ }
+ }
+ }
+}
+
+async function observeConsoleTest(browser) {
+ await spawnWithObserver(browser, testConsoleData, function(opts) {
+ let win = XPCNativeWrapper.unwrap(content.window);
+ expect("log", "arg");
+ win.console.log("arg");
+
+ expect("info", "arg", "extra arg");
+ win.console.info("arg", "extra arg");
+
+ expect("warn", "Lesson 1: PI is approximately equal to 3");
+ win.console.warn(
+ "Lesson %d: %s is approximately equal to %1.0f",
+ 1,
+ "PI",
+ 3.14159
+ );
+
+ expect("warn", "Lesson 1: PI is approximately equal to 3.14");
+ win.console.warn(
+ "Lesson %d: %s is approximately equal to %1.2f",
+ 1,
+ "PI",
+ 3.14159
+ );
+
+ expect("warn", "Lesson 1: PI is approximately equal to 3.141590");
+ win.console.warn(
+ "Lesson %d: %s is approximately equal to %f",
+ 1,
+ "PI",
+ 3.14159
+ );
+
+ expect("warn", "Lesson 1: PI is approximately equal to 3.1415900");
+ win.console.warn(
+ "Lesson %d: %s is approximately equal to %0.7f",
+ 1,
+ "PI",
+ 3.14159
+ );
+
+ expect("log", "%d, %s, %l");
+ win.console.log("%d, %s, %l");
+
+ expect("log", "%a %b %g");
+ win.console.log("%a %b %g");
+
+ expect("log", "%a %b %g", "a", "b");
+ win.console.log("%a %b %g", "a", "b");
+
+ expect("log", "2, a, %l", 3);
+ win.console.log("%d, %s, %l", 2, "a", 3);
+
+ // Bug #692550 handle null and undefined.
+ expect("log", "null, undefined");
+ win.console.log("%s, %s", null, undefined);
+
+ // Bug #696288 handle object as first argument.
+ let obj = { a: 1 };
+ expect("log", obj, "a");
+ win.console.log(obj, "a");
+
+ expect("dir", win.toString());
+ win.console.dir(win);
+
+ expect("error", "arg");
+ win.console.error("arg");
+
+ expect("exception", "arg");
+ win.console.exception("arg");
+
+ expect("log", "foobar");
+ gStyle = ["color:red;foobar;;"];
+ win.console.log("%cfoobar", gStyle[0]);
+
+ let obj4 = { d: 4 };
+ expect("warn", "foobar", obj4, "test", "bazbazstr", "last");
+ gStyle = [null, null, null, "color:blue;", "color:red"];
+ win.console.warn(
+ "foobar%Otest%cbazbaz%s%clast",
+ obj4,
+ gStyle[3],
+ "str",
+ gStyle[4]
+ );
+
+ let obj3 = { c: 3 };
+ expect("info", "foobar", "bazbaz", obj3, "%comg", "color:yellow");
+ gStyle = [null, "color:pink;"];
+ win.console.info(
+ "foobar%cbazbaz",
+ gStyle[1],
+ obj3,
+ "%comg",
+ "color:yellow"
+ );
+
+ gStyle = null;
+ let obj2 = { b: 2 };
+ expect("log", "omg ", obj, " foo ", 4, obj2);
+ win.console.log("omg %o foo %o", obj, 4, obj2);
+
+ expect("assert", "message");
+ win.console.assert(false, "message");
+
+ expect("count", { label: "label a", count: 1 });
+ win.console.count("label a");
+
+ expect("count", { label: "label b", count: 1 });
+ win.console.count("label b");
+
+ expect("count", { label: "label a", count: 2 });
+ win.console.count("label a");
+
+ expect("count", { label: "label b", count: 2 });
+ win.console.count("label b");
+ dump("Resolving\n");
+ resolve();
+ });
+ dump("There\n");
+}
+
+function testTraceConsoleData(aMessageObject) {
+ let messageWindow = Services.wm.getOuterWindowWithId(aMessageObject.ID);
+ is(messageWindow, gWindow, "found correct window by window ID");
+
+ is(aMessageObject.level, gLevel, "expected level received");
+ ok(aMessageObject.arguments, "we have arguments");
+
+ is(gLevel, "trace", "gLevel should be trace");
+ is(aMessageObject.arguments.length, 0, "arguments.length matches");
+ dump(aMessageObject.stacktrace.toSource() + "\n" + gArgs.toSource() + "\n");
+ is(
+ aMessageObject.stacktrace.toSource(),
+ gArgs.toSource(),
+ "stack trace is correct"
+ );
+ resolve();
+}
+
+async function startTraceTest(browser) {
+ dump("HERE\n");
+ await spawnWithObserver(browser, testTraceConsoleData, function(opts) {
+ dump("Observer attached\n");
+ gLevel = "trace";
+ gArgs = [
+ {
+ columnNumber: 9,
+ filename: TEST_URI,
+ functionName: "window.foobar585956c",
+ lineNumber: 6,
+ },
+ {
+ columnNumber: 16,
+ filename: TEST_URI,
+ functionName: "foobar585956b",
+ lineNumber: 11,
+ },
+ {
+ columnNumber: 16,
+ filename: TEST_URI,
+ functionName: "foobar585956a",
+ lineNumber: 15,
+ },
+ {
+ columnNumber: 1,
+ filename: TEST_URI,
+ functionName: "onclick",
+ lineNumber: 1,
+ },
+ ];
+ });
+
+ BrowserTestUtils.synthesizeMouseAtCenter("#test-trace", {}, browser);
+ await waitForResolve(browser);
+}
+
+function testLocationData(aMessageObject) {
+ let messageWindow = Services.wm.getOuterWindowWithId(aMessageObject.ID);
+ is(messageWindow, gWindow, "found correct window by window ID");
+
+ is(aMessageObject.level, gLevel, "expected level received");
+ ok(aMessageObject.arguments, "we have arguments");
+
+ is(aMessageObject.filename, gArgs[0].filename, "filename matches");
+ is(aMessageObject.lineNumber, gArgs[0].lineNumber, "lineNumber matches");
+ is(
+ aMessageObject.functionName,
+ gArgs[0].functionName,
+ "functionName matches"
+ );
+ is(
+ aMessageObject.arguments.length,
+ gArgs[0].arguments.length,
+ "arguments.length matches"
+ );
+ gArgs[0].arguments.forEach(function(a, i) {
+ is(aMessageObject.arguments[i], a, "correct arg " + i);
+ });
+
+ resolve();
+}
+
+async function startLocationTest(browser) {
+ await spawnWithObserver(browser, testLocationData, function(opts) {
+ gLevel = "log";
+ gArgs = [
+ {
+ filename: TEST_URI,
+ functionName: "foobar646025",
+ arguments: ["omg", "o", "d"],
+ lineNumber: 19,
+ },
+ ];
+ });
+
+ BrowserTestUtils.synthesizeMouseAtCenter("#test-location", {}, browser);
+ await waitForResolve(browser);
+}
+
+function testNativeCallback(aMessageObject) {
+ is(aMessageObject.level, "log", "expected level received");
+ is(aMessageObject.filename, "", "filename matches");
+ is(aMessageObject.lineNumber, 0, "lineNumber matches");
+ is(aMessageObject.functionName, "", "functionName matches");
+
+ resolve();
+}
+
+async function startNativeCallbackTest(browser) {
+ await spawnWithObserver(browser, testNativeCallback);
+
+ BrowserTestUtils.synthesizeMouseAtCenter("#test-nativeCallback", {}, browser);
+ await waitForResolve(browser);
+}
+
+function testConsoleGroup(aMessageObject) {
+ let messageWindow = Services.wm.getOuterWindowWithId(aMessageObject.ID);
+ is(messageWindow, gWindow, "found correct window by window ID");
+
+ ok(
+ aMessageObject.level == "group" ||
+ aMessageObject.level == "groupCollapsed" ||
+ aMessageObject.level == "groupEnd",
+ "expected level received"
+ );
+
+ is(aMessageObject.functionName, "testGroups", "functionName matches");
+ ok(
+ aMessageObject.lineNumber >= 46 && aMessageObject.lineNumber <= 50,
+ "lineNumber matches"
+ );
+ if (aMessageObject.level == "groupCollapsed") {
+ is(aMessageObject.groupName, "a group", "groupCollapsed groupName matches");
+ is(aMessageObject.arguments[0], "a", "groupCollapsed arguments[0] matches");
+ is(
+ aMessageObject.arguments[1],
+ "group",
+ "groupCollapsed arguments[0] matches"
+ );
+ } else if (aMessageObject.level == "group") {
+ is(aMessageObject.groupName, "b group", "group groupName matches");
+ is(aMessageObject.arguments[0], "b", "group arguments[0] matches");
+ is(aMessageObject.arguments[1], "group", "group arguments[1] matches");
+ } else if (aMessageObject.level == "groupEnd") {
+ is(aMessageObject.groupName, "b group", "groupEnd groupName matches");
+ }
+
+ if (aMessageObject.level == "groupEnd") {
+ resolve();
+ }
+}
+
+async function startGroupTest(browser) {
+ await spawnWithObserver(browser, testConsoleGroup);
+
+ BrowserTestUtils.synthesizeMouseAtCenter("#test-groups", {}, browser);
+ await waitForResolve(browser);
+}
+
+function testConsoleTime(aMessageObject) {
+ let messageWindow = Services.wm.getOuterWindowWithId(aMessageObject.ID);
+ is(messageWindow, gWindow, "found correct window by window ID");
+
+ is(aMessageObject.level, gLevel, "expected level received");
+
+ is(aMessageObject.filename, gArgs[0].filename, "filename matches");
+ is(aMessageObject.lineNumber, gArgs[0].lineNumber, "lineNumber matches");
+ is(
+ aMessageObject.functionName,
+ gArgs[0].functionName,
+ "functionName matches"
+ );
+ is(aMessageObject.timer.name, gArgs[0].timer.name, "timer.name matches");
+
+ gArgs[0].arguments.forEach(function(a, i) {
+ is(aMessageObject.arguments[i], a, "correct arg " + i);
+ });
+
+ resolve();
+}
+
+async function startTimeTest(browser) {
+ await spawnWithObserver(browser, testConsoleTime, function(opts) {
+ gLevel = "time";
+ gArgs = [
+ {
+ filename: TEST_URI,
+ lineNumber: 23,
+ functionName: "startTimer",
+ arguments: ["foo"],
+ timer: { name: "foo" },
+ },
+ ];
+ });
+
+ BrowserTestUtils.synthesizeMouseAtCenter("#test-time", {}, browser);
+ await waitForResolve(browser);
+}
+
+function testConsoleTimeEnd(aMessageObject) {
+ let messageWindow = Services.wm.getOuterWindowWithId(aMessageObject.ID);
+ is(messageWindow, gWindow, "found correct window by window ID");
+
+ is(aMessageObject.level, gLevel, "expected level received");
+ ok(aMessageObject.arguments, "we have arguments");
+
+ is(aMessageObject.filename, gArgs[0].filename, "filename matches");
+ is(aMessageObject.lineNumber, gArgs[0].lineNumber, "lineNumber matches");
+ is(
+ aMessageObject.functionName,
+ gArgs[0].functionName,
+ "functionName matches"
+ );
+ is(
+ aMessageObject.arguments.length,
+ gArgs[0].arguments.length,
+ "arguments.length matches"
+ );
+ is(aMessageObject.timer.name, gArgs[0].timer.name, "timer name matches");
+ is(
+ typeof aMessageObject.timer.duration,
+ "number",
+ "timer duration is a number"
+ );
+ info("timer duration: " + aMessageObject.timer.duration);
+ ok(aMessageObject.timer.duration >= 0, "timer duration is positive");
+
+ gArgs[0].arguments.forEach(function(a, i) {
+ is(aMessageObject.arguments[i], a, "correct arg " + i);
+ });
+
+ resolve();
+}
+
+async function startTimeEndTest(browser) {
+ await spawnWithObserver(browser, testConsoleTimeEnd, function(opts) {
+ gLevel = "timeEnd";
+ gArgs = [
+ {
+ filename: TEST_URI,
+ lineNumber: 27,
+ functionName: "stopTimer",
+ arguments: ["foo"],
+ timer: { name: "foo" },
+ },
+ ];
+ });
+
+ BrowserTestUtils.synthesizeMouseAtCenter("#test-timeEnd", {}, browser);
+ await waitForResolve(browser);
+}
+
+function testConsoleTimeStamp(aMessageObject) {
+ let messageWindow = Services.wm.getOuterWindowWithId(aMessageObject.ID);
+ is(messageWindow, gWindow, "found correct window by window ID");
+
+ is(aMessageObject.level, gLevel, "expected level received");
+
+ is(aMessageObject.filename, gArgs[0].filename, "filename matches");
+ is(aMessageObject.lineNumber, gArgs[0].lineNumber, "lineNumber matches");
+ is(
+ aMessageObject.functionName,
+ gArgs[0].functionName,
+ "functionName matches"
+ );
+ ok(aMessageObject.timeStamp > 0, "timeStamp is a positive value");
+
+ gArgs[0].arguments.forEach(function(a, i) {
+ is(aMessageObject.arguments[i], a, "correct arg " + i);
+ });
+
+ resolve();
+}
+
+async function startTimeStampTest(browser) {
+ await spawnWithObserver(browser, testConsoleTimeStamp, function() {
+ gLevel = "timeStamp";
+ gArgs = [
+ {
+ filename: TEST_URI,
+ lineNumber: 58,
+ functionName: "timeStamp",
+ arguments: ["!!!"],
+ },
+ ];
+ });
+
+ BrowserTestUtils.synthesizeMouseAtCenter("#test-timeStamp", {}, browser);
+ await waitForResolve(browser);
+}
+
+function testEmptyConsoleTimeStamp(aMessageObject) {
+ let messageWindow = Services.wm.getOuterWindowWithId(aMessageObject.ID);
+ is(messageWindow, gWindow, "found correct window by window ID");
+
+ is(aMessageObject.level, gLevel, "expected level received");
+
+ is(aMessageObject.filename, gArgs[0].filename, "filename matches");
+ is(aMessageObject.lineNumber, gArgs[0].lineNumber, "lineNumber matches");
+ is(
+ aMessageObject.functionName,
+ gArgs[0].functionName,
+ "functionName matches"
+ );
+ ok(aMessageObject.timeStamp > 0, "timeStamp is a positive value");
+ is(aMessageObject.arguments.length, 0, "we don't have arguments");
+
+ resolve();
+}
+
+async function startEmptyTimeStampTest(browser) {
+ await spawnWithObserver(browser, testEmptyConsoleTimeStamp, function() {
+ gLevel = "timeStamp";
+ gArgs = [
+ {
+ filename: TEST_URI,
+ lineNumber: 58,
+ functionName: "timeStamp",
+ arguments: [],
+ },
+ ];
+ });
+
+ BrowserTestUtils.synthesizeMouseAtCenter("#test-emptyTimeStamp", {}, browser);
+ await waitForResolve(browser);
+}
+
+function testEmptyTimer(aMessageObject) {
+ let messageWindow = Services.wm.getOuterWindowWithId(aMessageObject.ID);
+ is(messageWindow, gWindow, "found correct window by window ID");
+
+ ok(
+ aMessageObject.level == "time" || aMessageObject.level == "timeEnd",
+ "expected level received"
+ );
+ is(aMessageObject.arguments.length, 1, "we have the default argument");
+ is(aMessageObject.arguments[0], "default", "we have the default argument");
+ ok(aMessageObject.timer, "we have a timer");
+
+ is(aMessageObject.functionName, "namelessTimer", "functionName matches");
+ ok(
+ aMessageObject.lineNumber == 31 || aMessageObject.lineNumber == 32,
+ "lineNumber matches"
+ );
+
+ resolve();
+}
+
+async function startEmptyTimerTest(browser) {
+ await spawnWithObserver(browser, testEmptyTimer);
+
+ BrowserTestUtils.synthesizeMouseAtCenter("#test-namelessTimer", {}, browser);
+ await waitForResolve(browser);
+}