@testim/testim-cli
Version:
Command line interface for running Testing on you CI
499 lines (417 loc) • 39.5 kB
JavaScript
;
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"]}