@testim/testim-cli
Version:
Command line interface for running Testing on you CI
375 lines (295 loc) • 27.7 kB
JavaScript
;
var _createClass = require('babel-runtime/helpers/create-class')['default'];
var _classCallCheck = require('babel-runtime/helpers/class-call-check')['default'];
var _getIterator = require('babel-runtime/core-js/get-iterator')['default'];
var _regeneratorRuntime = require('babel-runtime/regenerator')['default'];
var _Promise = require('babel-runtime/core-js/promise')['default'];
var _interopRequireDefault = require('babel-runtime/helpers/interop-require-default')['default'];
Object.defineProperty(exports, '__esModule', {
value: true
});
var _child_process = require('child_process');
var _child_process2 = _interopRequireDefault(_child_process);
var _wdioDotReporter = require('wdio-dot-reporter');
var _wdioDotReporter2 = _interopRequireDefault(_wdioDotReporter);
var _utilsConfigParser = require('./utils/ConfigParser');
var _utilsConfigParser2 = _interopRequireDefault(_utilsConfigParser);
var _utilsBaseReporter = require('./utils/BaseReporter');
var _utilsBaseReporter2 = _interopRequireDefault(_utilsBaseReporter);
var Launcher = (function () {
function Launcher(configFile, argv) {
_classCallCheck(this, Launcher);
this.configParser = new _utilsConfigParser2['default']();
this.configParser.addConfigFile(configFile);
this.configParser.merge(argv);
this.reporter = this.initReporter();
this.argv = argv;
this.configFile = configFile;
this.exitCode = 0;
this.hasTriggeredExitRoutine = false;
this.hasStartedAnyProcess = false;
this.processes = [];
this.schedule = [];
}
/**
* check if multiremote or wdio test
*/
_createClass(Launcher, [{
key: 'isMultiremote',
value: function isMultiremote() {
var caps = this.configParser.getCapabilities();
return !Array.isArray(caps);
}
/**
* initialise reporter
*/
}, {
key: 'initReporter',
value: function initReporter() {
var reporter = new _utilsBaseReporter2['default']();
var config = this.configParser.getConfig();
/**
* if no reporter is set or config property is in a wrong format
* just use the dot reporter
*/
if (!config.reporter || !Array.isArray(config.reporter)) {
reporter.add(new _wdioDotReporter2['default'](reporter));
return reporter;
}
var _iteratorNormalCompletion = true;
var _didIteratorError = false;
var _iteratorError = undefined;
try {
for (var _iterator = _getIterator(config.reporter), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
var _reporter = _step.value;
try {
var Reporter = require('wdio-' + _reporter + '-reporter');
_reporter.add(new Reporter(_reporter));
} catch (e) {
throw new Error('reporter "wdio-' + _reporter + '-reporter" is not installed');
}
}
} catch (err) {
_didIteratorError = true;
_iteratorError = err;
} finally {
try {
if (!_iteratorNormalCompletion && _iterator['return']) {
_iterator['return']();
}
} finally {
if (_didIteratorError) {
throw _iteratorError;
}
}
}
return reporter;
}
/**
* run sequence
* @return {Promise} that only gets resolves with either an exitCode or an error
*/
}, {
key: 'run',
value: function run() {
var config, caps, _iteratorNormalCompletion2, _didIteratorError2, _iteratorError2, _iterator2, _step2, capabilities, exitCode;
return _regeneratorRuntime.async(function run$(context$2$0) {
var _this = this;
while (1) switch (context$2$0.prev = context$2$0.next) {
case 0:
config = this.configParser.getConfig();
caps = this.configParser.getCapabilities();
this.reporter.emit('start');
context$2$0.prev = 3;
context$2$0.next = 6;
return _regeneratorRuntime.awrap(config.onPrepare(config, caps));
case 6:
context$2$0.next = 11;
break;
case 8:
context$2$0.prev = 8;
context$2$0.t0 = context$2$0['catch'](3);
return context$2$0.abrupt('return', context$2$0.t0);
case 11:
if (!this.isMultiremote()) {
context$2$0.next = 13;
break;
}
return context$2$0.abrupt('return', this.startInstance(this.configParser.getSpecs(), caps));
case 13:
_iteratorNormalCompletion2 = true;
_didIteratorError2 = false;
_iteratorError2 = undefined;
context$2$0.prev = 16;
/**
* schedule test runs
*/
for (_iterator2 = _getIterator(caps); !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
capabilities = _step2.value;
this.schedule.push({
specs: this.configParser.getSpecs(capabilities.specs, capabilities.exclude),
availableInstances: capabilities.maxInstances || config.maxInstances || 1,
runningInstances: 0
});
}
/**
* catches ctrl+c event
*/
context$2$0.next = 24;
break;
case 20:
context$2$0.prev = 20;
context$2$0.t1 = context$2$0['catch'](16);
_didIteratorError2 = true;
_iteratorError2 = context$2$0.t1;
case 24:
context$2$0.prev = 24;
context$2$0.prev = 25;
if (!_iteratorNormalCompletion2 && _iterator2['return']) {
_iterator2['return']();
}
case 27:
context$2$0.prev = 27;
if (!_didIteratorError2) {
context$2$0.next = 30;
break;
}
throw _iteratorError2;
case 30:
return context$2$0.finish(27);
case 31:
return context$2$0.finish(24);
case 32:
process.on('SIGINT', this.exitHandler.bind(this));
/**
* make sure the program will not close instantly
*/
process.stdin.resume();
context$2$0.next = 36;
return _regeneratorRuntime.awrap(new _Promise(function (resolve, reject) {
_this.resolve = resolve;
_this.runSpecs();
}));
case 36:
exitCode = context$2$0.sent;
context$2$0.prev = 37;
context$2$0.next = 40;
return _regeneratorRuntime.awrap(config.onComplete(exitCode));
case 40:
context$2$0.next = 45;
break;
case 42:
context$2$0.prev = 42;
context$2$0.t2 = context$2$0['catch'](37);
return context$2$0.abrupt('return', context$2$0.t2);
case 45:
return context$2$0.abrupt('return', exitCode);
case 46:
case 'end':
return context$2$0.stop();
}
}, null, this, [[3, 8], [16, 20, 24, 32], [25,, 27, 31], [37, 42]]);
}
/**
* run multiple single remote tests
*/
}, {
key: 'runSpecs',
value: function runSpecs() {
var _this2 = this;
var specsLeft = 0;
var isRunning = false;
this.schedule.forEach(function (capability, cid) {
var specFiles = capability.specs.length;
specsLeft += specFiles;
for (var i = 0; i < capability.availableInstances && i < specFiles; i++) {
_this2.startInstance([capability.specs.pop()], cid);
capability.availableInstances--;
capability.runningInstances++;
}
isRunning = isRunning || capability.runningInstances > 0;
});
return specsLeft === 0 && !isRunning;
}
/**
* Start instance in a child process.
* @param {Object|Object[]} capabilities desired capabilities of instance
*/
}, {
key: 'startInstance',
value: function startInstance(specs, i) {
var childProcess = _child_process2['default'].fork(__dirname + '/runner.js', [], {
cwd: process.cwd()
});
this.processes.push(childProcess);
childProcess.on('message', this.messageHandler.bind(this)).on('exit', this.endHandler.bind(this));
childProcess.send({
cid: i,
command: 'run',
configFile: this.configFile,
argv: this.argv,
specs: specs,
isMultiremote: this.isMultiremote()
});
}
/**
* emit event from child process to reporter
* @param {Object} m event object
*/
}, {
key: 'messageHandler',
value: function messageHandler(m) {
this.hasStartedAnyProcess = true;
/**
* update schedule
*/
if (m.event === 'runner:end' || m.event === 'runner:error') {
this.schedule[m.cid].availableInstances++;
this.schedule[m.cid].runningInstances--;
}
if (m.event === 'runner:error') {
this.reporter.emit('error', m);
}
this.reporter.emit(m.event, m);
}
/**
* Closes test runner process once all instances finished and excited process.
* @param {Number} childProcessExitCode exit code of child process
*/
}, {
key: 'endHandler',
value: function endHandler(childProcessExitCode) {
this.exitCode = this.exitCode || childProcessExitCode;
if (!this.isMultiremote() && !this.runSpecs()) {
return;
}
this.reporter.emit('end', {
sigint: this.hasTriggeredExitRoutine,
exitCode: this.exitCode
});
this.resolve(this.exitCode);
}
/**
* Make sure all started selenium sessions get closed properly and prevent
* having dead driver processes. To do so let the runner end its Selenium
* session first before killing
*/
}, {
key: 'exitHandler',
value: function exitHandler() {
if (this.hasTriggeredExitRoutine || !this.hasStartedAnyProcess) {
console.log('\nKilling process, bye!');
return this.resolve(1);
}
console.log('\n\nEnd selenium sessions properly ...\n(press crtl+c again to hard kill the runner)\n');
this.hasTriggeredExitRoutine = true;
}
}]);
return Launcher;
})();
exports['default'] = Launcher;
module.exports = exports['default'];
/**
* if it is an object run multiremote test
*/
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../lib/launcher.js"],"names":[],"mappings":"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","file":"launcher.js","sourcesContent":["import child from 'child_process'\nimport DotReporter from 'wdio-dot-reporter'\n\nimport ConfigParser from './utils/ConfigParser'\nimport BaseReporter from './utils/BaseReporter'\n\n\nclass Launcher {\n    constructor (configFile, argv) {\n        this.configParser = new ConfigParser()\n        this.configParser.addConfigFile(configFile)\n        this.configParser.merge(argv)\n\n        this.reporter = this.initReporter()\n\n        this.argv = argv\n        this.configFile = configFile\n\n        this.exitCode = 0\n        this.hasTriggeredExitRoutine = false\n        this.hasStartedAnyProcess = false\n        this.processes = []\n        this.schedule = []\n    }\n\n    /**\n     * check if multiremote or wdio test\n     */\n    isMultiremote () {\n        let caps = this.configParser.getCapabilities()\n        return !Array.isArray(caps)\n    }\n\n    /**\n     * initialise reporter\n     */\n    initReporter () {\n        let reporter = new BaseReporter()\n        let config = this.configParser.getConfig()\n\n        /**\n         * if no reporter is set or config property is in a wrong format\n         * just use the dot reporter\n         */\n        if (!config.reporter || !Array.isArray(config.reporter)) {\n            reporter.add(new DotReporter(reporter))\n            return reporter\n        }\n\n        for (let reporter of config.reporter) {\n            try {\n                let Reporter = require(`wdio-${reporter}-reporter`)\n                reporter.add(new Reporter(reporter))\n            } catch (e) {\n                throw new Error(`reporter \"wdio-${reporter}-reporter\" is not installed`)\n            }\n        }\n\n        return reporter\n    }\n\n    /**\n     * run sequence\n     * @return  {Promise} that only gets resolves with either an exitCode or an error\n     */\n    async run () {\n        let config = this.configParser.getConfig()\n        let caps = this.configParser.getCapabilities()\n\n        this.reporter.emit('start')\n        try {\n            await config.onPrepare(config, caps)\n        } catch (e) {\n            return e\n        }\n\n        /**\n         * if it is an object run multiremote test\n         */\n        if (this.isMultiremote()) {\n            return this.startInstance(this.configParser.getSpecs(), caps)\n        }\n\n        /**\n         * schedule test runs\n         */\n        for (let capabilities of caps) {\n            this.schedule.push({\n                specs: this.configParser.getSpecs(capabilities.specs, capabilities.exclude),\n                availableInstances: capabilities.maxInstances || config.maxInstances || 1,\n                runningInstances: 0\n            })\n        }\n\n        /**\n         * catches ctrl+c event\n         */\n        process.on('SIGINT', this.exitHandler.bind(this))\n\n        /**\n         * make sure the program will not close instantly\n         */\n        process.stdin.resume()\n\n        let exitCode = await new Promise((resolve, reject) => {\n            this.resolve = resolve\n            this.runSpecs()\n        })\n\n        try {\n            await config.onComplete(exitCode)\n        } catch (e) {\n            return e\n        }\n        return exitCode\n    }\n\n    /**\n     * run multiple single remote tests\n     */\n    runSpecs () {\n        let specsLeft = 0\n        let isRunning = false\n\n        this.schedule.forEach((capability, cid) => {\n            let specFiles = capability.specs.length\n            specsLeft += specFiles\n\n            for (let i = 0; i < capability.availableInstances && i < specFiles; i++) {\n                this.startInstance([capability.specs.pop()], cid)\n                capability.availableInstances--\n                capability.runningInstances++\n            }\n            isRunning = isRunning || capability.runningInstances > 0\n        })\n\n        return specsLeft === 0 && !isRunning\n    }\n\n    /**\n     * Start instance in a child process.\n     * @param  {Object|Object[]} capabilities  desired capabilities of instance\n     */\n    startInstance (specs, i) {\n        let childProcess = child.fork(__dirname + '/runner.js', [], {\n            cwd: process.cwd()\n        })\n\n        this.processes.push(childProcess)\n\n        childProcess\n            .on('message', this.messageHandler.bind(this))\n            .on('exit', this.endHandler.bind(this))\n\n        childProcess.send({\n            cid: i,\n            command: 'run',\n            configFile: this.configFile,\n            argv: this.argv,\n            specs: specs,\n            isMultiremote: this.isMultiremote()\n        })\n    }\n\n    /**\n     * emit event from child process to reporter\n     * @param  {Object} m  event object\n     */\n    messageHandler (m) {\n        this.hasStartedAnyProcess = true\n\n        /**\n         * update schedule\n         */\n        if (m.event === 'runner:end' || m.event === 'runner:error') {\n            this.schedule[m.cid].availableInstances++\n            this.schedule[m.cid].runningInstances--\n        }\n\n        if (m.event === 'runner:error') {\n            this.reporter.emit('error', m)\n        }\n\n        this.reporter.emit(m.event, m)\n    }\n\n    /**\n     * Closes test runner process once all instances finished and excited process.\n     * @param  {Number} childProcessExitCode  exit code of child process\n     */\n    endHandler (childProcessExitCode) {\n        this.exitCode = this.exitCode || childProcessExitCode\n\n        if (!this.isMultiremote() && !this.runSpecs()) {\n            return\n        }\n\n        this.reporter.emit('end', {\n            sigint: this.hasTriggeredExitRoutine,\n            exitCode: this.exitCode\n        })\n\n        this.resolve(this.exitCode)\n    }\n\n    /**\n     * Make sure all started selenium sessions get closed properly and prevent\n     * having dead driver processes. To do so let the runner end its Selenium\n     * session first before killing\n     */\n    exitHandler () {\n        if (this.hasTriggeredExitRoutine || !this.hasStartedAnyProcess) {\n            console.log('\\nKilling process, bye!')\n            return this.resolve(1)\n        }\n\n        console.log(`\n\nEnd selenium sessions properly ...\n(press crtl+c again to hard kill the runner)\n`)\n\n        this.hasTriggeredExitRoutine = true\n    }\n}\n\nexport default Launcher\n"]}