chrome-devtools-frontend
Version:
Chrome DevTools UI
1,091 lines (948 loc) • 27.8 kB
JavaScript
// Copyright 2020 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 Platform from '../platform/platform.js'; // eslint-disable-line no-unused-vars
const originalConsole = console;
const originalAssert = console.assert;
/** @type {!URLSearchParams} */
const queryParamsObject = new URLSearchParams(location.search);
// The following variable are initialized all the way at the bottom of this file
/** @type {string} */
let importScriptPathPrefix;
let runtimePlatform = '';
/** @type {function(string):!Platform.UIString.LocalizedString} */
let l10nCallback;
/** @type {!Runtime|undefined} */
let runtimeInstance;
export function getRemoteBase(location = self.location.toString()) {
const url = new URL(location);
const remoteBase = url.searchParams.get('remoteBase');
if (!remoteBase) {
return null;
}
const version = /\/serve_file\/(@[0-9a-zA-Z]+)\/?$/.exec(remoteBase);
if (!version) {
return null;
}
return {base: `${url.origin}/remote/serve_file/${version[1]}/`, version: version[1]};
}
/** @type {!WeakMap<function(new:?), ?>} */
const constructedInstances = new WeakMap();
export class Runtime {
/**
* @private
* @param {!Array.<!ModuleDescriptor>} descriptors
*/
constructor(descriptors) {
/** @type {!Array<!Module>} */
this._modules = [];
/** @type {!Object<string, !Module>} */
this._modulesMap = {};
/** @type {!Array<!Extension>} */
this._extensions = [];
/** @type {!Object<string, function(new:Object):void>} */
this._cachedTypeClasses = {};
/** @type {!Object<string, !ModuleDescriptor>} */
this._descriptorsMap = {};
for (const descriptor of descriptors) {
this._registerModule(descriptor);
}
}
/**
* @param {{forceNew: ?boolean, moduleDescriptors: ?Array.<!ModuleDescriptor>}=} opts
* @return {!Runtime}
*/
static instance(opts = {forceNew: null, moduleDescriptors: null}) {
const {forceNew, moduleDescriptors} = opts;
if (!runtimeInstance || forceNew) {
if (!moduleDescriptors) {
throw new Error(`Unable to create runtime: moduleDescriptors must be provided: ${new Error().stack}`);
}
runtimeInstance = new Runtime(moduleDescriptors);
}
return runtimeInstance;
}
static removeInstance() {
runtimeInstance = undefined;
}
/**
* http://tools.ietf.org/html/rfc3986#section-5.2.4
* @param {string} path
* @return {string}
*/
static normalizePath(path) {
if (path.indexOf('..') === -1 && path.indexOf('.') === -1) {
return path;
}
const normalizedSegments = [];
const segments = path.split('/');
for (const segment of segments) {
if (segment === '.') {
continue;
} else if (segment === '..') {
normalizedSegments.pop();
} else if (segment) {
normalizedSegments.push(segment);
}
}
let normalizedPath = normalizedSegments.join('/');
if (normalizedPath[normalizedPath.length - 1] === '/') {
return normalizedPath;
}
if (path[0] === '/' && normalizedPath) {
normalizedPath = '/' + normalizedPath;
}
if ((path[path.length - 1] === '/') || (segments[segments.length - 1] === '.') ||
(segments[segments.length - 1] === '..')) {
normalizedPath = normalizedPath + '/';
}
return normalizedPath;
}
/**
* @param {string} name
* @return {?string}
*/
static queryParam(name) {
return queryParamsObject.get(name);
}
/**
* @return {!Object<string,boolean>}
*/
static _experimentsSetting() {
try {
return /** @type {!Object<string,boolean>} */ (
JSON.parse(self.localStorage && self.localStorage['experiments'] ? self.localStorage['experiments'] : '{}'));
} catch (e) {
console.error('Failed to parse localStorage[\'experiments\']');
return {};
}
}
/**
* @param {*} value
* @param {string} message
*/
static _assert(value, message) {
if (value) {
return;
}
originalAssert.call(originalConsole, value, message + ' ' + new Error().stack);
}
/**
* @param {string} platform
*/
static setPlatform(platform) {
runtimePlatform = platform;
}
static platform() {
return runtimePlatform;
}
/**
* @param {!{experiment: (?string|undefined), condition: (?string|undefined)}} descriptor
* @return {boolean}
*/
static isDescriptorEnabled(descriptor) {
const activatorExperiment = descriptor['experiment'];
if (activatorExperiment === '*') {
return true;
}
if (activatorExperiment && activatorExperiment.startsWith('!') &&
experiments.isEnabled(activatorExperiment.substring(1))) {
return false;
}
if (activatorExperiment && !activatorExperiment.startsWith('!') && !experiments.isEnabled(activatorExperiment)) {
return false;
}
const condition = descriptor['condition'];
if (condition && !condition.startsWith('!') && !Runtime.queryParam(condition)) {
return false;
}
if (condition && condition.startsWith('!') && Runtime.queryParam(condition.substring(1))) {
return false;
}
return true;
}
/**
* @param {string} path
* @return {string}
*/
static resolveSourceURL(path) {
let sourceURL = self.location.href;
if (self.location.search) {
sourceURL = sourceURL.replace(self.location.search, '');
}
sourceURL = sourceURL.substring(0, sourceURL.lastIndexOf('/') + 1) + path;
return '\n/*# sourceURL=' + sourceURL + ' */';
}
/**
* @param {function(string):!Platform.UIString.LocalizedString} localizationFunction
*/
static setL10nCallback(localizationFunction) {
l10nCallback = localizationFunction;
}
/**
* @param {string} moduleName
* @return {!Module}
*/
module(moduleName) {
return this._modulesMap[moduleName];
}
/**
* @param {!ModuleDescriptor} descriptor
*/
_registerModule(descriptor) {
const module = new Module(this, descriptor);
this._modules.push(module);
this._modulesMap[descriptor['name']] = module;
}
/**
* @param {string} moduleName
* @return {!Promise.<boolean>}
*/
loadModulePromise(moduleName) {
return this._modulesMap[moduleName]._loadPromise();
}
/**
* @param {!Array.<string>} moduleNames
* @return {!Promise.<!Array.<*>>}
*/
loadAutoStartModules(moduleNames) {
const promises = [];
for (const moduleName of moduleNames) {
promises.push(this.loadModulePromise(moduleName));
}
return Promise.all(promises);
}
/**
* @param {!Extension} extension
* @param {?function(function(new:Object)):boolean} predicate
* @return {boolean}
*/
_checkExtensionApplicability(extension, predicate) {
if (!predicate) {
return false;
}
const contextTypes = extension.descriptor().contextTypes;
if (!contextTypes) {
return true;
}
for (let i = 0; i < contextTypes.length; ++i) {
const contextType = this._resolve(contextTypes[i]);
const isMatching = contextType && predicate(contextType);
if (isMatching) {
return true;
}
}
return false;
}
/**
* @param {!Extension} extension
* @param {?Object} context
* @return {boolean}
*/
isExtensionApplicableToContext(extension, context) {
if (!context) {
return true;
}
return this._checkExtensionApplicability(extension, isInstanceOf);
/**
* @param {!Function} targetType
* @return {boolean}
*/
function isInstanceOf(targetType) {
return context instanceof targetType;
}
}
/**
* @param {!Extension} extension
* @param {!Set.<function(new:Object, ...?):void>} currentContextTypes
* @return {boolean}
*/
isExtensionApplicableToContextTypes(extension, currentContextTypes) {
if (!extension.descriptor().contextTypes) {
return true;
}
let callback = null;
if (currentContextTypes) {
/**
* @param {function(new:Object, ...?):void} targetType
* @return {boolean}
*/
callback = targetType => {
return currentContextTypes.has(targetType);
};
}
return this._checkExtensionApplicability(extension, callback);
}
/**
* @param {*} type
* @param {?Object=} context
* @param {boolean=} sortByTitle
* @return {!Array.<!Extension>}
*/
extensions(type, context, sortByTitle) {
return this._extensions.filter(filter).sort(sortByTitle ? titleComparator : orderComparator);
/**
* @param {!Extension} extension
* @return {boolean}
*/
function filter(extension) {
if (extension._type !== type && extension._typeClass() !== type) {
return false;
}
if (!extension.enabled()) {
return false;
}
return !context || extension.isApplicable(context);
}
/**
* @param {!Extension} extension1
* @param {!Extension} extension2
* @return {number}
*/
function orderComparator(extension1, extension2) {
const order1 = extension1.descriptor()['order'] || 0;
const order2 = extension2.descriptor()['order'] || 0;
return order1 - order2;
}
/**
* @param {!Extension} extension1
* @param {!Extension} extension2
* @return {number}
*/
function titleComparator(extension1, extension2) {
const title1 = extension1.title() || '';
const title2 = extension2.title() || '';
return title1.localeCompare(title2);
}
}
/**
* @param {*} type
* @param {?Object=} context
* @return {?Extension}
*/
extension(type, context) {
return this.extensions(type, context)[0] || null;
}
/**
* @param {*} type
* @param {?Object=} context
* @return {!Promise.<!Array.<!Object>>}
*/
allInstances(type, context) {
return Promise.all(this.extensions(type, context).map(extension => extension.instance()));
}
/**
* @param {string} typeName
* @return {?function(new:Object)}
*/
_resolve(typeName) {
if (!this._cachedTypeClasses[typeName]) {
/** @type {!Array<string>} */
const path = typeName.split('.');
/** @type {*} */
let object = self;
for (let i = 0; object && (i < path.length); ++i) {
object = object[path[i]];
}
if (object) {
this._cachedTypeClasses[typeName] = /** @type {function(new:Object):void} */ (object);
}
}
return this._cachedTypeClasses[typeName] || null;
}
/**
* @param {function(new:T)} constructorFunction
* @return {!T}
* @template T
*/
sharedInstance(constructorFunction) {
const instanceDescriptor = Object.getOwnPropertyDescriptor(constructorFunction, 'instance');
if (instanceDescriptor) {
const method = instanceDescriptor.value;
if (method instanceof Function) {
return method.call(null);
}
}
let instance = constructedInstances.get(constructorFunction);
if (!instance) {
instance = new constructorFunction();
constructedInstances.set(constructorFunction, instance);
}
return instance;
}
}
export class ModuleDescriptor {
constructor() {
/**
* @type {string}
*/
this.name;
/**
* @type {!Array.<!RuntimeExtensionDescriptor>}
*/
this.extensions;
/**
* @type {!Array.<string>|undefined}
*/
this.dependencies;
/**
* @type {!Array.<string>}
*/
this.scripts;
/**
* @type {!Array.<string>}
*/
this.modules;
/**
* @type {!Array.<string>}
*/
this.resources;
/**
* @type {string|undefined}
*/
this.condition;
/** @type {string|null} */
this.experiment;
}
}
// This class is named like this, because we already have an "ExtensionDescriptor" in the externs
// These two do not share the same structure
export class RuntimeExtensionDescriptor {
constructor() {
/** @type {string} */
this.type;
/**
* @type {string|undefined}
*/
this.className;
/**
* @type {string|undefined}
*/
this.factoryName;
/**
* @type {!Array.<string>|undefined}
*/
this.contextTypes;
/** @type {number} */
this.order;
/** @type {string|null} */
this.actionId;
/** @type {string|null} */
this.experiment;
/** @type {string|null} */
this.condition;
/** @type {string|null} */
this.startPage;
/** @type {string|null} */
this.name;
/** @type {string|null} */
this.destination;
/** @type {string|null} */
this.color;
/** @type {string|null} */
this.prefix;
/** @type {string|null} */
this.decoratorType;
/** @type {string|null} */
this.category;
/** @type {boolean|null|undefined} */
this.reloadRequired;
/** @type {string|null} */
this.location;
/** @type {!Array<string>|undefined} */
this.settings;
// This is an EmulatedDevice, but typing it as such introduces a
// circular dep between emulation and root.
/** @type {?} */
this.device;
/** @type {string|null} */
this.viewId;
/** @type {?string} */
this.persistence;
/** @type {?string} */
this.setting;
}
}
/**
* @typedef {{
* title: string,
* value: (string|boolean),
* raw: (boolean|undefined),
* text: (string|undefined),
* }}
*/
// @ts-ignore typedef
export let Option;
export class Module {
/**
* @param {!Runtime} manager
* @param {!ModuleDescriptor} descriptor
*/
constructor(manager, descriptor) {
this._manager = manager;
this._descriptor = descriptor;
this._name = descriptor.name;
/** @type {!Array<!Extension>} */
this._extensions = [];
/** @type {!Map<string, !Array<!Extension>>} */
this._extensionsByClassName = new Map();
const extensions = /** @type {?Array.<!RuntimeExtensionDescriptor>} */ (descriptor.extensions);
for (let i = 0; extensions && i < extensions.length; ++i) {
const extension = new Extension(this, extensions[i]);
this._manager._extensions.push(extension);
this._extensions.push(extension);
}
this._loadedForTest = false;
}
/**
* @return {string}
*/
name() {
return this._name;
}
/**
* @return {boolean}
*/
enabled() {
return Runtime.isDescriptorEnabled(this._descriptor);
}
/**
* @param {string} name
* @return {string}
*/
resource(name) {
const fullName = this._name + '/' + name;
const content = cachedResources.get(fullName);
if (!content) {
throw new Error(fullName + ' not preloaded. Check module.json');
}
return content;
}
/**
* @return {!Promise.<boolean>}
*/
_loadPromise() {
if (!this.enabled()) {
return Promise.reject(new Error('Module ' + this._name + ' is not enabled'));
}
if (this._pendingLoadPromise) {
return this._pendingLoadPromise;
}
const dependencies = this._descriptor.dependencies;
const dependencyPromises = [];
for (let i = 0; dependencies && i < dependencies.length; ++i) {
dependencyPromises.push(this._manager._modulesMap[dependencies[i]]._loadPromise());
}
this._pendingLoadPromise = Promise.all(dependencyPromises)
.then(this._loadModules.bind(this))
.then(() => {
this._loadedForTest = true;
return this._loadedForTest;
});
return this._pendingLoadPromise;
}
async _loadModules() {
const legacyFileName = `${this._name}-legacy.js`;
const moduleFileName = `${this._name}_module.js`;
const entrypointFileName = `${this._name}.js`;
// If a module has resources, they are part of the `_module.js` files that are generated
// by `build_release_applications`. These need to be loaded before any other code is
// loaded, to make sure that the resource content is properly cached in `cachedResources`.
if (this._descriptor.modules.includes(moduleFileName)) {
await import(`../${this._name}/${moduleFileName}`);
}
const fileName = this._descriptor.modules.includes(legacyFileName) ? legacyFileName : entrypointFileName;
await import(`../${this._name}/${fileName}`);
}
/**
* @param {string} resourceName
*/
_modularizeURL(resourceName) {
return Runtime.normalizePath(this._name + '/' + resourceName);
}
/**
* @param {string} resourceName
* @return {!Promise.<string>}
*/
fetchResource(resourceName) {
const sourceURL = getResourceURL(this._modularizeURL(resourceName));
return loadResourcePromise(sourceURL);
}
/**
* @param {string} value
* @return {string}
*/
substituteURL(value) {
return value.replace(/\(([^\)]*?)\)/g, convertURL.bind(this));
/**
* @param {string} match
* @param {string} url
* @this {Module}
*/
function convertURL(match, url) {
return importScriptPathPrefix + this._modularizeURL(url);
}
}
}
export class Extension {
/**
* @param {!Module} moduleParam
* @param {!RuntimeExtensionDescriptor} descriptor
*/
constructor(moduleParam, descriptor) {
this._module = moduleParam;
this._descriptor = descriptor;
this._type = descriptor.type;
this._hasTypeClass = this._type.charAt(0) === '@';
/**
* @type {?string}
*/
this._className = descriptor.className || null;
this._factoryName = descriptor.factoryName || null;
}
/**
* @return {!RuntimeExtensionDescriptor}
*/
descriptor() {
return this._descriptor;
}
/**
* @return {!Module}
*/
module() {
return this._module;
}
/**
* @return {boolean}
*/
enabled() {
return this._module.enabled() && Runtime.isDescriptorEnabled(this.descriptor());
}
/**
* @return {?function(new:Object)}
*/
_typeClass() {
if (!this._hasTypeClass) {
return null;
}
return this._module._manager._resolve(this._type.substring(1));
}
/**
* @param {?Object} context
* @return {boolean}
*/
isApplicable(context) {
return this._module._manager.isExtensionApplicableToContext(this, context);
}
/**
* @return {!Promise.<!Object>}
*/
instance() {
return this._module._loadPromise().then(this._createInstance.bind(this));
}
/**
* @return {boolean}
*/
canInstantiate() {
return Boolean(this._className || this._factoryName);
}
/**
* @return {!Object}
*/
_createInstance() {
const className = this._className || this._factoryName;
if (!className) {
throw new Error('Could not instantiate extension with no class');
}
const constructorFunction = self.eval(/** @type {string} */ (className));
if (!(constructorFunction instanceof Function)) {
throw new Error('Could not instantiate: ' + className);
}
if (this._className) {
return this._module._manager.sharedInstance(constructorFunction);
}
return new constructorFunction(this);
}
/**
* @return {!Platform.UIString.LocalizedString}
*/
title() {
// @ts-ignore Magic lookup for objects
const title = this._descriptor['title-' + runtimePlatform] || this._descriptor['title'];
if (title && l10nCallback) {
return l10nCallback(title);
}
return title;
}
/**
* @param {function(new:Object, ...?):void} contextType
* @return {boolean}
*/
hasContextType(contextType) {
const contextTypes = this.descriptor().contextTypes;
if (!contextTypes) {
return false;
}
for (let i = 0; i < contextTypes.length; ++i) {
if (contextType === this._module._manager._resolve(contextTypes[i])) {
return true;
}
}
return false;
}
}
export class ExperimentsSupport {
constructor() {
/** @type {!Array<!Experiment>} */
this._experiments = [];
/** @type {!Set<string>} */
this._experimentNames = new Set();
/** @type {!Set<string>} */
this._enabledTransiently = new Set();
/** @type {!Set<string>} */
this._enabledByDefault = new Set();
/** @type {!Set<string>} */
this._serverEnabled = new Set();
}
/**
* @return {!Array.<!Experiment>}
*/
allConfigurableExperiments() {
const result = [];
for (const experiment of this._experiments) {
if (!this._enabledTransiently.has(experiment.name)) {
result.push(experiment);
}
}
return result;
}
/**
* @return {!Array.<!Experiment>}
*/
enabledExperiments() {
return this._experiments.filter(experiment => experiment.isEnabled());
}
/**
* @param {!Object} value
*/
_setExperimentsSetting(value) {
if (!self.localStorage) {
return;
}
self.localStorage['experiments'] = JSON.stringify(value);
}
/**
* @param {string} experimentName
* @param {string} experimentTitle
* @param {boolean=} unstable
*/
register(experimentName, experimentTitle, unstable) {
Runtime._assert(
!this._experimentNames.has(experimentName), 'Duplicate registration of experiment ' + experimentName);
this._experimentNames.add(experimentName);
this._experiments.push(new Experiment(this, experimentName, experimentTitle, Boolean(unstable)));
}
/**
* @param {string} experimentName
* @return {boolean}
*/
isEnabled(experimentName) {
this._checkExperiment(experimentName);
// Check for explicitly disabled experiments first - the code could call setEnable(false) on the experiment enabled
// by default and we should respect that.
if (Runtime._experimentsSetting()[experimentName] === false) {
return false;
}
if (this._enabledTransiently.has(experimentName) || this._enabledByDefault.has(experimentName)) {
return true;
}
if (this._serverEnabled.has(experimentName)) {
return true;
}
return Boolean(Runtime._experimentsSetting()[experimentName]);
}
/**
* @param {string} experimentName
* @param {boolean} enabled
*/
setEnabled(experimentName, enabled) {
this._checkExperiment(experimentName);
const experimentsSetting = Runtime._experimentsSetting();
experimentsSetting[experimentName] = enabled;
this._setExperimentsSetting(experimentsSetting);
}
/**
* @param {!Array.<string>} experimentNames
*/
enableExperimentsTransiently(experimentNames) {
for (const experimentName of experimentNames) {
this._checkExperiment(experimentName);
this._enabledTransiently.add(experimentName);
}
}
/**
* @param {!Array.<string>} experimentNames
*/
enableExperimentsByDefault(experimentNames) {
for (const experimentName of experimentNames) {
this._checkExperiment(experimentName);
this._enabledByDefault.add(experimentName);
}
}
/**
* @param {!Array.<string>} experimentNames
*/
setServerEnabledExperiments(experimentNames) {
for (const experiment of experimentNames) {
this._checkExperiment(experiment);
this._serverEnabled.add(experiment);
}
}
/**
* @param {string} experimentName
*/
enableForTest(experimentName) {
this._checkExperiment(experimentName);
this._enabledTransiently.add(experimentName);
}
clearForTest() {
this._experiments = [];
this._experimentNames.clear();
this._enabledTransiently.clear();
this._enabledByDefault.clear();
this._serverEnabled.clear();
}
cleanUpStaleExperiments() {
const experimentsSetting = Runtime._experimentsSetting();
/** @type {!Object<string,boolean>} */
const cleanedUpExperimentSetting = {};
for (const {name: experimentName} of this._experiments) {
if (experimentsSetting.hasOwnProperty(experimentName)) {
const isEnabled = experimentsSetting[experimentName];
if (isEnabled || this._enabledByDefault.has(experimentName)) {
cleanedUpExperimentSetting[experimentName] = isEnabled;
}
}
}
this._setExperimentsSetting(cleanedUpExperimentSetting);
}
/**
* @param {string} experimentName
*/
_checkExperiment(experimentName) {
Runtime._assert(this._experimentNames.has(experimentName), 'Unknown experiment ' + experimentName);
}
}
export class Experiment {
/**
* @param {!ExperimentsSupport} experiments
* @param {string} name
* @param {string} title
* @param {boolean} unstable
*/
constructor(experiments, name, title, unstable) {
this.name = name;
this.title = title;
this.unstable = unstable;
this._experiments = experiments;
}
/**
* @return {boolean}
*/
isEnabled() {
return this._experiments.isEnabled(this.name);
}
/**
* @param {boolean} enabled
*/
setEnabled(enabled) {
this._experiments.setEnabled(this.name, enabled);
}
}
/**
* @param {string} url
* @return {!Promise.<string>}
*/
export function loadResourcePromise(url) {
return new Promise(load);
/**
* @param {function(?):void} fulfill
* @param {function(*):void} reject
*/
function load(fulfill, reject) {
const xhr = new XMLHttpRequest();
xhr.open('GET', url, true);
xhr.onreadystatechange = onreadystatechange;
/**
* @param {!Event} e
*/
function onreadystatechange(e) {
if (xhr.readyState !== XMLHttpRequest.DONE) {
return;
}
const {response} = /** @type {*} */ (e.target);
// DevTools Proxy server can mask 404s as 200s, check the body to be sure
const status = /^HTTP\/1.1 404/.test(response) ? 404 : xhr.status;
if ([0, 200, 304].indexOf(status) === -1) // Testing harness file:/// results in 0.
{
reject(new Error('While loading from url ' + url + ' server responded with a status of ' + status));
} else {
fulfill(response);
}
}
xhr.send(null);
}
}
/**
* @param {string} scriptName
* @param {string=} base
* @return {string}
*/
function getResourceURL(scriptName, base) {
const sourceURL = (base || importScriptPathPrefix) + scriptName;
const schemaIndex = sourceURL.indexOf('://') + 3;
let pathIndex = sourceURL.indexOf('/', schemaIndex);
if (pathIndex === -1) {
pathIndex = sourceURL.length;
}
return sourceURL.substring(0, pathIndex) + Runtime.normalizePath(sourceURL.substring(pathIndex));
}
(function() {
const baseUrl = self.location ? self.location.origin + self.location.pathname : '';
importScriptPathPrefix = baseUrl.substring(0, baseUrl.lastIndexOf('/') + 1);
})();
// This must be constructed after the query parameters have been parsed.
export const experiments = new ExperimentsSupport();
/**
* @type {!Map<string, string>}
*/
export const cachedResources = new Map();
// Only exported for LightHouse, which uses it in `report-generator.js`.
// Do not use this global in DevTools' implementation.
// TODO(crbug.com/1127292): remove this global
globalThis.EXPORTED_CACHED_RESOURCES_ONLY_FOR_LIGHTHOUSE = cachedResources;
/** @type {function():void} */
export let appStartedPromiseCallback;
/** @type {!Promise<void>} */
export const appStarted = new Promise(fulfill => {
appStartedPromiseCallback = fulfill;
});
/** @enum {string} */
export const ExperimentName = {
CAPTURE_NODE_CREATION_STACKS: 'captureNodeCreationStacks',
CSS_OVERVIEW: 'cssOverview',
LIVE_HEAP_PROFILE: 'liveHeapProfile',
DEVELOPER_RESOURCES_VIEW: 'developerResourcesView',
TIMELINE_REPLAY_EVENT: 'timelineReplayEvent',
CSP_VIOLATIONS_VIEW: 'cspViolationsView',
WASM_DWARF_DEBUGGING: 'wasmDWARFDebugging',
ALL: '*',
PROTOCOL_MONITOR: 'protocolMonitor',
WEBAUTHN_PANE: 'webauthnPane',
RECORDER: 'recorder',
};
/** @enum {string} */
export const ConditionName = {
CAN_DOCK: 'can_dock',
NOT_SOURCES_HIDE_ADD_FOLDER: '!sources.hide_add_folder',
};