fix(benchpress): benchpress fixes and a smoke test for Dart

This commit is contained in:
Yegor Jbanov 2015-02-20 17:44:23 -08:00
parent 0a0c0d8302
commit d1f03e509b
18 changed files with 263 additions and 61 deletions

View File

@ -19,6 +19,7 @@ var multicopy = require('./tools/build/multicopy');
var karma = require('karma').server; var karma = require('karma').server;
var minimist = require('minimist'); var minimist = require('minimist');
var es5build = require('./tools/build/es5build'); var es5build = require('./tools/build/es5build');
var runServerDartTests = require('./tools/build/run_server_dart_tests');
var DART_SDK = require('./tools/build/dartdetect')(gulp); var DART_SDK = require('./tools/build/dartdetect')(gulp);
// ----------------------- // -----------------------
@ -484,38 +485,46 @@ gulp.task('docs/serve', function() {
}); });
// ------------------ // ------------------
// tests // karma tests
// These tests run in the browser and are allowed to access
// HTML DOM APIs.
function getBrowsersFromCLI() { function getBrowsersFromCLI() {
var args = minimist(process.argv.slice(2)); var args = minimist(process.argv.slice(2));
return [args.browsers?args.browsers:'DartiumWithWebPlatform'] return [args.browsers?args.browsers:'DartiumWithWebPlatform']
} }
gulp.task('test.js', function (done) { gulp.task('test.unit.js', function (done) {
karma.start({configFile: __dirname + '/karma-js.conf.js'}, done); karma.start({configFile: __dirname + '/karma-js.conf.js'}, done);
}); });
gulp.task('test.dart', function (done) { gulp.task('test.unit.dart', function (done) {
karma.start({configFile: __dirname + '/karma-dart.conf.js'}, done); karma.start({configFile: __dirname + '/karma-dart.conf.js'}, done);
}); });
gulp.task('test.js/ci', function (done) { gulp.task('test.unit.js/ci', function (done) {
karma.start({configFile: __dirname + '/karma-js.conf.js', singleRun: true, reporters: ['dots'], browsers: getBrowsersFromCLI()}, done); karma.start({configFile: __dirname + '/karma-js.conf.js',
singleRun: true, reporters: ['dots'], browsers: getBrowsersFromCLI()}, done);
}); });
gulp.task('test.dart/ci', function (done) { gulp.task('test.unit.dart/ci', function (done) {
karma.start({configFile: __dirname + '/karma-dart.conf.js', singleRun: true, reporters: ['dots'], browsers: getBrowsersFromCLI()}, done); karma.start({configFile: __dirname + '/karma-dart.conf.js',
singleRun: true, reporters: ['dots'], browsers: getBrowsersFromCLI()}, done);
}); });
// ------------------
// server tests
// These tests run on the VM on the command-line and are
// allowed to access the file system and network.
gulp.task('test.server.dart', runServerDartTests(gulp, gulpPlugins, {
dest: 'dist/dart'
}));
// -----------------
// test builders
gulp.task('test.transpiler.unittest', function (done) { gulp.task('test.transpiler.unittest', function (done) {
return gulp.src('tools/transpiler/unittest/**/*.js') return gulp.src('tools/transpiler/unittest/**/*.js')
.pipe(jasmine({ .pipe(jasmine({
includeStackTrace: true includeStackTrace: true
})) }))
}); });
gulp.task('ci', function(done) {
runSequence(
'test.transpiler.unittest',
'test.js/ci',
'test.dart/ci',
done
);
});
// Copy test resources to dist
gulp.task('tests/transform.dart', function() { gulp.task('tests/transform.dart', function() {
return gulp.src('modules/angular2/test/transform/**') return gulp.src('modules/angular2/test/transform/**')
.pipe(gulp.dest('dist/dart/angular2/test/transform')); .pipe(gulp.dest('dist/dart/angular2/test/transform'));

View File

@ -95,6 +95,14 @@ class StringJoiner {
} }
class NumberWrapper { class NumberWrapper {
static String toFixed(num n, int fractionDigits) {
return n.toStringAsFixed(fractionDigits);
}
static bool equal(num a, num b) {
return a == b;
}
static int parseIntAutoRadix(String text) { static int parseIntAutoRadix(String text) {
return int.parse(text); return int.parse(text);
} }

View File

@ -138,6 +138,14 @@ export class NumberParseError extends Error {
export class NumberWrapper { export class NumberWrapper {
static toFixed(n:number, fractionDigits:int):string {
return n.toFixed(fractionDigits);
}
static equal(a, b):boolean {
return a === b;
}
static parseIntAutoRadix(text:string):int { static parseIntAutoRadix(text:string):int {
var result:int = parseInt(text); var result:int = parseInt(text);
if (isNaN(result)) { if (isNaN(result)) {

View File

@ -123,11 +123,11 @@ void _runTests() {
// Read in input & output files. // Read in input & output files.
config.assetPathToInputPath.forEach((key, value) { config.assetPathToInputPath.forEach((key, value) {
config.assetPathToInputPath[key] = config.assetPathToInputPath[key] =
cache.putIfAbsent(value, () => new File(value).readAsStringSync()); cache.putIfAbsent(value, () => new File('test/transform/${value}').readAsStringSync());
}); });
config.assetPathToExpectedOutputPath.forEach((key, value) { config.assetPathToExpectedOutputPath.forEach((key, value) {
config.assetPathToExpectedOutputPath[key] = cache.putIfAbsent(value, () { config.assetPathToExpectedOutputPath[key] = cache.putIfAbsent(value, () {
var code = new File(value).readAsStringSync(); var code = new File('test/transform/${value}').readAsStringSync();
return value.endsWith('dart') ? formatter.format(code) : code; return value.endsWith('dart') ? formatter.format(code) : code;
}); });
}); });

View File

@ -12,5 +12,6 @@ dependencies:
stack_trace: '>=1.1.1 <1.2.0' stack_trace: '>=1.1.1 <1.2.0'
angular2: angular2:
path: ../angular2 path: ../angular2
webdriver: ">=0.9.0 <0.10.0"
dev_dependencies: dev_dependencies:
guinness: ">=0.1.16 <0.2.0" guinness: ">=0.1.16 <0.2.0"

View File

@ -1,5 +1,5 @@
import { PromiseWrapper, Promise } from 'angular2/src/facade/async'; import { PromiseWrapper, Promise } from 'angular2/src/facade/async';
import { isPresent, isBlank, int, BaseException, StringWrapper } from 'angular2/src/facade/lang'; import { isPresent, isBlank, int, BaseException, StringWrapper, Math } from 'angular2/src/facade/lang';
import { ListWrapper, StringMap, StringMapWrapper } from 'angular2/src/facade/collection'; import { ListWrapper, StringMap, StringMapWrapper } from 'angular2/src/facade/collection';
import { bind, OpaqueToken } from 'angular2/di'; import { bind, OpaqueToken } from 'angular2/di';
@ -95,7 +95,12 @@ export class PerflogMetric extends Metric {
if (needSort) { if (needSort) {
// Need to sort because of the ph==='X' events // Need to sort because of the ph==='X' events
ListWrapper.sort(this._remainingEvents, (a,b) => { ListWrapper.sort(this._remainingEvents, (a,b) => {
return a['ts'] - b['ts']; var diff = a['ts'] - b['ts'];
return diff > 0
? 1
: diff < 0
? -1
: 0;
}); });
} }
} }

View File

@ -1,4 +1,4 @@
import { print, isPresent, isBlank } from 'angular2/src/facade/lang'; import { print, isPresent, isBlank, NumberWrapper } from 'angular2/src/facade/lang';
import { StringMapWrapper, ListWrapper, List } from 'angular2/src/facade/collection'; import { StringMapWrapper, ListWrapper, List } from 'angular2/src/facade/collection';
import { Promise, PromiseWrapper } from 'angular2/src/facade/async'; import { Promise, PromiseWrapper } from 'angular2/src/facade/async';
import { Math } from 'angular2/src/facade/math'; import { Math } from 'angular2/src/facade/math';
@ -28,14 +28,8 @@ export class ConsoleReporter extends Reporter {
return result + value; return result + value;
} }
static _formatNum(num) { static _formatNum(n) {
var result; return NumberWrapper.toFixed(n, 2);
if (num === 0) {
result = '000';
} else {
result = `${Math.floor(num * 100)}`;
}
return result.substring(0, result.length - 2) + '.' + result.substring(result.length-2);
} }
static _sortedProps(obj) { static _sortedProps(obj) {
@ -89,7 +83,8 @@ export class ConsoleReporter extends Reporter {
var sample = ListWrapper.map(validSample, (measureValues) => measureValues.values[metricName]); var sample = ListWrapper.map(validSample, (measureValues) => measureValues.values[metricName]);
var mean = Statistic.calculateMean(sample); var mean = Statistic.calculateMean(sample);
var cv = Statistic.calculateCoefficientOfVariation(sample, mean); var cv = Statistic.calculateCoefficientOfVariation(sample, mean);
return `${ConsoleReporter._formatNum(mean)}\u00B1${Math.floor(cv)}%`; var formattedCv = NumberWrapper.isNaN(cv) ? 'NaN' : Math.floor(cv);
return `${ConsoleReporter._formatNum(mean)}\u00B1${formattedCv}%`;
}) })
); );
return PromiseWrapper.resolve(null); return PromiseWrapper.resolve(null);

View File

@ -1,23 +1,32 @@
library benchpress.src.webdriver.async_webdriver_adapter_dart; library benchpress.src.webdriver.async_webdriver_adapter_dart;
import 'package:webdriver/webdriver.dart' show WebDriver, LogEntry;
import 'package:angular2/src/facade/async.dart' show Future; import 'package:angular2/src/facade/async.dart' show Future;
import '../web_driver_adapter.dart' show WebDriverAdapter; import '../web_driver_adapter.dart' show WebDriverAdapter;
class AsyncWebDriverAdapter extends WebDriverAdapter { class AsyncWebDriverAdapter extends WebDriverAdapter {
dynamic _driver; WebDriver _driver;
AsyncWebDriverAdapter(driver) { AsyncWebDriverAdapter(this._driver);
this._driver = driver;
}
Future waitFor(Function callback) { Future waitFor(Function callback) {
return callback(); return callback();
} }
Future executeScript(String script) { Future executeScript(String script) {
return this._driver.execute(script); return _driver.execute(script, const[]);
} }
Future capabilities() {
return this._driver.capabilities; Map capabilities() {
return _driver.capabilities;
} }
Future logs(String type) {
return this._driver.logs.get(type); Future<List<Map>> logs(String type) {
return _driver.logs.get(type)
.map((LogEntry entry) => {
'message': entry.message
})
.fold(<Map>[], (log, Map entry) {
return log..add(entry);
});
} }
} }

View File

@ -7,6 +7,8 @@ dependencies:
browser: '>=0.10.0 <0.11.0' browser: '>=0.10.0 <0.11.0'
dev_dependencies: dev_dependencies:
guinness: ">=0.1.16 <0.2.0" guinness: ">=0.1.16 <0.2.0"
benchpress:
path: ../benchpress
transformers: transformers:
- angular2: - angular2:
bootstrap_entry_point: web/src/hello_world/index_common.dart bootstrap_entry_point: web/src/hello_world/index_common.dart

View File

@ -0,0 +1,16 @@
<!doctype html>
<html>
<head>
<title>Benchpress test</title>
</head>
<body>
<button onclick="pleaseLog()">Click me</button>
<div id="log"></div>
<script type="text/javascript">
function pleaseLog() {
document.getElementById("log").innerHTML = "hi";
}
</script>
</body>
</html>

View File

@ -0,0 +1,58 @@
import 'dart:async';
import 'dart:io' show Platform;
import 'package:guinness/guinness.dart';
import 'package:benchpress/benchpress.dart';
import 'package:benchpress/src/webdriver/async_webdriver_adapter.dart';
import 'package:webdriver/webdriver.dart' show WebDriver, Capabilities, LogType, LogLevel, By;
main() {
describe('benchpress', () {
WebDriver driver;
Runner runner;
beforeEach(() async {
driver = await createTestDriver();
await driver.get('http://localhost:8002/examples/src/benchpress/index.html');
var bindings = [
bind(WebDriverAdapter).toFactory(() => new AsyncWebDriverAdapter(driver), [])
];
runner = new Runner(bindings);
});
afterEach(() async {
await driver.close();
});
it('should work', () {
return runner.sample(
id: 'benchpress smoke test',
execute: () async {
var button = await driver.findElement(const By.tagName('button'));
await button.click();
var logText = await (await driver.findElement(const By.id('log'))).text;
expect(logText, 'hi');
}
);
});
});
}
Future<WebDriver> createTestDriver() {
Map env = Platform.environment;
return WebDriver.createDriver(desiredCapabilities: {
'name': 'Dartium',
'browserName': 'chrome',
'chromeOptions': {
'binary': env['DARTIUM_BIN'],
'args': ['--js-flags=--expose-gc'],
'perfLoggingPrefs': {
'traceCategories': 'blink.console,disabled-by-default-devtools.timeline'
},
},
'loggingPrefs': {
'performance': 'ALL',
'browser': 'ALL',
}
});
}

View File

@ -10,4 +10,7 @@ cd $SCRIPT_DIR/../..
${SCRIPT_DIR}/build_$MODE.sh ${SCRIPT_DIR}/build_$MODE.sh
${SCRIPT_DIR}/test_unit_$MODE.sh ${SCRIPT_DIR}/test_unit_$MODE.sh
if [ "$MODE" == "dart" ]; then # JS doesn't yet have server tests
${SCRIPT_DIR}/test_server_$MODE.sh
fi
${SCRIPT_DIR}/test_e2e_$MODE.sh ${SCRIPT_DIR}/test_e2e_$MODE.sh

29
scripts/ci/test_server_dart.sh Executable file
View File

@ -0,0 +1,29 @@
#!/bin/bash
set -e
echo =============================================================================
# go to project dir
SCRIPT_DIR=$(dirname $0)
source $SCRIPT_DIR/env_dart.sh
cd $SCRIPT_DIR/../..
./node_modules/.bin/webdriver-manager update
./node_modules/.bin/webdriver-manager start&
webdriverServerPid=$!
ps -ef | grep webdriver-manager
./node_modules/.bin/gulp serve.js.dart2js&
serverPid=$!
function killAllServers () {
kill $serverPid
pkill -P $webdriverServerPid
}
trap killAllServers EXIT
# wait for server to come up!
sleep 3
./node_modules/.bin/gulp test.transpiler.unittest
./node_modules/.bin/gulp test.server.dart --browsers=$KARMA_BROWSERS

View File

@ -8,4 +8,4 @@ source $SCRIPT_DIR/env_dart.sh
cd $SCRIPT_DIR/../.. cd $SCRIPT_DIR/../..
./node_modules/.bin/gulp test.transpiler.unittest ./node_modules/.bin/gulp test.transpiler.unittest
./node_modules/.bin/gulp test.dart/ci --browsers=$KARMA_BROWSERS ./node_modules/.bin/gulp test.unit.dart/ci --browsers=$KARMA_BROWSERS

View File

@ -8,4 +8,4 @@ source $SCRIPT_DIR/env_dart.sh
cd $SCRIPT_DIR/../.. cd $SCRIPT_DIR/../..
./node_modules/.bin/gulp test.transpiler.unittest ./node_modules/.bin/gulp test.transpiler.unittest
./node_modules/.bin/gulp test.js/ci --browsers=$KARMA_BROWSERS ./node_modules/.bin/gulp test.unit.js/ci --browsers=$KARMA_BROWSERS

View File

@ -4,30 +4,32 @@ var spawn = require('child_process').spawn;
var path = require('path'); var path = require('path');
var glob = require('glob'); var glob = require('glob');
var fs = require('fs'); var fs = require('fs');
var util = require('./util');
module.exports = function(gulp, plugins, config) { module.exports = function(gulp, plugins, config) {
return function() { return function() {
var dartModuleFolders = [].slice.call(glob.sync(config.dest + '/*'));
var tempFile = '_analyzer.dart'; var tempFile = '_analyzer.dart';
// analyze in parallel! return util.forEachSubDir(
return Q.all(dartModuleFolders.map(function(dir) { config.dest,
var srcFiles = [].slice.call(glob.sync(dir + '{/lib,/web}/**/*.dart', { function(dir) {
cwd: dir var srcFiles = [].slice.call(glob.sync('{/lib,/web}/**/*.dart', {
})); cwd: dir
var testFiles = [].slice.call(glob.sync('test/**/*_spec.dart', { }));
cwd: dir var testFiles = [].slice.call(glob.sync('test/**/*_spec.dart', {
})); cwd: dir
var analyzeFile = ['library _analyzer;']; }));
srcFiles.concat(testFiles).forEach(function(fileName, index) { var analyzeFile = ['library _analyzer;'];
if (fileName !== tempFile && fileName.indexOf("/packages/") === -1) { srcFiles.concat(testFiles).forEach(function(fileName, index) {
analyzeFile.push('import "./'+fileName+'" as mod'+index+';'); if (fileName !== tempFile && fileName.indexOf("/packages/") === -1) {
} analyzeFile.push('import "./'+fileName+'" as mod'+index+';');
}); }
fs.writeFileSync(path.join(dir, tempFile), analyzeFile.join('\n')); });
var defer = Q.defer(); fs.writeFileSync(path.join(dir, tempFile), analyzeFile.join('\n'));
analyze(dir, defer.makeNodeResolver()); var defer = Q.defer();
return defer.promise; analyze(dir, defer.makeNodeResolver());
})); return defer.promise;
}
);
function analyze(dirName, done) { function analyze(dirName, done) {
//TODO remove --package-warnings once dartanalyzer handles transitive libraries //TODO remove --package-warnings once dartanalyzer handles transitive libraries

View File

@ -0,0 +1,48 @@
var Q = require('q');
var glob = require('glob');
var fs = require('fs');
var path = require('path');
var spawn = require('child_process').spawn;
var util = require('./util');
module.exports = function(gulp, plugins, config) {
return function() {
return util.forEachSubDir(
config.dest,
function(dir) {
var testDir = path.join(dir, 'test');
var relativeMasterTestFile = 'test/_all_tests.dart';
var testFiles = [].slice.call(glob.sync('**/*.server.spec.dart', {
cwd: testDir
}));
if (testFiles.length == 0) {
// No test files found
return;
}
var header = ['library _all_tests;', ''];
var main = ['main() {'];
testFiles.forEach(function(fileName, index) {
header.push('import "' + fileName + '" as test_' + index + ';');
main.push(' test_' + index + '.main();');
});
header.push('');
main.push('}');
var absMasterTestFile = path.join(dir, relativeMasterTestFile);
fs.writeFileSync(absMasterTestFile, header.concat(main).join('\n'));
var defer = Q.defer();
var done = defer.makeNodeResolver();
util.processToPromise(spawn('dart', ['-c', relativeMasterTestFile], {
stdio: 'inherit',
cwd: dir
})).then(
function() { done(); },
function(error) { done(error); }
);
return defer.promise;
}
);
};
};

View File

@ -1,15 +1,24 @@
var Q = require('q'); var Q = require('q');
var path = require('path'); var path = require('path');
var minimatch = require('minimatch'); var minimatch = require('minimatch');
var glob = require('glob');
module.exports = { module.exports = {
processToPromise: processToPromise, processToPromise: processToPromise,
streamToPromise: streamToPromise, streamToPromise: streamToPromise,
insertSrcFolder: insertSrcFolder, insertSrcFolder: insertSrcFolder,
filterByFile: filterByFile filterByFile: filterByFile,
forEachSubDir: forEachSubDir
}; };
function forEachSubDir(dir, callback) {
var moduleFolders = [].slice.call(glob.sync(dir + '/*'));
return Q.all(moduleFolders.map(function(subDir) {
return callback(subDir);
}));
};
function processToPromise(process) { function processToPromise(process) {
var defer = Q.defer(); var defer = Q.defer();
process.on('close', function(code) { process.on('close', function(code) {