forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			2209 lines
		
	
	
	
		
			69 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			2209 lines
		
	
	
	
		
			69 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* Any copyright is dedicated to the Public Domain.
 | |
|  * http://creativecommons.org/publicdomain/zero/1.0/ */
 | |
| /* eslint no-unused-vars: [2, {"vars": "local"}] */
 | |
| 
 | |
| /* import-globals-from ../../inspector/test/shared-head.js */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| // This shared-head.js file is used by most mochitests
 | |
| // and we start using it in xpcshell tests as well.
 | |
| // It contains various common helper functions.
 | |
| 
 | |
| const isMochitest = "gTestPath" in this;
 | |
| const isXpcshell = !isMochitest;
 | |
| if (isXpcshell) {
 | |
|   // gTestPath isn't exposed to xpcshell tests
 | |
|   // _TEST_FILE is an array for a unique string
 | |
|   /* global _TEST_FILE */
 | |
|   this.gTestPath = _TEST_FILE[0];
 | |
| }
 | |
| 
 | |
| const { Constructor: CC } = Components;
 | |
| 
 | |
| // Print allocation count if DEBUG_DEVTOOLS_ALLOCATIONS is set to "normal",
 | |
| // and allocation sites if DEBUG_DEVTOOLS_ALLOCATIONS is set to "verbose".
 | |
| const DEBUG_ALLOCATIONS = Services.env.get("DEBUG_DEVTOOLS_ALLOCATIONS");
 | |
| if (DEBUG_ALLOCATIONS) {
 | |
|   // Use a custom loader with `invisibleToDebugger` flag for the allocation tracker
 | |
|   // as it instantiates custom Debugger API instances and has to be running in a distinct
 | |
|   // compartments from DevTools and system scopes (JSMs, XPCOM,...)
 | |
|   const {
 | |
|     useDistinctSystemPrincipalLoader,
 | |
|     releaseDistinctSystemPrincipalLoader,
 | |
|   } = ChromeUtils.importESModule(
 | |
|     "resource://devtools/shared/loader/DistinctSystemPrincipalLoader.sys.mjs"
 | |
|   );
 | |
|   const requester = {};
 | |
|   const loader = useDistinctSystemPrincipalLoader(requester);
 | |
|   registerCleanupFunction(() =>
 | |
|     releaseDistinctSystemPrincipalLoader(requester)
 | |
|   );
 | |
| 
 | |
|   const { allocationTracker } = loader.require(
 | |
|     "resource://devtools/shared/test-helpers/allocation-tracker.js"
 | |
|   );
 | |
|   const tracker = allocationTracker({ watchAllGlobals: true });
 | |
|   registerCleanupFunction(() => {
 | |
|     if (DEBUG_ALLOCATIONS == "normal") {
 | |
|       tracker.logCount();
 | |
|     } else if (DEBUG_ALLOCATIONS == "verbose") {
 | |
|       tracker.logAllocationSites();
 | |
|     }
 | |
|     tracker.stop();
 | |
|   });
 | |
| }
 | |
| 
 | |
| const { loader, require } = ChromeUtils.importESModule(
 | |
|   "resource://devtools/shared/loader/Loader.sys.mjs"
 | |
| );
 | |
| const { sinon } = ChromeUtils.importESModule(
 | |
|   "resource://testing-common/Sinon.sys.mjs"
 | |
| );
 | |
| 
 | |
| // When loaded from xpcshell test, this file is loaded via xpcshell.ini's head property
 | |
| // and so it loaded first before anything else and isn't having access to Services global.
 | |
| // Whereas many head.js files from mochitest import this file via loadSubScript
 | |
| // and already expose Services as a global.
 | |
| 
 | |
| const {
 | |
|   gDevTools,
 | |
| } = require("resource://devtools/client/framework/devtools.js");
 | |
| const {
 | |
|   CommandsFactory,
 | |
| } = require("resource://devtools/shared/commands/commands-factory.js");
 | |
| const DevToolsUtils = require("resource://devtools/shared/DevToolsUtils.js");
 | |
| 
 | |
| const KeyShortcuts = require("resource://devtools/client/shared/key-shortcuts.js");
 | |
| 
 | |
| const { LocalizationHelper } = require("resource://devtools/shared/l10n.js");
 | |
| 
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "ResponsiveUIManager",
 | |
|   "resource://devtools/client/responsive/manager.js"
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "localTypes",
 | |
|   "resource://devtools/client/responsive/types.js"
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "ResponsiveMessageHelper",
 | |
|   "resource://devtools/client/responsive/utils/message.js"
 | |
| );
 | |
| 
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "FluentReact",
 | |
|   "resource://devtools/client/shared/vendor/fluent-react.js"
 | |
| );
 | |
| 
 | |
| const TEST_DIR = gTestPath.substr(0, gTestPath.lastIndexOf("/"));
 | |
| const CHROME_URL_ROOT = TEST_DIR + "/";
 | |
| const URL_ROOT = CHROME_URL_ROOT.replace(
 | |
|   "chrome://mochitests/content/",
 | |
|   "http://example.com/"
 | |
| );
 | |
| const URL_ROOT_SSL = CHROME_URL_ROOT.replace(
 | |
|   "chrome://mochitests/content/",
 | |
|   "https://example.com/"
 | |
| );
 | |
| 
 | |
| // Add aliases which make it more explicit that URL_ROOT uses a com TLD.
 | |
| const URL_ROOT_COM = URL_ROOT;
 | |
| const URL_ROOT_COM_SSL = URL_ROOT_SSL;
 | |
| 
 | |
| // Also expose http://example.org, http://example.net, https://example.org to
 | |
| // test Fission scenarios easily.
 | |
| // Note: example.net is not available for https.
 | |
| const URL_ROOT_ORG = CHROME_URL_ROOT.replace(
 | |
|   "chrome://mochitests/content/",
 | |
|   "http://example.org/"
 | |
| );
 | |
| const URL_ROOT_ORG_SSL = CHROME_URL_ROOT.replace(
 | |
|   "chrome://mochitests/content/",
 | |
|   "https://example.org/"
 | |
| );
 | |
| const URL_ROOT_NET = CHROME_URL_ROOT.replace(
 | |
|   "chrome://mochitests/content/",
 | |
|   "http://example.net/"
 | |
| );
 | |
| const URL_ROOT_NET_SSL = CHROME_URL_ROOT.replace(
 | |
|   "chrome://mochitests/content/",
 | |
|   "https://example.net/"
 | |
| );
 | |
| // mochi.test:8888 is the actual primary location where files are served.
 | |
| const URL_ROOT_MOCHI_8888 = CHROME_URL_ROOT.replace(
 | |
|   "chrome://mochitests/content/",
 | |
|   "http://mochi.test:8888/"
 | |
| );
 | |
| 
 | |
| try {
 | |
|   if (isMochitest) {
 | |
|     Services.scriptloader.loadSubScript(
 | |
|       "chrome://mochitests/content/browser/devtools/client/shared/test/telemetry-test-helpers.js",
 | |
|       this
 | |
|     );
 | |
|   }
 | |
| } catch (e) {
 | |
|   ok(
 | |
|     false,
 | |
|     "MISSING DEPENDENCY ON telemetry-test-helpers.js\n" +
 | |
|       "Please add the following line in browser.ini:\n" +
 | |
|       "  !/devtools/client/shared/test/telemetry-test-helpers.js\n"
 | |
|   );
 | |
|   throw e;
 | |
| }
 | |
| 
 | |
| // Force devtools to be initialized so menu items and keyboard shortcuts get installed
 | |
| require("resource://devtools/client/framework/devtools-browser.js");
 | |
| 
 | |
| // All tests are asynchronous
 | |
| if (isMochitest) {
 | |
|   waitForExplicitFinish();
 | |
| }
 | |
| 
 | |
| var EXPECTED_DTU_ASSERT_FAILURE_COUNT = 0;
 | |
| 
 | |
| registerCleanupFunction(function () {
 | |
|   if (
 | |
|     DevToolsUtils.assertionFailureCount !== EXPECTED_DTU_ASSERT_FAILURE_COUNT
 | |
|   ) {
 | |
|     ok(
 | |
|       false,
 | |
|       "Should have had the expected number of DevToolsUtils.assert() failures." +
 | |
|         " Expected " +
 | |
|         EXPECTED_DTU_ASSERT_FAILURE_COUNT +
 | |
|         ", got " +
 | |
|         DevToolsUtils.assertionFailureCount
 | |
|     );
 | |
|   }
 | |
| });
 | |
| 
 | |
| // Uncomment this pref to dump all devtools emitted events to the console.
 | |
| // Services.prefs.setBoolPref("devtools.dump.emit", true);
 | |
| 
 | |
| /**
 | |
|  * Watch console messages for failed propType definitions in React components.
 | |
|  */
 | |
| function onConsoleMessage(subject) {
 | |
|   const message = subject.wrappedJSObject.arguments[0];
 | |
| 
 | |
|   if (message && /Failed propType/.test(message.toString())) {
 | |
|     ok(false, message);
 | |
|   }
 | |
| }
 | |
| 
 | |
| const ConsoleAPIStorage = Cc["@mozilla.org/consoleAPI-storage;1"].getService(
 | |
|   Ci.nsIConsoleAPIStorage
 | |
| );
 | |
| 
 | |
| ConsoleAPIStorage.addLogEventListener(
 | |
|   onConsoleMessage,
 | |
|   Cc["@mozilla.org/systemprincipal;1"].createInstance(Ci.nsIPrincipal)
 | |
| );
 | |
| registerCleanupFunction(() => {
 | |
|   ConsoleAPIStorage.removeLogEventListener(onConsoleMessage);
 | |
| });
 | |
| 
 | |
| Services.prefs.setBoolPref("devtools.inspector.three-pane-enabled", true);
 | |
| 
 | |
| // Disable this preference to reduce exceptions related to pending `listWorkers`
 | |
| // requests occuring after a process is created/destroyed. See Bug 1620983.
 | |
| Services.prefs.setBoolPref("dom.ipc.processPrelaunch.enabled", false);
 | |
| 
 | |
| // Disable this preference to capture async stacks across all locations during
 | |
| // DevTools mochitests. Async stacks provide very valuable information to debug
 | |
| // intermittents, but come with a performance overhead, which is why they are
 | |
| // only captured in Debuggees by default.
 | |
| Services.prefs.setBoolPref(
 | |
|   "javascript.options.asyncstack_capture_debuggee_only",
 | |
|   false
 | |
| );
 | |
| 
 | |
| // On some Linux platforms, prefers-reduced-motion is enabled, which would
 | |
| // trigger the notification to be displayed in the toolbox. Dismiss the message
 | |
| // by default.
 | |
| Services.prefs.setBoolPref(
 | |
|   "devtools.inspector.simple-highlighters.message-dismissed",
 | |
|   true
 | |
| );
 | |
| 
 | |
| registerCleanupFunction(() => {
 | |
|   Services.prefs.clearUserPref("devtools.dump.emit");
 | |
|   Services.prefs.clearUserPref("devtools.inspector.three-pane-enabled");
 | |
|   Services.prefs.clearUserPref("dom.ipc.processPrelaunch.enabled");
 | |
|   Services.prefs.clearUserPref("devtools.toolbox.host");
 | |
|   Services.prefs.clearUserPref("devtools.toolbox.previousHost");
 | |
|   Services.prefs.clearUserPref("devtools.toolbox.splitconsoleEnabled");
 | |
|   Services.prefs.clearUserPref("devtools.toolbox.splitconsoleHeight");
 | |
|   Services.prefs.clearUserPref(
 | |
|     "javascript.options.asyncstack_capture_debuggee_only"
 | |
|   );
 | |
|   Services.prefs.clearUserPref(
 | |
|     "devtools.inspector.simple-highlighters.message-dismissed"
 | |
|   );
 | |
| });
 | |
| 
 | |
| var {
 | |
|   BrowserConsoleManager,
 | |
| } = require("resource://devtools/client/webconsole/browser-console-manager.js");
 | |
| 
 | |
| registerCleanupFunction(async function cleanup() {
 | |
|   // Closing the browser console if there's one
 | |
|   const browserConsole = BrowserConsoleManager.getBrowserConsole();
 | |
|   if (browserConsole) {
 | |
|     await safeCloseBrowserConsole({ clearOutput: true });
 | |
|   }
 | |
| 
 | |
|   // Close any tab opened by the test.
 | |
|   // There should be only one tab opened by default when firefox starts the test.
 | |
|   while (isMochitest && gBrowser.tabs.length > 1) {
 | |
|     await closeTabAndToolbox(gBrowser.selectedTab);
 | |
|   }
 | |
| 
 | |
|   // Note that this will run before cleanup functions registered by tests or other head.js files.
 | |
|   // So all connections must be cleaned up by the test when the test ends,
 | |
|   // before the harness starts invoking the cleanup functions
 | |
|   await waitForTick();
 | |
| 
 | |
|   // All connections must be cleaned up by the test when the test ends.
 | |
|   const {
 | |
|     DevToolsServer,
 | |
|   } = require("resource://devtools/server/devtools-server.js");
 | |
|   ok(
 | |
|     !DevToolsServer.hasConnection(),
 | |
|     "The main process DevToolsServer has no pending connection when the test ends"
 | |
|   );
 | |
|   // If there is still open connection, close all of them so that following tests
 | |
|   // could pass.
 | |
|   if (DevToolsServer.hasConnection()) {
 | |
|     for (const conn of Object.values(DevToolsServer._connections)) {
 | |
|       conn.close();
 | |
|     }
 | |
|   }
 | |
| });
 | |
| 
 | |
| async function safeCloseBrowserConsole({ clearOutput = false } = {}) {
 | |
|   const hud = BrowserConsoleManager.getBrowserConsole();
 | |
|   if (!hud) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (clearOutput) {
 | |
|     info("Clear the browser console output");
 | |
|     const { ui } = hud;
 | |
|     const promises = [ui.once("messages-cleared")];
 | |
|     // If there's an object inspector, we need to wait for the actors to be released.
 | |
|     if (ui.outputNode.querySelector(".object-inspector")) {
 | |
|       promises.push(ui.once("fronts-released"));
 | |
|     }
 | |
|     await ui.clearOutput(true);
 | |
|     await Promise.all(promises);
 | |
|     info("Browser console cleared");
 | |
|   }
 | |
| 
 | |
|   info("Wait for all Browser Console targets to be attached");
 | |
|   // It might happen that waitForAllTargetsToBeAttached does not resolve, so we set a
 | |
|   // timeout of 1s before closing
 | |
|   await Promise.race([
 | |
|     waitForAllTargetsToBeAttached(hud.commands.targetCommand),
 | |
|     wait(1000),
 | |
|   ]);
 | |
| 
 | |
|   info("Close the Browser Console");
 | |
|   await BrowserConsoleManager.closeBrowserConsole();
 | |
|   info("Browser Console closed");
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Observer code to register the test actor in every DevTools server which
 | |
|  * starts registering its own actors.
 | |
|  *
 | |
|  * We require immediately the highlighter test actor file, because it will force to load and
 | |
|  * register the front and the spec for HighlighterTestActor. Normally specs and fronts are
 | |
|  * in separate files registered in specs/index.js. But here to simplify the
 | |
|  * setup everything is in the same file and we force to load it here.
 | |
|  *
 | |
|  * DevToolsServer will emit "devtools-server-initialized" after finishing its
 | |
|  * initialization. We watch this observable to add our custom actor.
 | |
|  *
 | |
|  * As a single test may create several DevTools servers, we keep the observer
 | |
|  * alive until the test ends.
 | |
|  *
 | |
|  * To avoid leaks, the observer needs to be removed at the end of each test.
 | |
|  * The test cleanup will send the async message "remove-devtools-highlightertestactor-observer",
 | |
|  * we listen to this message to cleanup the observer.
 | |
|  */
 | |
| function highlighterTestActorBootstrap() {
 | |
|   /* eslint-env mozilla/process-script */
 | |
|   const HIGHLIGHTER_TEST_ACTOR_URL =
 | |
|     "chrome://mochitests/content/browser/devtools/client/shared/test/highlighter-test-actor.js";
 | |
| 
 | |
|   const { require: _require } = ChromeUtils.importESModule(
 | |
|     "resource://devtools/shared/loader/Loader.sys.mjs"
 | |
|   );
 | |
|   _require(HIGHLIGHTER_TEST_ACTOR_URL);
 | |
| 
 | |
|   const actorRegistryObserver = subject => {
 | |
|     const actorRegistry = subject.wrappedJSObject;
 | |
|     actorRegistry.registerModule(HIGHLIGHTER_TEST_ACTOR_URL, {
 | |
|       prefix: "highlighterTest",
 | |
|       constructor: "HighlighterTestActor",
 | |
|       type: { target: true },
 | |
|     });
 | |
|   };
 | |
|   Services.obs.addObserver(
 | |
|     actorRegistryObserver,
 | |
|     "devtools-server-initialized"
 | |
|   );
 | |
| 
 | |
|   const unloadListener = () => {
 | |
|     Services.cpmm.removeMessageListener(
 | |
|       "remove-devtools-testactor-observer",
 | |
|       unloadListener
 | |
|     );
 | |
|     Services.obs.removeObserver(
 | |
|       actorRegistryObserver,
 | |
|       "devtools-server-initialized"
 | |
|     );
 | |
|   };
 | |
|   Services.cpmm.addMessageListener(
 | |
|     "remove-devtools-testactor-observer",
 | |
|     unloadListener
 | |
|   );
 | |
| }
 | |
| 
 | |
| if (isMochitest) {
 | |
|   const highlighterTestActorBootstrapScript =
 | |
|     "data:,(" + highlighterTestActorBootstrap + ")()";
 | |
|   Services.ppmm.loadProcessScript(
 | |
|     highlighterTestActorBootstrapScript,
 | |
|     // Load this script in all processes (created or to be created)
 | |
|     true
 | |
|   );
 | |
| 
 | |
|   registerCleanupFunction(() => {
 | |
|     Services.ppmm.broadcastAsyncMessage("remove-devtools-testactor-observer");
 | |
|     Services.ppmm.removeDelayedProcessScript(
 | |
|       highlighterTestActorBootstrapScript
 | |
|     );
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Spawn an instance of the highlighter test actor for the given toolbox
 | |
|  *
 | |
|  * @param {Toolbox} toolbox
 | |
|  * @param {Object} options
 | |
|  * @param {Function} options.target: Optional target to get the highlighterTestFront for.
 | |
|  *        If not provided, the top level target will be used.
 | |
|  * @returns {HighlighterTestFront}
 | |
|  */
 | |
| async function getHighlighterTestFront(toolbox, { target } = {}) {
 | |
|   // Loading the Inspector panel in order to overwrite the TestActor getter for the
 | |
|   // highlighter instance with a method that points to the currently visible
 | |
|   // Box Model Highlighter managed by the Inspector panel.
 | |
|   const inspector = await toolbox.loadTool("inspector");
 | |
| 
 | |
|   const highlighterTestFront = await (target || toolbox.target).getFront(
 | |
|     "highlighterTest"
 | |
|   );
 | |
|   // Override the highligher getter with a method to return the active box model
 | |
|   // highlighter. Adaptation for multi-process scenarios where there can be multiple
 | |
|   // highlighters, one per process.
 | |
|   highlighterTestFront.highlighter = () => {
 | |
|     return inspector.highlighters.getActiveHighlighter(
 | |
|       inspector.highlighters.TYPES.BOXMODEL
 | |
|     );
 | |
|   };
 | |
|   return highlighterTestFront;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Spawn an instance of the highlighter test actor for the given tab, when we need the
 | |
|  * highlighter test front before opening or without a toolbox.
 | |
|  *
 | |
|  * @param {Tab} tab
 | |
|  * @returns {HighlighterTestFront}
 | |
|  */
 | |
| async function getHighlighterTestFrontWithoutToolbox(tab) {
 | |
|   const commands = await CommandsFactory.forTab(tab);
 | |
|   // Initialize the TargetCommands which require some async stuff to be done
 | |
|   // before being fully ready. This will define the `targetCommand.targetFront` attribute.
 | |
|   await commands.targetCommand.startListening();
 | |
| 
 | |
|   const targetFront = commands.targetCommand.targetFront;
 | |
|   return targetFront.getFront("highlighterTest");
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Returns a Promise that resolves when all the targets are fully attached.
 | |
|  *
 | |
|  * @param {TargetCommand} targetCommand
 | |
|  */
 | |
| function waitForAllTargetsToBeAttached(targetCommand) {
 | |
|   return Promise.allSettled(
 | |
|     targetCommand
 | |
|       .getAllTargets(targetCommand.ALL_TYPES)
 | |
|       .map(target => target.initialized)
 | |
|   );
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Add a new test tab in the browser and load the given url.
 | |
|  * @param {String} url The url to be loaded in the new tab
 | |
|  * @param {Object} options Object with various optional fields:
 | |
|  *   - {Boolean} background If true, open the tab in background
 | |
|  *   - {ChromeWindow} window Firefox top level window we should use to open the tab
 | |
|  *   - {Number} userContextId The userContextId of the tab.
 | |
|  *   - {String} preferredRemoteType
 | |
|  *   - {Boolean} waitForLoad Wait for the page in the new tab to load. (Defaults to true.)
 | |
|  * @return a promise that resolves to the tab object when the url is loaded
 | |
|  */
 | |
| async function addTab(url, options = {}) {
 | |
|   info("Adding a new tab with URL: " + url);
 | |
| 
 | |
|   const {
 | |
|     background = false,
 | |
|     userContextId,
 | |
|     preferredRemoteType,
 | |
|     waitForLoad = true,
 | |
|   } = options;
 | |
|   const { gBrowser } = options.window ? options.window : window;
 | |
| 
 | |
|   const tab = BrowserTestUtils.addTab(gBrowser, url, {
 | |
|     userContextId,
 | |
|     preferredRemoteType,
 | |
|   });
 | |
| 
 | |
|   if (!background) {
 | |
|     gBrowser.selectedTab = tab;
 | |
|   }
 | |
| 
 | |
|   if (waitForLoad) {
 | |
|     await BrowserTestUtils.browserLoaded(tab.linkedBrowser);
 | |
|     // Waiting for presShell helps with test timeouts in webrender platforms.
 | |
|     await waitForPresShell(tab.linkedBrowser);
 | |
|     info("Tab added and finished loading");
 | |
|   } else {
 | |
|     info("Tab added");
 | |
|   }
 | |
| 
 | |
|   return tab;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Remove the given tab.
 | |
|  * @param {Object} tab The tab to be removed.
 | |
|  * @return Promise<undefined> resolved when the tab is successfully removed.
 | |
|  */
 | |
| async function removeTab(tab) {
 | |
|   info("Removing tab.");
 | |
| 
 | |
|   const { gBrowser } = tab.ownerDocument.defaultView;
 | |
|   const onClose = once(gBrowser.tabContainer, "TabClose");
 | |
|   gBrowser.removeTab(tab);
 | |
|   await onClose;
 | |
| 
 | |
|   info("Tab removed and finished closing");
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Alias for navigateTo which will reuse the current URI of the provided browser
 | |
|  * to trigger a navigation.
 | |
|  */
 | |
| async function reloadBrowser({
 | |
|   browser = gBrowser.selectedBrowser,
 | |
|   isErrorPage = false,
 | |
|   waitForLoad = true,
 | |
| } = {}) {
 | |
|   return navigateTo(browser.currentURI.spec, {
 | |
|     browser,
 | |
|     isErrorPage,
 | |
|     waitForLoad,
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Navigate the currently selected tab to a new URL and wait for it to load.
 | |
|  * Also wait for the toolbox to attach to the new target, if we navigated
 | |
|  * to a new process.
 | |
|  *
 | |
|  * @param {String} url The url to be loaded in the current tab.
 | |
|  * @param {JSON} options Optional dictionary object with the following keys:
 | |
|  *        - {XULBrowser} browser
 | |
|  *          The browser element which should navigate. Defaults to the selected
 | |
|  *          browser.
 | |
|  *        - {Boolean} isErrorPage
 | |
|  *          You may pass `true` if the URL is an error page. Otherwise
 | |
|  *          BrowserTestUtils.browserLoaded will wait for 'load' event, which
 | |
|  *          never fires for error pages.
 | |
|  *        - {Boolean} waitForLoad
 | |
|  *          You may pass `false` if the page load is expected to be blocked by
 | |
|  *          a script or a breakpoint.
 | |
|  *
 | |
|  * @return a promise that resolves when the page has fully loaded.
 | |
|  */
 | |
| async function navigateTo(
 | |
|   uri,
 | |
|   {
 | |
|     browser = gBrowser.selectedBrowser,
 | |
|     isErrorPage = false,
 | |
|     waitForLoad = true,
 | |
|   } = {}
 | |
| ) {
 | |
|   const waitForDevToolsReload = await watchForDevToolsReload(browser, {
 | |
|     isErrorPage,
 | |
|     waitForLoad,
 | |
|   });
 | |
| 
 | |
|   uri = uri.replaceAll("\n", "");
 | |
|   info(`Navigating to "${uri}"`);
 | |
| 
 | |
|   const onBrowserLoaded = BrowserTestUtils.browserLoaded(
 | |
|     browser,
 | |
|     // includeSubFrames
 | |
|     false,
 | |
|     // resolve on this specific page to load (if null, it would be any page load)
 | |
|     loadedUrl => {
 | |
|       // loadedUrl is encoded, while uri might not be.
 | |
|       return loadedUrl === uri || decodeURI(loadedUrl) === uri;
 | |
|     },
 | |
|     isErrorPage
 | |
|   );
 | |
| 
 | |
|   // if we're navigating to the same page we're already on, use reloadTab instead as the
 | |
|   // behavior slightly differs from loadURI (e.g. scroll position isn't keps with the latter).
 | |
|   if (uri === browser.currentURI.spec) {
 | |
|     gBrowser.reloadTab(gBrowser.getTabForBrowser(browser));
 | |
|   } else {
 | |
|     BrowserTestUtils.loadURIString(browser, uri);
 | |
|   }
 | |
| 
 | |
|   if (waitForLoad) {
 | |
|     info(`Waiting for page to be loaded…`);
 | |
|     await onBrowserLoaded;
 | |
|     info(`→ page loaded`);
 | |
|   }
 | |
| 
 | |
|   await waitForDevToolsReload();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * This method should be used to watch for completion of any browser navigation
 | |
|  * performed with a DevTools UI.
 | |
|  *
 | |
|  * It should watch for:
 | |
|  * - Toolbox reload
 | |
|  * - Toolbox commands reload
 | |
|  * - RDM reload
 | |
|  * - RDM commands reload
 | |
|  *
 | |
|  * And it should work both for target switching or old-style navigations.
 | |
|  *
 | |
|  * This method, similarly to all the other watch* navigation methods in this file,
 | |
|  * is async but returns another method which should be called after the navigation
 | |
|  * is done. Browser navigation might be monitored differently depending on the
 | |
|  * situation, so it's up to the caller to handle it as needed.
 | |
|  *
 | |
|  * Typically, this would be used as follows:
 | |
|  * ```
 | |
|  *   async function someNavigationHelper(browser) {
 | |
|  *     const waitForDevToolsFn = await watchForDevToolsReload(browser);
 | |
|  *
 | |
|  *     // This step should wait for the load to be completed from the browser's
 | |
|  *     // point of view, so that waitForDevToolsFn can compare pIds, browsing
 | |
|  *     // contexts etc... and check if we should expect a target switch
 | |
|  *     await performBrowserNavigation(browser);
 | |
|  *
 | |
|  *     await waitForDevToolsFn();
 | |
|  *   }
 | |
|  * ```
 | |
|  */
 | |
| async function watchForDevToolsReload(
 | |
|   browser,
 | |
|   { isErrorPage = false, waitForLoad = true } = {}
 | |
| ) {
 | |
|   const waitForToolboxReload = await _watchForToolboxReload(browser, {
 | |
|     isErrorPage,
 | |
|     waitForLoad,
 | |
|   });
 | |
|   const waitForResponsiveReload = await _watchForResponsiveReload(browser, {
 | |
|     isErrorPage,
 | |
|     waitForLoad,
 | |
|   });
 | |
| 
 | |
|   return async function () {
 | |
|     info("Wait for the toolbox to reload");
 | |
|     await waitForToolboxReload();
 | |
| 
 | |
|     info("Wait for Responsive UI to reload");
 | |
|     await waitForResponsiveReload();
 | |
|   };
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Start watching for the toolbox reload to be completed:
 | |
|  * - watch for the toolbox's commands to be fully reloaded
 | |
|  * - watch for the toolbox's current panel to be reloaded
 | |
|  */
 | |
| async function _watchForToolboxReload(
 | |
|   browser,
 | |
|   { isErrorPage, waitForLoad } = {}
 | |
| ) {
 | |
|   const tab = gBrowser.getTabForBrowser(browser);
 | |
| 
 | |
|   const toolbox = gDevTools.getToolboxForTab(tab);
 | |
| 
 | |
|   if (!toolbox) {
 | |
|     // No toolbox to wait for
 | |
|     return function () {};
 | |
|   }
 | |
| 
 | |
|   const waitForCurrentPanelReload = watchForCurrentPanelReload(toolbox);
 | |
|   const waitForToolboxCommandsReload = await watchForCommandsReload(
 | |
|     toolbox.commands,
 | |
|     { isErrorPage, waitForLoad }
 | |
|   );
 | |
|   const checkTargetSwitching = await watchForTargetSwitching(
 | |
|     toolbox.commands,
 | |
|     browser
 | |
|   );
 | |
| 
 | |
|   return async function () {
 | |
|     const isTargetSwitching = checkTargetSwitching();
 | |
| 
 | |
|     info(`Waiting for toolbox commands to be reloaded…`);
 | |
|     await waitForToolboxCommandsReload(isTargetSwitching);
 | |
| 
 | |
|     // TODO: We should wait for all loaded panels to reload here, because some
 | |
|     // of them might still perform background updates.
 | |
|     if (waitForCurrentPanelReload) {
 | |
|       info(`Waiting for ${toolbox.currentToolId} to be reloaded…`);
 | |
|       await waitForCurrentPanelReload();
 | |
|       info(`→ panel reloaded`);
 | |
|     }
 | |
|   };
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Start watching for Responsive UI (RDM) reload to be completed:
 | |
|  * - watch for the Responsive UI's commands to be fully reloaded
 | |
|  * - watch for the Responsive UI's target switch to be done
 | |
|  */
 | |
| async function _watchForResponsiveReload(
 | |
|   browser,
 | |
|   { isErrorPage, waitForLoad } = {}
 | |
| ) {
 | |
|   const tab = gBrowser.getTabForBrowser(browser);
 | |
|   const ui = ResponsiveUIManager.getResponsiveUIForTab(tab);
 | |
| 
 | |
|   if (!ui) {
 | |
|     // No responsive UI to wait for
 | |
|     return function () {};
 | |
|   }
 | |
| 
 | |
|   const onResponsiveTargetSwitch = ui.once("responsive-ui-target-switch-done");
 | |
|   const waitForResponsiveCommandsReload = await watchForCommandsReload(
 | |
|     ui.commands,
 | |
|     { isErrorPage, waitForLoad }
 | |
|   );
 | |
|   const checkTargetSwitching = await watchForTargetSwitching(
 | |
|     ui.commands,
 | |
|     browser
 | |
|   );
 | |
| 
 | |
|   return async function () {
 | |
|     const isTargetSwitching = checkTargetSwitching();
 | |
| 
 | |
|     info(`Waiting for responsive ui commands to be reloaded…`);
 | |
|     await waitForResponsiveCommandsReload(isTargetSwitching);
 | |
| 
 | |
|     if (isTargetSwitching) {
 | |
|       await onResponsiveTargetSwitch;
 | |
|     }
 | |
|   };
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Watch for the current panel selected in the provided toolbox to be reloaded.
 | |
|  * Some panels implement custom events that should be expected for every reload.
 | |
|  *
 | |
|  * Note about returning a method instead of a promise:
 | |
|  * In general this pattern is useful so that we can check if a target switch
 | |
|  * occurred or not, and decide which events to listen for. So far no panel is
 | |
|  * behaving differently whether there was a target switch or not. But to remain
 | |
|  * consistent with other watch* methods we still return a function here.
 | |
|  *
 | |
|  * @param {Toolbox}
 | |
|  *        The Toolbox instance which is going to experience a reload
 | |
|  * @return {function} An async method to be called and awaited after the reload
 | |
|  *         started. Will return `null` for panels which don't implement any
 | |
|  *         specific reload event.
 | |
|  */
 | |
| function watchForCurrentPanelReload(toolbox) {
 | |
|   return _watchForPanelReload(toolbox, toolbox.currentToolId);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Watch for all the panels loaded in the provided toolbox to be reloaded.
 | |
|  * Some panels implement custom events that should be expected for every reload.
 | |
|  *
 | |
|  * Note about returning a method instead of a promise:
 | |
|  * See comment for watchForCurrentPanelReload
 | |
|  *
 | |
|  * @param {Toolbox}
 | |
|  *        The Toolbox instance which is going to experience a reload
 | |
|  * @return {function} An async method to be called and awaited after the reload
 | |
|  *         started.
 | |
|  */
 | |
| function watchForLoadedPanelsReload(toolbox) {
 | |
|   const waitForPanels = [];
 | |
|   for (const [id] of toolbox.getToolPanels()) {
 | |
|     // Store a watcher method for each panel already loaded.
 | |
|     waitForPanels.push(_watchForPanelReload(toolbox, id));
 | |
|   }
 | |
| 
 | |
|   return function () {
 | |
|     return Promise.all(
 | |
|       waitForPanels.map(async watchPanel => {
 | |
|         // Wait for all panels to be reloaded.
 | |
|         if (watchPanel) {
 | |
|           await watchPanel();
 | |
|         }
 | |
|       })
 | |
|     );
 | |
|   };
 | |
| }
 | |
| 
 | |
| function _watchForPanelReload(toolbox, toolId) {
 | |
|   const panel = toolbox.getPanel(toolId);
 | |
| 
 | |
|   if (toolId == "inspector") {
 | |
|     const markuploaded = panel.once("markuploaded");
 | |
|     const onNewRoot = panel.once("new-root");
 | |
|     const onUpdated = panel.once("inspector-updated");
 | |
|     const onReloaded = panel.once("reloaded");
 | |
| 
 | |
|     return async function () {
 | |
|       info("Waiting for markup view to load after navigation.");
 | |
|       await markuploaded;
 | |
| 
 | |
|       info("Waiting for new root.");
 | |
|       await onNewRoot;
 | |
| 
 | |
|       info("Waiting for inspector to update after new-root event.");
 | |
|       await onUpdated;
 | |
| 
 | |
|       info("Waiting for inspector updates after page reload");
 | |
|       await onReloaded;
 | |
|     };
 | |
|   } else if (
 | |
|     ["netmonitor", "accessibility", "webconsole", "jsdebugger"].includes(toolId)
 | |
|   ) {
 | |
|     const onReloaded = panel.once("reloaded");
 | |
|     return async function () {
 | |
|       info(`Waiting for ${toolId} updates after page reload`);
 | |
|       await onReloaded;
 | |
|     };
 | |
|   }
 | |
|   return null;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Watch for a Commands instance to be reloaded after a navigation.
 | |
|  *
 | |
|  * As for other navigation watch* methods, this should be called before the
 | |
|  * navigation starts, and the function it returns should be called after the
 | |
|  * navigation is done from a Browser point of view.
 | |
|  *
 | |
|  * !!! The wait function expects a `isTargetSwitching` argument to be provided,
 | |
|  * which needs to be monitored using watchForTargetSwitching !!!
 | |
|  */
 | |
| async function watchForCommandsReload(
 | |
|   commands,
 | |
|   { isErrorPage = false, waitForLoad = true } = {}
 | |
| ) {
 | |
|   // If we're switching origins, we need to wait for the 'switched-target'
 | |
|   // event to make sure everything is ready.
 | |
|   // Navigating from/to pages loaded in the parent process, like about:robots,
 | |
|   // also spawn new targets.
 | |
|   // (If target switching is disabled, the toolbox will reboot)
 | |
|   const onTargetSwitched = commands.targetCommand.once("switched-target");
 | |
| 
 | |
|   // Wait until we received a page load resource:
 | |
|   // - dom-complete if we can wait for a full page load
 | |
|   // - dom-loading otherwise
 | |
|   // This allows to wait for page load for consumers calling directly
 | |
|   // waitForDevTools instead of navigateTo/reloadBrowser.
 | |
|   // This is also useful as an alternative to target switching, when no target
 | |
|   // switch is supposed to happen.
 | |
|   const waitForCompleteLoad = waitForLoad && !isErrorPage;
 | |
|   const documentEventName = waitForCompleteLoad
 | |
|     ? "dom-complete"
 | |
|     : "dom-loading";
 | |
| 
 | |
|   const { onResource: onTopLevelDomEvent } =
 | |
|     await commands.resourceCommand.waitForNextResource(
 | |
|       commands.resourceCommand.TYPES.DOCUMENT_EVENT,
 | |
|       {
 | |
|         ignoreExistingResources: true,
 | |
|         predicate: resource =>
 | |
|           resource.targetFront.isTopLevel &&
 | |
|           resource.name === documentEventName,
 | |
|       }
 | |
|     );
 | |
| 
 | |
|   return async function (isTargetSwitching) {
 | |
|     if (typeof isTargetSwitching === "undefined") {
 | |
|       throw new Error("isTargetSwitching was not provided to the wait method");
 | |
|     }
 | |
| 
 | |
|     if (isTargetSwitching) {
 | |
|       info(`Waiting for target switch…`);
 | |
|       await onTargetSwitched;
 | |
|       info(`→ switched-target emitted`);
 | |
|     }
 | |
| 
 | |
|     info(`Waiting for '${documentEventName}' resource…`);
 | |
|     await onTopLevelDomEvent;
 | |
|     info(`→ '${documentEventName}' resource emitted`);
 | |
| 
 | |
|     return isTargetSwitching;
 | |
|   };
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Watch if an upcoming navigation will trigger a target switching, for the
 | |
|  * provided Commands instance and the provided Browser.
 | |
|  *
 | |
|  * As for other navigation watch* methods, this should be called before the
 | |
|  * navigation starts, and the function it returns should be called after the
 | |
|  * navigation is done from a Browser point of view.
 | |
|  */
 | |
| async function watchForTargetSwitching(commands, browser) {
 | |
|   browser = browser || gBrowser.selectedBrowser;
 | |
|   const currentPID = browser.browsingContext.currentWindowGlobal.osPid;
 | |
|   const currentBrowsingContextID = browser.browsingContext.id;
 | |
| 
 | |
|   // If the current top-level target follows the window global lifecycle, a
 | |
|   // target switch will occur regardless of process changes.
 | |
|   const targetFollowsWindowLifecycle =
 | |
|     commands.targetCommand.targetFront.targetForm.followWindowGlobalLifeCycle;
 | |
| 
 | |
|   return function () {
 | |
|     // Compare the PIDs (and not the toolbox's targets) as PIDs are updated also immediately,
 | |
|     // while target may be updated slightly later.
 | |
|     const switchedProcess =
 | |
|       currentPID !== browser.browsingContext.currentWindowGlobal.osPid;
 | |
|     const switchedBrowsingContext =
 | |
|       currentBrowsingContextID !== browser.browsingContext.id;
 | |
| 
 | |
|     return (
 | |
|       targetFollowsWindowLifecycle || switchedProcess || switchedBrowsingContext
 | |
|     );
 | |
|   };
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Create a Target for the provided tab and attach to it before resolving.
 | |
|  * This should only be used for tests which don't involve the frontend or a
 | |
|  * toolbox. Typically, retrieving the target and attaching to it should be
 | |
|  * handled at framework level when a Toolbox is used.
 | |
|  *
 | |
|  * @param {XULTab} tab
 | |
|  *        The tab for which a target should be created.
 | |
|  * @return {WindowGlobalTargetFront} The attached target front.
 | |
|  */
 | |
| async function createAndAttachTargetForTab(tab) {
 | |
|   info("Creating and attaching to a local tab target");
 | |
| 
 | |
|   const commands = await CommandsFactory.forTab(tab);
 | |
| 
 | |
|   // Initialize the TargetCommands which require some async stuff to be done
 | |
|   // before being fully ready. This will define the `targetCommand.targetFront` attribute.
 | |
|   await commands.targetCommand.startListening();
 | |
| 
 | |
|   const target = commands.targetCommand.targetFront;
 | |
|   return target;
 | |
| }
 | |
| 
 | |
| function isFissionEnabled() {
 | |
|   return SpecialPowers.useRemoteSubframes;
 | |
| }
 | |
| 
 | |
| function isEveryFrameTargetEnabled() {
 | |
|   return Services.prefs.getBoolPref(
 | |
|     "devtools.every-frame-target.enabled",
 | |
|     false
 | |
|   );
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Open the inspector in a tab with given URL.
 | |
|  * @param {string} url  The URL to open.
 | |
|  * @param {String} hostType Optional hostType, as defined in Toolbox.HostType
 | |
|  * @return A promise that is resolved once the tab and inspector have loaded
 | |
|  *         with an object: { tab, toolbox, inspector, highlighterTestFront }.
 | |
|  */
 | |
| async function openInspectorForURL(url, hostType) {
 | |
|   const tab = await addTab(url);
 | |
|   const { inspector, toolbox, highlighterTestFront } = await openInspector(
 | |
|     hostType
 | |
|   );
 | |
|   return { tab, inspector, toolbox, highlighterTestFront };
 | |
| }
 | |
| 
 | |
| function getActiveInspector() {
 | |
|   const toolbox = gDevTools.getToolboxForTab(gBrowser.selectedTab);
 | |
|   return toolbox.getPanel("inspector");
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Simulate a key event from an electron key shortcut string:
 | |
|  * https://github.com/electron/electron/blob/master/docs/api/accelerator.md
 | |
|  *
 | |
|  * @param {String} key
 | |
|  * @param {DOMWindow} target
 | |
|  *        Optional window where to fire the key event
 | |
|  */
 | |
| function synthesizeKeyShortcut(key, target) {
 | |
|   // parseElectronKey requires any window, just to access `KeyboardEvent`
 | |
|   const window = Services.appShell.hiddenDOMWindow;
 | |
|   const shortcut = KeyShortcuts.parseElectronKey(window, key);
 | |
|   const keyEvent = {
 | |
|     altKey: shortcut.alt,
 | |
|     ctrlKey: shortcut.ctrl,
 | |
|     metaKey: shortcut.meta,
 | |
|     shiftKey: shortcut.shift,
 | |
|   };
 | |
|   if (shortcut.keyCode) {
 | |
|     keyEvent.keyCode = shortcut.keyCode;
 | |
|   }
 | |
| 
 | |
|   info("Synthesizing key shortcut: " + key);
 | |
|   EventUtils.synthesizeKey(shortcut.key || "", keyEvent, target);
 | |
| }
 | |
| 
 | |
| var waitForTime = DevToolsUtils.waitForTime;
 | |
| 
 | |
| /**
 | |
|  * Wait for a tick.
 | |
|  * @return {Promise}
 | |
|  */
 | |
| function waitForTick() {
 | |
|   return new Promise(resolve => DevToolsUtils.executeSoon(resolve));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * This shouldn't be used in the tests, but is useful when writing new tests or
 | |
|  * debugging existing tests in order to introduce delays in the test steps
 | |
|  *
 | |
|  * @param {Number} ms
 | |
|  *        The time to wait
 | |
|  * @return A promise that resolves when the time is passed
 | |
|  */
 | |
| function wait(ms) {
 | |
|   return new Promise(resolve => {
 | |
|     setTimeout(resolve, ms);
 | |
|     info("Waiting " + ms / 1000 + " seconds.");
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Wait for a predicate to return a result.
 | |
|  *
 | |
|  * @param function condition
 | |
|  *        Invoked once in a while until it returns a truthy value. This should be an
 | |
|  *        idempotent function, since we have to run it a second time after it returns
 | |
|  *        true in order to return the value.
 | |
|  * @param string message [optional]
 | |
|  *        A message to output if the condition fails.
 | |
|  * @param number interval [optional]
 | |
|  *        How often the predicate is invoked, in milliseconds.
 | |
|  *        Can be set globally for a test via `waitFor.overrideIntervalForTestFile = someNumber;`.
 | |
|  * @param number maxTries [optional]
 | |
|  *        How many times the predicate is invoked before timing out.
 | |
|  *        Can be set globally for a test via `waitFor.overrideMaxTriesForTestFile = someNumber;`.
 | |
|  * @return object
 | |
|  *         A promise that is resolved with the result of the condition.
 | |
|  */
 | |
| async function waitFor(condition, message = "", interval = 10, maxTries = 500) {
 | |
|   // Update interval & maxTries if overrides are defined on the waitFor object.
 | |
|   interval =
 | |
|     typeof waitFor.overrideIntervalForTestFile !== "undefined"
 | |
|       ? waitFor.overrideIntervalForTestFile
 | |
|       : interval;
 | |
|   maxTries =
 | |
|     typeof waitFor.overrideMaxTriesForTestFile !== "undefined"
 | |
|       ? waitFor.overrideMaxTriesForTestFile
 | |
|       : maxTries;
 | |
| 
 | |
|   try {
 | |
|     const value = await BrowserTestUtils.waitForCondition(
 | |
|       condition,
 | |
|       message,
 | |
|       interval,
 | |
|       maxTries
 | |
|     );
 | |
|     return value;
 | |
|   } catch (e) {
 | |
|     const errorMessage = `Failed waitFor(): ${message} \nFailed condition: ${condition} \nException Message: ${e}`;
 | |
|     throw new Error(errorMessage);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Wait for eventName on target to be delivered a number of times.
 | |
|  *
 | |
|  * @param {Object} target
 | |
|  *        An observable object that either supports on/off or
 | |
|  *        addEventListener/removeEventListener
 | |
|  * @param {String} eventName
 | |
|  * @param {Number} numTimes
 | |
|  *        Number of deliveries to wait for.
 | |
|  * @param {Boolean} useCapture
 | |
|  *        Optional, for addEventListener/removeEventListener
 | |
|  * @return A promise that resolves when the event has been handled
 | |
|  */
 | |
| function waitForNEvents(target, eventName, numTimes, useCapture = false) {
 | |
|   info("Waiting for event: '" + eventName + "' on " + target + ".");
 | |
| 
 | |
|   let count = 0;
 | |
| 
 | |
|   return new Promise(resolve => {
 | |
|     for (const [add, remove] of [
 | |
|       ["on", "off"],
 | |
|       ["addEventListener", "removeEventListener"],
 | |
|       ["addListener", "removeListener"],
 | |
|       ["addMessageListener", "removeMessageListener"],
 | |
|     ]) {
 | |
|       if (add in target && remove in target) {
 | |
|         target[add](
 | |
|           eventName,
 | |
|           function onEvent(...args) {
 | |
|             if (typeof info === "function") {
 | |
|               info("Got event: '" + eventName + "' on " + target + ".");
 | |
|             }
 | |
| 
 | |
|             if (++count == numTimes) {
 | |
|               target[remove](eventName, onEvent, useCapture);
 | |
|               resolve(...args);
 | |
|             }
 | |
|           },
 | |
|           useCapture
 | |
|         );
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Wait for DOM change on target.
 | |
|  *
 | |
|  * @param {Object} target
 | |
|  *        The Node on which to observe DOM mutations.
 | |
|  * @param {String} selector
 | |
|  *        Given a selector to watch whether the expected element is changed
 | |
|  *        on target.
 | |
|  * @param {Number} expectedLength
 | |
|  *        Optional, default set to 1
 | |
|  *        There may be more than one element match an array match the selector,
 | |
|  *        give an expected length to wait for more elements.
 | |
|  * @return A promise that resolves when the event has been handled
 | |
|  */
 | |
| function waitForDOM(target, selector, expectedLength = 1) {
 | |
|   return new Promise(resolve => {
 | |
|     const observer = new MutationObserver(mutations => {
 | |
|       mutations.forEach(mutation => {
 | |
|         const elements = mutation.target.querySelectorAll(selector);
 | |
| 
 | |
|         if (elements.length === expectedLength) {
 | |
|           observer.disconnect();
 | |
|           resolve(elements);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     observer.observe(target, {
 | |
|       attributes: true,
 | |
|       childList: true,
 | |
|       subtree: true,
 | |
|     });
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Wait for eventName on target.
 | |
|  *
 | |
|  * @param {Object} target
 | |
|  *        An observable object that either supports on/off or
 | |
|  *        addEventListener/removeEventListener
 | |
|  * @param {String} eventName
 | |
|  * @param {Boolean} useCapture
 | |
|  *        Optional, for addEventListener/removeEventListener
 | |
|  * @return A promise that resolves when the event has been handled
 | |
|  */
 | |
| function once(target, eventName, useCapture = false) {
 | |
|   return waitForNEvents(target, eventName, 1, useCapture);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Some tests may need to import one or more of the test helper scripts.
 | |
|  * A test helper script is simply a js file that contains common test code that
 | |
|  * is either not common-enough to be in head.js, or that is located in a
 | |
|  * separate directory.
 | |
|  * The script will be loaded synchronously and in the test's scope.
 | |
|  * @param {String} filePath The file path, relative to the current directory.
 | |
|  *                 Examples:
 | |
|  *                 - "helper_attributes_test_runner.js"
 | |
|  */
 | |
| function loadHelperScript(filePath) {
 | |
|   const testDir = gTestPath.substr(0, gTestPath.lastIndexOf("/"));
 | |
|   Services.scriptloader.loadSubScript(testDir + "/" + filePath, this);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Open the toolbox in a given tab.
 | |
|  * @param {XULNode} tab The tab the toolbox should be opened in.
 | |
|  * @param {String} toolId Optional. The ID of the tool to be selected.
 | |
|  * @param {String} hostType Optional. The type of toolbox host to be used.
 | |
|  * @return {Promise} Resolves with the toolbox, when it has been opened.
 | |
|  */
 | |
| async function openToolboxForTab(tab, toolId, hostType) {
 | |
|   info("Opening the toolbox");
 | |
| 
 | |
|   // Check if the toolbox is already loaded.
 | |
|   let toolbox = gDevTools.getToolboxForTab(tab);
 | |
|   if (toolbox) {
 | |
|     if (!toolId || (toolId && toolbox.getPanel(toolId))) {
 | |
|       info("Toolbox is already opened");
 | |
|       return toolbox;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // If not, load it now.
 | |
|   toolbox = await gDevTools.showToolboxForTab(tab, { toolId, hostType });
 | |
| 
 | |
|   // Make sure that the toolbox frame is focused.
 | |
|   await new Promise(resolve => waitForFocus(resolve, toolbox.win));
 | |
| 
 | |
|   info("Toolbox opened and focused");
 | |
| 
 | |
|   return toolbox;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Add a new tab and open the toolbox in it.
 | |
|  * @param {String} url The URL for the tab to be opened.
 | |
|  * @param {String} toolId Optional. The ID of the tool to be selected.
 | |
|  * @param {String} hostType Optional. The type of toolbox host to be used.
 | |
|  * @return {Promise} Resolves when the tab has been added, loaded and the
 | |
|  * toolbox has been opened. Resolves to the toolbox.
 | |
|  */
 | |
| async function openNewTabAndToolbox(url, toolId, hostType) {
 | |
|   const tab = await addTab(url);
 | |
|   return openToolboxForTab(tab, toolId, hostType);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Close a tab and if necessary, the toolbox that belongs to it
 | |
|  * @param {Tab} tab The tab to close.
 | |
|  * @return {Promise} Resolves when the toolbox and tab have been destroyed and
 | |
|  * closed.
 | |
|  */
 | |
| async function closeTabAndToolbox(tab = gBrowser.selectedTab) {
 | |
|   if (gDevTools.hasToolboxForTab(tab)) {
 | |
|     await gDevTools.closeToolboxForTab(tab);
 | |
|   }
 | |
| 
 | |
|   await removeTab(tab);
 | |
| 
 | |
|   await new Promise(resolve => setTimeout(resolve, 0));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Close a toolbox and the current tab.
 | |
|  * @param {Toolbox} toolbox The toolbox to close.
 | |
|  * @return {Promise} Resolves when the toolbox and tab have been destroyed and
 | |
|  * closed.
 | |
|  */
 | |
| async function closeToolboxAndTab(toolbox) {
 | |
|   await toolbox.destroy();
 | |
|   await removeTab(gBrowser.selectedTab);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Waits until a predicate returns true.
 | |
|  *
 | |
|  * @param function predicate
 | |
|  *        Invoked once in a while until it returns true.
 | |
|  * @param number interval [optional]
 | |
|  *        How often the predicate is invoked, in milliseconds.
 | |
|  */
 | |
| function waitUntil(predicate, interval = 10) {
 | |
|   if (predicate()) {
 | |
|     return Promise.resolve(true);
 | |
|   }
 | |
|   return new Promise(resolve => {
 | |
|     setTimeout(function () {
 | |
|       waitUntil(predicate, interval).then(() => resolve(true));
 | |
|     }, interval);
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Variant of waitUntil that accepts a predicate returning a promise.
 | |
|  */
 | |
| async function asyncWaitUntil(predicate, interval = 10) {
 | |
|   let success = await predicate();
 | |
|   while (!success) {
 | |
|     // Wait for X milliseconds.
 | |
|     await new Promise(resolve => setTimeout(resolve, interval));
 | |
|     // Test the predicate again.
 | |
|     success = await predicate();
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Wait for a context menu popup to open.
 | |
|  *
 | |
|  * @param Element popup
 | |
|  *        The XUL popup you expect to open.
 | |
|  * @param Element button
 | |
|  *        The button/element that receives the contextmenu event. This is
 | |
|  *        expected to open the popup.
 | |
|  * @param function onShown
 | |
|  *        Function to invoke on popupshown event.
 | |
|  * @param function onHidden
 | |
|  *        Function to invoke on popuphidden event.
 | |
|  * @return object
 | |
|  *         A Promise object that is resolved after the popuphidden event
 | |
|  *         callback is invoked.
 | |
|  */
 | |
| function waitForContextMenu(popup, button, onShown, onHidden) {
 | |
|   return new Promise(resolve => {
 | |
|     function onPopupShown() {
 | |
|       info("onPopupShown");
 | |
|       popup.removeEventListener("popupshown", onPopupShown);
 | |
| 
 | |
|       onShown && onShown();
 | |
| 
 | |
|       // Use executeSoon() to get out of the popupshown event.
 | |
|       popup.addEventListener("popuphidden", onPopupHidden);
 | |
|       DevToolsUtils.executeSoon(() => popup.hidePopup());
 | |
|     }
 | |
|     function onPopupHidden() {
 | |
|       info("onPopupHidden");
 | |
|       popup.removeEventListener("popuphidden", onPopupHidden);
 | |
| 
 | |
|       onHidden && onHidden();
 | |
| 
 | |
|       resolve(popup);
 | |
|     }
 | |
| 
 | |
|     popup.addEventListener("popupshown", onPopupShown);
 | |
| 
 | |
|     info("wait for the context menu to open");
 | |
|     synthesizeContextMenuEvent(button);
 | |
|   });
 | |
| }
 | |
| 
 | |
| function synthesizeContextMenuEvent(el) {
 | |
|   el.scrollIntoView();
 | |
|   const eventDetails = { type: "contextmenu", button: 2 };
 | |
|   EventUtils.synthesizeMouse(
 | |
|     el,
 | |
|     5,
 | |
|     2,
 | |
|     eventDetails,
 | |
|     el.ownerDocument.defaultView
 | |
|   );
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Promise wrapper around SimpleTest.waitForClipboard
 | |
|  */
 | |
| function waitForClipboardPromise(setup, expected) {
 | |
|   return new Promise((resolve, reject) => {
 | |
|     SimpleTest.waitForClipboard(expected, setup, resolve, reject);
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Simple helper to push a temporary preference. Wrapper on SpecialPowers
 | |
|  * pushPrefEnv that returns a promise resolving when the preferences have been
 | |
|  * updated.
 | |
|  *
 | |
|  * @param {String} preferenceName
 | |
|  *        The name of the preference to updated
 | |
|  * @param {} value
 | |
|  *        The preference value, type can vary
 | |
|  * @return {Promise} resolves when the preferences have been updated
 | |
|  */
 | |
| function pushPref(preferenceName, value) {
 | |
|   const options = { set: [[preferenceName, value]] };
 | |
|   return SpecialPowers.pushPrefEnv(options);
 | |
| }
 | |
| 
 | |
| async function closeToolbox() {
 | |
|   await gDevTools.closeToolboxForTab(gBrowser.selectedTab);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Clean the logical clipboard content. This method only clears the OS clipboard on
 | |
|  * Windows (see Bug 666254).
 | |
|  */
 | |
| function emptyClipboard() {
 | |
|   const clipboard = Services.clipboard;
 | |
|   clipboard.emptyClipboard(clipboard.kGlobalClipboard);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Check if the current operating system is Windows.
 | |
|  */
 | |
| function isWindows() {
 | |
|   return Services.appinfo.OS === "WINNT";
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Create an HTTP server that can be used to simulate custom requests within
 | |
|  * a test.  It is automatically cleaned up when the test ends, so no need to
 | |
|  * call `destroy`.
 | |
|  *
 | |
|  * See https://developer.mozilla.org/en-US/docs/Httpd.js/HTTP_server_for_unit_tests
 | |
|  * for more information about how to register handlers.
 | |
|  *
 | |
|  * The server can be accessed like:
 | |
|  *
 | |
|  *   const server = createTestHTTPServer();
 | |
|  *   let url = "http://localhost: " + server.identity.primaryPort + "/path";
 | |
|  * @returns {HttpServer}
 | |
|  */
 | |
| function createTestHTTPServer() {
 | |
|   const { HttpServer } = ChromeUtils.importESModule(
 | |
|     "resource://testing-common/httpd.sys.mjs"
 | |
|   );
 | |
|   const server = new HttpServer();
 | |
| 
 | |
|   registerCleanupFunction(async function cleanup() {
 | |
|     await new Promise(resolve => server.stop(resolve));
 | |
|   });
 | |
| 
 | |
|   server.start(-1);
 | |
|   return server;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Register an actor in the content process of the current tab.
 | |
|  *
 | |
|  * Calling ActorRegistry.registerModule only registers the actor in the current process.
 | |
|  * As all test scripts are ran in the parent process, it is only registered here.
 | |
|  * This function helps register them in the content process used for the current tab.
 | |
|  *
 | |
|  * @param {string} url
 | |
|  *        Actor module URL or absolute require path
 | |
|  * @param {json} options
 | |
|  *        Arguments to be passed to DevToolsServer.registerModule
 | |
|  */
 | |
| async function registerActorInContentProcess(url, options) {
 | |
|   function convertChromeToFile(uri) {
 | |
|     return Cc["@mozilla.org/chrome/chrome-registry;1"]
 | |
|       .getService(Ci.nsIChromeRegistry)
 | |
|       .convertChromeURL(Services.io.newURI(uri)).spec;
 | |
|   }
 | |
|   // chrome://mochitests URI is registered only in the parent process, so convert these
 | |
|   // URLs to file:// one in order to work in the content processes
 | |
|   url = url.startsWith("chrome://mochitests") ? convertChromeToFile(url) : url;
 | |
|   return SpecialPowers.spawn(
 | |
|     gBrowser.selectedBrowser,
 | |
|     [{ url, options }],
 | |
|     args => {
 | |
|       // eslint-disable-next-line no-shadow
 | |
|       const { require } = ChromeUtils.importESModule(
 | |
|         "resource://devtools/shared/loader/Loader.sys.mjs"
 | |
|       );
 | |
|       const {
 | |
|         ActorRegistry,
 | |
|       } = require("resource://devtools/server/actors/utils/actor-registry.js");
 | |
|       ActorRegistry.registerModule(args.url, args.options);
 | |
|     }
 | |
|   );
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Move the provided Window to the provided left, top coordinates and wait for
 | |
|  * the window position to be updated.
 | |
|  */
 | |
| async function moveWindowTo(win, left, top) {
 | |
|   // Check that the expected coordinates are within the window available area.
 | |
|   left = Math.max(win.screen.availLeft, left);
 | |
|   left = Math.min(win.screen.width, left);
 | |
|   top = Math.max(win.screen.availTop, top);
 | |
|   top = Math.min(win.screen.height, top);
 | |
| 
 | |
|   info(`Moving window to {${left}, ${top}}`);
 | |
|   win.moveTo(left, top);
 | |
| 
 | |
|   // Bug 1600809: window move/resize can be async on Linux sometimes.
 | |
|   // Wait so that the anchor's position is correctly measured.
 | |
|   return waitUntil(() => {
 | |
|     info(
 | |
|       `Wait for window screenLeft and screenTop to be updated: (${win.screenLeft}, ${win.screenTop})`
 | |
|     );
 | |
|     return win.screenLeft === left && win.screenTop === top;
 | |
|   });
 | |
| }
 | |
| 
 | |
| function getCurrentTestFilePath() {
 | |
|   return gTestPath.replace("chrome://mochitests/content/browser/", "");
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Unregister all registered service workers.
 | |
|  *
 | |
|  * @param {DevToolsClient} client
 | |
|  */
 | |
| async function unregisterAllServiceWorkers(client) {
 | |
|   info("Wait until all workers have a valid registrationFront");
 | |
|   let workers;
 | |
|   await asyncWaitUntil(async function () {
 | |
|     workers = await client.mainRoot.listAllWorkers();
 | |
|     const allWorkersRegistered = workers.service.every(
 | |
|       worker => !!worker.registrationFront
 | |
|     );
 | |
|     return allWorkersRegistered;
 | |
|   });
 | |
| 
 | |
|   info("Unregister all service workers");
 | |
|   const promises = [];
 | |
|   for (const worker of workers.service) {
 | |
|     promises.push(worker.registrationFront.unregister());
 | |
|   }
 | |
|   await Promise.all(promises);
 | |
| }
 | |
| 
 | |
| /**********************
 | |
|  * Screenshot helpers *
 | |
|  **********************/
 | |
| 
 | |
| /**
 | |
|  * Returns an object containing the r,g and b colors of the provided image at
 | |
|  * the passed position
 | |
|  *
 | |
|  * @param {Image} image
 | |
|  * @param {Int} x
 | |
|  * @param {Int} y
 | |
|  * @returns Object with the following properties:
 | |
|  *           - {Int} r: The red component of the pixel
 | |
|  *           - {Int} g: The green component of the pixel
 | |
|  *           - {Int} b: The blue component of the pixel
 | |
|  */
 | |
| function colorAt(image, x, y) {
 | |
|   // Create a test canvas element.
 | |
|   const HTML_NS = "http://www.w3.org/1999/xhtml";
 | |
|   const canvas = document.createElementNS(HTML_NS, "canvas");
 | |
|   canvas.width = image.width;
 | |
|   canvas.height = image.height;
 | |
| 
 | |
|   // Draw the image in the canvas
 | |
|   const context = canvas.getContext("2d");
 | |
|   context.drawImage(image, 0, 0, image.width, image.height);
 | |
| 
 | |
|   // Return the color found at the provided x,y coordinates as a "r, g, b" string.
 | |
|   const [r, g, b] = context.getImageData(x, y, 1, 1).data;
 | |
|   return { r, g, b };
 | |
| }
 | |
| 
 | |
| let allDownloads = [];
 | |
| /**
 | |
|  * Returns a Promise that resolves when a new screenshot is available in the download folder.
 | |
|  *
 | |
|  * @param {Object} [options]
 | |
|  * @param {Boolean} options.isWindowPrivate: Set to true if the window from which the screenshot
 | |
|  *                  is taken is a private window. This will ensure that we check that the
 | |
|  *                  screenshot appears in the private window, not the non-private one (See Bug 1783373)
 | |
|  */
 | |
| async function waitUntilScreenshot({ isWindowPrivate = false } = {}) {
 | |
|   const { Downloads } = ChromeUtils.importESModule(
 | |
|     "resource://gre/modules/Downloads.sys.mjs"
 | |
|   );
 | |
|   const list = await Downloads.getList(Downloads.ALL);
 | |
| 
 | |
|   return new Promise(function (resolve) {
 | |
|     const view = {
 | |
|       onDownloadAdded: async download => {
 | |
|         await download.whenSucceeded();
 | |
|         if (allDownloads.includes(download)) {
 | |
|           return;
 | |
|         }
 | |
| 
 | |
|         is(
 | |
|           !!download.source.isPrivate,
 | |
|           isWindowPrivate,
 | |
|           `The download occured in the expected${
 | |
|             isWindowPrivate ? " private" : ""
 | |
|           } window`
 | |
|         );
 | |
| 
 | |
|         allDownloads.push(download);
 | |
|         resolve(download.target.path);
 | |
|         list.removeView(view);
 | |
|       },
 | |
|     };
 | |
| 
 | |
|     list.addView(view);
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Clear all the download references.
 | |
|  */
 | |
| async function resetDownloads() {
 | |
|   info("Reset downloads");
 | |
|   const { Downloads } = ChromeUtils.importESModule(
 | |
|     "resource://gre/modules/Downloads.sys.mjs"
 | |
|   );
 | |
|   const downloadList = await Downloads.getList(Downloads.ALL);
 | |
|   const downloads = await downloadList.getAll();
 | |
|   for (const download of downloads) {
 | |
|     downloadList.remove(download);
 | |
|     await download.finalize(true);
 | |
|   }
 | |
|   allDownloads = [];
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Return a screenshot of the currently selected node in the inspector (using the internal
 | |
|  * Inspector#screenshotNode method).
 | |
|  *
 | |
|  * @param {Inspector} inspector
 | |
|  * @returns {Image}
 | |
|  */
 | |
| async function takeNodeScreenshot(inspector) {
 | |
|   // Cleanup all downloads at the end of the test.
 | |
|   registerCleanupFunction(resetDownloads);
 | |
| 
 | |
|   info(
 | |
|     "Call screenshotNode() and wait until the screenshot is found in the Downloads"
 | |
|   );
 | |
|   const whenScreenshotSucceeded = waitUntilScreenshot();
 | |
|   inspector.screenshotNode();
 | |
|   const filePath = await whenScreenshotSucceeded;
 | |
| 
 | |
|   info("Create an image using the downloaded fileas source");
 | |
|   const image = new Image();
 | |
|   const onImageLoad = once(image, "load");
 | |
|   image.src = PathUtils.toFileURI(filePath);
 | |
|   await onImageLoad;
 | |
| 
 | |
|   info("Remove the downloaded screenshot file");
 | |
|   await IOUtils.remove(filePath);
 | |
| 
 | |
|   // See intermittent Bug 1508435. Even after removing the file, tests still manage to
 | |
|   // reuse files from the previous test if they have the same name. Since our file name
 | |
|   // is based on a timestamp that has "second" precision, wait for one second to make sure
 | |
|   // screenshots will have different names.
 | |
|   info(
 | |
|     "Wait for one second to make sure future screenshots will use a different name"
 | |
|   );
 | |
|   await new Promise(r => setTimeout(r, 1000));
 | |
| 
 | |
|   return image;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Check that the provided image has the expected width, height, and color.
 | |
|  * NOTE: This test assumes that the image is only made of a single color and will only
 | |
|  * check one pixel.
 | |
|  */
 | |
| async function assertSingleColorScreenshotImage(
 | |
|   image,
 | |
|   width,
 | |
|   height,
 | |
|   { r, g, b }
 | |
| ) {
 | |
|   info(`Assert ${image.src} content`);
 | |
|   const ratio = await SpecialPowers.spawn(
 | |
|     gBrowser.selectedBrowser,
 | |
|     [],
 | |
|     () => content.wrappedJSObject.devicePixelRatio
 | |
|   );
 | |
| 
 | |
|   is(
 | |
|     image.width,
 | |
|     ratio * width,
 | |
|     `node screenshot has the expected width (dpr = ${ratio})`
 | |
|   );
 | |
|   is(
 | |
|     image.height,
 | |
|     height * ratio,
 | |
|     `node screenshot has the expected height (dpr = ${ratio})`
 | |
|   );
 | |
| 
 | |
|   const color = colorAt(image, 0, 0);
 | |
|   is(color.r, r, "node screenshot has the expected red component");
 | |
|   is(color.g, g, "node screenshot has the expected green component");
 | |
|   is(color.b, b, "node screenshot has the expected blue component");
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Check that the provided image has the expected color at a given position
 | |
|  */
 | |
| function checkImageColorAt({ image, x = 0, y, expectedColor, label }) {
 | |
|   const color = colorAt(image, x, y);
 | |
|   is(`rgb(${Object.values(color).join(", ")})`, expectedColor, label);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Wait until the store has reached a state that matches the predicate.
 | |
|  * @param Store store
 | |
|  *        The Redux store being used.
 | |
|  * @param function predicate
 | |
|  *        A function that returns true when the store has reached the expected
 | |
|  *        state.
 | |
|  * @return Promise
 | |
|  *         Resolved once the store reaches the expected state.
 | |
|  */
 | |
| function waitUntilState(store, predicate) {
 | |
|   return new Promise(resolve => {
 | |
|     const unsubscribe = store.subscribe(check);
 | |
| 
 | |
|     info(`Waiting for state predicate "${predicate}"`);
 | |
|     function check() {
 | |
|       if (predicate(store.getState())) {
 | |
|         info(`Found state predicate "${predicate}"`);
 | |
|         unsubscribe();
 | |
|         resolve();
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Fire the check immediately in case the action has already occurred
 | |
|     check();
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Wait for a specific action type to be dispatched.
 | |
|  *
 | |
|  * If the action is async and defines a `status` property, this helper will wait
 | |
|  * for the status to reach either "error" or "done".
 | |
|  *
 | |
|  * @param {Object} store
 | |
|  *        Redux store where the action should be dispatched.
 | |
|  * @param {String} actionType
 | |
|  *        The actionType to wait for.
 | |
|  * @param {Number} repeat
 | |
|  *        Optional, number of time the action is expected to be dispatched.
 | |
|  *        Defaults to 1
 | |
|  * @return {Promise}
 | |
|  */
 | |
| function waitForDispatch(store, actionType, repeat = 1) {
 | |
|   let count = 0;
 | |
|   return new Promise(resolve => {
 | |
|     store.dispatch({
 | |
|       type: "@@service/waitUntil",
 | |
|       predicate: action => {
 | |
|         const isDone =
 | |
|           !action.status ||
 | |
|           action.status === "done" ||
 | |
|           action.status === "error";
 | |
| 
 | |
|         if (action.type === actionType && isDone && ++count == repeat) {
 | |
|           return true;
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|       },
 | |
|       run: (dispatch, getState, action) => {
 | |
|         resolve(action);
 | |
|       },
 | |
|     });
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Retrieve a browsing context in nested frames.
 | |
|  *
 | |
|  * @param {BrowsingContext|XULBrowser} browsingContext
 | |
|  *        The topmost browsing context under which we should search for the
 | |
|  *        browsing context.
 | |
|  * @param {Array<String>} selectors
 | |
|  *        Array of CSS selectors that form a path to a specific nested frame.
 | |
|  * @return {BrowsingContext} The nested browsing context.
 | |
|  */
 | |
| async function getBrowsingContextInFrames(browsingContext, selectors) {
 | |
|   let context = browsingContext;
 | |
| 
 | |
|   if (!Array.isArray(selectors)) {
 | |
|     throw new Error(
 | |
|       "getBrowsingContextInFrames called with an invalid selectors argument"
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   if (selectors.length === 0) {
 | |
|     throw new Error(
 | |
|       "getBrowsingContextInFrames called with an empty selectors array"
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   const clonedSelectors = [...selectors];
 | |
|   while (clonedSelectors.length) {
 | |
|     const selector = clonedSelectors.shift();
 | |
|     context = await SpecialPowers.spawn(context, [selector], _selector => {
 | |
|       return content.document.querySelector(_selector).browsingContext;
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   return context;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Synthesize a mouse event on an element, after ensuring that it is visible
 | |
|  * in the viewport.
 | |
|  *
 | |
|  * @param {String|Array} selector: The node selector to get the node target for the event.
 | |
|  *        To target an element in a specific iframe, pass an array of CSS selectors
 | |
|  *        (e.g. ["iframe", ".el-in-iframe"])
 | |
|  * @param {number} x
 | |
|  * @param {number} y
 | |
|  * @param {object} options: Options that will be passed to BrowserTestUtils.synthesizeMouse
 | |
|  */
 | |
| async function safeSynthesizeMouseEventInContentPage(
 | |
|   selector,
 | |
|   x,
 | |
|   y,
 | |
|   options = {}
 | |
| ) {
 | |
|   let context = gBrowser.selectedBrowser.browsingContext;
 | |
| 
 | |
|   // If an array of selector is passed, we need to retrieve the context in which the node
 | |
|   // lives in.
 | |
|   if (Array.isArray(selector)) {
 | |
|     if (selector.length === 1) {
 | |
|       selector = selector[0];
 | |
|     } else {
 | |
|       context = await getBrowsingContextInFrames(
 | |
|         context,
 | |
|         // only pass the iframe path
 | |
|         selector.slice(0, -1)
 | |
|       );
 | |
|       // retrieve the last item of the selector, which should be the one for the node we want.
 | |
|       selector = selector.at(-1);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   await scrollContentPageNodeIntoView(context, selector);
 | |
|   BrowserTestUtils.synthesizeMouse(selector, x, y, options, context);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Synthesize a mouse event at the center of an element, after ensuring that it is visible
 | |
|  * in the viewport.
 | |
|  *
 | |
|  * @param {String|Array} selector: The node selector to get the node target for the event.
 | |
|  *        To target an element in a specific iframe, pass an array of CSS selectors
 | |
|  *        (e.g. ["iframe", ".el-in-iframe"])
 | |
|  * @param {object} options: Options that will be passed to BrowserTestUtils.synthesizeMouse
 | |
|  */
 | |
| async function safeSynthesizeMouseEventAtCenterInContentPage(
 | |
|   selector,
 | |
|   options = {}
 | |
| ) {
 | |
|   let context = gBrowser.selectedBrowser.browsingContext;
 | |
| 
 | |
|   // If an array of selector is passed, we need to retrieve the context in which the node
 | |
|   // lives in.
 | |
|   if (Array.isArray(selector)) {
 | |
|     if (selector.length === 1) {
 | |
|       selector = selector[0];
 | |
|     } else {
 | |
|       context = await getBrowsingContextInFrames(
 | |
|         context,
 | |
|         // only pass the iframe path
 | |
|         selector.slice(0, -1)
 | |
|       );
 | |
|       // retrieve the last item of the selector, which should be the one for the node we want.
 | |
|       selector = selector.at(-1);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   await scrollContentPageNodeIntoView(context, selector);
 | |
|   BrowserTestUtils.synthesizeMouseAtCenter(selector, options, context);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Scroll into view an element in the content page matching the passed selector
 | |
|  *
 | |
|  * @param {BrowsingContext} browsingContext: The browsing context the element lives in.
 | |
|  * @param {String} selector: The node selector to get the node to scroll into view
 | |
|  * @returns {Promise}
 | |
|  */
 | |
| function scrollContentPageNodeIntoView(browsingContext, selector) {
 | |
|   return SpecialPowers.spawn(
 | |
|     browsingContext,
 | |
|     [selector],
 | |
|     function (innerSelector) {
 | |
|       const node =
 | |
|         content.wrappedJSObject.document.querySelector(innerSelector);
 | |
|       node.scrollIntoView();
 | |
|     }
 | |
|   );
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Change the zoom level of the selected page.
 | |
|  *
 | |
|  * @param {Number} zoomLevel
 | |
|  */
 | |
| function setContentPageZoomLevel(zoomLevel) {
 | |
|   gBrowser.selectedBrowser.fullZoom = zoomLevel;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Wait for the next DOCUMENT_EVENT dom-complete resource on a top-level target
 | |
|  *
 | |
|  * @param {Object} commands
 | |
|  * @return {Promise<Object>}
 | |
|  *         Return a promise which resolves once we fully settle the resource listener.
 | |
|  *         You should await for its resolution before doing the action which may fire
 | |
|  *         your resource.
 | |
|  *         This promise will resolve with an object containing a `onDomCompleteResource` property,
 | |
|  *         which is also a promise, that will resolve once a "top-level" DOCUMENT_EVENT dom-complete
 | |
|  *         is received.
 | |
|  */
 | |
| async function waitForNextTopLevelDomCompleteResource(commands) {
 | |
|   const { onResource: onDomCompleteResource } =
 | |
|     await commands.resourceCommand.waitForNextResource(
 | |
|       commands.resourceCommand.TYPES.DOCUMENT_EVENT,
 | |
|       {
 | |
|         ignoreExistingResources: true,
 | |
|         predicate: resource =>
 | |
|           resource.name === "dom-complete" && resource.targetFront.isTopLevel,
 | |
|       }
 | |
|     );
 | |
|   return { onDomCompleteResource };
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Wait for the provided context to have a valid presShell. This can be useful
 | |
|  * for tests which try to create popup panels or interact with the document very
 | |
|  * early.
 | |
|  *
 | |
|  * @param {BrowsingContext} context
 | |
|  **/
 | |
| function waitForPresShell(context) {
 | |
|   return SpecialPowers.spawn(context, [], async () => {
 | |
|     const winUtils = SpecialPowers.getDOMWindowUtils(content);
 | |
|     await ContentTaskUtils.waitForCondition(() => {
 | |
|       try {
 | |
|         return !!winUtils.getPresShellId();
 | |
|       } catch (e) {
 | |
|         return false;
 | |
|       }
 | |
|     }, "Waiting for a valid presShell");
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * In tests using Fluent localization, it is preferable to match DOM elements using
 | |
|  * a message ID rather than the raw string as:
 | |
|  *
 | |
|  *  1. It allows testing infrastructure to be multilingual if needed.
 | |
|  *  2. It isolates the tests from localization changes.
 | |
|  *
 | |
|  * @param {Array<string>} resourceIds A list of .ftl files to load.
 | |
|  * @returns {(id: string, args?: Record<string, FluentVariable>) => string}
 | |
|  */
 | |
| async function getFluentStringHelper(resourceIds) {
 | |
|   const locales = Services.locale.appLocalesAsBCP47;
 | |
|   const generator = L10nRegistry.getInstance().generateBundles(
 | |
|     locales,
 | |
|     resourceIds
 | |
|   );
 | |
| 
 | |
|   const bundles = [];
 | |
|   for await (const bundle of generator) {
 | |
|     bundles.push(bundle);
 | |
|   }
 | |
| 
 | |
|   const reactLocalization = new FluentReact.ReactLocalization(bundles);
 | |
| 
 | |
|   /**
 | |
|    * Get the string from a message id. It throws when the message is not found.
 | |
|    *
 | |
|    * @param {string} id
 | |
|    * @param {string} attributeName: attribute name if you need to access a specific attribute
 | |
|    *                 defined in the fluent string, e.g. setting "title" for this param
 | |
|    *                 will retrieve the `title` string in
 | |
|    *                    compatibility-issue-browsers-list =
 | |
|    *                      .title = This is the title
 | |
|    * @param {Record<string, FluentVariable>} [args] optional
 | |
|    * @returns {string}
 | |
|    */
 | |
|   return (id, attributeName, args) => {
 | |
|     let string;
 | |
| 
 | |
|     if (!attributeName) {
 | |
|       string = reactLocalization.getString(id, args);
 | |
|     } else {
 | |
|       for (const bundle of reactLocalization.bundles) {
 | |
|         const msg = bundle.getMessage(id);
 | |
|         if (msg?.attributes[attributeName]) {
 | |
|           string = bundle.formatPattern(
 | |
|             msg.attributes[attributeName],
 | |
|             args,
 | |
|             []
 | |
|           );
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (!string) {
 | |
|       throw new Error(
 | |
|         `Could not find a string for "${id}"${
 | |
|           attributeName ? ` and attribute "${attributeName}")` : ""
 | |
|         }. Was the correct resource bundle loaded?`
 | |
|       );
 | |
|     }
 | |
|     return string;
 | |
|   };
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Open responsive design mode for the given tab.
 | |
|  */
 | |
| async function openRDM(tab, { waitForDeviceList = true } = {}) {
 | |
|   info("Opening responsive design mode");
 | |
|   const manager = ResponsiveUIManager;
 | |
|   const ui = await manager.openIfNeeded(tab.ownerGlobal, tab, {
 | |
|     trigger: "test",
 | |
|   });
 | |
|   info("Responsive design mode opened");
 | |
| 
 | |
|   await ResponsiveMessageHelper.wait(ui.toolWindow, "post-init");
 | |
|   info("Responsive design initialized");
 | |
| 
 | |
|   await waitForRDMLoaded(ui, { waitForDeviceList });
 | |
| 
 | |
|   return { ui, manager };
 | |
| }
 | |
| 
 | |
| async function waitForRDMLoaded(ui, { waitForDeviceList = true } = {}) {
 | |
|   // Always wait for the viewport to be added.
 | |
|   const { store } = ui.toolWindow;
 | |
|   await waitUntilState(store, state => state.viewports.length == 1);
 | |
| 
 | |
|   if (waitForDeviceList) {
 | |
|     // Wait until the device list has been loaded.
 | |
|     await waitUntilState(
 | |
|       store,
 | |
|       state => state.devices.listState == localTypes.loadableState.LOADED
 | |
|     );
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Close responsive design mode for the given tab.
 | |
|  */
 | |
| async function closeRDM(tab, options) {
 | |
|   info("Closing responsive design mode");
 | |
|   const manager = ResponsiveUIManager;
 | |
|   await manager.closeIfNeeded(tab.ownerGlobal, tab, options);
 | |
|   info("Responsive design mode closed");
 | |
| }
 | |
| 
 | |
| function getInputStream(data) {
 | |
|   const BufferStream = Components.Constructor(
 | |
|     "@mozilla.org/io/arraybuffer-input-stream;1",
 | |
|     "nsIArrayBufferInputStream",
 | |
|     "setData"
 | |
|   );
 | |
|   const buffer = new TextEncoder().encode(data).buffer;
 | |
|   return new BufferStream(buffer, 0, buffer.byteLength);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Wait for a specific target to have been fully processed by targetCommand.
 | |
|  *
 | |
|  * @param {Commands} commands
 | |
|  *        The commands instance
 | |
|  * @param {Function} isExpectedTargetFn
 | |
|  *        Predicate which will be called with a target front argument. Should
 | |
|  *        return true if the target front is the expected one, false otherwise.
 | |
|  * @return {Promise}
 | |
|  *         Promise which resolves when a target matching `isExpectedTargetFn`
 | |
|  *         has been processed by targetCommand.
 | |
|  */
 | |
| function waitForTargetProcessed(commands, isExpectedTargetFn) {
 | |
|   return new Promise(resolve => {
 | |
|     const onProcessed = targetFront => {
 | |
|       try {
 | |
|         if (isExpectedTargetFn(targetFront)) {
 | |
|           commands.targetCommand.off("processed-available-target", onProcessed);
 | |
|           resolve();
 | |
|         }
 | |
|       } catch {
 | |
|         // Ignore errors from isExpectedTargetFn.
 | |
|       }
 | |
|     };
 | |
| 
 | |
|     commands.targetCommand.on("processed-available-target", onProcessed);
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Instantiate a HTTP Server that serves files from a given test folder.
 | |
|  * The test folder should be made of multiple sub folder named: v1, v2, v3,...
 | |
|  * We will serve the content from one of these sub folder
 | |
|  * and switch to the next one, each time `httpServer.switchToNextVersion()`
 | |
|  * is called.
 | |
|  *
 | |
|  * @return Object Test server with two functions:
 | |
|  *   - urlFor(path)
 | |
|  *     Returns the absolute url for a given file.
 | |
|  *   - switchToNextVersion()
 | |
|  *     Start serving files from the next available sub folder.
 | |
|  *   - backToFirstVersion()
 | |
|  *     When running more than one test, helps restart from the first folder.
 | |
|  */
 | |
| function createVersionizedHttpTestServer(testFolderName) {
 | |
|   const httpServer = createTestHTTPServer();
 | |
| 
 | |
|   let currentVersion = 1;
 | |
| 
 | |
|   httpServer.registerPrefixHandler("/", async (request, response) => {
 | |
|     response.processAsync();
 | |
|     response.setStatusLine(request.httpVersion, 200, "OK");
 | |
|     if (request.path.endsWith(".js")) {
 | |
|       response.setHeader("Content-Type", "application/javascript");
 | |
|     } else if (request.path.endsWith(".js.map")) {
 | |
|       response.setHeader("Content-Type", "application/json");
 | |
|     }
 | |
|     if (request.path == "/" || request.path.endsWith(".html")) {
 | |
|       response.setHeader("Content-Type", "text/html");
 | |
|     }
 | |
|     // If a query string is passed, lookup with a matching file, if available
 | |
|     // The '?' is replaced by '.'
 | |
|     let fetchResponse;
 | |
| 
 | |
|     if (request.queryString) {
 | |
|       const url = `${URL_ROOT_SSL}${testFolderName}/v${currentVersion}${request.path}.${request.queryString}`;
 | |
|       try {
 | |
|         fetchResponse = await fetch(url);
 | |
|         // Log this only if the request succeed
 | |
|         info(`[test-http-server] serving: ${url}`);
 | |
|       } catch (e) {
 | |
|         // Ignore any error and proceed without the query string
 | |
|         fetchResponse = null;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (!fetchResponse) {
 | |
|       const url = `${URL_ROOT_SSL}${testFolderName}/v${currentVersion}${request.path}`;
 | |
|       info(`[test-http-server] serving: ${url}`);
 | |
|       fetchResponse = await fetch(url);
 | |
|     }
 | |
| 
 | |
|     // Ensure forwarding the response headers generated by the other http server
 | |
|     // (this can be especially useful when query .sjs files)
 | |
|     for (const [name, value] of fetchResponse.headers.entries()) {
 | |
|       response.setHeader(name, value);
 | |
|     }
 | |
| 
 | |
|     // Override cache settings so that versionized requests are never cached
 | |
|     // and we get brand new content for any request.
 | |
|     response.setHeader("Cache-Control", "no-store");
 | |
| 
 | |
|     const text = await fetchResponse.text();
 | |
|     response.write(text);
 | |
|     response.finish();
 | |
|   });
 | |
| 
 | |
|   return {
 | |
|     switchToNextVersion() {
 | |
|       currentVersion++;
 | |
|     },
 | |
|     backToFirstVersion() {
 | |
|       currentVersion = 1;
 | |
|     },
 | |
|     urlFor(path) {
 | |
|       const port = httpServer.identity.primaryPort;
 | |
|       return `http://localhost:${port}/${path}`;
 | |
|     },
 | |
|   };
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Fake clicking a link and return the URL we would have navigated to.
 | |
|  * This function should be used to check external links since we can't access
 | |
|  * network in tests.
 | |
|  * This can also be used to test that a click will not be fired.
 | |
|  *
 | |
|  * @param ElementNode element
 | |
|  *        The <a> element we want to simulate click on.
 | |
|  * @returns Promise
 | |
|  *          A Promise that is resolved when the link click simulation occured or
 | |
|  *          when the click is not dispatched.
 | |
|  *          The promise resolves with an object that holds the following properties
 | |
|  *          - link: url of the link or null(if event not fired)
 | |
|  *          - where: "tab" if tab is active or "tabshifted" if tab is inactive
 | |
|  *            or null(if event not fired)
 | |
|  */
 | |
| function simulateLinkClick(element) {
 | |
|   const browserWindow = Services.wm.getMostRecentWindow(
 | |
|     gDevTools.chromeWindowType
 | |
|   );
 | |
| 
 | |
|   const onOpenLink = new Promise(resolve => {
 | |
|     const openLinkIn = (link, where) => resolve({ link, where });
 | |
|     sinon.replace(browserWindow, "openTrustedLinkIn", openLinkIn);
 | |
|     sinon.replace(browserWindow, "openWebLinkIn", openLinkIn);
 | |
|   });
 | |
| 
 | |
|   element.click();
 | |
| 
 | |
|   // Declare a timeout Promise that we can use to make sure spied methods were not called.
 | |
|   const onTimeout = new Promise(function (resolve) {
 | |
|     setTimeout(() => {
 | |
|       resolve({ link: null, where: null });
 | |
|     }, 1000);
 | |
|   });
 | |
| 
 | |
|   const raceResult = Promise.race([onOpenLink, onTimeout]);
 | |
|   sinon.restore();
 | |
|   return raceResult;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Since the MDN data is updated frequently, it might happen that the properties used in
 | |
|  * this test are not in the dataset anymore/now have URLs.
 | |
|  * This function will return properties in the dataset that don't have MDN url so you
 | |
|  * can easily find a replacement.
 | |
|  */
 | |
| function logCssCompatDataPropertiesWithoutMDNUrl() {
 | |
|   const cssPropertiesCompatData = require("resource://devtools/shared/compatibility/dataset/css-properties.json");
 | |
| 
 | |
|   function walk(node) {
 | |
|     for (const propertyName in node) {
 | |
|       const property = node[propertyName];
 | |
|       if (property.__compat) {
 | |
|         if (!property.__compat.mdn_url) {
 | |
|           dump(
 | |
|             `"${propertyName}" - MDN URL: ${
 | |
|               property.__compat.mdn_url || "❌"
 | |
|             } - Spec URL: ${property.__compat.spec_url || "❌"}\n`
 | |
|           );
 | |
|         }
 | |
|       } else if (typeof property == "object") {
 | |
|         walk(property);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   walk(cssPropertiesCompatData);
 | |
| }
 | 
