UNPKG

@testim/testim-cli

Version:

Command line interface for running Testing on you CI

384 lines (317 loc) 30.6 kB
'use strict'; 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"]}