210 lines
7 KiB
JavaScript
210 lines
7 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 { watcherSpec } = require("resource://devtools/shared/specs/watcher.js");
|
|
const {
|
|
FrontClassWithSpec,
|
|
registerFront,
|
|
} = require("resource://devtools/shared/protocol.js");
|
|
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"WindowGlobalTargetFront",
|
|
"resource://devtools/client/fronts/targets/window-global.js",
|
|
true
|
|
);
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"ContentProcessTargetFront",
|
|
"resource://devtools/client/fronts/targets/content-process.js",
|
|
true
|
|
);
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"WorkerTargetFront",
|
|
"resource://devtools/client/fronts/targets/worker.js",
|
|
true
|
|
);
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"ContentScriptTargetFront",
|
|
"resource://devtools/client/fronts/targets/content-script.js",
|
|
true
|
|
);
|
|
|
|
class WatcherFront extends FrontClassWithSpec(watcherSpec) {
|
|
constructor(client, targetFront, parentFront) {
|
|
super(client, targetFront, parentFront);
|
|
|
|
this._onTargetAvailable = this._onTargetAvailable.bind(this);
|
|
this._onTargetDestroyed = this._onTargetDestroyed.bind(this);
|
|
|
|
// Convert form, which is just JSON object to Fronts for these two events
|
|
this.on("target-available-form", this._onTargetAvailable);
|
|
this.on("target-destroyed-form", this._onTargetDestroyed);
|
|
}
|
|
|
|
form(json) {
|
|
this.actorID = json.actor;
|
|
this.traits = json.traits;
|
|
}
|
|
|
|
_onTargetAvailable(form) {
|
|
let front;
|
|
if (form.actor.includes("/contentProcessTarget")) {
|
|
front = new ContentProcessTargetFront(this.conn, null, this);
|
|
} else if (form.actor.includes("/workerTarget")) {
|
|
front = new WorkerTargetFront(this.conn, null, this);
|
|
} else if (form.actor.includes("/contentScriptTarget")) {
|
|
front = new ContentScriptTargetFront(this.conn, null, this);
|
|
} else {
|
|
front = new WindowGlobalTargetFront(this.conn, null, this);
|
|
}
|
|
front.actorID = form.actor;
|
|
front.form(form);
|
|
this.manage(front);
|
|
this.emit("target-available", front);
|
|
}
|
|
|
|
_onTargetDestroyed(form, options = {}) {
|
|
const front = this._getTargetFront(form);
|
|
|
|
// When server side target switching is off,
|
|
// the watcher may notify us about the top level target destruction a bit late.
|
|
// The descriptor (`this.parentFront`) already switched to the new target.
|
|
// Missing `target-destroyed` isn't critical when target switching is off
|
|
// as `TargetCommand.switchToTarget` will end calling `TargetCommand.onTargetDestroyed` for all
|
|
// existing targets.
|
|
// https://searchfox.org/mozilla-central/rev/af8e5d37fd56be90ccddae2203e7b875d3f3ae87/devtools/shared/commands/target/target-command.js#166-173
|
|
if (front) {
|
|
this.emit("target-destroyed", front, options);
|
|
}
|
|
}
|
|
|
|
_getTargetFront(form) {
|
|
let front = this.getActorByID(form.actor);
|
|
// For top level target, the target will be a child of the descriptor front,
|
|
// which happens to be the parent front of the watcher.
|
|
if (!front) {
|
|
front = this.parentFront.getActorByID(form.actor);
|
|
}
|
|
return front;
|
|
}
|
|
|
|
/**
|
|
* Retrieve the already existing WindowGlobalTargetFront for the parent
|
|
* BrowsingContext of the given BrowsingContext ID.
|
|
*/
|
|
async getParentWindowGlobalTarget(browsingContextID) {
|
|
const id = await this.getParentBrowsingContextID(browsingContextID);
|
|
if (!id) {
|
|
return null;
|
|
}
|
|
return this.getWindowGlobalTarget(id);
|
|
}
|
|
|
|
/**
|
|
* Memoized getter for the "blackboxing" actor
|
|
*/
|
|
async getBlackboxingActor() {
|
|
if (!this._blackboxingActor) {
|
|
this._blackboxingActor = await super.getBlackboxingActor();
|
|
}
|
|
return this._blackboxingActor;
|
|
}
|
|
/**
|
|
* Memoized getter for the "breakpoint-list" actor
|
|
*/
|
|
async getBreakpointListActor() {
|
|
if (!this._breakpointListActor) {
|
|
this._breakpointListActor = await super.getBreakpointListActor();
|
|
}
|
|
return this._breakpointListActor;
|
|
}
|
|
|
|
/**
|
|
* Memoized getter for the "target-configuration" actor
|
|
*/
|
|
async getTargetConfigurationActor() {
|
|
if (!this._targetConfigurationActor) {
|
|
this._targetConfigurationActor =
|
|
await super.getTargetConfigurationActor();
|
|
}
|
|
return this._targetConfigurationActor;
|
|
}
|
|
|
|
/**
|
|
* Memoized getter for the "thread-configuration" actor
|
|
*/
|
|
async getThreadConfigurationActor() {
|
|
if (!this._threadConfigurationActor) {
|
|
this._threadConfigurationActor =
|
|
await super.getThreadConfigurationActor();
|
|
}
|
|
return this._threadConfigurationActor;
|
|
}
|
|
|
|
/**
|
|
* For a given BrowsingContext ID, return the already existing WindowGlobalTargetFront
|
|
*/
|
|
async getWindowGlobalTarget(id) {
|
|
// First scan the watcher children as the watcher manages all the targets
|
|
for (const front of this.poolChildren()) {
|
|
if (front.browsingContextID == id) {
|
|
return front;
|
|
}
|
|
}
|
|
// But the top level target will be created by the Descriptor.getTarget() method
|
|
// and so be hosted in the Descriptor's pool.
|
|
// The parent front of the WatcherActor happens to be the Descriptor Actor.
|
|
// This code could go away or be simplified if the Descriptor starts fetch all
|
|
// the targets, including the top level one via the Watcher. i.e. drop Descriptor.getTarget().
|
|
const topLevelTarget = await this.parentFront.getTarget();
|
|
if (topLevelTarget?.browsingContextID == id) {
|
|
return topLevelTarget;
|
|
}
|
|
|
|
// If we could not find a window global target for the provided id, the
|
|
// window global might not be the topmost one of a given process (isProcessRoot == true).
|
|
// For now we only create targets for the top window global of each process,
|
|
// so we recursively check the parent browsing context ids
|
|
// until we find a valid target.
|
|
const parentBrowsingContextID = await this.getParentBrowsingContextID(id);
|
|
if (parentBrowsingContextID && parentBrowsingContextID !== id) {
|
|
return this.getWindowGlobalTarget(parentBrowsingContextID);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
getWindowGlobalTargetByInnerWindowId(innerWindowId) {
|
|
for (const front of this.poolChildren()) {
|
|
if (front.innerWindowId == innerWindowId) {
|
|
return front;
|
|
}
|
|
}
|
|
// Use getCachedTarget in order to have a fully synchronous method
|
|
// as the callsite in ResourceCommand benefit from being synchronous.
|
|
// Here we care only about already existing resource and do not need to
|
|
// wait for the next target to come.
|
|
const topLevelTarget = this.parentFront.getCachedTarget();
|
|
if (topLevelTarget?.innerWindowId == innerWindowId) {
|
|
return topLevelTarget;
|
|
}
|
|
console.error("Unable to find target with innerWindowId:" + innerWindowId);
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Memoized getter for the "networkParent" actor
|
|
*/
|
|
async getNetworkParentActor() {
|
|
if (!this._networkParentActor) {
|
|
this._networkParentActor = await super.getNetworkParentActor();
|
|
}
|
|
return this._networkParentActor;
|
|
}
|
|
}
|
|
registerFront(WatcherFront);
|