UNPKG

56.6 kBJavaScriptView Raw
1import { PromiseWrapper, TimerWrapper } from 'angular2/src/facade/async';
2import { isPresent, isBlank, StringWrapper, RegExpWrapper, NumberWrapper } from 'angular2/src/facade/lang';
3import { BaseException } from 'angular2/src/facade/exceptions';
4import { ListWrapper, StringMapWrapper } from 'angular2/src/facade/collection';
5import { bind, provide, OpaqueToken } from 'angular2/src/core/di';
6import { WebDriverExtension } from '../web_driver_extension';
7import { Metric } from '../metric';
8import { Options } from '../common_options';
9/**
10 * A metric that reads out the performance log
11 */
12export class PerflogMetric extends Metric {
13 /**
14 * @param driverExtension
15 * @param setTimeout
16 * @param microMetrics Name and description of metrics provided via console.time / console.timeEnd
17 **/
18 constructor(_driverExtension, _setTimeout, _microMetrics, _forceGc, _captureFrames, _receivedData, _requestCount) {
19 super();
20 this._driverExtension = _driverExtension;
21 this._setTimeout = _setTimeout;
22 this._microMetrics = _microMetrics;
23 this._forceGc = _forceGc;
24 this._captureFrames = _captureFrames;
25 this._receivedData = _receivedData;
26 this._requestCount = _requestCount;
27 this._remainingEvents = [];
28 this._measureCount = 0;
29 this._perfLogFeatures = _driverExtension.perfLogFeatures();
30 if (!this._perfLogFeatures.userTiming) {
31 // User timing is needed for navigationStart.
32 this._receivedData = false;
33 this._requestCount = false;
34 }
35 }
36 // TODO(tbosch): use static values when our transpiler supports them
37 static get BINDINGS() { return _PROVIDERS; }
38 // TODO(tbosch): use static values when our transpiler supports them
39 static get SET_TIMEOUT() { return _SET_TIMEOUT; }
40 describe() {
41 var res = {
42 'scriptTime': 'script execution time in ms, including gc and render',
43 'pureScriptTime': 'script execution time in ms, without gc nor render'
44 };
45 if (this._perfLogFeatures.render) {
46 res['renderTime'] = 'render time in ms';
47 }
48 if (this._perfLogFeatures.gc) {
49 res['gcTime'] = 'gc time in ms';
50 res['gcAmount'] = 'gc amount in kbytes';
51 res['majorGcTime'] = 'time of major gcs in ms';
52 if (this._forceGc) {
53 res['forcedGcTime'] = 'forced gc time in ms';
54 res['forcedGcAmount'] = 'forced gc amount in kbytes';
55 }
56 }
57 if (this._receivedData) {
58 res['receivedData'] = 'encoded bytes received since navigationStart';
59 }
60 if (this._requestCount) {
61 res['requestCount'] = 'count of requests sent since navigationStart';
62 }
63 if (this._captureFrames) {
64 if (!this._perfLogFeatures.frameCapture) {
65 var warningMsg = 'WARNING: Metric requested, but not supported by driver';
66 // using dot syntax for metric name to keep them grouped together in console reporter
67 res['frameTime.mean'] = warningMsg;
68 res['frameTime.worst'] = warningMsg;
69 res['frameTime.best'] = warningMsg;
70 res['frameTime.smooth'] = warningMsg;
71 }
72 else {
73 res['frameTime.mean'] = 'mean frame time in ms (target: 16.6ms for 60fps)';
74 res['frameTime.worst'] = 'worst frame time in ms';
75 res['frameTime.best'] = 'best frame time in ms';
76 res['frameTime.smooth'] = 'percentage of frames that hit 60fps';
77 }
78 }
79 StringMapWrapper.forEach(this._microMetrics, (desc, name) => { StringMapWrapper.set(res, name, desc); });
80 return res;
81 }
82 beginMeasure() {
83 var resultPromise = PromiseWrapper.resolve(null);
84 if (this._forceGc) {
85 resultPromise = resultPromise.then((_) => this._driverExtension.gc());
86 }
87 return resultPromise.then((_) => this._beginMeasure());
88 }
89 endMeasure(restart) {
90 if (this._forceGc) {
91 return this._endPlainMeasureAndMeasureForceGc(restart);
92 }
93 else {
94 return this._endMeasure(restart);
95 }
96 }
97 _endPlainMeasureAndMeasureForceGc(restartMeasure) {
98 return this._endMeasure(true).then((measureValues) => {
99 // disable frame capture for measurements during forced gc
100 var originalFrameCaptureValue = this._captureFrames;
101 this._captureFrames = false;
102 return this._driverExtension.gc()
103 .then((_) => this._endMeasure(restartMeasure))
104 .then((forceGcMeasureValues) => {
105 this._captureFrames = originalFrameCaptureValue;
106 StringMapWrapper.set(measureValues, 'forcedGcTime', forceGcMeasureValues['gcTime']);
107 StringMapWrapper.set(measureValues, 'forcedGcAmount', forceGcMeasureValues['gcAmount']);
108 return measureValues;
109 });
110 });
111 }
112 _beginMeasure() {
113 return this._driverExtension.timeBegin(this._markName(this._measureCount++));
114 }
115 _endMeasure(restart) {
116 var markName = this._markName(this._measureCount - 1);
117 var nextMarkName = restart ? this._markName(this._measureCount++) : null;
118 return this._driverExtension.timeEnd(markName, nextMarkName)
119 .then((_) => this._readUntilEndMark(markName));
120 }
121 _readUntilEndMark(markName, loopCount = 0, startEvent = null) {
122 if (loopCount > _MAX_RETRY_COUNT) {
123 throw new BaseException(`Tried too often to get the ending mark: ${loopCount}`);
124 }
125 return this._driverExtension.readPerfLog().then((events) => {
126 this._addEvents(events);
127 var result = this._aggregateEvents(this._remainingEvents, markName);
128 if (isPresent(result)) {
129 this._remainingEvents = events;
130 return result;
131 }
132 var completer = PromiseWrapper.completer();
133 this._setTimeout(() => completer.resolve(this._readUntilEndMark(markName, loopCount + 1)), 100);
134 return completer.promise;
135 });
136 }
137 _addEvents(events) {
138 var needSort = false;
139 events.forEach(event => {
140 if (StringWrapper.equals(event['ph'], 'X')) {
141 needSort = true;
142 var startEvent = {};
143 var endEvent = {};
144 StringMapWrapper.forEach(event, (value, prop) => {
145 startEvent[prop] = value;
146 endEvent[prop] = value;
147 });
148 startEvent['ph'] = 'B';
149 endEvent['ph'] = 'E';
150 endEvent['ts'] = startEvent['ts'] + startEvent['dur'];
151 this._remainingEvents.push(startEvent);
152 this._remainingEvents.push(endEvent);
153 }
154 else {
155 this._remainingEvents.push(event);
156 }
157 });
158 if (needSort) {
159 // Need to sort because of the ph==='X' events
160 ListWrapper.sort(this._remainingEvents, (a, b) => {
161 var diff = a['ts'] - b['ts'];
162 return diff > 0 ? 1 : diff < 0 ? -1 : 0;
163 });
164 }
165 }
166 _aggregateEvents(events, markName) {
167 var result = { 'scriptTime': 0, 'pureScriptTime': 0 };
168 if (this._perfLogFeatures.gc) {
169 result['gcTime'] = 0;
170 result['majorGcTime'] = 0;
171 result['gcAmount'] = 0;
172 }
173 if (this._perfLogFeatures.render) {
174 result['renderTime'] = 0;
175 }
176 if (this._captureFrames) {
177 result['frameTime.mean'] = 0;
178 result['frameTime.best'] = 0;
179 result['frameTime.worst'] = 0;
180 result['frameTime.smooth'] = 0;
181 }
182 StringMapWrapper.forEach(this._microMetrics, (desc, name) => { result[name] = 0; });
183 if (this._receivedData) {
184 result['receivedData'] = 0;
185 }
186 if (this._requestCount) {
187 result['requestCount'] = 0;
188 }
189 var markStartEvent = null;
190 var markEndEvent = null;
191 var gcTimeInScript = 0;
192 var renderTimeInScript = 0;
193 var frameTimestamps = [];
194 var frameTimes = [];
195 var frameCaptureStartEvent = null;
196 var frameCaptureEndEvent = null;
197 var intervalStarts = {};
198 var intervalStartCount = {};
199 events.forEach((event) => {
200 var ph = event['ph'];
201 var name = event['name'];
202 var microIterations = 1;
203 var microIterationsMatch = RegExpWrapper.firstMatch(_MICRO_ITERATIONS_REGEX, name);
204 if (isPresent(microIterationsMatch)) {
205 name = microIterationsMatch[1];
206 microIterations = NumberWrapper.parseInt(microIterationsMatch[2], 10);
207 }
208 if (StringWrapper.equals(ph, 'b') && StringWrapper.equals(name, markName)) {
209 markStartEvent = event;
210 }
211 else if (StringWrapper.equals(ph, 'e') && StringWrapper.equals(name, markName)) {
212 markEndEvent = event;
213 }
214 let isInstant = StringWrapper.equals(ph, 'I') || StringWrapper.equals(ph, 'i');
215 if (this._requestCount && StringWrapper.equals(name, 'sendRequest')) {
216 result['requestCount'] += 1;
217 }
218 else if (this._receivedData && StringWrapper.equals(name, 'receivedData') && isInstant) {
219 result['receivedData'] += event['args']['encodedDataLength'];
220 }
221 else if (StringWrapper.equals(name, 'navigationStart')) {
222 // We count data + requests since the last navigationStart
223 // (there might be chrome extensions loaded by selenium before our page, so there
224 // will likely be more than one navigationStart).
225 if (this._receivedData) {
226 result['receivedData'] = 0;
227 }
228 if (this._requestCount) {
229 result['requestCount'] = 0;
230 }
231 }
232 if (isPresent(markStartEvent) && isBlank(markEndEvent) &&
233 event['pid'] === markStartEvent['pid']) {
234 if (StringWrapper.equals(ph, 'b') && StringWrapper.equals(name, _MARK_NAME_FRAME_CAPUTRE)) {
235 if (isPresent(frameCaptureStartEvent)) {
236 throw new BaseException('can capture frames only once per benchmark run');
237 }
238 if (!this._captureFrames) {
239 throw new BaseException('found start event for frame capture, but frame capture was not requested in benchpress');
240 }
241 frameCaptureStartEvent = event;
242 }
243 else if (StringWrapper.equals(ph, 'e') &&
244 StringWrapper.equals(name, _MARK_NAME_FRAME_CAPUTRE)) {
245 if (isBlank(frameCaptureStartEvent)) {
246 throw new BaseException('missing start event for frame capture');
247 }
248 frameCaptureEndEvent = event;
249 }
250 if (isInstant) {
251 if (isPresent(frameCaptureStartEvent) && isBlank(frameCaptureEndEvent) &&
252 StringWrapper.equals(name, 'frame')) {
253 frameTimestamps.push(event['ts']);
254 if (frameTimestamps.length >= 2) {
255 frameTimes.push(frameTimestamps[frameTimestamps.length - 1] -
256 frameTimestamps[frameTimestamps.length - 2]);
257 }
258 }
259 }
260 if (StringWrapper.equals(ph, 'B') || StringWrapper.equals(ph, 'b')) {
261 if (isBlank(intervalStarts[name])) {
262 intervalStartCount[name] = 1;
263 intervalStarts[name] = event;
264 }
265 else {
266 intervalStartCount[name]++;
267 }
268 }
269 else if ((StringWrapper.equals(ph, 'E') || StringWrapper.equals(ph, 'e')) &&
270 isPresent(intervalStarts[name])) {
271 intervalStartCount[name]--;
272 if (intervalStartCount[name] === 0) {
273 var startEvent = intervalStarts[name];
274 var duration = (event['ts'] - startEvent['ts']);
275 intervalStarts[name] = null;
276 if (StringWrapper.equals(name, 'gc')) {
277 result['gcTime'] += duration;
278 var amount = (startEvent['args']['usedHeapSize'] - event['args']['usedHeapSize']) / 1000;
279 result['gcAmount'] += amount;
280 var majorGc = event['args']['majorGc'];
281 if (isPresent(majorGc) && majorGc) {
282 result['majorGcTime'] += duration;
283 }
284 if (isPresent(intervalStarts['script'])) {
285 gcTimeInScript += duration;
286 }
287 }
288 else if (StringWrapper.equals(name, 'render')) {
289 result['renderTime'] += duration;
290 if (isPresent(intervalStarts['script'])) {
291 renderTimeInScript += duration;
292 }
293 }
294 else if (StringWrapper.equals(name, 'script')) {
295 result['scriptTime'] += duration;
296 }
297 else if (isPresent(this._microMetrics[name])) {
298 result[name] += duration / microIterations;
299 }
300 }
301 }
302 }
303 });
304 if (!isPresent(markStartEvent) || !isPresent(markEndEvent)) {
305 // not all events have been received, no further processing for now
306 return null;
307 }
308 if (isPresent(markEndEvent) && isPresent(frameCaptureStartEvent) &&
309 isBlank(frameCaptureEndEvent)) {
310 throw new BaseException('missing end event for frame capture');
311 }
312 if (this._captureFrames && isBlank(frameCaptureStartEvent)) {
313 throw new BaseException('frame capture requested in benchpress, but no start event was found');
314 }
315 if (frameTimes.length > 0) {
316 this._addFrameMetrics(result, frameTimes);
317 }
318 result['pureScriptTime'] = result['scriptTime'] - gcTimeInScript - renderTimeInScript;
319 return result;
320 }
321 _addFrameMetrics(result, frameTimes) {
322 result['frameTime.mean'] = frameTimes.reduce((a, b) => a + b, 0) / frameTimes.length;
323 var firstFrame = frameTimes[0];
324 result['frameTime.worst'] = frameTimes.reduce((a, b) => a > b ? a : b, firstFrame);
325 result['frameTime.best'] = frameTimes.reduce((a, b) => a < b ? a : b, firstFrame);
326 result['frameTime.smooth'] =
327 frameTimes.filter(t => t < _FRAME_TIME_SMOOTH_THRESHOLD).length / frameTimes.length;
328 }
329 _markName(index) { return `${_MARK_NAME_PREFIX}${index}`; }
330}
331var _MICRO_ITERATIONS_REGEX = /(.+)\*(\d+)$/g;
332var _MAX_RETRY_COUNT = 20;
333var _MARK_NAME_PREFIX = 'benchpress';
334var _SET_TIMEOUT = new OpaqueToken('PerflogMetric.setTimeout');
335var _MARK_NAME_FRAME_CAPUTRE = 'frameCapture';
336// using 17ms as a somewhat looser threshold, instead of 16.6666ms
337var _FRAME_TIME_SMOOTH_THRESHOLD = 17;
338var _PROVIDERS = [
339 bind(PerflogMetric)
340 .toFactory((driverExtension, setTimeout, microMetrics, forceGc, captureFrames, receivedData, requestCount) => new PerflogMetric(driverExtension, setTimeout, microMetrics, forceGc, captureFrames, receivedData, requestCount), [
341 WebDriverExtension,
342 _SET_TIMEOUT,
343 Options.MICRO_METRICS,
344 Options.FORCE_GC,
345 Options.CAPTURE_FRAMES,
346 Options.RECEIVED_DATA,
347 Options.REQUEST_COUNT
348 ]),
349 provide(_SET_TIMEOUT, { useValue: (fn, millis) => TimerWrapper.setTimeout(fn, millis) })
350];
351//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"perflog_metric.js","sourceRoot":"","sources":["diffing_plugin_wrapper-output_path-xBLIBrVR.tmp/benchpress/src/metric/perflog_metric.ts"],"names":[],"mappings":"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","sourcesContent":["import {PromiseWrapper, TimerWrapper} from 'angular2/src/facade/async';\nimport {\n  isPresent,\n  isBlank,\n  StringWrapper,\n  Math,\n  RegExpWrapper,\n  NumberWrapper\n} from 'angular2/src/facade/lang';\nimport {BaseException, WrappedException} from 'angular2/src/facade/exceptions';\nimport {ListWrapper, StringMapWrapper} from 'angular2/src/facade/collection';\nimport {bind, provide, Provider, OpaqueToken} from 'angular2/src/core/di';\n\nimport {WebDriverExtension, PerfLogFeatures} from '../web_driver_extension';\nimport {Metric} from '../metric';\nimport {Options} from '../common_options';\n\n/**\n * A metric that reads out the performance log\n */\nexport class PerflogMetric extends Metric {\n  // TODO(tbosch): use static values when our transpiler supports them\n  static get BINDINGS(): Provider[] { return _PROVIDERS; }\n  // TODO(tbosch): use static values when our transpiler supports them\n  static get SET_TIMEOUT(): OpaqueToken { return _SET_TIMEOUT; }\n\n  private _remainingEvents: Array<{[key: string]: any}>;\n  private _measureCount: number;\n  _perfLogFeatures: PerfLogFeatures;\n\n\n  /**\n   * @param driverExtension\n   * @param setTimeout\n   * @param microMetrics Name and description of metrics provided via console.time / console.timeEnd\n   **/\n  constructor(private _driverExtension: WebDriverExtension, private _setTimeout: Function,\n              private _microMetrics: {[key: string]: any}, private _forceGc: boolean,\n              private _captureFrames: boolean, private _receivedData: boolean,\n              private _requestCount: 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  describe(): {[key: string]: any} {\n    var res = {\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        var 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    StringMapWrapper.forEach(this._microMetrics,\n                             (desc, name) => { StringMapWrapper.set(res, name, desc); });\n    return res;\n  }\n\n  beginMeasure(): Promise<any> {\n    var resultPromise = PromiseWrapper.resolve(null);\n    if (this._forceGc) {\n      resultPromise = resultPromise.then((_) => this._driverExtension.gc());\n    }\n    return resultPromise.then((_) => this._beginMeasure());\n  }\n\n  endMeasure(restart: boolean): Promise<{[key: string]: any}> {\n    if (this._forceGc) {\n      return this._endPlainMeasureAndMeasureForceGc(restart);\n    } else {\n      return this._endMeasure(restart);\n    }\n  }\n\n  _endPlainMeasureAndMeasureForceGc(restartMeasure: boolean) {\n    return this._endMeasure(true).then((measureValues) => {\n      // disable frame capture for measurements during forced gc\n      var 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            StringMapWrapper.set(measureValues, 'forcedGcTime', forceGcMeasureValues['gcTime']);\n            StringMapWrapper.set(measureValues, 'forcedGcAmount', forceGcMeasureValues['gcAmount']);\n            return measureValues;\n          });\n    });\n  }\n\n  _beginMeasure(): Promise<any> {\n    return this._driverExtension.timeBegin(this._markName(this._measureCount++));\n  }\n\n  _endMeasure(restart: boolean): Promise<{[key: string]: any}> {\n    var markName = this._markName(this._measureCount - 1);\n    var nextMarkName = restart ? this._markName(this._measureCount++) : null;\n    return this._driverExtension.timeEnd(markName, nextMarkName)\n        .then((_) => this._readUntilEndMark(markName));\n  }\n\n  _readUntilEndMark(markName: string, loopCount: number = 0, startEvent = null) {\n    if (loopCount > _MAX_RETRY_COUNT) {\n      throw new BaseException(`Tried too often to get the ending mark: ${loopCount}`);\n    }\n    return this._driverExtension.readPerfLog().then((events) => {\n      this._addEvents(events);\n      var result = this._aggregateEvents(this._remainingEvents, markName);\n      if (isPresent(result)) {\n        this._remainingEvents = events;\n        return result;\n      }\n      var completer = PromiseWrapper.completer();\n      this._setTimeout(() => completer.resolve(this._readUntilEndMark(markName, loopCount + 1)),\n                       100);\n      return completer.promise;\n    });\n  }\n\n  _addEvents(events: { [key: string]: string }[]) {\n    var needSort = false;\n    events.forEach(event => {\n      if (StringWrapper.equals(event['ph'], 'X')) {\n        needSort = true;\n        var startEvent = {};\n        var endEvent = {};\n        StringMapWrapper.forEach(event, (value, prop) => {\n          startEvent[prop] = value;\n          endEvent[prop] = value;\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      ListWrapper.sort(this._remainingEvents, (a, b) => {\n        var diff = a['ts'] - b['ts'];\n        return diff > 0 ? 1 : diff < 0 ? -1 : 0;\n      });\n    }\n  }\n\n  _aggregateEvents(events: Array<{[key: string]: any}>, markName): {[key: string]: any} {\n    var result = {'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    StringMapWrapper.forEach(this._microMetrics, (desc, name) => { result[name] = 0; });\n    if (this._receivedData) {\n      result['receivedData'] = 0;\n    }\n    if (this._requestCount) {\n      result['requestCount'] = 0;\n    }\n\n    var markStartEvent = null;\n    var markEndEvent = null;\n    var gcTimeInScript = 0;\n    var renderTimeInScript = 0;\n\n    var frameTimestamps = [];\n    var frameTimes = [];\n    var frameCaptureStartEvent = null;\n    var frameCaptureEndEvent = null;\n\n    var intervalStarts: {[key: string]: any} = {};\n    var intervalStartCount: {[key: string]: number} = {};\n    events.forEach((event) => {\n      var ph = event['ph'];\n      var name = event['name'];\n      var microIterations = 1;\n      var microIterationsMatch = RegExpWrapper.firstMatch(_MICRO_ITERATIONS_REGEX, name);\n      if (isPresent(microIterationsMatch)) {\n        name = microIterationsMatch[1];\n        microIterations = NumberWrapper.parseInt(microIterationsMatch[2], 10);\n      }\n\n      if (StringWrapper.equals(ph, 'b') && StringWrapper.equals(name, markName)) {\n        markStartEvent = event;\n      } else if (StringWrapper.equals(ph, 'e') && StringWrapper.equals(name, markName)) {\n        markEndEvent = event;\n      }\n\n      let isInstant = StringWrapper.equals(ph, 'I') || StringWrapper.equals(ph, 'i');\n      if (this._requestCount && StringWrapper.equals(name, 'sendRequest')) {\n        result['requestCount'] += 1;\n      } else if (this._receivedData && StringWrapper.equals(name, 'receivedData') && isInstant) {\n        result['receivedData'] += event['args']['encodedDataLength'];\n      } else if (StringWrapper.equals(name, 'navigationStart')) {\n        // We count data + requests since the last navigationStart\n        // (there might be chrome extensions loaded by selenium before our page, so there\n        // will likely be more than one navigationStart).\n        if (this._receivedData) {\n          result['receivedData'] = 0;\n        }\n        if (this._requestCount) {\n          result['requestCount'] = 0;\n        }\n      }\n      if (isPresent(markStartEvent) && isBlank(markEndEvent) &&\n          event['pid'] === markStartEvent['pid']) {\n        if (StringWrapper.equals(ph, 'b') && StringWrapper.equals(name, _MARK_NAME_FRAME_CAPUTRE)) {\n          if (isPresent(frameCaptureStartEvent)) {\n            throw new BaseException('can capture frames only once per benchmark run');\n          }\n          if (!this._captureFrames) {\n            throw new BaseException(\n                'found start event for frame capture, but frame capture was not requested in benchpress')\n          }\n          frameCaptureStartEvent = event;\n        } else if (StringWrapper.equals(ph, 'e') &&\n                   StringWrapper.equals(name, _MARK_NAME_FRAME_CAPUTRE)) {\n          if (isBlank(frameCaptureStartEvent)) {\n            throw new BaseException('missing start event for frame capture');\n          }\n          frameCaptureEndEvent = event;\n        }\n\n        if (isInstant) {\n          if (isPresent(frameCaptureStartEvent) && isBlank(frameCaptureEndEvent) &&\n              StringWrapper.equals(name, 'frame')) {\n            frameTimestamps.push(event['ts']);\n            if (frameTimestamps.length >= 2) {\n              frameTimes.push(frameTimestamps[frameTimestamps.length - 1] -\n                              frameTimestamps[frameTimestamps.length - 2]);\n            }\n          }\n        }\n\n        if (StringWrapper.equals(ph, 'B') || StringWrapper.equals(ph, 'b')) {\n          if (isBlank(intervalStarts[name])) {\n            intervalStartCount[name] = 1;\n            intervalStarts[name] = event;\n          } else {\n            intervalStartCount[name]++;\n          }\n        } else if ((StringWrapper.equals(ph, 'E') || StringWrapper.equals(ph, 'e')) &&\n                   isPresent(intervalStarts[name])) {\n          intervalStartCount[name]--;\n          if (intervalStartCount[name] === 0) {\n            var startEvent = intervalStarts[name];\n            var duration = (event['ts'] - startEvent['ts']);\n            intervalStarts[name] = null;\n            if (StringWrapper.equals(name, 'gc')) {\n              result['gcTime'] += duration;\n              var amount =\n                  (startEvent['args']['usedHeapSize'] - event['args']['usedHeapSize']) / 1000;\n              result['gcAmount'] += amount;\n              var majorGc = event['args']['majorGc'];\n              if (isPresent(majorGc) && majorGc) {\n                result['majorGcTime'] += duration;\n              }\n              if (isPresent(intervalStarts['script'])) {\n                gcTimeInScript += duration;\n              }\n            } else if (StringWrapper.equals(name, 'render')) {\n              result['renderTime'] += duration;\n              if (isPresent(intervalStarts['script'])) {\n                renderTimeInScript += duration;\n              }\n            } else if (StringWrapper.equals(name, 'script')) {\n              result['scriptTime'] += duration;\n            } else if (isPresent(this._microMetrics[name])) {\n              result[name] += duration / microIterations;\n            }\n          }\n        }\n      }\n    });\n    if (!isPresent(markStartEvent) || !isPresent(markEndEvent)) {\n      // not all events have been received, no further processing for now\n      return null;\n    }\n\n    if (isPresent(markEndEvent) && isPresent(frameCaptureStartEvent) &&\n        isBlank(frameCaptureEndEvent)) {\n      throw new BaseException('missing end event for frame capture');\n    }\n    if (this._captureFrames && isBlank(frameCaptureStartEvent)) {\n      throw new BaseException(\n          'frame capture requested in benchpress, but no start event was found');\n    }\n    if (frameTimes.length > 0) {\n      this._addFrameMetrics(result, frameTimes);\n    }\n    result['pureScriptTime'] = result['scriptTime'] - gcTimeInScript - renderTimeInScript;\n    return result;\n  }\n\n  _addFrameMetrics(result: {[key: string]: any}, frameTimes: any[]) {\n    result['frameTime.mean'] = frameTimes.reduce((a, b) => a + b, 0) / frameTimes.length;\n    var firstFrame = frameTimes[0];\n    result['frameTime.worst'] = frameTimes.reduce((a, b) => a > b ? a : b, firstFrame);\n    result['frameTime.best'] = frameTimes.reduce((a, b) => a < b ? a : b, firstFrame);\n    result['frameTime.smooth'] =\n        frameTimes.filter(t => t < _FRAME_TIME_SMOOTH_THRESHOLD).length / frameTimes.length;\n  }\n\n  _markName(index) { return `${_MARK_NAME_PREFIX}${index}`; }\n}\n\nvar _MICRO_ITERATIONS_REGEX = /(.+)\\*(\\d+)$/g;\n\nvar _MAX_RETRY_COUNT = 20;\nvar _MARK_NAME_PREFIX = 'benchpress';\nvar _SET_TIMEOUT = new OpaqueToken('PerflogMetric.setTimeout');\n\nvar _MARK_NAME_FRAME_CAPUTRE = 'frameCapture';\n// using 17ms as a somewhat looser threshold, instead of 16.6666ms\nvar _FRAME_TIME_SMOOTH_THRESHOLD = 17;\n\nvar _PROVIDERS = [\n  bind(PerflogMetric)\n      .toFactory(\n          (driverExtension, setTimeout, microMetrics, forceGc, captureFrames, receivedData,\n           requestCount) => new PerflogMetric(driverExtension, setTimeout, microMetrics, forceGc,\n                                              captureFrames, receivedData, requestCount),\n          [\n            WebDriverExtension,\n            _SET_TIMEOUT,\n            Options.MICRO_METRICS,\n            Options.FORCE_GC,\n            Options.CAPTURE_FRAMES,\n            Options.RECEIVED_DATA,\n            Options.REQUEST_COUNT\n          ]),\n  provide(_SET_TIMEOUT, {useValue: (fn, millis) => TimerWrapper.setTimeout(fn, millis)})\n];\n"]}
\No newline at end of file