mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-11-06 11:18:19 +02:00
188 lines
6.3 KiB
JavaScript
188 lines
6.3 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
const lazy = {};
|
|
|
|
ChromeUtils.defineESModuleGetters(lazy, {
|
|
ContextDescriptorType:
|
|
"chrome://remote/content/shared/messagehandler/MessageHandler.sys.mjs",
|
|
isBrowsingContextCompatible:
|
|
"chrome://remote/content/shared/messagehandler/transports/BrowsingContextUtils.sys.mjs",
|
|
Log: "chrome://remote/content/shared/Log.sys.mjs",
|
|
MessageHandlerFrameActor:
|
|
"chrome://remote/content/shared/messagehandler/transports/js-window-actors/MessageHandlerFrameActor.sys.mjs",
|
|
TabManager: "chrome://remote/content/shared/TabManager.sys.mjs",
|
|
});
|
|
|
|
ChromeUtils.defineLazyGetter(lazy, "logger", () => lazy.Log.get());
|
|
|
|
const MAX_RETRY_ATTEMPTS = 10;
|
|
|
|
/**
|
|
* RootTransport is intended to be used from a ROOT MessageHandler to communicate
|
|
* with WINDOW_GLOBAL MessageHandlers via the MessageHandlerFrame JSWindow
|
|
* actors.
|
|
*/
|
|
export class RootTransport {
|
|
/**
|
|
* @param {MessageHandler} messageHandler
|
|
* The MessageHandler instance which owns this RootTransport instance.
|
|
*/
|
|
constructor(messageHandler) {
|
|
this._messageHandler = messageHandler;
|
|
|
|
// RootTransport will rely on the MessageHandlerFrame JSWindow actors.
|
|
// Make sure they are registered when instanciating a RootTransport.
|
|
lazy.MessageHandlerFrameActor.register();
|
|
}
|
|
|
|
/**
|
|
* Forward the provided command to WINDOW_GLOBAL MessageHandlers via the
|
|
* MessageHandlerFrame actors.
|
|
*
|
|
* @param {Command} command
|
|
* The command to forward. See type definition in MessageHandler.js
|
|
* @returns {Promise}
|
|
* Returns a promise that resolves with the result of the command after
|
|
* being processed by WINDOW_GLOBAL MessageHandlers.
|
|
*/
|
|
forwardCommand(command) {
|
|
if (command.destination.id && command.destination.contextDescriptor) {
|
|
throw new Error(
|
|
"Invalid command destination with both 'id' and 'contextDescriptor' properties"
|
|
);
|
|
}
|
|
|
|
// With an id given forward the command to only this specific destination.
|
|
if (command.destination.id) {
|
|
const browsingContext = BrowsingContext.get(command.destination.id);
|
|
if (!browsingContext) {
|
|
throw new Error(
|
|
"Unable to find a BrowsingContext for id " + command.destination.id
|
|
);
|
|
}
|
|
return this._sendCommandToBrowsingContext(command, browsingContext);
|
|
}
|
|
|
|
// ... otherwise broadcast to destinations matching the contextDescriptor.
|
|
if (command.destination.contextDescriptor) {
|
|
return this._broadcastCommand(command);
|
|
}
|
|
|
|
throw new Error(
|
|
"Unrecognized command destination, missing 'id' or 'contextDescriptor' properties"
|
|
);
|
|
}
|
|
|
|
_broadcastCommand(command) {
|
|
const { contextDescriptor } = command.destination;
|
|
const browsingContexts =
|
|
this._getBrowsingContextsForDescriptor(contextDescriptor);
|
|
|
|
return Promise.all(
|
|
browsingContexts.map(async browsingContext => {
|
|
try {
|
|
return await this._sendCommandToBrowsingContext(
|
|
command,
|
|
browsingContext
|
|
);
|
|
} catch (e) {
|
|
console.error(
|
|
`Failed to broadcast a command to browsingContext ${browsingContext.id}`,
|
|
e
|
|
);
|
|
return null;
|
|
}
|
|
})
|
|
);
|
|
}
|
|
|
|
async _sendCommandToBrowsingContext(command, browsingContext) {
|
|
const name = `${command.moduleName}.${command.commandName}`;
|
|
|
|
// The browsing context might be destroyed by a navigation. Keep a reference
|
|
// to the webProgress, which will persist, and always use it to retrieve the
|
|
// currently valid browsing context.
|
|
const webProgress = browsingContext.webProgress;
|
|
|
|
const { retryOnAbort = false } = command;
|
|
|
|
let attempts = 0;
|
|
while (true) {
|
|
try {
|
|
return await webProgress.browsingContext.currentWindowGlobal
|
|
.getActor("MessageHandlerFrame")
|
|
.sendCommand(command, this._messageHandler.sessionId);
|
|
} catch (e) {
|
|
if (!retryOnAbort || e.name != "AbortError") {
|
|
// Only retry if the command supports retryOnAbort and when the
|
|
// JSWindowActor pair gets destroyed.
|
|
throw e;
|
|
}
|
|
|
|
if (++attempts > MAX_RETRY_ATTEMPTS) {
|
|
lazy.logger.trace(
|
|
`RootTransport reached the limit of retry attempts (${MAX_RETRY_ATTEMPTS})` +
|
|
` for command ${name} and browsing context ${webProgress.browsingContext.id}.`
|
|
);
|
|
throw e;
|
|
}
|
|
|
|
lazy.logger.trace(
|
|
`RootTransport retrying command ${name} for ` +
|
|
`browsing context ${webProgress.browsingContext.id}, attempt: ${attempts}.`
|
|
);
|
|
await new Promise(resolve => Services.tm.dispatchToMainThread(resolve));
|
|
}
|
|
}
|
|
}
|
|
|
|
toString() {
|
|
return `[object ${this.constructor.name} ${this._messageHandler.name}]`;
|
|
}
|
|
|
|
_getBrowsingContextsForDescriptor(contextDescriptor) {
|
|
const { id, type } = contextDescriptor;
|
|
|
|
if (type === lazy.ContextDescriptorType.All) {
|
|
return this._getBrowsingContexts();
|
|
}
|
|
|
|
if (type === lazy.ContextDescriptorType.TopBrowsingContext) {
|
|
return this._getBrowsingContexts({ browserId: id });
|
|
}
|
|
|
|
// TODO: Handle other types of context descriptors.
|
|
throw new Error(
|
|
`Unsupported contextDescriptor type for broadcasting: ${type}`
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Get all browsing contexts, optionally matching the provided options.
|
|
*
|
|
* @param {object} options
|
|
* @param {string=} options.browserId
|
|
* The id of the browser to filter the browsing contexts by (optional).
|
|
* @returns {Array<BrowsingContext>}
|
|
* The browsing contexts matching the provided options or all browsing contexts
|
|
* if no options are provided.
|
|
*/
|
|
_getBrowsingContexts(options = {}) {
|
|
// extract browserId from options
|
|
const { browserId } = options;
|
|
let browsingContexts = [];
|
|
|
|
// Fetch all tab related browsing contexts for top-level windows.
|
|
for (const { browsingContext } of lazy.TabManager.browsers) {
|
|
if (lazy.isBrowsingContextCompatible(browsingContext, { browserId })) {
|
|
browsingContexts = browsingContexts.concat(
|
|
browsingContext.getAllBrowsingContextsInSubtree()
|
|
);
|
|
}
|
|
}
|
|
|
|
return browsingContexts;
|
|
}
|
|
}
|