1 | import { PromiseWrapper, TimerWrapper } from 'angular2/src/facade/async';
|
2 | import { isPresent, isBlank, StringWrapper, RegExpWrapper, NumberWrapper } from 'angular2/src/facade/lang';
|
3 | import { BaseException } from 'angular2/src/facade/exceptions';
|
4 | import { ListWrapper, StringMapWrapper } from 'angular2/src/facade/collection';
|
5 | import { bind, provide, OpaqueToken } from 'angular2/src/core/di';
|
6 | import { WebDriverExtension } from '../web_driver_extension';
|
7 | import { Metric } from '../metric';
|
8 | import { Options } from '../common_options';
|
9 | /**
|
10 | * A metric that reads out the performance log
|
11 | */
|
12 | export 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 | }
|
331 | var _MICRO_ITERATIONS_REGEX = /(.+)\*(\d+)$/g;
|
332 | var _MAX_RETRY_COUNT = 20;
|
333 | var _MARK_NAME_PREFIX = 'benchpress';
|
334 | var _SET_TIMEOUT = new OpaqueToken('PerflogMetric.setTimeout');
|
335 | var _MARK_NAME_FRAME_CAPUTRE = 'frameCapture';
|
336 | // using 17ms as a somewhat looser threshold, instead of 16.6666ms
|
337 | var _FRAME_TIME_SMOOTH_THRESHOLD = 17;
|
338 | var _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 |