@testim/testim-cli
Version:
Command line interface for running Testing on you CI
384 lines (317 loc) • 30.6 kB
JavaScript
;
var _createClass = require('babel-runtime/helpers/create-class')['default'];
var _classCallCheck = require('babel-runtime/helpers/class-call-check')['default'];
var _regeneratorRuntime = require('babel-runtime/regenerator')['default'];
var _getIterator = require('babel-runtime/core-js/get-iterator')['default'];
var _Object$keys = require('babel-runtime/core-js/object/keys')['default'];
var _interopRequireDefault = require('babel-runtime/helpers/interop-require-default')['default'];
var _deepmerge = require('deepmerge');
var _deepmerge2 = _interopRequireDefault(_deepmerge);
var _utilsConfigParser = require('./utils/ConfigParser');
var _utilsConfigParser2 = _interopRequireDefault(_utilsConfigParser);
var _ = require('../');
var Runner = (function () {
function Runner() {
_classCallCheck(this, Runner);
this.haltSIGINT = false;
this.sigintWasCalled = false;
this.hasSessionID = false;
this.failures = 0;
}
_createClass(Runner, [{
key: 'run',
value: function run(m) {
var config, capabilities;
return _regeneratorRuntime.async(function run$(context$2$0) {
var _this = this;
while (1) switch (context$2$0.prev = context$2$0.next) {
case 0:
this.cid = m.cid;
this.configParser = new _utilsConfigParser2['default']();
this.configParser.addConfigFile(m.configFile);
this.configParser.merge(m.argv);
config = this.configParser.getConfig();
capabilities = this.configParser.getCapabilities(m.cid);
this.framework = this.initialiseFramework(config);
global.browser = this.initialiseInstance(m.isMultiremote, capabilities);
this.initialisePlugins(config);
/**
* store end method before it gets fiberised by wdio-sync
*/
this.endSession = global.browser.end.bind(global.browser);
/**
* initialisation successful, send start message
*/
process.send({
event: 'runner:start',
cid: m.cid,
capabilities: capabilities,
config: config
});
/**
* register runner events
*/
global.browser.on('init', function (payload) {
process.send({
event: 'runner:init',
cid: m.cid,
sessionID: payload.sessionID,
options: payload.options,
desiredCapabilities: payload.desiredCapabilities
});
_this.hasSessionID = true;
});
global.browser.on('command', function (payload) {
process.send({
event: 'runner:command',
cid: m.cid,
method: payload.method,
uri: payload.uri,
data: payload.data
});
});
global.browser.on('result', function (payload) {
process.send({
event: 'runner:result',
cid: m.cid,
requestData: payload.requestData,
requestOptions: payload.requestOptions,
body: payload.body // ToDo figure out if this slows down the execution time
});
});
global.browser.on('error', function (payload) {
process.send({
event: 'runner:error',
cid: m.cid,
err: payload.err,
requestData: payload.requestData,
requestOptions: payload.requestOptions,
body: payload.body
});
});
this.haltSIGINT = true;
context$2$0.prev = 16;
context$2$0.next = 19;
return _regeneratorRuntime.awrap(global.browser.init());
case 19:
this.haltSIGINT = false;
/**
* kill session of SIGINT signal showed up while trying to
* get a session ID
*/
if (!this.sigintWasCalled) {
context$2$0.next = 23;
break;
}
context$2$0.next = 23;
return _regeneratorRuntime.awrap(this.end(1));
case 23:
context$2$0.next = 25;
return _regeneratorRuntime.awrap(this.framework.run(m.cid, config, m.specs, capabilities));
case 25:
this.failures = context$2$0.sent;
context$2$0.next = 28;
return _regeneratorRuntime.awrap(this.end(this.failures));
case 28:
context$2$0.next = 35;
break;
case 30:
context$2$0.prev = 30;
context$2$0.t0 = context$2$0['catch'](16);
process.send({
event: 'error',
cid: this.cid,
capabilities: capabilities,
error: {
message: context$2$0.t0.message,
stack: context$2$0.t0.stack
}
});
context$2$0.next = 35;
return _regeneratorRuntime.awrap(this.end(1));
case 35:
case 'end':
return context$2$0.stop();
}
}, null, this, [[16, 30]]);
}
/**
* end test runner instance and exit process
*/
}, {
key: 'end',
value: function end() {
var failures = arguments.length <= 0 || arguments[0] === undefined ? 0 : arguments[0];
return _regeneratorRuntime.async(function end$(context$2$0) {
while (1) switch (context$2$0.prev = context$2$0.next) {
case 0:
if (!this.hasSessionID) {
context$2$0.next = 3;
break;
}
context$2$0.next = 3;
return _regeneratorRuntime.awrap(this.endSession());
case 3:
process.send({
event: 'runner:end',
failures: failures,
cid: this.cid
});
process.exit(failures === 0 ? 0 : 1);
case 5:
case 'end':
return context$2$0.stop();
}
}, null, this);
}
}, {
key: 'sigintHandler',
value: function sigintHandler() {
if (this.sigintWasCalled) {
return;
}
this.sigintWasCalled = true;
if (this.haltSIGINT) {
return;
}
global.browser.removeAllListeners();
this.end(1);
}
}, {
key: 'initialiseFramework',
value: function initialiseFramework(config) {
if (typeof config.framework !== 'string') {
throw new Error('You haven\'t defined a valid framework. ' + 'Please checkout http://webdriver.io/guide/testrunner/frameworks.html');
}
var frameworkLibrary = config.framework.toLowerCase();
try {
return require('wdio-' + frameworkLibrary + '-framework');
} catch (e) {
throw new Error('Couldn\'t load "' + frameworkLibrary + '" framework. You need to install ' + ('it with `$ npm install wdio-' + frameworkLibrary + '-framework`!'));
}
}
}, {
key: 'initialiseInstance',
value: function initialiseInstance(isMultiremote, capabilities) {
var config = this.configParser.getConfig();
if (!isMultiremote) {
config.desiredCapabilities = capabilities;
return (0, _.remote)(config);
}
var options = capabilities;
var _iteratorNormalCompletion = true;
var _didIteratorError = false;
var _iteratorError = undefined;
try {
for (var _iterator = _getIterator(_Object$keys(options)), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
var browserName = _step.value;
options[browserName] = (0, _deepmerge2['default'])(config, options[browserName]);
}
} catch (err) {
_didIteratorError = true;
_iteratorError = err;
} finally {
try {
if (!_iteratorNormalCompletion && _iterator['return']) {
_iterator['return']();
}
} finally {
if (_didIteratorError) {
throw _iteratorError;
}
}
}
var browser = (0, _.multiremote)(options);
var _iteratorNormalCompletion2 = true;
var _didIteratorError2 = false;
var _iteratorError2 = undefined;
try {
for (var _iterator2 = _getIterator(_Object$keys(options)), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
var browserName = _step2.value;
global[browserName] = browser.select(browserName);
}
} catch (err) {
_didIteratorError2 = true;
_iteratorError2 = err;
} finally {
try {
if (!_iteratorNormalCompletion2 && _iterator2['return']) {
_iterator2['return']();
}
} finally {
if (_didIteratorError2) {
throw _iteratorError2;
}
}
}
return browser;
}
/**
* initialise WebdriverIO compliant plugins
*/
}, {
key: 'initialisePlugins',
value: function initialisePlugins(config) {
if (typeof config.plugins !== 'object') {
return;
}
var _iteratorNormalCompletion3 = true;
var _didIteratorError3 = false;
var _iteratorError3 = undefined;
try {
for (var _iterator3 = _getIterator(_Object$keys(config.plugins)), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) {
var pluginName = _step3.value;
var plugin = undefined;
try {
plugin = require(pluginName);
} catch (e) {
throw new Error('Couldn\'t find plugin "' + pluginName + '". You need to install it ' + ('with `$ npm install ' + pluginName + '`!'));
}
if (typeof plugin.init !== 'function') {
throw new Error('The plugin "' + pluginName + '" is not WebdriverIO compliant!');
}
plugin.init(global.browser, config.plugins[pluginName]);
}
} catch (err) {
_didIteratorError3 = true;
_iteratorError3 = err;
} finally {
try {
if (!_iteratorNormalCompletion3 && _iterator3['return']) {
_iterator3['return']();
}
} finally {
if (_didIteratorError3) {
throw _iteratorError3;
}
}
}
}
}]);
return Runner;
})();
var runner = new Runner();
process.on('message', function (m) {
runner[m.command](m)['catch'](function (e) {
/**
* custom exit code to propagate initialisation error
*/
process.send({
event: 'runner:error',
error: {
message: e.message,
stack: e.stack
},
capabilities: runner.configParser.getCapabilities(runner.cid),
cid: runner.cid
});
process.exit(1);
});
});
/**
* catches ctrl+c event
*/
process.on('SIGINT', function () {
runner.sigintHandler();
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../lib/runner.js"],"names":[],"mappings":"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","file":"runner.js","sourcesContent":["import merge from 'deepmerge'\n\nimport ConfigParser from './utils/ConfigParser'\nimport { remote, multiremote } from '../'\n\nclass Runner {\n    constructor () {\n        this.haltSIGINT = false\n        this.sigintWasCalled = false\n        this.hasSessionID = false\n        this.failures = 0\n    }\n\n    async run (m) {\n        this.cid = m.cid\n\n        this.configParser = new ConfigParser()\n        this.configParser.addConfigFile(m.configFile)\n        this.configParser.merge(m.argv)\n\n        let config = this.configParser.getConfig()\n        let capabilities = this.configParser.getCapabilities(m.cid)\n\n        this.framework = this.initialiseFramework(config)\n        global.browser = this.initialiseInstance(m.isMultiremote, capabilities)\n        this.initialisePlugins(config)\n\n        /**\n         * store end method before it gets fiberised by wdio-sync\n         */\n        this.endSession = global.browser.end.bind(global.browser)\n\n        /**\n         * initialisation successful, send start message\n         */\n        process.send({\n            event: 'runner:start',\n            cid: m.cid,\n            capabilities: capabilities,\n            config: config\n        })\n\n        /**\n         * register runner events\n         */\n        global.browser.on('init', (payload) => {\n            process.send({\n                event: 'runner:init',\n                cid: m.cid,\n                sessionID: payload.sessionID,\n                options: payload.options,\n                desiredCapabilities: payload.desiredCapabilities\n            })\n\n            this.hasSessionID = true\n        })\n\n        global.browser.on('command', (payload) => {\n            process.send({\n                event: 'runner:command',\n                cid: m.cid,\n                method: payload.method,\n                uri: payload.uri,\n                data: payload.data\n            })\n        })\n\n        global.browser.on('result', (payload) => {\n            process.send({\n                event: 'runner:result',\n                cid: m.cid,\n                requestData: payload.requestData,\n                requestOptions: payload.requestOptions,\n                body: payload.body // ToDo figure out if this slows down the execution time\n            })\n        })\n\n        global.browser.on('error', (payload) => {\n            process.send({\n                event: 'runner:error',\n                cid: m.cid,\n                err: payload.err,\n                requestData: payload.requestData,\n                requestOptions: payload.requestOptions,\n                body: payload.body\n            })\n        })\n\n        this.haltSIGINT = true\n\n        try {\n            await global.browser.init()\n            this.haltSIGINT = false\n\n            /**\n             * kill session of SIGINT signal showed up while trying to\n             * get a session ID\n             */\n            if (this.sigintWasCalled) {\n                await this.end(1)\n            }\n\n            this.failures = await this.framework.run(m.cid, config, m.specs, capabilities)\n            await this.end(this.failures)\n        } catch (e) {\n            process.send({\n                event: 'error',\n                cid: this.cid,\n                capabilities,\n                error: {\n                    message: e.message,\n                    stack: e.stack\n                }\n            })\n\n            await this.end(1)\n        }\n    }\n\n    /**\n     * end test runner instance and exit process\n     */\n    async end (failures = 0) {\n        if (this.hasSessionID) {\n            await this.endSession()\n        }\n\n        process.send({\n            event: 'runner:end',\n            failures: failures,\n            cid: this.cid\n        })\n        process.exit(failures === 0 ? 0 : 1)\n    }\n\n    sigintHandler () {\n        if (this.sigintWasCalled) {\n            return\n        }\n\n        this.sigintWasCalled = true\n\n        if (this.haltSIGINT) {\n            return\n        }\n\n        global.browser.removeAllListeners()\n        this.end(1)\n    }\n\n    initialiseFramework (config) {\n        if (typeof config.framework !== 'string') {\n            throw new Error(\n                `You haven't defined a valid framework. ` +\n                `Please checkout http://webdriver.io/guide/testrunner/frameworks.html`\n            )\n        }\n\n        let frameworkLibrary = config.framework.toLowerCase()\n        try {\n            return require(`wdio-${frameworkLibrary}-framework`)\n        } catch (e) {\n            throw new Error(\n                `Couldn't load \"${frameworkLibrary}\" framework. You need to install ` +\n                `it with \\`$ npm install wdio-${frameworkLibrary}-framework\\`!`\n            )\n        }\n    }\n\n    initialiseInstance (isMultiremote, capabilities) {\n        let config = this.configParser.getConfig()\n\n        if (!isMultiremote) {\n            config.desiredCapabilities = capabilities\n            return remote(config)\n        }\n\n        let options = capabilities\n        for (let browserName of Object.keys(options)) {\n            options[browserName] = merge(config, options[browserName])\n        }\n\n        let browser = multiremote(options)\n        for (let browserName of Object.keys(options)) {\n            global[browserName] = browser.select(browserName)\n        }\n        return browser\n    }\n\n    /**\n     * initialise WebdriverIO compliant plugins\n     */\n    initialisePlugins (config) {\n        if (typeof config.plugins !== 'object') {\n            return\n        }\n\n        for (let pluginName of Object.keys(config.plugins)) {\n            let plugin\n\n            try {\n                plugin = require(pluginName)\n            } catch (e) {\n                throw new Error(\n                    `Couldn't find plugin \"${pluginName}\". You need to install it ` +\n                    `with \\`$ npm install ${pluginName}\\`!`\n                )\n            }\n\n            if (typeof plugin.init !== 'function') {\n                throw new Error(`The plugin \"${pluginName}\" is not WebdriverIO compliant!`)\n            }\n\n            plugin.init(global.browser, config.plugins[pluginName])\n        }\n    }\n}\n\nlet runner = new Runner()\n\nprocess.on('message', (m) => {\n    runner[m.command](m).catch((e) => {\n        /**\n         * custom exit code to propagate initialisation error\n         */\n        process.send({\n            event: 'runner:error',\n            error: {\n                message: e.message,\n                stack: e.stack\n            },\n            capabilities: runner.configParser.getCapabilities(runner.cid),\n            cid: runner.cid\n        })\n        process.exit(1)\n    })\n})\n\n/**\n * catches ctrl+c event\n */\nprocess.on('SIGINT', () => {\n    runner.sigintHandler()\n})\n"]}