@angular/benchpress
Version:
Benchpress - a framework for e2e performance tests
394 lines • 55.6 kB
JavaScript
/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
var PerflogMetric_1;
import { __decorate, __metadata, __param } from "tslib";
import { Inject, Injectable, InjectionToken } from '@angular/core';
import { Options } from '../common_options';
import { Metric } from '../metric';
import { WebDriverExtension } from '../web_driver_extension';
/**
* A metric that reads out the performance log
*/
let PerflogMetric = PerflogMetric_1 = class PerflogMetric extends Metric {
/**
* @param driverExtension
* @param setTimeout
* @param microMetrics Name and description of metrics provided via console.time / console.timeEnd
* @param ignoreNavigation If true, don't measure from navigationStart events. These events are
* usually triggered by a page load, but can also be triggered when adding iframes to the DOM.
**/
constructor(_driverExtension, _setTimeout, _microMetrics, _forceGc, _captureFrames, _receivedData, _requestCount, _ignoreNavigation) {
super();
this._driverExtension = _driverExtension;
this._setTimeout = _setTimeout;
this._microMetrics = _microMetrics;
this._forceGc = _forceGc;
this._captureFrames = _captureFrames;
this._receivedData = _receivedData;
this._requestCount = _requestCount;
this._ignoreNavigation = _ignoreNavigation;
this._remainingEvents = [];
this._measureCount = 0;
this._perfLogFeatures = _driverExtension.perfLogFeatures();
if (!this._perfLogFeatures.userTiming) {
// User timing is needed for navigationStart.
this._receivedData = false;
this._requestCount = false;
}
}
describe() {
const res = {
'scriptTime': 'script execution time in ms, including gc and render',
'pureScriptTime': 'script execution time in ms, without gc nor render'
};
if (this._perfLogFeatures.render) {
res['renderTime'] = 'render time in ms';
}
if (this._perfLogFeatures.gc) {
res['gcTime'] = 'gc time in ms';
res['gcAmount'] = 'gc amount in kbytes';
res['majorGcTime'] = 'time of major gcs in ms';
if (this._forceGc) {
res['forcedGcTime'] = 'forced gc time in ms';
res['forcedGcAmount'] = 'forced gc amount in kbytes';
}
}
if (this._receivedData) {
res['receivedData'] = 'encoded bytes received since navigationStart';
}
if (this._requestCount) {
res['requestCount'] = 'count of requests sent since navigationStart';
}
if (this._captureFrames) {
if (!this._perfLogFeatures.frameCapture) {
const warningMsg = 'WARNING: Metric requested, but not supported by driver';
// using dot syntax for metric name to keep them grouped together in console reporter
res['frameTime.mean'] = warningMsg;
res['frameTime.worst'] = warningMsg;
res['frameTime.best'] = warningMsg;
res['frameTime.smooth'] = warningMsg;
}
else {
res['frameTime.mean'] = 'mean frame time in ms (target: 16.6ms for 60fps)';
res['frameTime.worst'] = 'worst frame time in ms';
res['frameTime.best'] = 'best frame time in ms';
res['frameTime.smooth'] = 'percentage of frames that hit 60fps';
}
}
for (const name in this._microMetrics) {
res[name] = this._microMetrics[name];
}
return res;
}
beginMeasure() {
let resultPromise = Promise.resolve(null);
if (this._forceGc) {
resultPromise = resultPromise.then((_) => this._driverExtension.gc());
}
return resultPromise.then((_) => this._beginMeasure());
}
endMeasure(restart) {
if (this._forceGc) {
return this._endPlainMeasureAndMeasureForceGc(restart);
}
else {
return this._endMeasure(restart);
}
}
/** @internal */
_endPlainMeasureAndMeasureForceGc(restartMeasure) {
return this._endMeasure(true).then((measureValues) => {
// disable frame capture for measurements during forced gc
const originalFrameCaptureValue = this._captureFrames;
this._captureFrames = false;
return this._driverExtension.gc()
.then((_) => this._endMeasure(restartMeasure))
.then((forceGcMeasureValues) => {
this._captureFrames = originalFrameCaptureValue;
measureValues['forcedGcTime'] = forceGcMeasureValues['gcTime'];
measureValues['forcedGcAmount'] = forceGcMeasureValues['gcAmount'];
return measureValues;
});
});
}
_beginMeasure() {
return this._driverExtension.timeBegin(this._markName(this._measureCount++));
}
_endMeasure(restart) {
const markName = this._markName(this._measureCount - 1);
const nextMarkName = restart ? this._markName(this._measureCount++) : null;
return this._driverExtension.timeEnd(markName, nextMarkName)
.then((_) => this._readUntilEndMark(markName));
}
_readUntilEndMark(markName, loopCount = 0, startEvent = null) {
if (loopCount > _MAX_RETRY_COUNT) {
throw new Error(`Tried too often to get the ending mark: ${loopCount}`);
}
return this._driverExtension.readPerfLog().then((events) => {
this._addEvents(events);
const result = this._aggregateEvents(this._remainingEvents, markName);
if (result) {
this._remainingEvents = events;
return result;
}
let resolve;
const promise = new Promise(res => {
resolve = res;
});
this._setTimeout(() => resolve(this._readUntilEndMark(markName, loopCount + 1)), 100);
return promise;
});
}
_addEvents(events) {
let needSort = false;
events.forEach(event => {
if (event['ph'] === 'X') {
needSort = true;
const startEvent = {};
const endEvent = {};
for (const prop in event) {
startEvent[prop] = event[prop];
endEvent[prop] = event[prop];
}
startEvent['ph'] = 'B';
endEvent['ph'] = 'E';
endEvent['ts'] = startEvent['ts'] + startEvent['dur'];
this._remainingEvents.push(startEvent);
this._remainingEvents.push(endEvent);
}
else {
this._remainingEvents.push(event);
}
});
if (needSort) {
// Need to sort because of the ph==='X' events
this._remainingEvents.sort((a, b) => {
const diff = a['ts'] - b['ts'];
return diff > 0 ? 1 : diff < 0 ? -1 : 0;
});
}
}
_aggregateEvents(events, markName) {
const result = { 'scriptTime': 0, 'pureScriptTime': 0 };
if (this._perfLogFeatures.gc) {
result['gcTime'] = 0;
result['majorGcTime'] = 0;
result['gcAmount'] = 0;
}
if (this._perfLogFeatures.render) {
result['renderTime'] = 0;
}
if (this._captureFrames) {
result['frameTime.mean'] = 0;
result['frameTime.best'] = 0;
result['frameTime.worst'] = 0;
result['frameTime.smooth'] = 0;
}
for (const name in this._microMetrics) {
result[name] = 0;
}
if (this._receivedData) {
result['receivedData'] = 0;
}
if (this._requestCount) {
result['requestCount'] = 0;
}
let markStartEvent = null;
let markEndEvent = null;
events.forEach((event) => {
const ph = event['ph'];
const name = event['name'];
// Here we are determining if this is the event signaling the start or end of our performance
// testing (this is triggered by us calling #timeBegin and #timeEnd).
//
// Previously, this was done by checking that the event name matched our mark name and that
// the phase was either "B" or "E" ("begin" or "end"). However, since Chrome v90 this is
// showing up as "-bpstart" and "-bpend" ("benchpress start/end"), which is what one would
// actually expect since that is the mark name used in ChromeDriverExtension - see the
// #timeBegin and #timeEnd implementations in chrome_driver_extension.ts. For
// backwards-compatibility with Chrome v89 (and older), we do both checks: the phase-based
// one ("B" or "E") and event name-based (the "-bp(start/end)" suffix).
const isStartEvent = (ph === 'B' && name === markName) || name === markName + '-bpstart';
const isEndEvent = (ph === 'E' && name === markName) || name === markName + '-bpend';
if (isStartEvent) {
markStartEvent = event;
}
else if (ph === 'I' && name === 'navigationStart' && !this._ignoreNavigation) {
// if a benchmark measures reload of a page, use the last
// navigationStart as begin event
markStartEvent = event;
}
else if (isEndEvent) {
markEndEvent = event;
}
});
if (!markStartEvent || !markEndEvent) {
// not all events have been received, no further processing for now
return null;
}
if (markStartEvent.pid !== markEndEvent.pid) {
result['invalid'] = 1;
}
let gcTimeInScript = 0;
let renderTimeInScript = 0;
const frameTimestamps = [];
const frameTimes = [];
let frameCaptureStartEvent = null;
let frameCaptureEndEvent = null;
const intervalStarts = {};
const intervalStartCount = {};
let inMeasureRange = false;
events.forEach((event) => {
const ph = event['ph'];
let name = event['name'];
let microIterations = 1;
const microIterationsMatch = name.match(_MICRO_ITERATIONS_REGEX);
if (microIterationsMatch) {
name = microIterationsMatch[1];
microIterations = parseInt(microIterationsMatch[2], 10);
}
if (event === markStartEvent) {
inMeasureRange = true;
}
else if (event === markEndEvent) {
inMeasureRange = false;
}
if (!inMeasureRange || event['pid'] !== markStartEvent['pid']) {
return;
}
if (this._requestCount && name === 'sendRequest') {
result['requestCount'] += 1;
}
else if (this._receivedData && name === 'receivedData' && ph === 'I') {
result['receivedData'] += event['args']['encodedDataLength'];
}
if (ph === 'B' && name === _MARK_NAME_FRAME_CAPTURE) {
if (frameCaptureStartEvent) {
throw new Error('can capture frames only once per benchmark run');
}
if (!this._captureFrames) {
throw new Error('found start event for frame capture, but frame capture was not requested in benchpress');
}
frameCaptureStartEvent = event;
}
else if (ph === 'E' && name === _MARK_NAME_FRAME_CAPTURE) {
if (!frameCaptureStartEvent) {
throw new Error('missing start event for frame capture');
}
frameCaptureEndEvent = event;
}
if (ph === 'I' && frameCaptureStartEvent && !frameCaptureEndEvent && name === 'frame') {
frameTimestamps.push(event['ts']);
if (frameTimestamps.length >= 2) {
frameTimes.push(frameTimestamps[frameTimestamps.length - 1] -
frameTimestamps[frameTimestamps.length - 2]);
}
}
if (ph === 'B') {
if (!intervalStarts[name]) {
intervalStartCount[name] = 1;
intervalStarts[name] = event;
}
else {
intervalStartCount[name]++;
}
}
else if ((ph === 'E') && intervalStarts[name]) {
intervalStartCount[name]--;
if (intervalStartCount[name] === 0) {
const startEvent = intervalStarts[name];
const duration = (event['ts'] - startEvent['ts']);
intervalStarts[name] = null;
if (name === 'gc') {
result['gcTime'] += duration;
const amount = (startEvent['args']['usedHeapSize'] - event['args']['usedHeapSize']) / 1000;
result['gcAmount'] += amount;
const majorGc = event['args']['majorGc'];
if (majorGc && majorGc) {
result['majorGcTime'] += duration;
}
if (intervalStarts['script']) {
gcTimeInScript += duration;
}
}
else if (name === 'render') {
result['renderTime'] += duration;
if (intervalStarts['script']) {
renderTimeInScript += duration;
}
}
else if (name === 'script') {
result['scriptTime'] += duration;
}
else if (this._microMetrics[name]) {
result[name] += duration / microIterations;
}
}
}
});
if (frameCaptureStartEvent && !frameCaptureEndEvent) {
throw new Error('missing end event for frame capture');
}
if (this._captureFrames && !frameCaptureStartEvent) {
throw new Error('frame capture requested in benchpress, but no start event was found');
}
if (frameTimes.length > 0) {
this._addFrameMetrics(result, frameTimes);
}
result['pureScriptTime'] = result['scriptTime'] - gcTimeInScript - renderTimeInScript;
return result;
}
_addFrameMetrics(result, frameTimes) {
result['frameTime.mean'] = frameTimes.reduce((a, b) => a + b, 0) / frameTimes.length;
const firstFrame = frameTimes[0];
result['frameTime.worst'] = frameTimes.reduce((a, b) => a > b ? a : b, firstFrame);
result['frameTime.best'] = frameTimes.reduce((a, b) => a < b ? a : b, firstFrame);
result['frameTime.smooth'] =
frameTimes.filter(t => t < _FRAME_TIME_SMOOTH_THRESHOLD).length / frameTimes.length;
}
_markName(index) {
return `${_MARK_NAME_PREFIX}${index}`;
}
};
PerflogMetric.SET_TIMEOUT = new InjectionToken('PerflogMetric.setTimeout');
PerflogMetric.IGNORE_NAVIGATION = new InjectionToken('PerflogMetric.ignoreNavigation');
PerflogMetric.PROVIDERS = [
{
provide: PerflogMetric_1,
deps: [
WebDriverExtension, PerflogMetric_1.SET_TIMEOUT, Options.MICRO_METRICS, Options.FORCE_GC,
Options.CAPTURE_FRAMES, Options.RECEIVED_DATA, Options.REQUEST_COUNT,
PerflogMetric_1.IGNORE_NAVIGATION
]
},
{
provide: PerflogMetric_1.SET_TIMEOUT,
useValue: (fn, millis) => setTimeout(fn, millis)
},
{ provide: PerflogMetric_1.IGNORE_NAVIGATION, useValue: false }
];
PerflogMetric = PerflogMetric_1 = __decorate([
Injectable(),
__param(1, Inject(PerflogMetric_1.SET_TIMEOUT)),
__param(2, Inject(Options.MICRO_METRICS)),
__param(3, Inject(Options.FORCE_GC)),
__param(4, Inject(Options.CAPTURE_FRAMES)),
__param(5, Inject(Options.RECEIVED_DATA)),
__param(6, Inject(Options.REQUEST_COUNT)),
__param(7, Inject(PerflogMetric_1.IGNORE_NAVIGATION)),
__metadata("design:paramtypes", [WebDriverExtension,
Function, Object, Boolean, Boolean, Boolean, Boolean, Boolean])
], PerflogMetric);
export { PerflogMetric };
const _MICRO_ITERATIONS_REGEX = /(.+)\*(\d+)$/;
const _MAX_RETRY_COUNT = 20;
const _MARK_NAME_PREFIX = 'benchpress';
const _MARK_NAME_FRAME_CAPTURE = 'frameCapture';
// using 17ms as a somewhat looser threshold, instead of 16.6666ms
const _FRAME_TIME_SMOOTH_THRESHOLD = 17;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"perflog_metric.js","sourceRoot":"","sources":["../../../../../../../packages/benchpress/src/metric/perflog_metric.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {Inject, Injectable, InjectionToken} from '@angular/core';\n\nimport {Options} from '../common_options';\nimport {Metric} from '../metric';\nimport {PerfLogEvent, PerfLogFeatures, WebDriverExtension} from '../web_driver_extension';\n\n\n/**\n * A metric that reads out the performance log\n */\n@Injectable()\nexport class PerflogMetric extends Metric {\n  static SET_TIMEOUT = new InjectionToken('PerflogMetric.setTimeout');\n  static IGNORE_NAVIGATION = new InjectionToken('PerflogMetric.ignoreNavigation');\n  static PROVIDERS = [\n    {\n      provide: PerflogMetric,\n      deps:\n          [\n            WebDriverExtension, PerflogMetric.SET_TIMEOUT, Options.MICRO_METRICS, Options.FORCE_GC,\n            Options.CAPTURE_FRAMES, Options.RECEIVED_DATA, Options.REQUEST_COUNT,\n            PerflogMetric.IGNORE_NAVIGATION\n          ]\n    },\n    {\n      provide: PerflogMetric.SET_TIMEOUT,\n      useValue: (fn: Function, millis: number) => <any>setTimeout(fn, millis)\n    },\n    {provide: PerflogMetric.IGNORE_NAVIGATION, useValue: false}\n  ];\n\n  private _remainingEvents: PerfLogEvent[];\n  private _measureCount: number;\n  private _perfLogFeatures: PerfLogFeatures;\n\n  /**\n   * @param driverExtension\n   * @param setTimeout\n   * @param microMetrics Name and description of metrics provided via console.time / console.timeEnd\n   * @param ignoreNavigation If true, don't measure from navigationStart events. These events are\n   *   usually triggered by a page load, but can also be triggered when adding iframes to the DOM.\n   **/\n  constructor(\n      private _driverExtension: WebDriverExtension,\n      @Inject(PerflogMetric.SET_TIMEOUT) private _setTimeout: Function,\n      @Inject(Options.MICRO_METRICS) private _microMetrics: {[key: string]: string},\n      @Inject(Options.FORCE_GC) private _forceGc: boolean,\n      @Inject(Options.CAPTURE_FRAMES) private _captureFrames: boolean,\n      @Inject(Options.RECEIVED_DATA) private _receivedData: boolean,\n      @Inject(Options.REQUEST_COUNT) private _requestCount: boolean,\n      @Inject(PerflogMetric.IGNORE_NAVIGATION) private _ignoreNavigation: boolean) {\n    super();\n\n    this._remainingEvents = [];\n    this._measureCount = 0;\n    this._perfLogFeatures = _driverExtension.perfLogFeatures();\n    if (!this._perfLogFeatures.userTiming) {\n      // User timing is needed for navigationStart.\n      this._receivedData = false;\n      this._requestCount = false;\n    }\n  }\n\n  override describe(): {[key: string]: string} {\n    const res: {[key: string]: any} = {\n      'scriptTime': 'script execution time in ms, including gc and render',\n      'pureScriptTime': 'script execution time in ms, without gc nor render'\n    };\n    if (this._perfLogFeatures.render) {\n      res['renderTime'] = 'render time in ms';\n    }\n    if (this._perfLogFeatures.gc) {\n      res['gcTime'] = 'gc time in ms';\n      res['gcAmount'] = 'gc amount in kbytes';\n      res['majorGcTime'] = 'time of major gcs in ms';\n      if (this._forceGc) {\n        res['forcedGcTime'] = 'forced gc time in ms';\n        res['forcedGcAmount'] = 'forced gc amount in kbytes';\n      }\n    }\n    if (this._receivedData) {\n      res['receivedData'] = 'encoded bytes received since navigationStart';\n    }\n    if (this._requestCount) {\n      res['requestCount'] = 'count of requests sent since navigationStart';\n    }\n    if (this._captureFrames) {\n      if (!this._perfLogFeatures.frameCapture) {\n        const warningMsg = 'WARNING: Metric requested, but not supported by driver';\n        // using dot syntax for metric name to keep them grouped together in console reporter\n        res['frameTime.mean'] = warningMsg;\n        res['frameTime.worst'] = warningMsg;\n        res['frameTime.best'] = warningMsg;\n        res['frameTime.smooth'] = warningMsg;\n      } else {\n        res['frameTime.mean'] = 'mean frame time in ms (target: 16.6ms for 60fps)';\n        res['frameTime.worst'] = 'worst frame time in ms';\n        res['frameTime.best'] = 'best frame time in ms';\n        res['frameTime.smooth'] = 'percentage of frames that hit 60fps';\n      }\n    }\n    for (const name in this._microMetrics) {\n      res[name] = this._microMetrics[name];\n    }\n    return res;\n  }\n\n  override beginMeasure(): Promise<any> {\n    let resultPromise = Promise.resolve(null);\n    if (this._forceGc) {\n      resultPromise = resultPromise.then((_) => this._driverExtension.gc());\n    }\n    return resultPromise.then((_) => this._beginMeasure());\n  }\n\n  override endMeasure(restart: boolean): Promise<{[key: string]: number}> {\n    if (this._forceGc) {\n      return this._endPlainMeasureAndMeasureForceGc(restart);\n    } else {\n      return this._endMeasure(restart);\n    }\n  }\n\n  /** @internal */\n  private _endPlainMeasureAndMeasureForceGc(restartMeasure: boolean) {\n    return this._endMeasure(true).then((measureValues) => {\n      // disable frame capture for measurements during forced gc\n      const originalFrameCaptureValue = this._captureFrames;\n      this._captureFrames = false;\n      return this._driverExtension.gc()\n          .then((_) => this._endMeasure(restartMeasure))\n          .then((forceGcMeasureValues) => {\n            this._captureFrames = originalFrameCaptureValue;\n            measureValues['forcedGcTime'] = forceGcMeasureValues['gcTime'];\n            measureValues['forcedGcAmount'] = forceGcMeasureValues['gcAmount'];\n            return measureValues;\n          });\n    });\n  }\n\n  private _beginMeasure(): Promise<any> {\n    return this._driverExtension.timeBegin(this._markName(this._measureCount++));\n  }\n\n  private _endMeasure(restart: boolean): Promise<{[key: string]: number}> {\n    const markName = this._markName(this._measureCount - 1);\n    const nextMarkName = restart ? this._markName(this._measureCount++) : null;\n    return this._driverExtension.timeEnd(markName, nextMarkName)\n        .then((_: any) => this._readUntilEndMark(markName));\n  }\n\n  private _readUntilEndMark(\n      markName: string, loopCount: number = 0, startEvent: PerfLogEvent|null = null) {\n    if (loopCount > _MAX_RETRY_COUNT) {\n      throw new Error(`Tried too often to get the ending mark: ${loopCount}`);\n    }\n    return this._driverExtension.readPerfLog().then((events) => {\n      this._addEvents(events);\n      const result = this._aggregateEvents(this._remainingEvents, markName);\n      if (result) {\n        this._remainingEvents = events;\n        return result;\n      }\n      let resolve: (result: any) => void;\n      const promise = new Promise<{[key: string]: number}>(res => {\n        resolve = res;\n      });\n      this._setTimeout(() => resolve(this._readUntilEndMark(markName, loopCount + 1)), 100);\n      return promise;\n    });\n  }\n\n  private _addEvents(events: PerfLogEvent[]) {\n    let needSort = false;\n    events.forEach(event => {\n      if (event['ph'] === 'X') {\n        needSort = true;\n        const startEvent: PerfLogEvent = {};\n        const endEvent: PerfLogEvent = {};\n        for (const prop in event) {\n          startEvent[prop] = event[prop];\n          endEvent[prop] = event[prop];\n        }\n        startEvent['ph'] = 'B';\n        endEvent['ph'] = 'E';\n        endEvent['ts'] = startEvent['ts']! + startEvent['dur']!;\n        this._remainingEvents.push(startEvent);\n        this._remainingEvents.push(endEvent);\n      } else {\n        this._remainingEvents.push(event);\n      }\n    });\n    if (needSort) {\n      // Need to sort because of the ph==='X' events\n      this._remainingEvents.sort((a, b) => {\n        const diff = a['ts']! - b['ts']!;\n        return diff > 0 ? 1 : diff < 0 ? -1 : 0;\n      });\n    }\n  }\n\n  private _aggregateEvents(events: PerfLogEvent[], markName: string): {[key: string]: number}|null {\n    const result: {[key: string]: number} = {'scriptTime': 0, 'pureScriptTime': 0};\n    if (this._perfLogFeatures.gc) {\n      result['gcTime'] = 0;\n      result['majorGcTime'] = 0;\n      result['gcAmount'] = 0;\n    }\n    if (this._perfLogFeatures.render) {\n      result['renderTime'] = 0;\n    }\n    if (this._captureFrames) {\n      result['frameTime.mean'] = 0;\n      result['frameTime.best'] = 0;\n      result['frameTime.worst'] = 0;\n      result['frameTime.smooth'] = 0;\n    }\n    for (const name in this._microMetrics) {\n      result[name] = 0;\n    }\n    if (this._receivedData) {\n      result['receivedData'] = 0;\n    }\n    if (this._requestCount) {\n      result['requestCount'] = 0;\n    }\n\n    let markStartEvent: PerfLogEvent = null!;\n    let markEndEvent: PerfLogEvent = null!;\n    events.forEach((event) => {\n      const ph = event['ph'];\n      const name = event['name'];\n\n      // Here we are determining if this is the event signaling the start or end of our performance\n      // testing (this is triggered by us calling #timeBegin and #timeEnd).\n      //\n      // Previously, this was done by checking that the event name matched our mark name and that\n      // the phase was either \"B\" or \"E\" (\"begin\" or \"end\"). However, since Chrome v90 this is\n      // showing up as \"-bpstart\" and \"-bpend\" (\"benchpress start/end\"), which is what one would\n      // actually expect since that is the mark name used in ChromeDriverExtension - see the\n      // #timeBegin and #timeEnd implementations in chrome_driver_extension.ts. For\n      // backwards-compatibility with Chrome v89 (and older), we do both checks: the phase-based\n      // one (\"B\" or \"E\") and event name-based (the \"-bp(start/end)\" suffix).\n      const isStartEvent = (ph === 'B' && name === markName) || name === markName + '-bpstart';\n      const isEndEvent = (ph === 'E' && name === markName) || name === markName + '-bpend';\n      if (isStartEvent) {\n        markStartEvent = event;\n      } else if (ph === 'I' && name === 'navigationStart' && !this._ignoreNavigation) {\n        // if a benchmark measures reload of a page, use the last\n        // navigationStart as begin event\n        markStartEvent = event;\n      } else if (isEndEvent) {\n        markEndEvent = event;\n      }\n    });\n    if (!markStartEvent || !markEndEvent) {\n      // not all events have been received, no further processing for now\n      return null;\n    }\n    if (markStartEvent.pid !== markEndEvent.pid) {\n      result['invalid'] = 1;\n    }\n\n    let gcTimeInScript = 0;\n    let renderTimeInScript = 0;\n\n    const frameTimestamps: number[] = [];\n    const frameTimes: number[] = [];\n    let frameCaptureStartEvent: PerfLogEvent|null = null;\n    let frameCaptureEndEvent: PerfLogEvent|null = null;\n\n    const intervalStarts: {[key: string]: PerfLogEvent} = {};\n    const intervalStartCount: {[key: string]: number} = {};\n\n    let inMeasureRange = false;\n    events.forEach((event) => {\n      const ph = event['ph'];\n      let name = event['name']!;\n      let microIterations = 1;\n      const microIterationsMatch = name.match(_MICRO_ITERATIONS_REGEX);\n      if (microIterationsMatch) {\n        name = microIterationsMatch[1];\n        microIterations = parseInt(microIterationsMatch[2], 10);\n      }\n      if (event === markStartEvent) {\n        inMeasureRange = true;\n      } else if (event === markEndEvent) {\n        inMeasureRange = false;\n      }\n      if (!inMeasureRange ||