chrome-devtools-frontend
Version:
Chrome DevTools UI
448 lines (395 loc) • 15.5 kB
text/typescript
// Copyright 2021 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import * as Common from '../common/common.js';
import * as Platform from '../platform/platform.js';
import type * as ProtocolClient from '../protocol_client/protocol_client.js';
import type * as Protocol from '../../generated/protocol.js';
import {Type as TargetType} from './Target.js';
import {Target} from './Target.js';
import {SDKModel} from './SDKModel.js';
import * as Root from '../root/root.js';
import * as Host from '../host/host.js';
import {assertNotNullOrUndefined} from '../platform/platform.js';
let targetManagerInstance: TargetManager|undefined;
type ModelClass<T = SDKModel> = new (arg1: Target) => T;
export class TargetManager extends Common.ObjectWrapper.ObjectWrapper<EventTypes> {
#targetsInternal: Set<Target>;
readonly #observers: Set<Observer>;
/* eslint-disable @typescript-eslint/no-explicit-any */
#modelListeners: Platform.MapUtilities.Multimap<string|symbol|number, {
modelClass: ModelClass,
thisObject: Object|undefined,
listener: Common.EventTarget.EventListener<any, any>,
wrappedListener: Common.EventTarget.EventListener<any, any>,
}>;
readonly #modelObservers: Platform.MapUtilities.Multimap<ModelClass, SDKModelObserver<any>>;
#scopedObservers: WeakSet<Observer|SDKModelObserver<any>>;
/* eslint-enable @typescript-eslint/no-explicit-any */
#isSuspended: boolean;
#browserTargetInternal: Target|null;
#scopeTarget: Target|null;
#defaultScopeSet: boolean;
readonly #scopeChangeListeners: Set<() => void>;
private constructor() {
super();
this.#targetsInternal = new Set();
this.#observers = new Set();
this.#modelListeners = new Platform.MapUtilities.Multimap();
this.#modelObservers = new Platform.MapUtilities.Multimap();
this.#isSuspended = false;
this.#browserTargetInternal = null;
this.#scopeTarget = null;
this.#scopedObservers = new WeakSet();
this.#defaultScopeSet = false;
this.#scopeChangeListeners = new Set();
}
static instance({forceNew}: {
forceNew: boolean,
} = {forceNew: false}): TargetManager {
if (!targetManagerInstance || forceNew) {
targetManagerInstance = new TargetManager();
}
return targetManagerInstance;
}
static removeInstance(): void {
targetManagerInstance = undefined;
}
onInspectedURLChange(target: Target): void {
this.dispatchEventToListeners(Events.InspectedURLChanged, target);
}
onNameChange(target: Target): void {
this.dispatchEventToListeners(Events.NameChanged, target);
}
async suspendAllTargets(reason?: string): Promise<void> {
if (this.#isSuspended) {
return;
}
this.#isSuspended = true;
this.dispatchEventToListeners(Events.SuspendStateChanged);
const suspendPromises = Array.from(this.#targetsInternal.values(), target => target.suspend(reason));
await Promise.all(suspendPromises);
}
async resumeAllTargets(): Promise<void> {
if (!this.#isSuspended) {
return;
}
this.#isSuspended = false;
this.dispatchEventToListeners(Events.SuspendStateChanged);
const resumePromises = Array.from(this.#targetsInternal.values(), target => target.resume());
await Promise.all(resumePromises);
}
allTargetsSuspended(): boolean {
return this.#isSuspended;
}
models<T extends SDKModel>(modelClass: ModelClass<T>, opts?: {scoped: boolean}): T[] {
const result = [];
for (const target of this.#targetsInternal) {
if (opts?.scoped && !this.isInScope(target)) {
continue;
}
const model = target.model(modelClass);
if (!model) {
continue;
}
result.push(model);
}
return result;
}
inspectedURL(): string {
const mainTarget = this.primaryPageTarget();
return mainTarget ? mainTarget.inspectedURL() : '';
}
observeModels<T extends SDKModel>(modelClass: ModelClass<T>, observer: SDKModelObserver<T>, opts?: {scoped: boolean}):
void {
const models = this.models(modelClass, opts);
this.#modelObservers.set(modelClass, observer);
if (opts?.scoped) {
this.#scopedObservers.add(observer);
}
for (const model of models) {
observer.modelAdded(model);
}
}
unobserveModels<T extends SDKModel>(modelClass: ModelClass<T>, observer: SDKModelObserver<T>): void {
this.#modelObservers.delete(modelClass, observer);
this.#scopedObservers.delete(observer);
}
modelAdded(target: Target, modelClass: ModelClass, model: SDKModel, inScope: boolean): void {
for (const observer of this.#modelObservers.get(modelClass).values()) {
if (!this.#scopedObservers.has(observer) || inScope) {
observer.modelAdded(model);
}
}
}
private modelRemoved(target: Target, modelClass: ModelClass, model: SDKModel, inScope: boolean): void {
for (const observer of this.#modelObservers.get(modelClass).values()) {
if (!this.#scopedObservers.has(observer) || inScope) {
observer.modelRemoved(model);
}
}
}
addModelListener<Events, T extends keyof Events>(
modelClass: ModelClass<SDKModel<Events>>, eventType: T, listener: Common.EventTarget.EventListener<Events, T>,
thisObject?: Object, opts?: {scoped: boolean}): void {
const wrappedListener = (event: Common.EventTarget.EventTargetEvent<Events[T], Events>): void => {
if (!opts?.scoped || this.isInScope(event)) {
listener.call(thisObject, event);
}
};
for (const model of this.models(modelClass)) {
model.addEventListener(eventType, wrappedListener);
}
this.#modelListeners.set(eventType, {modelClass, thisObject, listener, wrappedListener});
}
removeModelListener<Events, T extends keyof Events>(
modelClass: ModelClass<SDKModel<Events>>, eventType: T, listener: Common.EventTarget.EventListener<Events, T>,
thisObject?: Object): void {
if (!this.#modelListeners.has(eventType)) {
return;
}
let wrappedListener = null;
for (const info of this.#modelListeners.get(eventType)) {
if (info.modelClass === modelClass && info.listener === listener && info.thisObject === thisObject) {
wrappedListener = info.wrappedListener;
this.#modelListeners.delete(eventType, info);
}
}
if (wrappedListener) {
for (const model of this.models(modelClass)) {
model.removeEventListener(eventType, wrappedListener);
}
}
}
observeTargets(targetObserver: Observer, opts?: {scoped: boolean}): void {
if (this.#observers.has(targetObserver)) {
throw new Error('Observer can only be registered once');
}
if (opts?.scoped) {
this.#scopedObservers.add(targetObserver);
}
for (const target of this.#targetsInternal) {
if (!opts?.scoped || this.isInScope(target)) {
targetObserver.targetAdded(target);
}
}
this.#observers.add(targetObserver);
}
unobserveTargets(targetObserver: Observer): void {
this.#observers.delete(targetObserver);
this.#scopedObservers.delete(targetObserver);
}
createTarget(
id: Protocol.Target.TargetID|'main', name: string, type: TargetType, parentTarget: Target|null,
sessionId?: string, waitForDebuggerInPage?: boolean, connection?: ProtocolClient.InspectorBackend.Connection,
targetInfo?: Protocol.Target.TargetInfo): Target {
const target = new Target(
this, id, name, type, parentTarget, sessionId || '', this.#isSuspended, connection || null, targetInfo);
if (waitForDebuggerInPage) {
void target.pageAgent().invoke_waitForDebugger();
}
target.createModels(new Set(this.#modelObservers.keysArray()));
this.#targetsInternal.add(target);
const inScope = this.isInScope(target);
// Iterate over a copy. #observers might be modified during iteration.
for (const observer of [...this.#observers]) {
if (!this.#scopedObservers.has(observer) || inScope) {
observer.targetAdded(target);
}
}
for (const [modelClass, model] of target.models().entries()) {
this.modelAdded(target, modelClass, model, inScope);
}
for (const key of this.#modelListeners.keysArray()) {
for (const info of this.#modelListeners.get(key)) {
const model = target.model(info.modelClass);
if (model) {
model.addEventListener(key, info.wrappedListener);
}
}
}
if ((target === target.outermostTarget() &&
(target.type() !== TargetType.Frame || target === this.primaryPageTarget())) &&
!this.#defaultScopeSet) {
this.setScopeTarget(target);
}
return target;
}
removeTarget(target: Target): void {
if (!this.#targetsInternal.has(target)) {
return;
}
const inScope = this.isInScope(target);
this.#targetsInternal.delete(target);
for (const modelClass of target.models().keys()) {
const model = target.models().get(modelClass);
assertNotNullOrUndefined(model);
this.modelRemoved(target, modelClass, model, inScope);
}
// Iterate over a copy. #observers might be modified during iteration.
for (const observer of [...this.#observers]) {
if (!this.#scopedObservers.has(observer) || inScope) {
observer.targetRemoved(target);
}
}
for (const key of this.#modelListeners.keysArray()) {
for (const info of this.#modelListeners.get(key)) {
const model = target.model(info.modelClass);
if (model) {
model.removeEventListener(key, info.wrappedListener);
}
}
}
}
targets(): Target[] {
return [...this.#targetsInternal];
}
targetById(id: string): Target|null {
// TODO(dgozman): add a map #id -> #target.
return this.targets().find(target => target.id() === id) || null;
}
rootTarget(): Target|null {
return this.#targetsInternal.size ? this.#targetsInternal.values().next().value : null;
}
primaryPageTarget(): Target|null {
let target = this.rootTarget();
if (target?.type() === TargetType.Tab) {
target =
this.targets().find(
t => t.parentTarget() === target && t.type() === TargetType.Frame && !t.targetInfo()?.subtype?.length) ||
null;
}
return target;
}
browserTarget(): Target|null {
return this.#browserTargetInternal;
}
async maybeAttachInitialTarget(): Promise<boolean> {
if (!Boolean(Root.Runtime.Runtime.queryParam('browserConnection'))) {
return false;
}
if (!this.#browserTargetInternal) {
this.#browserTargetInternal = new Target(
this, /* #id*/ 'main', /* #name*/ 'browser', TargetType.Browser, /* #parentTarget*/ null,
/* #sessionId */ '', /* suspended*/ false, /* #connection*/ null, /* targetInfo*/ undefined);
this.#browserTargetInternal.createModels(new Set(this.#modelObservers.keysArray()));
}
const targetId =
await Host.InspectorFrontendHost.InspectorFrontendHostInstance.initialTargetId() as Protocol.Target.TargetID;
// Do not await for Target.autoAttachRelated to return, as it goes throguh the renderer and we don't want to block early
// at front-end initialization if a renderer is stuck. The rest of #target discovery and auto-attach process should happen
// asynchronously upon Target.attachedToTarget.
void this.#browserTargetInternal.targetAgent().invoke_autoAttachRelated({
targetId,
waitForDebuggerOnStart: true,
});
return true;
}
clearAllTargetsForTest(): void {
this.#targetsInternal.clear();
}
// eslint-disable-next-line @typescript-eslint/no-explicit-any
isInScope(arg: SDKModel|Target|Common.EventTarget.EventTargetEvent<any, any>|null): boolean {
if (!arg) {
return false;
}
if (isSDKModelEvent(arg)) {
arg = arg.source as SDKModel;
}
if (arg instanceof SDKModel) {
arg = arg.target();
}
while (arg && arg !== this.#scopeTarget) {
arg = arg.parentTarget();
}
return Boolean(arg) && arg === this.#scopeTarget;
}
// Sets a root of a scope substree.
// TargetManager API invoked with `scoped: true` will behave as if targets
// outside of the scope subtree don't exist. Concretely this means that
// target observers, model observers and model listeners won't be invoked for targets outside of the
// scope tree. This method will invoke targetRemoved and modelRemoved for
// objects in the previous scope, as if they disappear and then will invoke
// targetAdded and modelAdded as if they just appeared.
// Note that scopeTarget could be null, which will effectively prevent scoped
// observes from getting any events.
setScopeTarget(scopeTarget: Target|null): void {
if (scopeTarget === this.#scopeTarget) {
return;
}
for (const target of this.targets()) {
if (!this.isInScope(target)) {
continue;
}
for (const modelClass of this.#modelObservers.keysArray()) {
const model = target.models().get(modelClass);
if (!model) {
continue;
}
for (const observer of [...this.#modelObservers.get(modelClass)].filter(o => this.#scopedObservers.has(o))) {
observer.modelRemoved(model);
}
}
// Iterate over a copy. #observers might be modified during iteration.
for (const observer of [...this.#observers].filter(o => this.#scopedObservers.has(o))) {
observer.targetRemoved(target);
}
}
this.#scopeTarget = scopeTarget;
for (const target of this.targets()) {
if (!this.isInScope(target)) {
continue;
}
for (const observer of [...this.#observers].filter(o => this.#scopedObservers.has(o))) {
observer.targetAdded(target);
}
for (const [modelClass, model] of target.models().entries()) {
for (const observer of [...this.#modelObservers.get(modelClass)].filter(o => this.#scopedObservers.has(o))) {
observer.modelAdded(model);
}
}
}
for (const scopeChangeListener of this.#scopeChangeListeners) {
scopeChangeListener();
}
}
addScopeChangeListener(listener: () => void): void {
this.#scopeChangeListeners.add(listener);
}
removeScopeChangeListener(listener: () => void): void {
this.#scopeChangeListeners.delete(listener);
}
scopeTarget(): Target|null {
return this.#scopeTarget;
}
}
// TODO(crbug.com/1167717): Make this a const enum again
// eslint-disable-next-line rulesdir/const_enum
export enum Events {
AvailableTargetsChanged = 'AvailableTargetsChanged',
InspectedURLChanged = 'InspectedURLChanged',
NameChanged = 'NameChanged',
SuspendStateChanged = 'SuspendStateChanged',
}
export type EventTypes = {
[Events.AvailableTargetsChanged]: Protocol.Target.TargetInfo[],
[Events.InspectedURLChanged]: Target,
[Events.NameChanged]: Target,
[Events.SuspendStateChanged]: void,
};
export class Observer {
targetAdded(_target: Target): void {
}
targetRemoved(_target: Target): void {
}
}
export class SDKModelObserver<T> {
modelAdded(_model: T): void {
}
modelRemoved(_model: T): void {
}
}
// eslint-disable-next-line @typescript-eslint/no-explicit-any
function isSDKModelEvent(arg: any): arg is Common.EventTarget.EventTargetEvent<any, any> {
return 'source' in arg && arg.source instanceof SDKModel;
}