/* Any copyright is dedicated to the Public Domain.
http://creativecommons.org/publicdomain/zero/1.0/ */
"use strict";
ChromeUtils.defineESModuleGetters(
this, {
NetworkObserver:
"resource://devtools/shared/network-observer/NetworkObserver.sys.mjs",
});
const TEST_DIR = gTestPath.substr(0, gTestPath.lastIndexOf(
"/"));
const CHROME_URL_ROOT = TEST_DIR +
"/";
const URL_ROOT = CHROME_URL_ROOT.replace(
"chrome://mochitests/content/",
"https://example.com/"
);
/**
* Load the provided url in an existing browser.
* Returns a promise which will resolve when the page is loaded.
*
* @param {Browser} browser
* The browser element where the URL should be loaded.
* @param {String} url
* The URL to load in the new tab
*/
async
function loadURL(browser, url) {
const loaded = BrowserTestUtils.browserLoaded(browser);
BrowserTestUtils.startLoadingURIString(browser, url);
return loaded;
}
/**
* Create a new foreground tab loading the provided url.
* Returns a promise which will resolve when the page is loaded.
*
* @param {String} url
* The URL to load in the new tab
*/
async
function addTab(url) {
const tab = await BrowserTestUtils.openNewForegroundTab(gBrowser, url);
registerCleanupFunction(() => {
gBrowser.removeTab(tab);
});
return tab;
}
/**
* Base network event owner class implementing all mandatory callbacks and
* keeping track of which callbacks have been called.
*/
class NetworkEventOwner {
hasCacheDetails =
false;
hasEventTimings =
false;
hasRawHeaders =
false;
hasResponseCache =
false;
hasResponseContent =
false;
hasResponseStart =
false;
hasSecurityInfo =
false;
hasServerTimings =
false;
addCacheDetails() {
this.hasCacheDetails =
true;
}
addEventTimings() {
this.hasEventTimings =
true;
}
addRawHeaders() {
this.hasRawHeaders =
true;
}
addResponseCache() {
this.hasResponseCache =
true;
}
addResponseContent() {
this.hasResponseContent =
true;
}
addResponseStart() {
this.hasResponseStart =
true;
}
addSecurityInfo() {
this.hasSecurityInfo =
true;
}
addServerTimings() {
this.hasServerTimings =
true;
}
addServiceWorkerTimings() {
this.hasServiceWorkerTimings =
true;
}
}
/**
* Create a simple network event owner, with mock implementations of all
* the expected APIs for a NetworkEventOwner.
*/
function createNetworkEventOwner() {
return new NetworkEventOwner();
}
/**
* Wait for network events matching the provided URL, until the count reaches
* the provided expected count.
*
* @param {string|null} expectedUrl
* The URL which should be monitored by the NetworkObserver.If set to null watch for
* all requests
* @param {number} expectedRequestsCount
* How many different events (requests) are expected.
* @returns {Promise}
* A promise which will resolve with an array of network event owners, when
* the expected event count is reached.
*/
async
function waitForNetworkEvents(expectedUrl =
null, expectedRequestsCount) {
const events = [];
const networkObserver =
new NetworkObserver({
ignoreChannelFunction: channel =>
expectedUrl ? channel.URI.spec !== expectedUrl :
false,
onNetworkEvent: () => {
info(
"waitForNetworkEvents received a new event");
const owner = createNetworkEventOwner();
events.push(owner);
return owner;
},
});
registerCleanupFunction(() => networkObserver.destroy());
info(
"Wait until the events count reaches " + expectedRequestsCount);
await BrowserTestUtils.waitForCondition(
() => events.length >= expectedRequestsCount
);
return events;
}