Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[No QA] [TS migration] Migrate 'Performance.js' lib to TypeScript #28271

Merged
Merged
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
146 changes: 92 additions & 54 deletions src/libs/Performance.js → src/libs/Performance.tsx
Original file line number Diff line number Diff line change
@@ -1,39 +1,75 @@
import _ from 'underscore';
import lodashTransform from 'lodash/transform';
import React, {Profiler, forwardRef} from 'react';
import {Alert, InteractionManager} from 'react-native';
import lodashTransform from 'lodash/transform';
import isObject from 'lodash/isObject';
import isEqual from 'lodash/isEqual';
import {Performance as RNPerformance, PerformanceEntry, PerformanceMark, PerformanceMeasure} from 'react-native-performance';
import {PerformanceObserverEntryList} from 'react-native-performance/lib/typescript/performance-observer';

import * as Metrics from './Metrics';
import getComponentDisplayName from './getComponentDisplayName';
import CONST from '../CONST';
import isE2ETestSession from './E2E/isE2ETestSession';

/** @type {import('react-native-performance').Performance} */
let rnPerformance;
type WrappedComponentConfig = {id: string};

type PerformanceEntriesCallback = (entry: PerformanceEntry) => void;

type Phase = 'mount' | 'update';

type WithRenderTraceHOC = <P extends Record<string, unknown>>(
WrappedComponent: React.ComponentType<P>,
) => React.ForwardRefExoticComponent<React.PropsWithoutRef<P> & React.RefAttributes<unknown>>;

type BlankHOC = <P extends Record<string, unknown>>(Component: React.ComponentType<P>) => React.ComponentType<P>;

type SetupPerformanceObserver = () => void;
type DiffObject = (object: Record<string, unknown>, base: Record<string, unknown>) => Record<string, unknown>;
type GetPerformanceMetrics = () => PerformanceEntry[];
type PrintPerformanceMetrics = () => void;
type MarkStart = (name: string, detail?: Record<string, unknown>) => PerformanceMark | void;
type MarkEnd = (name: string, detail?: Record<string, unknown>) => PerformanceMark | void;
type MeasureFailSafe = (measureName: string, startOrMeasureOptions: string, endMark: string) => void;
type MeasureTTI = (endMark: string) => void;
type TraceRender = (id: string, phase: Phase, actualDuration: number, baseDuration: number, startTime: number, commitTime: number, interactions: Set<unknown>) => PerformanceMeasure | void;
type WithRenderTrace = ({id}: WrappedComponentConfig) => WithRenderTraceHOC | BlankHOC;
type SubscribeToMeasurements = (callback: PerformanceEntriesCallback) => void;

type PerformanceModule = {
diffObject: DiffObject;
setupPerformanceObserver: SetupPerformanceObserver;
getPerformanceMetrics: GetPerformanceMetrics;
printPerformanceMetrics: PrintPerformanceMetrics;
markStart: MarkStart;
markEnd: MarkEnd;
measureFailSafe: MeasureFailSafe;
measureTTI: MeasureTTI;
traceRender: TraceRender;
withRenderTrace: WithRenderTrace;
subscribeToMeasurements: SubscribeToMeasurements;
};

let rnPerformance: RNPerformance;

/**
* Deep diff between two objects. Useful for figuring out what changed about an object from one render to the next so
* that state and props updates can be optimized.
*
* @param {Object} object
* @param {Object} base
* @return {Object}
*/
function diffObject(object, base) {
function changes(obj, comparisonObject) {
function diffObject(object: Record<string, unknown>, base: Record<string, unknown>): Record<string, unknown> {
function changes(obj: Record<string, unknown>, comparisonObject: Record<string, unknown>): Record<string, unknown> {
return lodashTransform(obj, (result, value, key) => {
if (_.isEqual(value, comparisonObject[key])) {
if (isEqual(value, comparisonObject[key])) {
return;
}

// eslint-disable-next-line no-param-reassign
result[key] = _.isObject(value) && _.isObject(comparisonObject[key]) ? changes(value, comparisonObject[key]) : value;
result[key] = isObject(value) && isObject(comparisonObject[key]) ? changes(value as Record<string, unknown>, comparisonObject[key] as Record<string, unknown>) : value;
});
}
return changes(object, base);
}

const Performance = {
const Performance: PerformanceModule = {
// When performance monitoring is disabled the implementations are blank
diffObject,
setupPerformanceObserver: () => {},
Expand All @@ -44,7 +80,11 @@ const Performance = {
measureFailSafe: () => {},
measureTTI: () => {},
traceRender: () => {},
withRenderTrace: () => (Component) => Component,
withRenderTrace:
() =>
// eslint-disable-next-line @typescript-eslint/naming-convention
<P extends Record<string, unknown>>(Component: React.ComponentType<P>): React.ComponentType<P> =>
Component,
subscribeToMeasurements: () => {},
};

Expand All @@ -53,20 +93,21 @@ if (Metrics.canCapturePerformanceMetrics()) {
perfModule.setResourceLoggingEnabled(true);
rnPerformance = perfModule.default;

Performance.measureFailSafe = (measureName, startOrMeasureOptions, endMark) => {
Performance.measureFailSafe = (measureName: string, startOrMeasureOptions: string, endMark: string) => {
try {
rnPerformance.measure(measureName, startOrMeasureOptions, endMark);
} catch (error) {
// Sometimes there might be no start mark recorded and the measure will fail with an error
console.debug(error.message);
if (error instanceof Error) {
console.debug(error.message);
}
}
};

/**
* Measures the TTI time. To be called when the app is considered to be interactive.
* @param {String} [endMark] Optional end mark name
*/
Performance.measureTTI = (endMark) => {
Performance.measureTTI = (endMark: string) => {
// Make sure TTI is captured when the app is really usable
InteractionManager.runAfterInteractions(() => {
requestAnimationFrame(() => {
Expand All @@ -88,8 +129,8 @@ if (Metrics.canCapturePerformanceMetrics()) {
performanceReported.setupDefaultFlipperReporter();

// Monitor some native marks that we want to put on the timeline
new perfModule.PerformanceObserver((list, observer) => {
list.getEntries().forEach((entry) => {
new perfModule.PerformanceObserver((list: PerformanceObserverEntryList, observer: PerformanceObserver) => {
list.getEntries().forEach((entry: PerformanceEntry) => {
if (entry.name === 'nativeLaunchEnd') {
Performance.measureFailSafe('nativeLaunch', 'nativeLaunchStart', 'nativeLaunchEnd');
}
Expand All @@ -108,8 +149,8 @@ if (Metrics.canCapturePerformanceMetrics()) {
}).observe({type: 'react-native-mark', buffered: true});

// Monitor for "_end" marks and capture "_start" to "_end" measures
new perfModule.PerformanceObserver((list) => {
list.getEntriesByType('mark').forEach((mark) => {
new perfModule.PerformanceObserver((list: PerformanceObserverEntryList) => {
list.getEntriesByType('mark').forEach((mark: PerformanceEntry) => {
if (mark.name.endsWith('_end')) {
const end = mark.name;
const name = end.replace(/_end$/, '');
Expand All @@ -125,65 +166,64 @@ if (Metrics.canCapturePerformanceMetrics()) {
}).observe({type: 'mark', buffered: true});
};

Performance.getPerformanceMetrics = () =>
_.chain([
Performance.getPerformanceMetrics = (): PerformanceEntry[] =>
[
...rnPerformance.getEntriesByName('nativeLaunch'),
...rnPerformance.getEntriesByName('runJsBundle'),
...rnPerformance.getEntriesByName('jsBundleDownload'),
...rnPerformance.getEntriesByName('TTI'),
...rnPerformance.getEntriesByName('regularAppStart'),
...rnPerformance.getEntriesByName('appStartedToReady'),
])
.filter((entry) => entry.duration > 0)
.value();
].filter((entry) => entry.duration > 0);

/**
* Outputs performance stats. We alert these so that they are easy to access in release builds.
*/
Performance.printPerformanceMetrics = () => {
const stats = Performance.getPerformanceMetrics();
const statsAsText = _.map(stats, (entry) => `\u2022 ${entry.name}: ${entry.duration.toFixed(1)}ms`).join('\n');
const statsAsText = stats.map((entry) => `\u2022 ${entry.name}: ${entry.duration.toFixed(1)}ms`).join('\n');

if (stats.length > 0) {
Alert.alert('Performance', statsAsText);
}
};

Performance.subscribeToMeasurements = (callback) => {
new perfModule.PerformanceObserver((list) => {
Performance.subscribeToMeasurements = (callback: PerformanceEntriesCallback) => {
new perfModule.PerformanceObserver((list: PerformanceObserverEntryList) => {
list.getEntriesByType('measure').forEach(callback);
}).observe({type: 'measure', buffered: true});
};

/**
* Add a start mark to the performance entries
* @param {string} name
* @param {Object} [detail]
* @returns {PerformanceMark}
*/
Performance.markStart = (name, detail) => rnPerformance.mark(`${name}_start`, {detail});
Performance.markStart = (name: string, detail?: Record<string, unknown>): PerformanceMark => rnPerformance.mark(`${name}_start`, {detail});

/**
* Add an end mark to the performance entries
* A measure between start and end is captured automatically
* @param {string} name
* @param {Object} [detail]
* @returns {PerformanceMark}
*/
Performance.markEnd = (name, detail) => rnPerformance.mark(`${name}_end`, {detail});
Performance.markEnd = (name: string, detail?: Record<string, unknown>): PerformanceMark => rnPerformance.mark(`${name}_end`, {detail});

/**
* Put data emitted by Profiler components on the timeline
* @param {string} id the "id" prop of the Profiler tree that has just committed
* @param {'mount'|'update'} phase either "mount" (if the tree just mounted) or "update" (if it re-rendered)
* @param {number} actualDuration time spent rendering the committed update
* @param {number} baseDuration estimated time to render the entire subtree without memoization
* @param {number} startTime when React began rendering this update
* @param {number} commitTime when React committed this update
* @param {Set} interactions the Set of interactions belonging to this update
* @returns {PerformanceMeasure}
* @param id the "id" prop of the Profiler tree that has just committed
* @param phase either "mount" (if the tree just mounted) or "update" (if it re-rendered)
* @param actualDuration time spent rendering the committed update
* @param baseDuration estimated time to render the entire subtree without memoization
* @param startTime when React began rendering this update
* @param commitTime when React committed this update
* @param interactions the Set of interactions belonging to this update
*/
Performance.traceRender = (id, phase, actualDuration, baseDuration, startTime, commitTime, interactions) =>
Performance.traceRender = (
id: string,
phase: Phase,
actualDuration: number,
baseDuration: number,
startTime: number,
commitTime: number,
interactions: Set<unknown>,
): PerformanceMeasure =>
rnPerformance.measure(id, {
start: startTime,
duration: actualDuration,
Expand All @@ -197,14 +237,12 @@ if (Metrics.canCapturePerformanceMetrics()) {

/**
* A HOC that captures render timings of the Wrapped component
* @param {object} config
* @param {string} config.id
* @returns {function(React.Component): React.FunctionComponent}
*/
Performance.withRenderTrace =
({id}) =>
VickyStash marked this conversation as resolved.
Show resolved Hide resolved
(WrappedComponent) => {
const WithRenderTrace = forwardRef((props, ref) => (
({id}: WrappedComponentConfig) =>
// eslint-disable-next-line @typescript-eslint/naming-convention
<P extends Record<string, unknown>>(WrappedComponent: React.ComponentType<P>): React.ForwardRefExoticComponent<React.PropsWithoutRef<P> & React.RefAttributes<unknown>> => {
VickyStash marked this conversation as resolved.
Show resolved Hide resolved
const WithRenderTrace = forwardRef((props: P, ref) => (
<Profiler
id={id}
onRender={Performance.traceRender}
Expand All @@ -217,7 +255,7 @@ if (Metrics.canCapturePerformanceMetrics()) {
</Profiler>
));

WithRenderTrace.displayName = `withRenderTrace(${getComponentDisplayName(WrappedComponent)})`;
WithRenderTrace.displayName = `withRenderTrace(${getComponentDisplayName(WrappedComponent as React.ComponentType)})`;
return WithRenderTrace;
};
}
Expand Down
Loading