gecko-dev/devtools/shared/fronts/root.js

449 строки
15 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/. */
"use strict";
const { Ci } = require("chrome");
const { rootSpec } = require("devtools/shared/specs/root");
const {
FrontClassWithSpec,
registerFront,
} = require("devtools/shared/protocol");
loader.lazyRequireGetter(this, "getFront", "devtools/shared/protocol", true);
loader.lazyRequireGetter(
this,
"ProcessDescriptorFront",
"devtools/shared/fronts/descriptors/process",
true
);
loader.lazyRequireGetter(
this,
"FrameDescriptorFront",
"devtools/shared/fronts/descriptors/frame",
true
);
loader.lazyRequireGetter(
this,
"BrowsingContextTargetFront",
"devtools/shared/fronts/targets/browsing-context",
true
);
loader.lazyRequireGetter(
this,
"ContentProcessTargetFront",
"devtools/shared/fronts/targets/content-process",
true
);
loader.lazyRequireGetter(
this,
"LocalTabTargetFront",
"devtools/shared/fronts/targets/local-tab",
true
);
class RootFront extends FrontClassWithSpec(rootSpec) {
constructor(client, form) {
super(client);
// Root Front is a special Front. It is the only one to set its actor ID manually
// out of the form object returned by RootActor.sayHello which is called when calling
// DebuggerClient.connect().
this.actorID = form.from;
this.applicationType = form.applicationType;
this.traits = form.traits;
// Cache root form as this will always be the same value.
Object.defineProperty(this, "rootForm", {
get() {
delete this.rootForm;
this.rootForm = this.getRoot();
return this.rootForm;
},
configurable: true,
});
// Cache of already created global scoped fronts
// [typeName:string => Front instance]
this.fronts = new Map();
this._client = client;
}
/**
* Retrieve all service worker registrations as well as workers from the parent and
* content processes. Listing service workers involves merging information coming from
* registrations and workers, this method will combine this information to present a
* unified array of serviceWorkers. If you are only interested in other workers, use
* listWorkers.
*
* @return {Object}
* - {Array} service
* array of form-like objects for serviceworkers
* - {Array} shared
* Array of WorkerTargetActor forms, containing shared workers.
* - {Array} other
* Array of WorkerTargetActor forms, containing other workers.
*/
async listAllWorkers() {
let registrations = [];
let workers = [];
try {
// List service worker registrations
({ registrations } = await this.listServiceWorkerRegistrations());
// List workers from the Parent process
({ workers } = await this.listWorkers());
// And then from the Child processes
const { processes } = await this.listProcesses();
for (const processDescriptorFront of processes) {
// Ignore parent process
if (processDescriptorFront.isParent) {
continue;
}
const front = await processDescriptorFront.getTarget();
const response = await front.listWorkers();
workers = workers.concat(response.workers);
}
} catch (e) {
// Something went wrong, maybe our client is disconnected?
}
const result = {
service: [],
shared: [],
other: [],
};
registrations.forEach(front => {
const { activeWorker, waitingWorker, installingWorker } = front;
const newestWorker = activeWorker || waitingWorker || installingWorker;
// All the information is simply mirrored from the registration front.
// However since registering workers will fetch similar information from the worker
// target front and will not have a service worker registration front, consumers
// should not read meta data directly on the registration front instance.
result.service.push({
active: front.active,
fetch: front.fetch,
id: front.id,
lastUpdateTime: front.lastUpdateTime,
name: front.url,
registrationFront: front,
scope: front.scope,
url: front.url,
newestWorkerId: newestWorker && newestWorker.id,
});
});
workers.forEach(front => {
const worker = {
id: front.id,
name: front.url,
url: front.url,
workerTargetFront: front,
};
switch (front.type) {
case Ci.nsIWorkerDebugger.TYPE_SERVICE:
const registration = result.service.find(r => {
/**
* Older servers will not define `ServiceWorkerFront.id` (the value
* of `r.newestWorkerId`), and a `ServiceWorkerFront`'s ID will only
* match its corresponding WorkerTargetFront's ID if their
* underlying actors are "connected" - this is only guaranteed with
* parent-intercept mode. The `if` statement is for backward
* compatibility and can be removed when the release channel is
* >= FF69 _and_ parent-intercept is stable (which definitely won't
* happen when the release channel is < FF69).
*/
const { isParentInterceptEnabled } = r.registrationFront.traits;
if (!r.newestWorkerId || !isParentInterceptEnabled) {
return r.scope === front.scope;
}
return r.newestWorkerId === front.id;
});
if (registration) {
// XXX: Race, sometimes a ServiceWorkerRegistrationInfo doesn't
// have a scriptSpec, but its associated WorkerDebugger does.
if (!registration.url) {
registration.name = registration.url = front.url;
}
registration.workerTargetFront = front;
} else {
worker.fetch = front.fetch;
// If a service worker registration could not be found, this means we are in
// e10s, and registrations are not forwarded to other processes until they
// reach the activated state. Augment the worker as a registration worker to
// display it in aboutdebugging.
worker.scope = front.scope;
worker.active = false;
result.service.push(worker);
}
break;
case Ci.nsIWorkerDebugger.TYPE_SHARED:
result.shared.push(worker);
break;
default:
result.other.push(worker);
}
});
return result;
}
async listProcesses() {
const { processes } = await super.listProcesses();
const processDescriptors = processes.map(form => {
if (form.actor && form.actor.includes("processDescriptor")) {
return this._getProcessDescriptorFront(form);
}
// Support FF69 and older
return {
id: form.id,
isParent: form.parent,
getTarget: () => {
return this.getProcess(form.id);
},
};
});
return { processes: processDescriptors };
}
/**
* Fetch the ParentProcessTargetActor for the main process.
*
* `getProcess` requests allows to fetch the target actor for any process
* and the main process is having the process ID zero.
*/
getMainProcess() {
return this.getProcess(0);
}
async getProcess(id) {
const { form } = await super.getProcess(id);
if (form.actor && form.actor.includes("processDescriptor")) {
// The server currently returns a form, when we can drop backwards compatibility,
// we can use automatic marshalling here instead, making the next line unnecessary
const processDescriptorFront = this._getProcessDescriptorFront(form);
return processDescriptorFront.getTarget();
}
// Backwards compatibility for servers up to FF69.
// Do not use specification automatic marshalling as getProcess may return
// two different type: ParentProcessTargetActor or ContentProcessTargetActor.
// Also, we do want to memoize the fronts and return already existing ones.
let front = this.actor(form.actor);
if (front) {
return front;
}
// getProcess may return a ContentProcessTargetActor or a ParentProcessTargetActor
// In most cases getProcess(0) will return the main process target actor,
// which is a ParentProcessTargetActor, but not in xpcshell, which uses a
// ContentProcessTargetActor. So select the right front based on the actor ID.
if (form.actor.includes("contentProcessTarget")) {
front = new ContentProcessTargetFront(this._client, null, this);
} else {
// ParentProcessTargetActor doesn't have a specific front, instead it uses
// BrowsingContextTargetFront on the client side.
front = new BrowsingContextTargetFront(this._client, null, this);
}
// As these fronts aren't instantiated by protocol.js, we have to set their actor ID
// manually like that:
front.actorID = form.actor;
front.form(form);
this.manage(front);
return front;
}
/**
* This exists as a polyfill for now for tabTargets, which do not have descriptors.
* The mainRoot fills the role of the descriptor
*/
/**
* Get the previous frame descriptor front if it exists, create a new one if not
*/
_getFrameDescriptorFront(form) {
let front = this.actor(form.actor);
if (front) {
return front;
}
front = new FrameDescriptorFront(this._client, null, this);
front.form(form);
front.actorID = form.actor;
this.manage(front);
return front;
}
/**
* Get the previous process descriptor front if it exists, create a new one if not.
*
* If we are using a modern server, we will get a form for a processDescriptorFront.
* Until we can switch to auto marshalling, we need to marshal this into a process
* descriptor front ourselves.
*/
_getProcessDescriptorFront(form) {
let front = this.actor(form.actor);
if (front) {
return front;
}
front = new ProcessDescriptorFront(this._client, null, this);
front.form(form);
front.actorID = form.actor;
this.manage(front);
return front;
}
async getBrowsingContextDescriptor(id) {
const form = await super.getBrowsingContextDescriptor(id);
if (form.actor && form.actor.includes("processDescriptor")) {
return this._getProcessDescriptorFront(form);
}
return this._getFrameDescriptorFront(form);
}
/**
* Override default listTabs request in order to return a list of
* BrowsingContextTargetFronts while updating their selected state.
*/
async listTabs(options) {
const { selected, tabs } = await super.listTabs(options);
for (const i in tabs) {
tabs[i].setIsSelected(i == selected);
}
return tabs;
}
/**
* Fetch the target actor for the currently selected tab, or for a specific
* tab given as first parameter.
*
* @param [optional] object filter
* A dictionary object with following optional attributes:
* - outerWindowID: used to match tabs in parent process
* - tabId: used to match tabs in child processes
* - tab: a reference to xul:tab element
* If nothing is specified, returns the actor for the currently
* selected tab.
*/
async getTab(filter) {
const packet = {};
if (filter) {
if (typeof filter.outerWindowID == "number") {
packet.outerWindowID = filter.outerWindowID;
} else if (typeof filter.tabId == "number") {
packet.tabId = filter.tabId;
} else if ("tab" in filter) {
const browser = filter.tab.linkedBrowser;
if (browser.frameLoader.remoteTab) {
// Tabs in child process
packet.tabId = browser.frameLoader.remoteTab.tabId;
} else if (browser.outerWindowID) {
// <xul:browser> tabs in parent process
packet.outerWindowID = browser.outerWindowID;
} else {
// <iframe mozbrowser> tabs in parent process
const windowUtils = browser.contentWindow.windowUtils;
packet.outerWindowID = windowUtils.outerWindowID;
}
} else {
// Throw if a filter object have been passed but without
// any clearly idenfified filter.
throw new Error("Unsupported argument given to getTab request");
}
}
const form = await super.getTab(packet);
let front = this.actor(form.actor);
if (front) {
front.form(form);
return front;
}
// Instanciate a specialized class for a local tab as it needs some more
// client side integration with the Firefox frontend.
// But ignore the fake `tab` object we receive, where there is only a
// `linkedBrowser` attribute, but this isn't a real <tab> element.
// devtools/client/framework/test/browser_toolbox_target.js is passing such
// a fake tab.
if (filter && filter.tab && filter.tab.tagName == "tab") {
front = new LocalTabTargetFront(this._client, null, this, filter.tab);
} else {
front = new BrowsingContextTargetFront(this._client, null, this);
}
// As these fronts aren't instantiated by protocol.js, we have to set their actor ID
// manually like that:
front.actorID = form.actor;
front.form(form);
this.manage(front);
return front;
}
/**
* Fetch the target front for a given add-on.
* This is just an helper on top of `listAddons` request.
*
* @param object filter
* A dictionary object with following attribute:
* - id: used to match the add-on to connect to.
*/
async getAddon({ id }) {
const addons = await this.listAddons();
const webextensionDescriptorFront = addons.find(addon => addon.id === id);
return webextensionDescriptorFront;
}
/**
* Fetch the target front for a given worker.
* This is just an helper on top of `listAllWorkers` request.
*
* @param id
*/
async getWorker(id) {
const { service, shared, other } = await this.listAllWorkers();
const worker = [...service, ...shared, ...other].find(w => w.id === id);
if (!worker) {
return null;
}
return worker.workerTargetFront || worker.registrationFront;
}
/**
* Test request that returns the object passed as first argument.
*
* `echo` is special as all the property of the given object have to be passed
* on the packet object. That's not something that can be achieve by requester helper.
*/
echo(packet) {
packet.type = "echo";
return this.request(packet);
}
/*
* This function returns a protocol.js Front for any root actor.
* i.e. the one directly served from RootActor.listTabs or getRoot.
*
* @param String typeName
* The type name used in protocol.js's spec for this actor.
*/
async getFront(typeName) {
let front = this.fronts.get(typeName);
if (front) {
return front;
}
const rootForm = await this.rootForm;
front = getFront(this._client, typeName, rootForm);
this.fronts.set(typeName, front);
return front;
}
}
exports.RootFront = RootFront;
registerFront(RootFront);