feat(build): initial SauceLabs setup

Closes #2347
This commit is contained in:
Marc Laval 2015-06-02 16:29:09 +02:00 committed by Tobias Bosch
parent bb50cda181
commit eebd736cfe
12 changed files with 320 additions and 39 deletions

View File

@ -17,6 +17,8 @@ env:
- KARMA_BROWSERS=DartiumWithWebPlatform - KARMA_BROWSERS=DartiumWithWebPlatform
- E2E_BROWSERS=Dartium - E2E_BROWSERS=Dartium
- LOGS_DIR=/tmp/angular-build/logs - LOGS_DIR=/tmp/angular-build/logs
- SAUCE_USERNAME=angular-ci
- SAUCE_ACCESS_KEY=9b988f434ff8-fbca-8aa4-4ae3-35442987
- ARCH=linux-x64 - ARCH=linux-x64
# Token for tsd to increase github rate limit # Token for tsd to increase github rate limit
# See https://github.com/DefinitelyTyped/tsd#tsdrc # See https://github.com/DefinitelyTyped/tsd#tsdrc
@ -30,6 +32,11 @@ env:
- MODE=dart DART_CHANNEL=stable - MODE=dart DART_CHANNEL=stable
- MODE=dart DART_CHANNEL=dev - MODE=dart DART_CHANNEL=dev
- MODE=dart_experimental DART_CHANNEL=dev - MODE=dart_experimental DART_CHANNEL=dev
- MODE=saucelabs DART_CHANNEL=dev
matrix:
allow_failures:
- env: "MODE=saucelabs DART_CHANNEL=dev"
addons: addons:
firefox: "38.0" firefox: "38.0"

View File

@ -40,6 +40,7 @@ var uglify = require('gulp-uglify');
var shouldLog = require('./tools/build/logging'); var shouldLog = require('./tools/build/logging');
var tslint = require('gulp-tslint'); var tslint = require('gulp-tslint');
var dartSdk = require('./tools/build/dart'); var dartSdk = require('./tools/build/dart');
var sauceConf = require('./sauce.conf');
require('./tools/check-environment')({ require('./tools/check-environment')({
requiredNpmVersion: '>=2.9.0', requiredNpmVersion: '>=2.9.0',
@ -530,13 +531,35 @@ gulp.task('test.all.dart', shell.task(['./scripts/ci/test_dart.sh']))
// These tests run in the browser and are allowed to access // These tests run in the browser and are allowed to access
// HTML DOM APIs. // HTML DOM APIs.
function getBrowsersFromCLI() { function getBrowsersFromCLI() {
var isSauce = false;
var args = minimist(process.argv.slice(2)); var args = minimist(process.argv.slice(2));
return [args.browsers?args.browsers:'DartiumWithWebPlatform'] var rawInput = args.browsers?args.browsers:'DartiumWithWebPlatform';
var inputList = rawInput.replace(' ', '').split(',');
var outputList = [];
for (var i = 0; i < inputList.length; i++) {
var input = inputList[i];
if (sauceConf.customLaunchers.hasOwnProperty(input)) {
//Non-sauce browsers case: overrides everything, ignoring other options
outputList = [input];
isSauce = false;
break;
} else if (sauceConf.customLaunchers.hasOwnProperty("SL_" + input.toUpperCase())) {
isSauce = true;
outputList.push("SL_" + input.toUpperCase());
} else if (sauceConf.aliases.hasOwnProperty(input.toUpperCase())) {
outputList = outputList.concat(sauceConf.aliases[input]);
isSauce = true;
} else {
throw new Error('ERROR: unknown browser found in getBrowsersFromCLI()');
}
}
return {
browsersToRun: outputList.filter(function(item, pos, self) {return self.indexOf(item) == pos;}),
isSauce: isSauce
}
} }
gulp.task('test.unit.js', ['build.js.dev'], function (done) {
gulp.task('test.unit.js', ['build.js.dev'], function (neverDone) {
runSequence( runSequence(
'!test.unit.js/karma-server', '!test.unit.js/karma-server',
function() { function() {
@ -548,6 +571,16 @@ gulp.task('test.unit.js', ['build.js.dev'], function (neverDone) {
); );
}); });
gulp.task('test.unit.js.sauce', ['build.js.dev'], function (done) {
var browserConf = getBrowsersFromCLI();
if (browserConf.isSauce) {
karma.server.start({configFile: __dirname + '/karma-js.conf.js',
singleRun: true, browserNoActivityTimeout: 240000, captureTimeout: 120000, reporters: ['dots'], browsers: browserConf.browsersToRun},
function(err) {done(); process.exit(err ? 1 : 0)});
} else {
throw new Error('ERROR: no Saucelabs browsers provided, add them with the --browsers option');
}
});
gulp.task('!test.unit.js/karma-server', function() { gulp.task('!test.unit.js/karma-server', function() {
karma.server.start({configFile: __dirname + '/karma-js.conf.js', reporters: 'dots'}); karma.server.start({configFile: __dirname + '/karma-js.conf.js', reporters: 'dots'});
@ -598,13 +631,21 @@ gulp.task('!test.unit.dart/karma-server', function() {
gulp.task('test.unit.js/ci', function (done) { gulp.task('test.unit.js/ci', function (done) {
var browserConf = getBrowsersFromCLI();
karma.server.start({configFile: __dirname + '/karma-js.conf.js', karma.server.start({configFile: __dirname + '/karma-js.conf.js',
singleRun: true, reporters: ['dots'], browsers: getBrowsersFromCLI()}, done); singleRun: true, reporters: ['dots'], browsers: browserConf.browsersToRun}, done);
});
gulp.task('test.unit.js.sauce/ci', function (done) {
karma.server.start({configFile: __dirname + '/karma-js.conf.js',
singleRun: true, browserNoActivityTimeout: 240000, captureTimeout: 120000, reporters: ['dots', 'saucelabs'], browsers: sauceConf.aliases.CI},
function(err) {done(); process.exit(err ? 1 : 0)});
}); });
gulp.task('test.unit.dart/ci', function (done) { gulp.task('test.unit.dart/ci', function (done) {
var browserConf = getBrowsersFromCLI();
karma.server.start({configFile: __dirname + '/karma-dart.conf.js', karma.server.start({configFile: __dirname + '/karma-dart.conf.js',
singleRun: true, reporters: ['dots'], browsers: getBrowsersFromCLI()}, done); singleRun: true, reporters: ['dots'], browsers: browserConf.browsersToRun}, done);
}); });

View File

@ -1,3 +1,5 @@
var sauceConf = require('./sauce.conf');
// Karma configuration // Karma configuration
// Generated on Thu Sep 25 2014 11:52:02 GMT-0700 (PDT) // Generated on Thu Sep 25 2014 11:52:02 GMT-0700 (PDT)
module.exports = function(config) { module.exports = function(config) {
@ -48,11 +50,7 @@ module.exports = function(config) {
'/packages/examples': '/base/dist/dart/examples/lib' '/packages/examples': '/base/dist/dart/examples/lib'
}, },
customLaunchers: { customLaunchers: sauceConf.customLaunchers,
DartiumWithWebPlatform: {
base: 'Dartium',
flags: ['--enable-experimental-web-platform-features'] }
},
browsers: ['DartiumWithWebPlatform'], browsers: ['DartiumWithWebPlatform'],
port: 9877 port: 9877

View File

@ -1,3 +1,5 @@
var sauceConf = require('./sauce.conf');
// Karma configuration // Karma configuration
// Generated on Thu Sep 25 2014 11:52:02 GMT-0700 (PDT) // Generated on Thu Sep 25 2014 11:52:02 GMT-0700 (PDT)
module.exports = function(config) { module.exports = function(config) {
@ -25,23 +27,42 @@ module.exports = function(config) {
'node_modules/reflect-metadata/Reflect.js', 'node_modules/reflect-metadata/Reflect.js',
'tools/build/file2modulename.js', 'tools/build/file2modulename.js',
'test-main.js', 'test-main.js',
{pattern: 'modules/**/test/**/static_assets/**', included: false, watched: false} {pattern: 'modules/**/test/**/static_assets/**', included: false, watched: false},
'modules/angular2/src/test_lib/shims_for_IE.ts'
], ],
exclude: [ exclude: [
'dist/js/dev/es5/**/e2e_test/**', 'dist/js/dev/es5/**/e2e_test/**',
], ],
customLaunchers: { customLaunchers: sauceConf.customLaunchers,
DartiumWithWebPlatform: {
base: 'Dartium', sauceLabs: {
flags: ['--enable-experimental-web-platform-features'] }, testName: 'Angular2',
ChromeNoSandbox: { startConnect: false,
base: 'Chrome', recordVideo: false,
flags: ['--no-sandbox'] } recordScreenshots: false,
options: {
'selenium-version': '2.45.0',
'command-timeout': 600,
'idle-timeout': 600,
'max-duration': 5400
}
}, },
browsers: ['ChromeCanary'], browsers: ['ChromeCanary'],
port: 9876 port: 9876
}); });
if (process.env.TRAVIS) {
config.sauceLabs.build = 'TRAVIS #' + process.env.TRAVIS_BUILD_NUMBER + ' (' + process.env.TRAVIS_BUILD_ID + ')';
config.sauceLabs.tunnelIdentifier = process.env.TRAVIS_JOB_NUMBER;
// TODO(mlaval): remove once SauceLabs supports websockets.
// This speeds up the capturing a bit, as browsers don't even try to use websocket.
config.transports = ['xhr-polling'];
}
}; };

View File

@ -177,15 +177,15 @@ void _it(gnsFn, name, fn) {
}); });
} }
void it(name, fn) { void it(name, fn, [timeOut = null]) {
_it(gns.it, name, fn); _it(gns.it, name, fn);
} }
void iit(name, fn) { void iit(name, fn, [timeOut = null]) {
_it(gns.iit, name, fn); _it(gns.iit, name, fn);
} }
void xit(name, fn) { void xit(name, fn, [timeOut = null]) {
_it(gns.xit, name, fn); _it(gns.xit, name, fn);
} }

View File

@ -129,7 +129,7 @@ export function beforeEachBindings(fn) {
}); });
} }
function _it(jsmFn, name, fn) { function _it(jsmFn, name, fn, timeOut) {
var runner = runnerStack[runnerStack.length - 1]; var runner = runnerStack[runnerStack.length - 1];
jsmFn(name, function(done) { jsmFn(name, function(done) {
@ -156,19 +156,19 @@ function _it(jsmFn, name, fn) {
inIt = false; inIt = false;
if (!async) done(); if (!async) done();
}); }, timeOut);
} }
export function it(name, fn) { export function it(name, fn, timeOut = null) {
return _it(jsmIt, name, fn); return _it(jsmIt, name, fn, timeOut);
} }
export function xit(name, fn) { export function xit(name, fn, timeOut = null) {
return _it(jsmXIt, name, fn); return _it(jsmXIt, name, fn, timeOut);
} }
export function iit(name, fn) { export function iit(name, fn, timeOut = null) {
return _it(jsmIIt, name, fn); return _it(jsmIIt, name, fn, timeOut);
} }
// Some Map polyfills don't polyfill Map.toString correctly, which // Some Map polyfills don't polyfill Map.toString correctly, which

View File

@ -27,7 +27,7 @@ export function main() {
expect(text.trim()).toEqual('<p>hey</p>'); expect(text.trim()).toEqual('<p>hey</p>');
async.done(); async.done();
}); });
})); }), 10000);
it('should reject the Promise on failure', inject([AsyncTestCompleter], (async) => { it('should reject the Promise on failure', inject([AsyncTestCompleter], (async) => {
PromiseWrapper.catchError(xhr.get(url404), (e) => { PromiseWrapper.catchError(xhr.get(url404), (e) => {
@ -35,6 +35,6 @@ export function main() {
async.done(); async.done();
return null; return null;
}); });
})); }), 10000);
}); });
} }

138
sauce.conf.js Normal file
View File

@ -0,0 +1,138 @@
var customLaunchers = {
'DartiumWithWebPlatform': {
base: 'Dartium',
flags: ['--enable-experimental-web-platform-features'] },
'ChromeNoSandbox': {
base: 'Chrome',
flags: ['--no-sandbox'] },
'SL_CHROME': {
base: 'SauceLabs',
browserName: 'chrome',
version: '43'
},
'SL_CHROMEBETA': {
base: 'SauceLabs',
browserName: 'chrome',
version: 'beta'
},
'SL_CHROMEDEV': {
base: 'SauceLabs',
browserName: 'chrome',
version: 'dev'
},
'SL_FIREFOX': {
base: 'SauceLabs',
browserName: 'firefox',
version: '37'
},
'SL_FIREFOXBETA': {
base: 'SauceLabs',
browserName: 'firefox',
version: 'beta'
},
'SL_FIREFOXDEV': {
base: 'SauceLabs',
browserName: 'firefox',
version: 'dev'
},
'SL_SAFARI7': {
base: 'SauceLabs',
browserName: 'safari',
platform: 'OS X 10.9',
version: '7'
},
'SL_SAFARI8': {
base: 'SauceLabs',
browserName: 'safari',
platform: 'OS X 10.10',
version: '8'
},
'SL_IOS7': {
base: 'SauceLabs',
browserName: 'iphone',
platform: 'OS X 10.10',
version: '7.1'
},
'SL_IOS8': {
base: 'SauceLabs',
browserName: 'iphone',
platform: 'OS X 10.10',
version: '8.2'
},
'SL_IE9': {
base: 'SauceLabs',
browserName: 'internet explorer',
platform: 'Windows 2008',
version: '9'
},
'SL_IE10': {
base: 'SauceLabs',
browserName: 'internet explorer',
platform: 'Windows 2012',
version: '10'
},
'SL_IE11': {
base: 'SauceLabs',
browserName: 'internet explorer',
platform: 'Windows 8.1',
version: '11'
},
'SL_ANDROID5.1': {
base: 'SauceLabs',
browserName: 'android',
platform: 'Linux',
version: '5.1'
},
'SL_ANDROID4.4': {
base: 'SauceLabs',
browserName: 'android',
platform: 'Linux',
version: '4.4'
},
'SL_ANDROID4.3': {
base: 'SauceLabs',
browserName: 'android',
platform: 'Linux',
version: '4.3'
},
'SL_ANDROID4.2': {
base: 'SauceLabs',
browserName: 'android',
platform: 'Linux',
version: '4.2'
},
'SL_ANDROID4.1': {
base: 'SauceLabs',
browserName: 'android',
platform: 'Linux',
version: '4.1'
},
'SL_ANDROID4.0': {
base: 'SauceLabs',
browserName: 'android',
platform: 'Linux',
version: '4.0'
}
};
var aliases = {
'ALL': Object.keys(customLaunchers).filter(function(item) {return customLaunchers[item].base == 'SauceLabs';}),
'DESKTOP': ['SL_CHROME', 'SL_FIREFOX', 'SL_IE9', 'SL_IE10', 'SL_IE11', 'SL_SAFARI7', 'SL_SAFARI8'],
'MOBILE': ['SL_ANDROID4.0', 'SL_ANDROID4.1', 'SL_ANDROID4.2', 'SL_ANDROID4.3', 'SL_ANDROID4.4', 'SL_ANDROID5.1', 'SL_IOS7', 'SL_IOS8'],
'ANDROID': ['SL_ANDROID4.0', 'SL_ANDROID4.1', 'SL_ANDROID4.2', 'SL_ANDROID4.3', 'SL_ANDROID4.4', 'SL_ANDROID5.1'],
'IE': ['SL_IE9', 'SL_IE10', 'SL_IE11'],
'IOS': ['SL_IOS7', 'SL_IOS8'],
'SAFARI': ['SL_SAFARI7', 'SL_SAFARI8'],
'BETA': ['SL_CHROMEBETA', 'SL_FIREFOXBETA'],
'DEV': ['SL_CHROMEDEV', 'SL_FIREFOXDEV'],
'CI': ['SL_CHROME', 'SL_FIREFOX']
};
module.exports = {
customLaunchers: customLaunchers,
aliases: aliases
}
if (process.env.TRAVIS) {
process.env.SAUCE_ACCESS_KEY = process.env.SAUCE_ACCESS_KEY.split('').reverse().join('');
}

View File

@ -12,7 +12,12 @@ if [ "$MODE" = "dart_experimental" ]
then then
${SCRIPT_DIR}/build_$MODE.sh ${SCRIPT_DIR}/build_$MODE.sh
else else
if [ "$MODE" = "saucelabs" ]
then
${SCRIPT_DIR}/test_$MODE.sh
else
${SCRIPT_DIR}/build_$MODE.sh ${SCRIPT_DIR}/build_$MODE.sh
mkdir deploy; tar -czpf deploy/dist.tgz -C dist . mkdir deploy; tar -czpf deploy/dist.tgz -C dist .
${SCRIPT_DIR}/test_$MODE.sh ${SCRIPT_DIR}/test_$MODE.sh
fi
fi fi

12
scripts/ci/test_saucelabs.sh Executable file
View File

@ -0,0 +1,12 @@
#!/bin/bash
set -e
echo =============================================================================
# go to project dir
SCRIPT_DIR=$(dirname $0)
cd $SCRIPT_DIR/../..
./scripts/sauce/sauce_connect_setup.sh
./scripts/sauce/sauce_connect_block.sh
./node_modules/.bin/gulp build.js.dev
./node_modules/.bin/gulp test.unit.js.sauce/ci

View File

@ -0,0 +1,10 @@
#!/bin/bash
# Wait for Connect to be ready before exiting
printf "Connecting to Sauce."
while [ ! -f $BROWSER_PROVIDER_READY_FILE ]; do
printf "."
#dart2js takes longer than the travis 10 min timeout to complete
sleep .5
done
echo "Connected"

View File

@ -0,0 +1,49 @@
#!/bin/bash
set -e -o pipefail
# Setup and start Sauce Connect for your TravisCI build
# This script requires your .travis.yml to include the following two private env variables:
# SAUCE_USERNAME
# SAUCE_ACCESS_KEY
# Follow the steps at https://saucelabs.com/opensource/travis to set that up.
#
# Curl and run this script as part of your .travis.yml before_script section:
# before_script:
# - curl https://gist.github.com/santiycr/5139565/raw/sauce_connect_setup.sh | bash
CONNECT_URL="https://saucelabs.com/downloads/sc-4.3.8-linux.tar.gz"
CONNECT_DIR="/tmp/sauce-connect-$RANDOM"
CONNECT_DOWNLOAD="sc-latest-linux.tar.gz"
CONNECT_LOG="$LOGS_DIR/sauce-connect"
CONNECT_STDOUT="$LOGS_DIR/sauce-connect.stdout"
CONNECT_STDERR="$LOGS_DIR/sauce-connect.stderr"
# Get Connect and start it
mkdir -p $CONNECT_DIR
cd $CONNECT_DIR
curl $CONNECT_URL -o $CONNECT_DOWNLOAD 2> /dev/null 1> /dev/null
mkdir sauce-connect
tar --extract --file=$CONNECT_DOWNLOAD --strip-components=1 --directory=sauce-connect > /dev/null
rm $CONNECT_DOWNLOAD
SAUCE_ACCESS_KEY=`echo $SAUCE_ACCESS_KEY | rev`
ARGS=""
# Set tunnel-id only on Travis, to make local testing easier.
if [ ! -z "$TRAVIS_JOB_NUMBER" ]; then
ARGS="$ARGS --tunnel-identifier $TRAVIS_JOB_NUMBER"
fi
if [ ! -z "$BROWSER_PROVIDER_READY_FILE" ]; then
ARGS="$ARGS --readyfile $BROWSER_PROVIDER_READY_FILE"
fi
echo "Starting Sauce Connect in the background, logging into:"
echo " $CONNECT_LOG"
echo " $CONNECT_STDOUT"
echo " $CONNECT_STDERR"
sauce-connect/bin/sc -u $SAUCE_USERNAME -k $SAUCE_ACCESS_KEY $ARGS \
--logfile $CONNECT_LOG 2> $CONNECT_STDERR 1> $CONNECT_STDOUT &