From 6e51d6e03154350cf7c2e35147fa45d2059a384e Mon Sep 17 00:00:00 2001 From: Edmond Chui <1967998+EdmondChuiHW@users.noreply.github.com> Date: Fri, 11 Oct 2024 12:58:28 +0100 Subject: [PATCH] Add 'panel shown' perf metrics (#96) --- front_end/core/host/RNPerfMetrics.ts | 46 +++++++++++++++++++++++++--- front_end/core/host/UserMetrics.ts | 4 +++ 2 files changed, 46 insertions(+), 4 deletions(-) diff --git a/front_end/core/host/RNPerfMetrics.ts b/front_end/core/host/RNPerfMetrics.ts index 41c13264d92..cee3f2f7e57 100644 --- a/front_end/core/host/RNPerfMetrics.ts +++ b/front_end/core/host/RNPerfMetrics.ts @@ -18,11 +18,14 @@ export function getInstance(): RNPerfMetrics { return instance; } +type PanelLocation = 'main'|'drawer'; type UnsubscribeFn = () => void; class RNPerfMetrics { readonly #consoleErrorMethod = 'error'; #listeners: Set = new Set(); #launchId: string|null = null; + // map of panel location to panel name + #currentPanels: Map = new Map(); addEventListener(listener: RNReliabilityEventListener): UnsubscribeFn { this.#listeners.add(listener); @@ -243,10 +246,28 @@ class RNPerfMetrics { }); } + panelShown(_panelName: string, _isLaunching?: boolean): void { + // no-op + // We only care about the "main" and "drawer" panels for now via panelShownInLocation(…) + // (This function is called for other "sub"-panels) + } + + panelClosed(panelName: string): void { + this.sendEvent({eventName: 'PanelClosed', params: {panelName}}); + } + + panelShownInLocation(panelName: string, location: PanelLocation): void { + // The current panel name will be sent along via #decorateEvent(…) + this.sendEvent({eventName: 'PanelShown', params: {location, newPanelName: panelName}}); + // So we should only update the current panel name to the new one after sending the event + this.#currentPanels.set(location, panelName); + } + #decorateEvent(event: ReactNativeChromeDevToolsEvent): Readonly { const commonFields: CommonEventFields = { timestamp: getPerfTimestamp(), launchId: this.#launchId, + currentPanels: this.#currentPanels, }; return { @@ -278,6 +299,7 @@ function maybeWrapError(baseMessage: string, error: unknown): [string, Error] { type CommonEventFields = Readonly<{ timestamp: DOMHighResTimeStamp, launchId: string | void | null, + currentPanels: Map, }>; type EntryPoint = 'rn_fusebox'|'rn_inspector'; @@ -367,9 +389,25 @@ export type MemoryPanelActionFinishedEvent = Readonly<{ }>, }>; -export type ReactNativeChromeDevToolsEvent = EntrypointLoadingStartedEvent|EntrypointLoadingFinishedEvent| - DebuggerReadyEvent|BrowserVisibilityChangeEvent|BrowserErrorEvent|RemoteDebuggingTerminatedEvent| - DeveloperResourceLoadingStartedEvent|DeveloperResourceLoadingFinishedEvent|FuseboxSetClientMetadataStartedEvent| - FuseboxSetClientMetadataFinishedEvent|MemoryPanelActionStartedEvent|MemoryPanelActionFinishedEvent; +export type PanelShownEvent = Readonly<{ + eventName: 'PanelShown', + params: Readonly<{ + location: PanelLocation, + newPanelName: string, + }>, +}>; + +export type PanelClosedEvent = Readonly<{ + eventName: 'PanelClosed', + params: Readonly<{ + panelName: string, + }>, +}>; + +export type ReactNativeChromeDevToolsEvent = + EntrypointLoadingStartedEvent|EntrypointLoadingFinishedEvent|DebuggerReadyEvent|BrowserVisibilityChangeEvent| + BrowserErrorEvent|RemoteDebuggingTerminatedEvent|DeveloperResourceLoadingStartedEvent| + DeveloperResourceLoadingFinishedEvent|FuseboxSetClientMetadataStartedEvent|FuseboxSetClientMetadataFinishedEvent| + MemoryPanelActionStartedEvent|MemoryPanelActionFinishedEvent|PanelShownEvent|PanelClosedEvent; export type DecoratedReactNativeChromeDevToolsEvent = CommonEventFields&ReactNativeChromeDevToolsEvent; diff --git a/front_end/core/host/UserMetrics.ts b/front_end/core/host/UserMetrics.ts index ce3cb6a3dac..f80330eda05 100644 --- a/front_end/core/host/UserMetrics.ts +++ b/front_end/core/host/UserMetrics.ts @@ -30,6 +30,7 @@ import {InspectorFrontendHostInstance} from './InspectorFrontendHost.js'; import {EnumeratedHistogram} from './InspectorFrontendHostAPI.js'; +import * as RNPerfMetrics from './RNPerfMetrics.js'; export class UserMetrics { #panelChangedSinceLaunch: boolean; @@ -67,6 +68,7 @@ export class UserMetrics { if (!isLaunching) { this.#panelChangedSinceLaunch = true; } + RNPerfMetrics.getInstance().panelShown(panelName, isLaunching); } /** @@ -77,6 +79,7 @@ export class UserMetrics { InspectorFrontendHostInstance.recordEnumeratedHistogram(EnumeratedHistogram.PanelClosed, code, PanelCodes.MaxValue); // Store that the user has changed the panel so we know launch histograms should not be fired. this.#panelChangedSinceLaunch = true; + RNPerfMetrics.getInstance().panelClosed(panelName); } panelShownInLocation(panelName: string, location: 'main'|'drawer'): void { @@ -87,6 +90,7 @@ export class UserMetrics { panelWithLocation, PanelWithLocation.MaxValue, ); + RNPerfMetrics.getInstance().panelShownInLocation(panelName, location); } elementsSidebarTabShown(sidebarPaneName: string): void {