blob: 8cb0d258d6f05bd3c1f37c87e06500f3e149caba [file] [log] [blame]
// Copyright (C) 2018 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// Keep this import first.
import '../base/disposable_polyfill';
import '../base/static_initializers';
import '../gen/all_plugins';
import '../gen/all_core_plugins';
import {Draft} from 'immer';
import m from 'mithril';
import {defer} from '../base/deferred';
import {addErrorHandler, reportError} from '../base/logging';
import {Store} from '../base/store';
import {Actions, DeferredAction, StateActions} from '../common/actions';
import {traceEvent} from '../common/metatracing';
import {pluginManager} from '../common/plugins';
import {State} from '../common/state';
import {initController, runControllers} from '../controller';
import {isGetCategoriesResponse} from '../controller/chrome_proxy_record_controller';
import {RECORDING_V2_FLAG, featureFlags} from '../core/feature_flags';
import {initLiveReload} from '../core/live_reload';
import {raf} from '../core/raf_scheduler';
import {initWasm} from '../trace_processor/wasm_engine_proxy';
import {setScheduleFullRedraw} from '../widgets/raf';
import {UiMain} from './ui_main';
import {initCssConstants} from './css_constants';
import {registerDebugGlobals} from './debug';
import {maybeShowErrorDialog} from './error_dialog';
import {installFileDropHandler} from './file_drop_handler';
import {FlagsPage} from './flags_page';
import {globals} from './globals';
import {HomePage} from './home_page';
import {InsightsPage} from './insights_page';
import {MetricsPage} from './metrics_page';
import {PluginsPage} from './plugins_page';
import {postMessageHandler} from './post_message_handler';
import {QueryPage} from './query_page';
import {RecordPage, updateAvailableAdbDevices} from './record_page';
import {RecordPageV2} from './record_page_v2';
import {Route, Router} from './router';
import {CheckHttpRpcConnection} from './rpc_http_dialog';
import {TraceInfoPage} from './trace_info_page';
import {maybeOpenTraceFromRoute} from './trace_url_handler';
import {ViewerPage} from './viewer_page';
import {VizPage} from './viz_page';
import {WidgetsPage} from './widgets_page';
import {HttpRpcEngine} from '../trace_processor/http_rpc_engine';
import {showModal} from '../widgets/modal';
import {initAnalytics} from './analytics';
import {IdleDetector} from './idle_detector';
import {IdleDetectorWindow} from './idle_detector_interface';
import {pageWithTrace} from './pages';
import {AppImpl} from '../core/app_impl';
import {setAddSqlTableTabImplFunction} from './sql_table_tab_interface';
import {addSqlTableTabImpl} from './sql_table_tab';
const EXTENSION_ID = 'lfmkphfpdbjijhpomgecfikhfohaoine';
const CSP_WS_PERMISSIVE_PORT = featureFlags.register({
id: 'cspAllowAnyWebsocketPort',
name: 'Relax Content Security Policy for 127.0.0.1:*',
description:
'Allows simultaneous usage of several trace_processor_shell ' +
'-D --http-port 1234 by opening ' +
'https://ui.perfetto.dev/#!/?rpc_port=1234',
defaultValue: false,
});
function setExtensionAvailability(available: boolean) {
globals.dispatch(
Actions.setExtensionAvailable({
available,
}),
);
}
function routeChange(route: Route) {
raf.scheduleFullRedraw();
maybeOpenTraceFromRoute(route);
if (route.fragment) {
// This needs to happen after the next redraw call. It's not enough
// to use setTimeout(..., 0); since that may occur before the
// redraw scheduled above.
raf.addPendingCallback(() => {
const e = document.getElementById(route.fragment);
if (e) {
e.scrollIntoView();
}
});
}
}
function setupContentSecurityPolicy() {
// Note: self and sha-xxx must be quoted, urls data: and blob: must not.
let rpcPolicy = [
'http://127.0.0.1:9001', // For trace_processor_shell --httpd.
'ws://127.0.0.1:9001', // Ditto, for the websocket RPC.
];
if (CSP_WS_PERMISSIVE_PORT.get()) {
const route = Router.parseUrl(window.location.href);
if (/^\d+$/.exec(route.args.rpc_port ?? '')) {
rpcPolicy = [
`http://127.0.0.1:${route.args.rpc_port}`,
`ws://127.0.0.1:${route.args.rpc_port}`,
];
}
}
const policy = {
'default-src': [
`'self'`,
// Google Tag Manager bootstrap.
`'sha256-LirUKeorCU4uRNtNzr8tlB11uy8rzrdmqHCX38JSwHY='`,
],
'script-src': [
`'self'`,
// TODO(b/201596551): this is required for Wasm after crrev.com/c/3179051
// and should be replaced with 'wasm-unsafe-eval'.
`'unsafe-eval'`,
'https://*.google.com',
'https://*.googleusercontent.com',
'https://www.googletagmanager.com',
'https://*.google-analytics.com',
],
'object-src': ['none'],
'connect-src': [
`'self'`,
'ws://127.0.0.1:8037', // For the adb websocket server.
'https://*.google-analytics.com',
'https://*.googleapis.com', // For Google Cloud Storage fetches.
'blob:',
'data:',
].concat(rpcPolicy),
'img-src': [
`'self'`,
'data:',
'blob:',
'https://*.google-analytics.com',
'https://www.googletagmanager.com',
'https://*.googleapis.com',
],
'style-src': [`'self'`, `'unsafe-inline'`],
'navigate-to': ['https://*.perfetto.dev', 'self'],
};
const meta = document.createElement('meta');
meta.httpEquiv = 'Content-Security-Policy';
let policyStr = '';
for (const [key, list] of Object.entries(policy)) {
policyStr += `${key} ${list.join(' ')}; `;
}
meta.content = policyStr;
document.head.appendChild(meta);
}
function setupExtentionPort(extensionLocalChannel: MessageChannel) {
// We proxy messages between the extension and the controller because the
// controller's worker can't access chrome.runtime.
const extensionPort =
// eslint-disable-next-line @typescript-eslint/strict-boolean-expressions
window.chrome && chrome.runtime
? chrome.runtime.connect(EXTENSION_ID)
: undefined;
setExtensionAvailability(extensionPort !== undefined);
if (extensionPort) {
// Send messages to keep-alive the extension port.
const interval = setInterval(() => {
extensionPort.postMessage({
method: 'ExtensionVersion',
});
}, 25000);
extensionPort.onDisconnect.addListener((_) => {
setExtensionAvailability(false);
clearInterval(interval);
void chrome.runtime.lastError; // Needed to not receive an error log.
});
// This forwards the messages from the extension to the controller.
extensionPort.onMessage.addListener(
(message: object, _port: chrome.runtime.Port) => {
if (isGetCategoriesResponse(message)) {
globals.dispatch(Actions.setChromeCategories(message));
return;
}
extensionLocalChannel.port2.postMessage(message);
},
);
}
// This forwards the messages from the controller to the extension
extensionLocalChannel.port2.onmessage = ({data}) => {
if (extensionPort) extensionPort.postMessage(data);
};
}
function main() {
// Wire up raf for widgets.
setScheduleFullRedraw(() => raf.scheduleFullRedraw());
setupContentSecurityPolicy();
// Load the css. The load is asynchronous and the CSS is not ready by the time
// appendChild returns.
const cssLoadPromise = defer<void>();
const css = document.createElement('link');
css.rel = 'stylesheet';
css.href = globals.root + 'perfetto.css';
css.onload = () => cssLoadPromise.resolve();
css.onerror = (err) => cssLoadPromise.reject(err);
const favicon = document.head.querySelector('#favicon');
if (favicon instanceof HTMLLinkElement) {
favicon.href = globals.root + 'assets/favicon.png';
}
// Load the script to detect if this is a Googler (see comments on globals.ts)
// and initialize GA after that (or after a timeout if something goes wrong).
const script = document.createElement('script');
script.src =
'https://storage.cloud.google.com/perfetto-ui-internal/is_internal_user.js';
script.async = true;
script.onerror = () => globals.logging.initialize();
script.onload = () => globals.logging.initialize();
setTimeout(() => globals.logging.initialize(), 5000);
document.head.append(script, css);
// Route errors to both the UI bugreport dialog and Analytics (if enabled).
addErrorHandler(maybeShowErrorDialog);
addErrorHandler((e) => globals.logging.logError(e));
// Add Error handlers for JS error and for uncaught exceptions in promises.
window.addEventListener('error', (e) => reportError(e));
window.addEventListener('unhandledrejection', (e) => reportError(e));
const extensionLocalChannel = new MessageChannel();
initWasm(globals.root);
initController(extensionLocalChannel.port1);
// These need to be set before globals.initialize.
const route = Router.parseUrl(window.location.href);
globals.embeddedMode = route.args.mode === 'embedded';
globals.hideSidebar = route.args.hideSidebar === true;
globals.initialize(stateActionDispatcher, initAnalytics);
globals.serviceWorkerController.install();
globals.store.subscribe(scheduleRafAndRunControllersOnStateChange);
globals.publishRedraw = () => raf.scheduleFullRedraw();
setupExtentionPort(extensionLocalChannel);
// Put debug variables in the global scope for better debugging.
registerDebugGlobals();
// Prevent pinch zoom.
document.body.addEventListener(
'wheel',
(e: MouseEvent) => {
if (e.ctrlKey) e.preventDefault();
},
{passive: false},
);
cssLoadPromise.then(() => onCssLoaded());
if (globals.testing) {
document.body.classList.add('testing');
}
(window as {} as IdleDetectorWindow).waitForPerfettoIdle = (ms?: number) => {
return new IdleDetector().waitForPerfettoIdle(ms);
};
}
function onCssLoaded() {
initCssConstants();
// Clear all the contents of the initial page (e.g. the <pre> error message)
// And replace it with the root <main> element which will be used by mithril.
document.body.innerHTML = '';
const router = new Router({
'/': HomePage,
'/flags': FlagsPage,
'/info': pageWithTrace(TraceInfoPage),
'/insights': pageWithTrace(InsightsPage),
'/metrics': pageWithTrace(MetricsPage),
'/plugins': PluginsPage,
'/query': pageWithTrace(QueryPage),
'/record': RECORDING_V2_FLAG.get() ? RecordPageV2 : RecordPage,
'/viewer': pageWithTrace(ViewerPage),
'/viz': pageWithTrace(VizPage),
'/widgets': WidgetsPage,
});
router.onRouteChanged = routeChange;
raf.domRedraw = () => {
m.render(document.body, m(UiMain, router.resolve()));
};
if (
(location.origin.startsWith('http://localhost:') ||
location.origin.startsWith('http://127.0.0.1:')) &&
!globals.embeddedMode &&
!globals.testing
) {
initLiveReload();
}
if (!RECORDING_V2_FLAG.get()) {
updateAvailableAdbDevices();
try {
navigator.usb.addEventListener('connect', () =>
updateAvailableAdbDevices(),
);
navigator.usb.addEventListener('disconnect', () =>
updateAvailableAdbDevices(),
);
} catch (e) {
console.error('WebUSB API not supported');
}
}
// Will update the chip on the sidebar footer that notifies that the RPC is
// connected. Has no effect on the controller (which will repeat this check
// before creating a new engine).
// Don't auto-open any trace URLs until we get a response here because we may
// accidentially clober the state of an open trace processor instance
// otherwise.
maybeChangeRpcPortFromFragment();
CheckHttpRpcConnection().then(() => {
const route = Router.parseUrl(window.location.href);
globals.dispatch(
Actions.maybeSetPendingDeeplink({
ts: route.args.ts,
tid: route.args.tid,
dur: route.args.dur,
pid: route.args.pid,
query: route.args.query,
visStart: route.args.visStart,
visEnd: route.args.visEnd,
}),
);
if (!globals.embeddedMode) {
installFileDropHandler();
}
// Don't allow postMessage or opening trace from route when the user says
// that they want to reuse the already loaded trace in trace processor.
const traceSource = AppImpl.instance.trace?.traceInfo.source;
if (traceSource && traceSource.type === 'HTTP_RPC') {
return;
}
// Add support for opening traces from postMessage().
window.addEventListener('message', postMessageHandler, {passive: true});
// Handles the initial ?local_cache_key=123 or ?s=permalink or ?url=...
// cases.
routeChange(route);
});
// Force one initial render to get everything in place
m.render(document.body, m(UiMain, router.resolve()));
// TODO(primiano): this injection is to break a cirular dependency. See
// comment in sql_table_tab_interface.ts. Remove once we add an extension
// point for context menus.
setAddSqlTableTabImplFunction(addSqlTableTabImpl);
// Initialize plugins, now that we are ready to go
pluginManager.initialize();
const route = Router.parseUrl(window.location.href);
for (const pluginId of (route.args.enablePlugins ?? '').split(',')) {
if (pluginManager.hasPlugin(pluginId)) {
pluginManager.activatePlugin(pluginId);
}
}
}
// If the URL is /#!?rpc_port=1234, change the default RPC port.
// For security reasons, this requires toggling a flag. Detect this and tell the
// user what to do in this case.
function maybeChangeRpcPortFromFragment() {
const route = Router.parseUrl(window.location.href);
if (route.args.rpc_port !== undefined) {
if (!CSP_WS_PERMISSIVE_PORT.get()) {
showModal({
title: 'Using a different port requires a flag change',
content: m(
'div',
m(
'span',
'For security reasons before connecting to a non-standard ' +
'TraceProcessor port you need to manually enable the flag to ' +
'relax the Content Security Policy and restart the UI.',
),
),
buttons: [
{
text: 'Take me to the flags page',
primary: true,
action: () => Router.navigate('#!/flags/cspAllowAnyWebsocketPort'),
},
],
});
} else {
HttpRpcEngine.rpcPort = route.args.rpc_port;
}
}
}
function stateActionDispatcher(actions: DeferredAction[]) {
const edits = actions.map((action) => {
return traceEvent(`action.${action.type}`, () => {
return (draft: Draft<State>) => {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
(StateActions as any)[action.type](draft, action.args);
};
});
});
globals.store.edit(edits);
}
function scheduleRafAndRunControllersOnStateChange(
store: Store<State>,
oldState: State,
) {
// Only redraw if something actually changed
if (oldState !== store.state) {
raf.scheduleFullRedraw();
}
// Run in a separate task to avoid avoid reentry.
setTimeout(runControllers, 0);
}
main();