UNPKG

@testim/testim-cli

Version:

Command line interface for running Testing on you CI

499 lines (417 loc) 39.5 kB
'use strict'; var _get = require('babel-runtime/helpers/get')['default']; var _inherits = require('babel-runtime/helpers/inherits')['default']; 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 _Object$keys = require('babel-runtime/core-js/object/keys')['default']; var _interopRequireDefault = require('babel-runtime/helpers/interop-require-default')['default']; Object.defineProperty(exports, '__esModule', { value: true }); var _tty = require('tty'); var _tty2 = _interopRequireDefault(_tty); var _events = require('events'); var _events2 = _interopRequireDefault(_events); var _supportsColor = require('supports-color'); var _supportsColor2 = _interopRequireDefault(_supportsColor); var ISATTY = _tty2['default'].isatty(1) && _tty2['default'].isatty(2); var COLORS = { 'pass': 90, 'fail': 31, 'bright pass': 92, 'bright fail': 91, 'bright yellow': 93, 'pending': 36, 'suite': 0, 'error title': 0, 'error message': 31, 'error stack': 90, 'checkmark': 32, 'fast': 90, 'medium': 33, 'slow': 31, 'green': 32, 'light': 90, 'diff gutter': 90, 'diff added': 32, 'diff removed': 31 }; var SYMBOLS_WIN = { ok: '√', err: '×', dot: '.', error: 'F' }; var SYMBOLS = { ok: '✓', err: '✖', dot: '․', error: 'F' }; var BaseReporter = (function (_events$EventEmitter) { _inherits(BaseReporter, _events$EventEmitter); function BaseReporter() { var _this = this; _classCallCheck(this, BaseReporter); _get(Object.getPrototypeOf(BaseReporter.prototype), 'constructor', this).call(this); this.stats = { suites: 0, tests: 0, passes: 0, pending: 0, failures: 0, runner: {} }; this.failures = []; this.reporters = []; this.printEpilogue = true; this.cursor = new Cursor(); this.on('start', function () { _this.stats.start = new Date(); _this.propagateEvent('start'); }); this.on('runner:start', function (runner) { if (!_this.stats.runner[runner.cid]) { _this.stats.runner[runner.cid] = { start: new Date(), capabilities: runner.capabilities, config: runner.config, tests: [] }; } _this.propagateEvent('runner:start', runner); }); this.on('runner:init', function (runner) { _this.stats.runner[runner.cid].sessionID = runner.sessionID; _this.propagateEvent('runner:init', runner); }); this.on('suite:start', function (suite) { suite.root || _this.stats.suites++; _this.propagateEvent('suite:start', suite); }); this.on('suite:end', function (suite) { _this.propagateEvent('suite:end', suite); }); this.on('test:end', function (test) { _this.stats.tests++; _this.propagateEvent('test:end', test); }); this.on('test:pass', function (test) { _this.stats.runner[test.cid].tests.push(null); _this.stats.passes++; _this.propagateEvent('test:pass', test); }); this.on('test:fail', function (test) { _this.stats.failures++; _this.stats.runner[test.cid].tests.push(test.err); /** * check if error also happened in other runners */ var duplicateError = false; var _iteratorNormalCompletion = true; var _didIteratorError = false; var _iteratorError = undefined; try { for (var _iterator = _getIterator(_this.failures), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { var failure = _step.value; if (test.err.message !== failure.err.message || failure.title !== test.title) { continue; } duplicateError = true; failure.runner[test.cid] = test.runner[test.cid]; } } catch (err) { _didIteratorError = true; _iteratorError = err; } finally { try { if (!_iteratorNormalCompletion && _iterator['return']) { _iterator['return'](); } } finally { if (_didIteratorError) { throw _iteratorError; } } } if (!duplicateError) { _this.failures.push(test); } _this.propagateEvent('test:fail', test); }); this.on('test:pending', function (test) { _this.stats.pending++; _this.propagateEvent('test:pending', test); }); this.on('runner:end', function (runner) { _this.stats.runner[runner.cid].end = new Date(); _this.propagateEvent('runner:end', runner); }); this.on('end', function (args) { _this.stats.end = new Date(); _this.stats.duration = new Date() - _this.stats.start; _this.printEpilogue = _this.printEpilogue && !args.sigint; _this.propagateEvent('end', args); }); this.on('error', function (m) { _this.printEpilogue = false; var fmt = _this.color('error message', 'ERROR: %s'); console.log(fmt, m.error.message); var sanitizedCaps = []; var _iteratorNormalCompletion2 = true; var _didIteratorError2 = false; var _iteratorError2 = undefined; try { for (var _iterator2 = _getIterator(_Object$keys(m.capabilities)), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) { var capability = _step2.value; /** * we don't need all capability types to recognise a vm */ if (['browserName', 'platform', 'version', 'platformVersion', 'deviceName', 'app'].indexOf(capability) === -1) { continue; } sanitizedCaps.push(capability + ': ' + JSON.stringify(m.capabilities[capability])); } } catch (err) { _didIteratorError2 = true; _iteratorError2 = err; } finally { try { if (!_iteratorNormalCompletion2 && _iterator2['return']) { _iterator2['return'](); } } finally { if (_didIteratorError2) { throw _iteratorError2; } } } fmt = _this.color('bright yellow', sanitizedCaps.join(', ')); console.log(fmt); if (m.error.stack) { fmt = _this.color('error stack', m.error.stack.replace('Error: ' + m.error.message + '\n', '')); } else { fmt = _this.color('error stack', ' no stack available'); } console.log(fmt); _this.propagateEvent('error', m); }); } /** * Expose some basic cursor interactions * that are common among reporters. */ /** * Color `str` with the given `type`, * allowing colors to be disabled, * as well as user-defined color * schemes. * * @param {String} type * @param {String} str * @return {String} * @api private */ _createClass(BaseReporter, [{ key: 'color', value: function color(type, str) { if (!_supportsColor2['default']) return String(str); return '\u001b[' + COLORS[type] + 'm' + str + '\u001b[0m'; } /** * Output common epilogue used by many of * the bundled reporters. * * @api public */ }, { key: 'epilogue', value: function epilogue() { var fmt = null; if (!this.printEpilogue) { return; } console.log('\n'); // passes fmt = this.color('green', '%d passing') + this.color('light', ' (%ss)'); console.log(fmt, this.stats.passes || 0, (Math.round(this.stats.duration / 100) / 10).toFixed(2)); // pending if (this.stats.pending) { fmt = this.color('pending', '%d pending'); console.log(fmt, this.stats.pending); } // failures if (this.stats.failures) { fmt = this.color('fail', '%d failing'); console.log(fmt, this.stats.failures); this.listFailures(); } console.log(); } /** * Outut the given failures as a list */ }, { key: 'listFailures', value: function listFailures() { var _this2 = this; console.log(); this.failures.forEach(function (test, i) { var runningBrowser = ''; var _iteratorNormalCompletion3 = true; var _didIteratorError3 = false; var _iteratorError3 = undefined; try { for (var _iterator3 = _getIterator(_Object$keys(test.runner)), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) { var pid = _step3.value; var caps = test.runner[pid]; runningBrowser += '\nrunning'; if (caps.browserName) { runningBrowser += ' ' + caps.browserName; } if (caps.version) { runningBrowser += ' (v' + caps.version + ')'; } if (caps.platform) { runningBrowser += ' on ' + caps.platform; } var host = _this2.stats.runner[pid].config.host; if (host && host.indexOf('saucelabs') > -1) { runningBrowser += '\nCheck out job at https://saucelabs.com/tests/' + _this2.stats.runner[pid].sessionID; } } // format } catch (err) { _didIteratorError3 = true; _iteratorError3 = err; } finally { try { if (!_iteratorNormalCompletion3 && _iterator3['return']) { _iterator3['return'](); } } finally { if (_didIteratorError3) { throw _iteratorError3; } } } var fmt = _this2.color('error title', '%s) %s:\n') + _this2.color('error message', '%s') + _this2.color('bright yellow', '%s') + _this2.color('error stack', '\n%s\n'); console.log(fmt, i + 1, test.title, test.err.message, runningBrowser, test.err.stack); }); } }, { key: 'add', value: function add(reporter) { this.reporters.push(reporter); } }, { key: 'propagateEvent', value: function propagateEvent() { if (this.reporters.length === 0) { return; } var _iteratorNormalCompletion4 = true; var _didIteratorError4 = false; var _iteratorError4 = undefined; try { for (var _len = arguments.length, args = Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } for (var _iterator4 = _getIterator(this.reporters), _step4; !(_iteratorNormalCompletion4 = (_step4 = _iterator4.next()).done); _iteratorNormalCompletion4 = true) { var reporter = _step4.value; /** * skip reporter if * - he isn't an eventemitter * - event is not registered */ if (typeof reporter.emit !== 'function' || _Object$keys(reporter._events).indexOf(args[0]) < 0) { continue; } reporter.emit.apply(reporter, args); } } catch (err) { _didIteratorError4 = true; _iteratorError4 = err; } finally { try { if (!_iteratorNormalCompletion4 && _iterator4['return']) { _iterator4['return'](); } } finally { if (_didIteratorError4) { throw _iteratorError4; } } } } /** * Default color map. */ }, { key: 'colors', get: function get() { return COLORS; } /** * Default symbol map. */ }, { key: 'symbols', get: function get() { /** * With node.js on Windows: use symbols available in terminal default fonts */ if (process.platform === 'win32') { return SYMBOLS_WIN; } return SYMBOLS; } }]); return BaseReporter; })(_events2['default'].EventEmitter); var Cursor = (function () { function Cursor() { _classCallCheck(this, Cursor); } _createClass(Cursor, [{ key: 'hide', value: function hide() { ISATTY && process.stdout.write('\u001b[?25l'); } }, { key: 'show', value: function show() { ISATTY && process.stdout.write('\u001b[?25h'); } }, { key: 'deleteLine', value: function deleteLine() { ISATTY && process.stdout.write('\u001b[2K'); } }, { key: 'beginningOfLine', value: function beginningOfLine() { ISATTY && process.stdout.write('\u001b[0G'); } }, { key: 'CR', value: function CR() { if (ISATTY) { this.deleteLine(); this.beginningOfLine(); } else { process.stdout.write('\r'); } } }, { key: 'isatty', get: function get() { return ISATTY; } }]); return Cursor; })(); exports['default'] = BaseReporter; exports.Cursor = Cursor; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../lib/utils/BaseReporter.js"],"names":[],"mappings":"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","file":"BaseReporter.js","sourcesContent":["import tty from 'tty'\nimport events from 'events'\nimport supportsColor from 'supports-color'\n\nconst ISATTY = tty.isatty(1) && tty.isatty(2)\n\nconst COLORS = {\n    'pass': 90,\n    'fail': 31,\n    'bright pass': 92,\n    'bright fail': 91,\n    'bright yellow': 93,\n    'pending': 36,\n    'suite': 0,\n    'error title': 0,\n    'error message': 31,\n    'error stack': 90,\n    'checkmark': 32,\n    'fast': 90,\n    'medium': 33,\n    'slow': 31,\n    'green': 32,\n    'light': 90,\n    'diff gutter': 90,\n    'diff added': 32,\n    'diff removed': 31\n}\n\nconst SYMBOLS_WIN = {\n    ok: '\\u221A',\n    err: '\\u00D7',\n    dot: '.',\n    error: 'F'\n}\n\nconst SYMBOLS = {\n    ok: '✓',\n    err: '✖',\n    dot: '․',\n    error: 'F'\n}\n\nclass BaseReporter extends events.EventEmitter {\n    constructor () {\n        super()\n\n        this.stats = {\n            suites: 0,\n            tests: 0,\n            passes: 0,\n            pending: 0,\n            failures: 0,\n            runner: {}\n        }\n        this.failures = []\n        this.reporters = []\n        this.printEpilogue = true\n        this.cursor = new Cursor()\n\n        this.on('start', () => {\n            this.stats.start = new Date()\n            this.propagateEvent('start')\n        })\n\n        this.on('runner:start', (runner) => {\n            if (!this.stats.runner[runner.cid]) {\n                this.stats.runner[runner.cid] = {\n                    start: new Date(),\n                    capabilities: runner.capabilities,\n                    config: runner.config,\n                    tests: []\n                }\n            }\n            this.propagateEvent('runner:start', runner)\n        })\n\n        this.on('runner:init', (runner) => {\n            this.stats.runner[runner.cid].sessionID = runner.sessionID\n            this.propagateEvent('runner:init', runner)\n        })\n\n        this.on('suite:start', (suite) => {\n            suite.root || this.stats.suites++\n            this.propagateEvent('suite:start', suite)\n        })\n\n        this.on('suite:end', (suite) => {\n            this.propagateEvent('suite:end', suite)\n        })\n\n        this.on('test:end', (test) => {\n            this.stats.tests++\n            this.propagateEvent('test:end', test)\n        })\n\n        this.on('test:pass', (test) => {\n            this.stats.runner[test.cid].tests.push(null)\n            this.stats.passes++\n            this.propagateEvent('test:pass', test)\n        })\n\n        this.on('test:fail', (test) => {\n            this.stats.failures++\n            this.stats.runner[test.cid].tests.push(test.err)\n\n            /**\n             * check if error also happened in other runners\n             */\n            var duplicateError = false\n            for (let failure of this.failures) {\n                if (test.err.message !== failure.err.message || failure.title !== test.title) {\n                    continue\n                }\n                duplicateError = true\n                failure.runner[test.cid] = test.runner[test.cid]\n            }\n\n            if (!duplicateError) {\n                this.failures.push(test)\n            }\n\n            this.propagateEvent('test:fail', test)\n        })\n\n        this.on('test:pending', (test) => {\n            this.stats.pending++\n            this.propagateEvent('test:pending', test)\n        })\n\n        this.on('runner:end', (runner) => {\n            this.stats.runner[runner.cid].end = new Date()\n            this.propagateEvent('runner:end', runner)\n        })\n\n        this.on('end', (args) => {\n            this.stats.end = new Date()\n            this.stats.duration = new Date() - this.stats.start\n            this.printEpilogue = this.printEpilogue && !args.sigint\n            this.propagateEvent('end', args)\n        })\n\n        this.on('error', (m) => {\n            this.printEpilogue = false\n\n            var fmt = this.color('error message', 'ERROR: %s')\n            console.log(fmt, m.error.message)\n\n            var sanitizedCaps = []\n            for (let capability of Object.keys(m.capabilities)) {\n                /**\n                 * we don't need all capability types to recognise a vm\n                 */\n                if (['browserName', 'platform', 'version', 'platformVersion', 'deviceName', 'app'].indexOf(capability) === -1) {\n                    continue\n                }\n\n                sanitizedCaps.push(capability + ': ' + JSON.stringify(m.capabilities[capability]))\n            }\n\n            fmt = this.color('bright yellow', sanitizedCaps.join(', '))\n            console.log(fmt)\n\n            if (m.error.stack) {\n                fmt = this.color('error stack', m.error.stack.replace(`Error: ${m.error.message}\\n`, ''))\n            } else {\n                fmt = this.color('error stack', '    no stack available')\n            }\n            console.log(fmt)\n\n            this.propagateEvent('error', m)\n        })\n    }\n\n    /**\n     * Color `str` with the given `type`,\n     * allowing colors to be disabled,\n     * as well as user-defined color\n     * schemes.\n     *\n     * @param {String} type\n     * @param {String} str\n     * @return {String}\n     * @api private\n     */\n    color (type, str) {\n        if (!supportsColor) return String(str)\n        return `\\u001b[${COLORS[type]}m${str}\\u001b[0m`\n    }\n\n    /**\n     * Output common epilogue used by many of\n     * the bundled reporters.\n     *\n     * @api public\n     */\n    epilogue () {\n        let fmt = null\n\n        if (!this.printEpilogue) {\n            return\n        }\n\n        console.log('\\n')\n\n        // passes\n        fmt = this.color('green', '%d passing') + this.color('light', ' (%ss)')\n        console.log(fmt, this.stats.passes || 0, ((Math.round(this.stats.duration / 100)) / 10).toFixed(2))\n\n        // pending\n        if (this.stats.pending) {\n            fmt = this.color('pending', '%d pending')\n            console.log(fmt, this.stats.pending)\n        }\n\n        // failures\n        if (this.stats.failures) {\n            fmt = this.color('fail', '%d failing')\n            console.log(fmt, this.stats.failures)\n            this.listFailures()\n        }\n\n        console.log()\n    }\n\n    /**\n     * Outut the given failures as a list\n     */\n    listFailures () {\n        console.log()\n\n        this.failures.forEach((test, i) => {\n            let runningBrowser = ''\n            for (let pid of Object.keys(test.runner)) {\n                let caps = test.runner[pid]\n                runningBrowser += '\\nrunning'\n\n                if (caps.browserName) {\n                    runningBrowser += ` ${caps.browserName}`\n                }\n                if (caps.version) {\n                    runningBrowser += ` (v${caps.version})`\n                }\n                if (caps.platform) {\n                    runningBrowser += ` on ${caps.platform}`\n                }\n\n                var host = this.stats.runner[pid].config.host\n                if (host && host.indexOf('saucelabs') > -1) {\n                    runningBrowser += '\\nCheck out job at https://saucelabs.com/tests/' + this.stats.runner[pid].sessionID\n                }\n            }\n\n            // format\n            let fmt = this.color('error title', '%s) %s:\\n') +\n                      this.color('error message', '%s') +\n                      this.color('bright yellow', '%s') +\n                      this.color('error stack', '\\n%s\\n')\n\n            console.log(fmt, (i + 1), test.title, test.err.message, runningBrowser, test.err.stack)\n        })\n    }\n\n    add (reporter) {\n        this.reporters.push(reporter)\n    }\n\n    propagateEvent (...args) {\n        if (this.reporters.length === 0) {\n            return\n        }\n\n        for (const reporter of this.reporters) {\n            /**\n             * skip reporter if\n             * \t- he isn't an eventemitter\n             * \t- event is not registered\n             */\n            if (typeof reporter.emit !== 'function' || Object.keys(reporter._events).indexOf(args[0]) < 0) {\n                continue\n            }\n\n            reporter.emit.apply(reporter, args)\n        }\n    }\n\n    /**\n     * Default color map.\n     */\n    get colors () {\n        return COLORS\n    }\n\n    /**\n     * Default symbol map.\n     */\n    get symbols () {\n        /**\n         * With node.js on Windows: use symbols available in terminal default fonts\n         */\n        if (process.platform === 'win32') {\n            return SYMBOLS_WIN\n        }\n\n        return SYMBOLS\n    }\n}\n\n/**\n * Expose some basic cursor interactions\n * that are common among reporters.\n */\nclass Cursor {\n    hide () {\n        ISATTY && process.stdout.write('\\u001b[?25l')\n    }\n\n    show () {\n        ISATTY && process.stdout.write('\\u001b[?25h')\n    }\n\n    deleteLine () {\n        ISATTY && process.stdout.write('\\u001b[2K')\n    }\n\n    beginningOfLine () {\n        ISATTY && process.stdout.write('\\u001b[0G')\n    }\n\n    CR () {\n        if (ISATTY) {\n            this.deleteLine()\n            this.beginningOfLine()\n        } else {\n            process.stdout.write('\\r')\n        }\n    }\n\n    get isatty () {\n        return ISATTY\n    }\n}\n\nexport default BaseReporter\nexport { Cursor }\n"]}