benchpress
Version:
Benchpress - a framework for e2e performance tests
230 lines • 42.1 kB
JavaScript
import { bind } from 'angular2/src/core/di';
import { ListWrapper, StringMapWrapper } from 'angular2/src/facade/collection';
import { Json, isPresent, isBlank, StringWrapper, NumberWrapper } from 'angular2/src/facade/lang';
import { BaseException } from 'angular2/src/facade/exceptions';
import { WebDriverExtension, PerfLogFeatures } from '../web_driver_extension';
import { WebDriverAdapter } from '../web_driver_adapter';
import { Options } from '../common_options';
/**
* Set the following 'traceCategories' to collect metrics in Chrome:
* 'v8,blink.console,disabled-by-default-devtools.timeline,devtools.timeline'
*
* In order to collect the frame rate related metrics, add 'benchmark'
* to the list above.
*/
export class ChromeDriverExtension extends WebDriverExtension {
constructor(_driver, userAgent) {
super();
this._driver = _driver;
this._majorChromeVersion = this._parseChromeVersion(userAgent);
}
// TODO(tbosch): use static values when our transpiler supports them
static get BINDINGS() { return _PROVIDERS; }
_parseChromeVersion(userAgent) {
if (isBlank(userAgent)) {
return -1;
}
var v = StringWrapper.split(userAgent, /Chrom(e|ium)\//g)[2];
if (isBlank(v)) {
return -1;
}
v = v.split('.')[0];
if (isBlank(v)) {
return -1;
}
return NumberWrapper.parseInt(v, 10);
}
gc() { return this._driver.executeScript('window.gc()'); }
timeBegin(name) {
return this._driver.executeScript(`console.time('${name}');`);
}
timeEnd(name, restartName = null) {
var script = `console.timeEnd('${name}');`;
if (isPresent(restartName)) {
script += `console.time('${restartName}');`;
}
return this._driver.executeScript(script);
}
// See [Chrome Trace Event
// Format](https://docs.google.com/document/d/1CvAClvFfyA5R-PhYUmn5OOQtYMH4h6I0nSsKchNAySU/edit)
readPerfLog() {
// TODO(tbosch): Chromedriver bug https://code.google.com/p/chromedriver/issues/detail?id=1098
// Need to execute at least one command so that the browser logs can be read out!
return this._driver.executeScript('1+1')
.then((_) => this._driver.logs('performance'))
.then((entries) => {
var events = [];
entries.forEach(entry => {
var message = Json.parse(entry['message'])['message'];
if (StringWrapper.equals(message['method'], 'Tracing.dataCollected')) {
events.push(message['params']);
}
if (StringWrapper.equals(message['method'], 'Tracing.bufferUsage')) {
throw new BaseException('The DevTools trace buffer filled during the test!');
}
});
return this._convertPerfRecordsToEvents(events);
});
}
_convertPerfRecordsToEvents(chromeEvents, normalizedEvents = null) {
if (isBlank(normalizedEvents)) {
normalizedEvents = [];
}
var majorGCPids = {};
chromeEvents.forEach((event) => {
var categories = this._parseCategories(event['cat']);
var name = event['name'];
if (this._isEvent(categories, name, ['blink.console'])) {
normalizedEvents.push(normalizeEvent(event, { 'name': name }));
}
else if (this._isEvent(categories, name, ['benchmark'], 'BenchmarkInstrumentation::ImplThreadRenderingStats')) {
// TODO(goderbauer): Instead of BenchmarkInstrumentation::ImplThreadRenderingStats the
// following events should be used (if available) for more accurate measurments:
// 1st choice: vsync_before - ground truth on Android
// 2nd choice: BenchmarkInstrumentation::DisplayRenderingStats - available on systems with
// new surfaces framework (not broadly enabled yet)
// 3rd choice: BenchmarkInstrumentation::ImplThreadRenderingStats - fallback event that is
// always available if something is rendered
var frameCount = event['args']['data']['frame_count'];
if (frameCount > 1) {
throw new BaseException('multi-frame render stats not supported');
}
if (frameCount == 1) {
normalizedEvents.push(normalizeEvent(event, { 'name': 'frame' }));
}
}
else if (this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'], 'Rasterize') ||
this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'], 'CompositeLayers')) {
normalizedEvents.push(normalizeEvent(event, { 'name': 'render' }));
}
else if (this._majorChromeVersion < 45) {
var normalizedEvent = this._processAsPreChrome45Event(event, categories, majorGCPids);
if (normalizedEvent != null)
normalizedEvents.push(normalizedEvent);
}
else {
var normalizedEvent = this._processAsPostChrome44Event(event, categories);
if (normalizedEvent != null)
normalizedEvents.push(normalizedEvent);
}
});
return normalizedEvents;
}
_processAsPreChrome45Event(event, categories, majorGCPids) {
var name = event['name'];
var args = event['args'];
var pid = event['pid'];
var ph = event['ph'];
if (this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'], 'FunctionCall') &&
(isBlank(args) || isBlank(args['data']) ||
!StringWrapper.equals(args['data']['scriptName'], 'InjectedScript'))) {
return normalizeEvent(event, { 'name': 'script' });
}
else if (this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'], 'RecalculateStyles') ||
this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'], 'Layout') ||
this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'], 'UpdateLayerTree') ||
this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'], 'Paint')) {
return normalizeEvent(event, { 'name': 'render' });
}
else if (this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'], 'GCEvent')) {
var normArgs = {
'usedHeapSize': isPresent(args['usedHeapSizeAfter']) ? args['usedHeapSizeAfter'] :
args['usedHeapSizeBefore']
};
if (StringWrapper.equals(ph, 'E')) {
normArgs['majorGc'] = isPresent(majorGCPids[pid]) && majorGCPids[pid];
}
majorGCPids[pid] = false;
return normalizeEvent(event, { 'name': 'gc', 'args': normArgs });
}
else if (this._isEvent(categories, name, ['v8'], 'majorGC') &&
StringWrapper.equals(ph, 'B')) {
majorGCPids[pid] = true;
}
return null; // nothing useful in this event
}
_processAsPostChrome44Event(event, categories) {
var name = event['name'];
var args = event['args'];
if (this._isEvent(categories, name, ['devtools.timeline', 'v8'], 'MajorGC')) {
var normArgs = {
'majorGc': true,
'usedHeapSize': isPresent(args['usedHeapSizeAfter']) ? args['usedHeapSizeAfter'] :
args['usedHeapSizeBefore']
};
return normalizeEvent(event, { 'name': 'gc', 'args': normArgs });
}
else if (this._isEvent(categories, name, ['devtools.timeline', 'v8'], 'MinorGC')) {
var normArgs = {
'majorGc': false,
'usedHeapSize': isPresent(args['usedHeapSizeAfter']) ? args['usedHeapSizeAfter'] :
args['usedHeapSizeBefore']
};
return normalizeEvent(event, { 'name': 'gc', 'args': normArgs });
}
else if (this._isEvent(categories, name, ['devtools.timeline', 'v8'], 'FunctionCall') &&
(isBlank(args) || isBlank(args['data']) ||
(!StringWrapper.equals(args['data']['scriptName'], 'InjectedScript') &&
!StringWrapper.equals(args['data']['scriptName'], '')))) {
return normalizeEvent(event, { 'name': 'script' });
}
else if (this._isEvent(categories, name, ['devtools.timeline', 'blink'], 'UpdateLayoutTree')) {
return normalizeEvent(event, { 'name': 'render' });
}
else if (this._isEvent(categories, name, ['devtools.timeline'], 'UpdateLayerTree') ||
this._isEvent(categories, name, ['devtools.timeline'], 'Layout') ||
this._isEvent(categories, name, ['devtools.timeline'], 'Paint')) {
return normalizeEvent(event, { 'name': 'render' });
}
else if (this._isEvent(categories, name, ['devtools.timeline'], 'ResourceReceivedData')) {
let normArgs = { 'encodedDataLength': args['data']['encodedDataLength'] };
return normalizeEvent(event, { 'name': 'receivedData', 'args': normArgs });
}
else if (this._isEvent(categories, name, ['devtools.timeline'], 'ResourceSendRequest')) {
let data = args['data'];
let normArgs = { 'url': data['url'], 'method': data['requestMethod'] };
return normalizeEvent(event, { 'name': 'sendRequest', 'args': normArgs });
}
else if (this._isEvent(categories, name, ['blink.user_timing'], 'navigationStart')) {
return normalizeEvent(event, { 'name': name });
}
return null; // nothing useful in this event
}
_parseCategories(categories) { return categories.split(','); }
_isEvent(eventCategories, eventName, expectedCategories, expectedName = null) {
var hasCategories = expectedCategories.reduce((value, cat) => { return value && ListWrapper.contains(eventCategories, cat); }, true);
return isBlank(expectedName) ? hasCategories :
hasCategories && StringWrapper.equals(eventName, expectedName);
}
perfLogFeatures() {
return new PerfLogFeatures({ render: true, gc: true, frameCapture: true, userTiming: true });
}
supports(capabilities) {
return this._majorChromeVersion != -1 &&
StringWrapper.equals(capabilities['browserName'].toLowerCase(), 'chrome');
}
}
function normalizeEvent(chromeEvent, data) {
var ph = chromeEvent['ph'];
if (StringWrapper.equals(ph, 'S')) {
ph = 'b';
}
else if (StringWrapper.equals(ph, 'F')) {
ph = 'e';
}
var result = { 'pid': chromeEvent['pid'], 'ph': ph, 'cat': 'timeline', 'ts': chromeEvent['ts'] / 1000 };
if (chromeEvent['ph'] === 'X') {
var dur = chromeEvent['dur'];
if (isBlank(dur)) {
dur = chromeEvent['tdur'];
}
result['dur'] = isBlank(dur) ? 0.0 : dur / 1000;
}
StringMapWrapper.forEach(data, (value, prop) => { result[prop] = value; });
return result;
}
var _PROVIDERS = [
bind(ChromeDriverExtension)
.toFactory((driver, userAgent) => new ChromeDriverExtension(driver, userAgent), [WebDriverAdapter, Options.USER_AGENT])
];
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chrome_driver_extension.js","sourceRoot":"","sources":["diffing_plugin_wrapper-output_path-xBLIBrVR.tmp/benchpress/src/webdriver/chrome_driver_extension.ts"],"names":[],"mappings":"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","sourcesContent":["import {bind, provide, Provider} from 'angular2/src/core/di';\nimport {ListWrapper, StringMapWrapper} from 'angular2/src/facade/collection';\nimport {\n  Json,\n  isPresent,\n  isBlank,\n  RegExpWrapper,\n  StringWrapper,\n  NumberWrapper\n} from 'angular2/src/facade/lang';\nimport {BaseException, WrappedException} from 'angular2/src/facade/exceptions';\n\nimport {WebDriverExtension, PerfLogFeatures} from '../web_driver_extension';\nimport {WebDriverAdapter} from '../web_driver_adapter';\nimport {Options} from '../common_options';\n\n/**\n * Set the following 'traceCategories' to collect metrics in Chrome:\n * 'v8,blink.console,disabled-by-default-devtools.timeline,devtools.timeline'\n *\n * In order to collect the frame rate related metrics, add 'benchmark'\n * to the list above.\n */\nexport class ChromeDriverExtension extends WebDriverExtension {\n  // TODO(tbosch): use static values when our transpiler supports them\n  static get BINDINGS(): Provider[] { return _PROVIDERS; }\n\n  private _majorChromeVersion: number;\n\n  constructor(private _driver: WebDriverAdapter, userAgent: string) {\n    super();\n    this._majorChromeVersion = this._parseChromeVersion(userAgent);\n  }\n\n  private _parseChromeVersion(userAgent: string): number {\n    if (isBlank(userAgent)) {\n      return -1;\n    }\n    var v = StringWrapper.split(userAgent, /Chrom(e|ium)\\//g)[2];\n    if (isBlank(v)) {\n      return -1;\n    }\n    v = v.split('.')[0];\n    if (isBlank(v)) {\n      return -1;\n    }\n    return NumberWrapper.parseInt(v, 10);\n  }\n\n  gc() { return this._driver.executeScript('window.gc()'); }\n\n  timeBegin(name: string): Promise<any> {\n    return this._driver.executeScript(`console.time('${name}');`);\n  }\n\n  timeEnd(name: string, restartName: string = null): Promise<any> {\n    var script = `console.timeEnd('${name}');`;\n    if (isPresent(restartName)) {\n      script += `console.time('${restartName}');`\n    }\n    return this._driver.executeScript(script);\n  }\n\n  // See [Chrome Trace Event\n  // Format](https://docs.google.com/document/d/1CvAClvFfyA5R-PhYUmn5OOQtYMH4h6I0nSsKchNAySU/edit)\n  readPerfLog(): Promise<any> {\n    // TODO(tbosch): Chromedriver bug https://code.google.com/p/chromedriver/issues/detail?id=1098\n    // Need to execute at least one command so that the browser logs can be read out!\n    return this._driver.executeScript('1+1')\n        .then((_) => this._driver.logs('performance'))\n        .then((entries) => {\n          var events = [];\n          entries.forEach(entry => {\n            var message = Json.parse(entry['message'])['message'];\n            if (StringWrapper.equals(message['method'], 'Tracing.dataCollected')) {\n              events.push(message['params']);\n            }\n            if (StringWrapper.equals(message['method'], 'Tracing.bufferUsage')) {\n              throw new BaseException('The DevTools trace buffer filled during the test!');\n            }\n          });\n          return this._convertPerfRecordsToEvents(events);\n        });\n  }\n\n  private _convertPerfRecordsToEvents(chromeEvents: Array<{[key: string]: any}>,\n                                      normalizedEvents: Array<{[key: string]: any}> = null) {\n    if (isBlank(normalizedEvents)) {\n      normalizedEvents = [];\n    }\n    var majorGCPids = {};\n    chromeEvents.forEach((event) => {\n      var categories = this._parseCategories(event['cat']);\n      var name = event['name'];\n      if (this._isEvent(categories, name, ['blink.console'])) {\n        normalizedEvents.push(normalizeEvent(event, {'name': name}));\n      } else if (this._isEvent(categories, name, ['benchmark'],\n                               'BenchmarkInstrumentation::ImplThreadRenderingStats')) {\n        // TODO(goderbauer): Instead of BenchmarkInstrumentation::ImplThreadRenderingStats the\n        // following events should be used (if available) for more accurate measurments:\n        //   1st choice: vsync_before - ground truth on Android\n        //   2nd choice: BenchmarkInstrumentation::DisplayRenderingStats - available on systems with\n        //               new surfaces framework (not broadly enabled yet)\n        //   3rd choice: BenchmarkInstrumentation::ImplThreadRenderingStats - fallback event that is\n        //               always available if something is rendered\n        var frameCount = event['args']['data']['frame_count'];\n        if (frameCount > 1) {\n          throw new BaseException('multi-frame render stats not supported');\n        }\n        if (frameCount == 1) {\n          normalizedEvents.push(normalizeEvent(event, {'name': 'frame'}));\n        }\n      } else if (this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'],\n                               'Rasterize') ||\n                 this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'],\n                               'CompositeLayers')) {\n        normalizedEvents.push(normalizeEvent(event, {'name': 'render'}));\n      } else if (this._majorChromeVersion < 45) {\n        var normalizedEvent = this._processAsPreChrome45Event(event, categories, majorGCPids);\n        if (normalizedEvent != null) normalizedEvents.push(normalizedEvent);\n      } else {\n        var normalizedEvent = this._processAsPostChrome44Event(event, categories);\n        if (normalizedEvent != null) normalizedEvents.push(normalizedEvent);\n      }\n    });\n    return normalizedEvents;\n  }\n\n  private _processAsPreChrome45Event(event, categories, majorGCPids) {\n    var name = event['name'];\n    var args = event['args'];\n    var pid = event['pid'];\n    var ph = event['ph'];\n    if (this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'],\n                      'FunctionCall') &&\n        (isBlank(args) || isBlank(args['data']) ||\n         !StringWrapper.equals(args['data']['scriptName'], 'InjectedScript'))) {\n      return normalizeEvent(event, {'name': 'script'});\n    } else if (this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'],\n                             'RecalculateStyles') ||\n               this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'],\n                             'Layout') ||\n               this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'],\n                             'UpdateLayerTree') ||\n               this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'],\n                             'Paint')) {\n      return normalizeEvent(event, {'name': 'render'});\n    } else if (this._isEvent(categories, name, ['disabled-by-default-devtools.timeline'],\n                             'GCEvent')) {\n      var normArgs = {\n        'usedHeapSize': isPresent(args['usedHeapSizeAfter']) ? args['usedHeapSizeAfter'] :\n                                                               args['usedHeapSizeBefore']\n      };\n      if (StringWrapper.equals(ph, 'E')) {\n        normArgs['majorGc'] = isPresent(majorGCPids[pid]) && majorGCPids[pid];\n      }\n      majorGCPids[pid] = false;\n      return normalizeEvent(event, {'name': 'gc', 'args': normArgs});\n    } else if (this._isEvent(categories, name, ['v8'], 'majorGC') &&\n               StringWrapper.equals(ph, 'B')) {\n      majorGCPids[pid] = true;\n    }\n    return null;  // nothing useful in this event\n  }\n\n  private _processAsPostChrome44Event(event, categories) {\n    var name = event['name'];\n    var args = event['args'];\n    if (this._isEvent(categories, name, ['devtools.timeline', 'v8'], 'MajorGC')) {\n      var normArgs = {\n        'majorGc': true,\n        'usedHeapSize': isPresent(args['usedHeapSizeAfter']) ? args['usedHeapSizeAfter'] :\n                                                               args['usedHeapSizeBefore']\n      };\n      return normalizeEvent(event, {'name': 'gc', 'args': normArgs});\n    } else if (this._isEvent(categories, name, ['devtools.timeline', 'v8'], 'MinorGC')) {\n      var normArgs = {\n        'majorGc': false,\n        'usedHeapSize': isPresent(args['usedHeapSizeAfter']) ? args['usedHeapSizeAfter'] :\n                                                               args['usedHeapSizeBefore']\n      };\n      return normalizeEvent(event, {'name': 'gc', 'args': normArgs});\n    } else if (this._isEvent(categories, name, ['devtools.timeline', 'v8'], 'FunctionCall') &&\n               (isBlank(args) || isBlank(args['data']) ||\n                (!StringWrapper.equals(args['data']['scriptName'], 'InjectedScript') &&\n                 !StringWrapper.equals(args['data']['scriptName'], '')))) {\n      return normalizeEvent(event, {'name': 'script'});\n    } else if (this._isEvent(categories, name, ['devtools.timeline', 'blink'],\n                             'UpdateLayoutTree')) {\n      return normalizeEvent(event, {'name': 'render'});\n    } else if (this._isEvent(categories, name, ['devtools.timeline'], 'UpdateLayerTree') ||\n               this._isEvent(categories, name, ['devtools.timeline'], 'Layout') ||\n               this._isEvent(categories, name, ['devtools.timeline'], 'Paint')) {\n      return normalizeEvent(event, {'name': 'render'});\n    } else if (this._isEvent(categories, name, ['devtools.timeline'], 'ResourceReceivedData')) {\n      let normArgs = {'encodedDataLength': args['data']['encodedDataLength']};\n      return normalizeEvent(event, {'name': 'receivedData', 'args': normArgs});\n    } else if (this._isEvent(categories, name, ['devtools.timeline'], 'ResourceSendRequest')) {\n      let data = args['data'];\n      let normArgs = {'url': data['url'], 'method': data['requestMethod']};\n      return normalizeEvent(event, {'name': 'sendRequest', 'args': normArgs});\n    } else if (this._isEvent(categories, name, ['blink.user_timing'], 'navigationStart')) {\n      return normalizeEvent(event, {'name': name});\n    }\n    return null;  // nothing useful in this event\n  }\n\n  private _parseCategories(categories: string): string[] { return categories.split(','); }\n\n  private _isEvent(eventCategories: string[], eventName: string, expectedCategories: string[],\n                   expectedName: string = null): boolean {\n    var hasCategories = expectedCategories.reduce(\n        (value, cat) => { return value && ListWrapper.contains(eventCategories, cat); }, true);\n    return isBlank(expectedName) ? hasCategories :\n                                   hasCategories && StringWrapper.equals(eventName, expectedName);\n  }\n\n  perfLogFeatures(): PerfLogFeatures {\n    return new PerfLogFeatures({render: true, gc: true, frameCapture: true, userTiming: true});\n  }\n\n  supports(capabilities: {[key: string]: any}): boolean {\n    return this._majorChromeVersion != -1 &&\n           StringWrapper.equals(capabilities['browserName'].toLowerCase(), 'chrome');\n  }\n}\n\nfunction normalizeEvent(chromeEvent: {[key: string]: any},\n                        data: {[key: string]: any}): {[key: string]: any} {\n  var ph = chromeEvent['ph'];\n  if (StringWrapper.equals(ph, 'S')) {\n    ph = 'b';\n  } else if (StringWrapper.equals(ph, 'F')) {\n    ph = 'e';\n  }\n  var result =\n      {'pid': chromeEvent['pid'], 'ph': ph, 'cat': 'timeline', 'ts': chromeEvent['ts'] / 1000};\n  if (chromeEvent['ph'] === 'X') {\n    var dur = chromeEvent['dur'];\n    if (isBlank(dur)) {\n      dur = chromeEvent['tdur'];\n    }\n    result['dur'] = isBlank(dur) ? 0.0 : dur / 1000;\n  }\n  StringMapWrapper.forEach(data, (value, prop) => { result[prop] = value; });\n  return result;\n}\n\nvar _PROVIDERS = [\n  bind(ChromeDriverExtension)\n      .toFactory((driver, userAgent) => new ChromeDriverExtension(driver, userAgent),\n                 [WebDriverAdapter, Options.USER_AGENT])\n];\n"]}