fune/browser/base/content/test/performance/browser_startup_content.js
Kris Maglione 977f86fb80 Bug 1645862: Part 1 - Split TelemetryController into parent and content modules. r=chutten
Most of TelemetryController is only used in the parent process, but is
currently loaded in all processes, where it needlessly consumes tens of KB of
memory. This patch splits the parent- and content-specific parts, and only
loads each in the processes where they're needed.

Differential Revision: https://phabricator.services.mozilla.com/D79870
2020-06-24 17:33:49 +00:00

256 lines
7.8 KiB
JavaScript

/* Any copyright is dedicated to the Public Domain.
http://creativecommons.org/publicdomain/zero/1.0/ */
/* This test records which services, JS components, frame scripts, process
* scripts, and JS modules are loaded when creating a new content process.
*
* If you made changes that cause this test to fail, it's likely because you
* are loading more JS code during content process startup. Please try to
* avoid this.
*
* If your code isn't strictly required to show a page, consider loading it
* lazily. If you can't, consider delaying its load until after we have started
* handling user events.
*/
"use strict";
/* Set this to true only for debugging purpose; it makes the output noisy. */
const kDumpAllStacks = false;
const known_scripts = {
modules: new Set([
"chrome://mochikit/content/ShutdownLeaksCollector.jsm",
"resource://gre/modules/ContentProcessSingleton.jsm",
// General utilities
"resource://gre/modules/AppConstants.jsm",
"resource://gre/modules/DeferredTask.jsm",
"resource://gre/modules/Services.jsm", // bug 1464542
"resource://gre/modules/Timer.jsm",
"resource://gre/modules/XPCOMUtils.jsm",
// Logging related
"resource://gre/modules/Log.jsm",
// Session store
"resource:///modules/sessionstore/ContentSessionStore.jsm",
// Browser front-end
"resource:///actors/AboutReaderChild.jsm",
"resource:///actors/BrowserTabChild.jsm",
"resource:///actors/LinkHandlerChild.jsm",
"resource:///actors/SearchTelemetryChild.jsm",
"resource://gre/actors/AutoCompleteChild.jsm",
"resource://gre/modules/ActorManagerChild.jsm",
"resource://gre/modules/E10SUtils.jsm",
"resource://gre/modules/Readerable.jsm",
// Telemetry
"resource://gre/modules/TelemetryController.jsm", // bug 1470339
"resource://gre/modules/TelemetryControllerBase.jsm", // bug 1470339
"resource://gre/modules/TelemetryControllerContent.jsm", // bug 1470339
"resource://gre/modules/TelemetryUtils.jsm", // bug 1470339
// Extensions
"resource://gre/modules/ExtensionProcessScript.jsm",
"resource://gre/modules/ExtensionUtils.jsm",
"resource://gre/modules/MessageChannel.jsm",
]),
frameScripts: new Set([
// Test related
"chrome://mochikit/content/shutdown-leaks-collector.js",
// Browser front-end
"chrome://global/content/browser-content.js",
// Extensions
"resource://gre/modules/addons/Content.js",
]),
processScripts: new Set([
"chrome://global/content/process-content.js",
"resource:///modules/ContentObservers.js",
"resource://gre/modules/extensionProcessScriptLoader.js",
]),
};
// Items on this list *might* load when creating the process, as opposed to
// items in the main list, which we expect will always load.
const intermittently_loaded_scripts = {
modules: new Set([
"resource://gre/modules/nsAsyncShutdown.jsm",
"resource://gre/modules/sessionstore/Utils.jsm",
// Session store.
"resource://gre/modules/sessionstore/SessionHistory.jsm",
// Webcompat about:config front-end. This is part of a system add-on which
// may not load early enough for the test.
"resource://webcompat/AboutCompat.jsm",
// Test related
"resource://testing-common/BrowserTestUtilsChild.jsm",
"resource://testing-common/ContentEventListenerChild.jsm",
"resource://specialpowers/SpecialPowersChild.jsm",
"resource://specialpowers/WrapPrivileged.jsm",
]),
frameScripts: new Set([]),
processScripts: new Set([
// Webcompat about:config front-end. This is presently nightly-only and
// part of a system add-on which may not load early enough for the test.
"resource://webcompat/aboutPageProcessScript.js",
]),
};
const forbiddenScripts = {
services: new Set([
"@mozilla.org/base/telemetry-startup;1",
"@mozilla.org/embedcomp/default-tooltiptextprovider;1",
"@mozilla.org/push/Service;1",
]),
};
add_task(async function() {
SimpleTest.requestCompleteLog();
let tab = await BrowserTestUtils.openNewForegroundTab({
gBrowser,
forceNewProcess: true,
});
let mm = gBrowser.selectedBrowser.messageManager;
let promise = BrowserTestUtils.waitForMessage(mm, "Test:LoadedScripts");
// Load a custom frame script to avoid using ContentTask which loads Task.jsm
mm.loadFrameScript(
"data:text/javascript,(" +
function() {
/* eslint-env mozilla/frame-script */
const Cm = Components.manager;
Cm.QueryInterface(Ci.nsIServiceManager);
const { AppConstants } = ChromeUtils.import(
"resource://gre/modules/AppConstants.jsm"
);
let collectStacks = AppConstants.NIGHTLY_BUILD || AppConstants.DEBUG;
let components = {};
for (let component of Cu.loadedComponents) {
/* Keep only the file name for components, as the path is an absolute file
URL rather than a resource:// URL like for modules. */
components[component.replace(/.*\//, "")] = collectStacks
? Cu.getComponentLoadStack(component)
: "";
}
let modules = {};
for (let module of Cu.loadedModules) {
modules[module] = collectStacks
? Cu.getModuleImportStack(module)
: "";
}
let services = {};
for (let contractID of Object.keys(Cc)) {
try {
if (
Cm.isServiceInstantiatedByContractID(contractID, Ci.nsISupports)
) {
services[contractID] = "";
}
} catch (e) {}
}
sendAsyncMessage("Test:LoadedScripts", {
components,
modules,
services,
});
} +
")()",
false
);
let loadedInfo = await promise;
// Gather loaded frame scripts.
loadedInfo.frameScripts = {};
for (let [uri] of Services.mm.getDelayedFrameScripts()) {
loadedInfo.frameScripts[uri] = "";
}
// Gather loaded process scripts.
loadedInfo.processScripts = {};
for (let [uri] of Services.ppmm.getDelayedProcessScripts()) {
loadedInfo.processScripts[uri] = "";
}
let loadedList = {};
for (let scriptType in known_scripts) {
loadedList[scriptType] = Object.keys(loadedInfo[scriptType]).filter(c => {
if (!known_scripts[scriptType].has(c)) {
return true;
}
known_scripts[scriptType].delete(c);
return false;
});
loadedList[scriptType] = loadedList[scriptType].filter(c => {
return !intermittently_loaded_scripts[scriptType].has(c);
});
is(
loadedList[scriptType].length,
0,
`should have no unexpected ${scriptType} loaded on content process startup`
);
for (let script of loadedList[scriptType]) {
record(
false,
`Unexpected ${scriptType} loaded during content process startup: ${script}`,
undefined,
loadedInfo[scriptType][script]
);
}
is(
known_scripts[scriptType].size,
0,
`all known ${scriptType} scripts should have been loaded`
);
for (let script of known_scripts[scriptType]) {
ok(
false,
`${scriptType} is expected to load for content process startup but wasn't: ${script}`
);
}
if (kDumpAllStacks) {
info(`Stacks for all loaded ${scriptType}:`);
for (let file in loadedInfo[scriptType]) {
if (loadedInfo[scriptType][file]) {
info(
`${file}\n------------------------------------\n` +
loadedInfo[scriptType][file] +
"\n"
);
}
}
}
}
for (let scriptType in forbiddenScripts) {
for (let script of forbiddenScripts[scriptType]) {
let loaded = script in loadedInfo[scriptType];
if (loaded) {
record(
false,
`Forbidden ${scriptType} loaded during content process startup: ${script}`,
undefined,
loadedInfo[scriptType][script]
);
}
}
}
BrowserTestUtils.removeTab(tab);
});