UNPKG

@testim/testim-cli

Version:

Command line interface for running Testing on you CI

375 lines (295 loc) 27.7 kB
'use strict'; 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":";;;;;;;;;;;;;;;;;;6BAAkB,eAAe;;;;+BACT,mBAAmB;;;;iCAElB,sBAAsB;;;;iCACtB,sBAAsB;;;;IAGzC,QAAQ;AACE,aADV,QAAQ,CACG,UAAU,EAAE,IAAI,EAAE;8BAD7B,QAAQ;;AAEN,YAAI,CAAC,YAAY,GAAG,oCAAkB,CAAA;AACtC,YAAI,CAAC,YAAY,CAAC,aAAa,CAAC,UAAU,CAAC,CAAA;AAC3C,YAAI,CAAC,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC,CAAA;;AAE7B,YAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,YAAY,EAAE,CAAA;;AAEnC,YAAI,CAAC,IAAI,GAAG,IAAI,CAAA;AAChB,YAAI,CAAC,UAAU,GAAG,UAAU,CAAA;;AAE5B,YAAI,CAAC,QAAQ,GAAG,CAAC,CAAA;AACjB,YAAI,CAAC,uBAAuB,GAAG,KAAK,CAAA;AACpC,YAAI,CAAC,oBAAoB,GAAG,KAAK,CAAA;AACjC,YAAI,CAAC,SAAS,GAAG,EAAE,CAAA;AACnB,YAAI,CAAC,QAAQ,GAAG,EAAE,CAAA;KACrB;;;;;;iBAhBC,QAAQ;;eAqBI,yBAAG;AACb,gBAAI,IAAI,GAAG,IAAI,CAAC,YAAY,CAAC,eAAe,EAAE,CAAA;AAC9C,mBAAO,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAA;SAC9B;;;;;;;eAKY,wBAAG;AACZ,gBAAI,QAAQ,GAAG,oCAAkB,CAAA;AACjC,gBAAI,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,SAAS,EAAE,CAAA;;;;;;AAM1C,gBAAI,CAAC,MAAM,CAAC,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE;AACrD,wBAAQ,CAAC,GAAG,CAAC,iCAAgB,QAAQ,CAAC,CAAC,CAAA;AACvC,uBAAO,QAAQ,CAAA;aAClB;;;;;;;AAED,kDAAqB,MAAM,CAAC,QAAQ,4GAAE;wBAA7B,SAAQ;;AACb,wBAAI;AACA,4BAAI,QAAQ,GAAG,OAAO,WAAS,SAAQ,eAAY,CAAA;AACnD,iCAAQ,CAAC,GAAG,CAAC,IAAI,QAAQ,CAAC,SAAQ,CAAC,CAAC,CAAA;qBACvC,CAAC,OAAO,CAAC,EAAE;AACR,8BAAM,IAAI,KAAK,qBAAmB,SAAQ,iCAA8B,CAAA;qBAC3E;iBACJ;;;;;;;;;;;;;;;;AAED,mBAAO,QAAQ,CAAA;SAClB;;;;;;;;eAMS;gBACF,MAAM,EACN,IAAI,uFAmBC,YAAY,EAkBjB,QAAQ;;;;;;;AAtCR,8BAAM,GAAG,IAAI,CAAC,YAAY,CAAC,SAAS,EAAE;AACtC,4BAAI,GAAG,IAAI,CAAC,YAAY,CAAC,eAAe,EAAE;;AAE9C,4BAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAA;;;yDAEjB,MAAM,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,CAAC;;;;;;;;;;;;6BAQpC,IAAI,CAAC,aAAa,EAAE;;;;;4DACb,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,EAAE,IAAI,CAAC;;;;;;;;;;;AAMjE,uDAAyB,IAAI,yGAAE;AAAtB,wCAAY;;AACjB,gCAAI,CAAC,QAAQ,CAAC,IAAI,CAAC;AACf,qCAAK,EAAE,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,YAAY,CAAC,KAAK,EAAE,YAAY,CAAC,OAAO,CAAC;AAC3E,kDAAkB,EAAE,YAAY,CAAC,YAAY,IAAI,MAAM,CAAC,YAAY,IAAI,CAAC;AACzE,gDAAgB,EAAE,CAAC;6BACtB,CAAC,CAAA;yBACL;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAKD,+BAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAA;;;;;AAKjD,+BAAO,CAAC,KAAK,CAAC,MAAM,EAAE,CAAA;;;yDAED,aAAY,UAAC,OAAO,EAAE,MAAM,EAAK;AAClD,kCAAK,OAAO,GAAG,OAAO,CAAA;AACtB,kCAAK,QAAQ,EAAE,CAAA;yBAClB,CAAC;;;AAHE,gCAAQ;;;yDAMF,MAAM,CAAC,UAAU,CAAC,QAAQ,CAAC;;;;;;;;;;;;4DAI9B,QAAQ;;;;;;;SAClB;;;;;;;eAKQ,oBAAG;;;AACR,gBAAI,SAAS,GAAG,CAAC,CAAA;AACjB,gBAAI,SAAS,GAAG,KAAK,CAAA;;AAErB,gBAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,UAAC,UAAU,EAAE,GAAG,EAAK;AACvC,oBAAI,SAAS,GAAG,UAAU,CAAC,KAAK,CAAC,MAAM,CAAA;AACvC,yBAAS,IAAI,SAAS,CAAA;;AAEtB,qBAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,CAAC,kBAAkB,IAAI,CAAC,GAAG,SAAS,EAAE,CAAC,EAAE,EAAE;AACrE,2BAAK,aAAa,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,EAAE,GAAG,CAAC,CAAA;AACjD,8BAAU,CAAC,kBAAkB,EAAE,CAAA;AAC/B,8BAAU,CAAC,gBAAgB,EAAE,CAAA;iBAChC;AACD,yBAAS,GAAG,SAAS,IAAI,UAAU,CAAC,gBAAgB,GAAG,CAAC,CAAA;aAC3D,CAAC,CAAA;;AAEF,mBAAO,SAAS,KAAK,CAAC,IAAI,CAAC,SAAS,CAAA;SACvC;;;;;;;;eAMa,uBAAC,KAAK,EAAE,CAAC,EAAE;AACrB,gBAAI,YAAY,GAAG,2BAAM,IAAI,CAAC,SAAS,GAAG,YAAY,EAAE,EAAE,EAAE;AACxD,mBAAG,EAAE,OAAO,CAAC,GAAG,EAAE;aACrB,CAAC,CAAA;;AAEF,gBAAI,CAAC,SAAS,CAAC,IAAI,CAAC,YAAY,CAAC,CAAA;;AAEjC,wBAAY,CACP,EAAE,CAAC,SAAS,EAAE,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAC7C,EAAE,CAAC,MAAM,EAAE,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAA;;AAE3C,wBAAY,CAAC,IAAI,CAAC;AACd,mBAAG,EAAE,CAAC;AACN,uBAAO,EAAE,KAAK;AACd,0BAAU,EAAE,IAAI,CAAC,UAAU;AAC3B,oBAAI,EAAE,IAAI,CAAC,IAAI;AACf,qBAAK,EAAE,KAAK;AACZ,6BAAa,EAAE,IAAI,CAAC,aAAa,EAAE;aACtC,CAAC,CAAA;SACL;;;;;;;;eAMc,wBAAC,CAAC,EAAE;AACf,gBAAI,CAAC,oBAAoB,GAAG,IAAI,CAAA;;;;;AAKhC,gBAAI,CAAC,CAAC,KAAK,KAAK,YAAY,IAAI,CAAC,CAAC,KAAK,KAAK,cAAc,EAAE;AACxD,oBAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,kBAAkB,EAAE,CAAA;AACzC,oBAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,gBAAgB,EAAE,CAAA;aAC1C;;AAED,gBAAI,CAAC,CAAC,KAAK,KAAK,cAAc,EAAE;AAC5B,oBAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,CAAA;aACjC;;AAED,gBAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,CAAA;SACjC;;;;;;;;eAMU,oBAAC,oBAAoB,EAAE;AAC9B,gBAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,IAAI,oBAAoB,CAAA;;AAErD,gBAAI,CAAC,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE;AAC3C,uBAAM;aACT;;AAED,gBAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,EAAE;AACtB,sBAAM,EAAE,IAAI,CAAC,uBAAuB;AACpC,wBAAQ,EAAE,IAAI,CAAC,QAAQ;aAC1B,CAAC,CAAA;;AAEF,gBAAI,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;SAC9B;;;;;;;;;eAOW,uBAAG;AACX,gBAAI,IAAI,CAAC,uBAAuB,IAAI,CAAC,IAAI,CAAC,oBAAoB,EAAE;AAC5D,uBAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAA;AACtC,uBAAO,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAA;aACzB;;AAED,mBAAO,CAAC,GAAG,0FAIjB,CAAA;;AAEM,gBAAI,CAAC,uBAAuB,GAAG,IAAI,CAAA;SACtC;;;WAxNC,QAAQ;;;qBA2NC,QAAQ","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"]}