summaryrefslogtreecommitdiffstats
path: root/toolkit/components/backgroundtasks/BackgroundTasksManager.sys.mjs
diff options
context:
space:
mode:
Diffstat (limited to 'toolkit/components/backgroundtasks/BackgroundTasksManager.sys.mjs')
-rw-r--r--toolkit/components/backgroundtasks/BackgroundTasksManager.sys.mjs329
1 files changed, 329 insertions, 0 deletions
diff --git a/toolkit/components/backgroundtasks/BackgroundTasksManager.sys.mjs b/toolkit/components/backgroundtasks/BackgroundTasksManager.sys.mjs
new file mode 100644
index 0000000000..0c2f277a23
--- /dev/null
+++ b/toolkit/components/backgroundtasks/BackgroundTasksManager.sys.mjs
@@ -0,0 +1,329 @@
+/* -*- indent-tabs-mode: nil; js-indent-level: 2 -*-
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs";
+
+const lazy = {};
+
+ChromeUtils.defineESModuleGetters(lazy, {
+ setTimeout: "resource://gre/modules/Timer.sys.mjs",
+});
+
+XPCOMUtils.defineLazyGetter(lazy, "log", () => {
+ let { ConsoleAPI } = ChromeUtils.importESModule(
+ "resource://gre/modules/Console.sys.mjs"
+ );
+ let consoleOptions = {
+ // tip: set maxLogLevel to "debug" and use log.debug() to create detailed
+ // messages during development. See LOG_LEVELS in Console.sys.mjs for details.
+ maxLogLevel: "error",
+ maxLogLevelPref: "toolkit.backgroundtasks.loglevel",
+ prefix: "BackgroundTasksManager",
+ };
+ return new ConsoleAPI(consoleOptions);
+});
+
+XPCOMUtils.defineLazyGetter(lazy, "DevToolsStartup", () => {
+ return Cc["@mozilla.org/devtools/startup-clh;1"].getService(
+ Ci.nsICommandLineHandler
+ ).wrappedJSObject;
+});
+
+// The default timing settings can be overriden by the preferences
+// toolkit.backgroundtasks.defaultTimeoutSec and
+// toolkit.backgroundtasks.defaultMinTaskRuntimeMS for all background tasks
+// and individually per module by
+// export const backgroundTaskTimeoutSec = X;
+// export const backgroundTaskMinRuntimeMS = Y;
+let timingSettings = {
+ minTaskRuntimeMS: 500,
+ maxTaskRuntimeSec: 600, // 10 minutes.
+};
+
+// Map resource://testing-common/ to the shared test modules directory. This is
+// a transliteration of `register_modules_protocol_handler` from
+// https://searchfox.org/mozilla-central/rev/f081504642a115cb8236bea4d8250e5cb0f39b02/testing/xpcshell/head.js#358-389.
+function registerModulesProtocolHandler() {
+ let _TESTING_MODULES_URI = Services.env.get(
+ "XPCSHELL_TESTING_MODULES_URI",
+ ""
+ );
+ if (!_TESTING_MODULES_URI) {
+ return false;
+ }
+
+ let protocolHandler = Services.io
+ .getProtocolHandler("resource")
+ .QueryInterface(Ci.nsIResProtocolHandler);
+
+ protocolHandler.setSubstitution(
+ "testing-common",
+ Services.io.newURI(_TESTING_MODULES_URI)
+ );
+ // Log loudly so that when testing, we always actually use the
+ // console logging mechanism and therefore deterministically load that code.
+ lazy.log.error(
+ `Substitution set: resource://testing-common aliases ${_TESTING_MODULES_URI}`
+ );
+
+ return true;
+}
+
+function locationsForBackgroundTaskNamed(name) {
+ const subModules = [
+ "resource:///modules", // App-specific first.
+ "resource://gre/modules", // Toolkit/general second.
+ ];
+
+ if (registerModulesProtocolHandler()) {
+ subModules.push("resource://testing-common"); // Test-only third.
+ }
+
+ let locations = [];
+ for (const subModule of subModules) {
+ let URI = `${subModule}/backgroundtasks/BackgroundTask_${name}.sys.mjs`;
+ locations.push(URI);
+ }
+
+ return locations;
+}
+
+/**
+ * Find an ES module named like `backgroundtasks/BackgroundTask_${name}.sys.mjs`,
+ * import it, and return the whole module.
+ *
+ * When testing, allow to load from `XPCSHELL_TESTING_MODULES_URI`,
+ * which is registered at `resource://testing-common`, the standard
+ * location for test-only modules.
+ *
+ * @return {Object} The imported module.
+ * @throws NS_ERROR_NOT_AVAILABLE if a background task with the given `name` is
+ * not found.
+ */
+function findBackgroundTaskModule(name) {
+ for (const URI of locationsForBackgroundTaskNamed(name)) {
+ lazy.log.debug(`Looking for background task at URI: ${URI}`);
+
+ try {
+ const taskModule = ChromeUtils.importESModule(URI);
+ lazy.log.info(`Found background task at URI: ${URI}`);
+ return taskModule;
+ } catch (ex) {
+ if (ex.result != Cr.NS_ERROR_FILE_NOT_FOUND) {
+ throw ex;
+ }
+ }
+ }
+
+ lazy.log.warn(`No backgroundtask named '${name}' registered`);
+ throw new Components.Exception(
+ `No backgroundtask named '${name}' registered`,
+ Cr.NS_ERROR_NOT_AVAILABLE
+ );
+}
+
+export class BackgroundTasksManager {
+ get helpInfo() {
+ const bts = Cc["@mozilla.org/backgroundtasks;1"].getService(
+ Ci.nsIBackgroundTasks
+ );
+
+ if (bts.isBackgroundTaskMode) {
+ return lazy.DevToolsStartup.jsdebuggerHelpInfo;
+ }
+
+ return "";
+ }
+
+ handle(commandLine) {
+ const bts = Cc["@mozilla.org/backgroundtasks;1"].getService(
+ Ci.nsIBackgroundTasks
+ );
+
+ if (!bts.isBackgroundTaskMode) {
+ lazy.log.info(
+ `${Services.appinfo.processID}: !isBackgroundTaskMode, exiting`
+ );
+ return;
+ }
+
+ const name = bts.backgroundTaskName();
+ lazy.log.info(
+ `${Services.appinfo.processID}: Preparing to run background task named '${name}'` +
+ ` (with ${commandLine.length} arguments)`
+ );
+
+ if (!("@mozilla.org/devtools/startup-clh;1" in Cc)) {
+ return;
+ }
+
+ // Check this before the devtools startup flow handles and removes it.
+ const CASE_INSENSITIVE = false;
+ if (
+ commandLine.findFlag("jsdebugger", CASE_INSENSITIVE) < 0 &&
+ commandLine.findFlag("start-debugger-server", CASE_INSENSITIVE) < 0
+ ) {
+ lazy.log.info(
+ `${Services.appinfo.processID}: No devtools flag found; not preparing devtools thread`
+ );
+ return;
+ }
+
+ const waitFlag =
+ commandLine.findFlag("wait-for-jsdebugger", CASE_INSENSITIVE) != -1;
+ if (waitFlag) {
+ function onDevtoolsThreadReady(subject, topic, data) {
+ lazy.log.info(
+ `${Services.appinfo.processID}: Setting breakpoints for background task named '${name}'` +
+ ` (with ${commandLine.length} arguments)`
+ );
+
+ const threadActor = subject.wrappedJSObject;
+ threadActor.setBreakpointOnLoad(locationsForBackgroundTaskNamed(name));
+
+ Services.obs.removeObserver(onDevtoolsThreadReady, topic);
+ }
+
+ Services.obs.addObserver(onDevtoolsThreadReady, "devtools-thread-ready");
+ }
+
+ const DevToolsStartup = Cc[
+ "@mozilla.org/devtools/startup-clh;1"
+ ].getService(Ci.nsICommandLineHandler);
+ DevToolsStartup.handle(commandLine);
+ }
+
+ async runBackgroundTaskNamed(name, commandLine) {
+ function addMarker(markerName) {
+ return ChromeUtils.addProfilerMarker(markerName, undefined, name);
+ }
+ addMarker("BackgroundTasksManager:AfterRunBackgroundTaskNamed");
+
+ lazy.log.info(
+ `${Services.appinfo.processID}: Running background task named '${name}'` +
+ ` (with ${commandLine.length} arguments)`
+ );
+ lazy.log.debug(
+ `${Services.appinfo.processID}: Background task using profile` +
+ ` '${Services.dirsvc.get("ProfD", Ci.nsIFile).path}'`
+ );
+
+ let exitCode = EXIT_CODE.NOT_FOUND;
+ try {
+ let taskModule = findBackgroundTaskModule(name);
+ addMarker("BackgroundTasksManager:AfterFindRunBackgroundTask");
+
+ // Get timing configuration. First check for default preferences,
+ // then for per module overrides.
+ timingSettings.minTaskRuntimeMS = Services.prefs.getIntPref(
+ "toolkit.backgroundtasks.defaultMinTaskRuntimeMS",
+ timingSettings.minTaskRuntimeMS
+ );
+ if (taskModule.backgroundTaskMinRuntimeMS) {
+ timingSettings.minTaskRuntimeMS = taskModule.backgroundTaskMinRuntimeMS;
+ }
+ timingSettings.maxTaskRuntimeSec = Services.prefs.getIntPref(
+ "toolkit.backgroundtasks.defaultTimeoutSec",
+ timingSettings.maxTaskRuntimeSec
+ );
+ if (taskModule.backgroundTaskTimeoutSec) {
+ timingSettings.maxTaskRuntimeSec = taskModule.backgroundTaskTimeoutSec;
+ }
+
+ try {
+ let minimumReached = false;
+ let minRuntime = new Promise(resolve =>
+ lazy.setTimeout(() => {
+ minimumReached = true;
+ resolve(true);
+ }, timingSettings.minTaskRuntimeMS)
+ );
+ exitCode = await Promise.race([
+ new Promise(resolve =>
+ lazy.setTimeout(() => {
+ lazy.log.error(`Background task named '${name}' timed out`);
+ resolve(EXIT_CODE.TIMEOUT);
+ }, timingSettings.maxTaskRuntimeSec * 1000)
+ ),
+ taskModule.runBackgroundTask(commandLine),
+ ]);
+ if (!minimumReached) {
+ lazy.log.debug(
+ `Backgroundtask named '${name}' waiting for minimum runtime.`
+ );
+ await minRuntime;
+ }
+ lazy.log.info(
+ `Backgroundtask named '${name}' completed with exit code ${exitCode}`
+ );
+ } catch (e) {
+ lazy.log.error(`Backgroundtask named '${name}' threw exception`, e);
+ exitCode = EXIT_CODE.EXCEPTION;
+ }
+ } finally {
+ addMarker("BackgroundTasksManager:AfterAwaitRunBackgroundTask");
+
+ lazy.log.info(`Invoking Services.startup.quit(..., ${exitCode})`);
+ Services.startup.quit(Ci.nsIAppStartup.eForceQuit, exitCode);
+ }
+
+ return exitCode;
+ }
+
+ classID = Components.ID("{4d48c536-e16f-4699-8f9c-add4f28f92f0}");
+ QueryInterface = ChromeUtils.generateQI([
+ "nsIBackgroundTasksManager",
+ "nsICommandLineHandler",
+ ]);
+}
+
+/**
+ * Background tasks should standard exit code conventions where 0 denotes
+ * success and non-zero denotes failure and/or an error. In addition, since
+ * background tasks have limited channels to communicate with consumers, the
+ * special values `NOT_FOUND` (integer 2) and `THREW_EXCEPTION` (integer 3) are
+ * distinguished.
+ *
+ * If you extend this to add background task-specific exit codes, use exit codes
+ * greater than 10 to allow for additional shared exit codes to be added here.
+ * Exit codes should be between 0 and 127 to be safe across platforms.
+ */
+export const EXIT_CODE = {
+ /**
+ * The task succeeded.
+ *
+ * The `runBackgroundTask(...)` promise resolved to 0.
+ */
+ SUCCESS: 0,
+
+ /**
+ * The task with the specified name could not be found or imported.
+ *
+ * The corresponding `runBackgroundTask` method could not be found.
+ */
+ NOT_FOUND: 2,
+
+ /**
+ * The task failed with an uncaught exception.
+ *
+ * The `runBackgroundTask(...)` promise rejected with an exception.
+ */
+ EXCEPTION: 3,
+
+ /**
+ * The task took too long and timed out.
+ *
+ * The default timeout is controlled by the pref:
+ * "toolkit.backgroundtasks.defaultTimeoutSec", but tasks can override this
+ * by exporting a non-zero `backgroundTaskTimeoutSec` value.
+ */
+ TIMEOUT: 4,
+
+ /**
+ * The last exit code reserved by this structure. Use codes larger than this
+ * code for background task-specific exit codes.
+ */
+ LAST_RESERVED: 10,
+};