Simon Priet 9e2991e668 init
2021-09-05 22:53:58 +02:00

168 lines
16 KiB
JavaScript

'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck');
var _classCallCheck3 = _interopRequireDefault(_classCallCheck2);
var _createClass2 = require('babel-runtime/helpers/createClass');
var _createClass3 = _interopRequireDefault(_createClass2);
var _lodash = require('lodash');
var _lodash2 = _interopRequireDefault(_lodash);
var _child_process = require('child_process');
var _child_process2 = _interopRequireDefault(_child_process);
var _command_types = require('./command_types');
var _command_types2 = _interopRequireDefault(_command_types);
var _path = require('path');
var _path2 = _interopRequireDefault(_path);
var _readline = require('readline');
var _readline2 = _interopRequireDefault(_readline);
var _status = require('../../status');
var _status2 = _interopRequireDefault(_status);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
var slaveCommand = _path2.default.resolve(__dirname, '..', '..', '..', 'bin', 'run_slave');
var Master = function () {
// options - {dryRun, failFast, filterStacktraces, strict}
function Master(_ref) {
var eventBroadcaster = _ref.eventBroadcaster,
options = _ref.options,
supportCodePaths = _ref.supportCodePaths,
supportCodeRequiredModules = _ref.supportCodeRequiredModules,
testCases = _ref.testCases;
(0, _classCallCheck3.default)(this, Master);
this.eventBroadcaster = eventBroadcaster;
this.options = options || {};
this.supportCodePaths = supportCodePaths;
this.supportCodeRequiredModules = supportCodeRequiredModules;
this.testCases = testCases || [];
this.nextTestCaseIndex = 0;
this.testCasesCompleted = 0;
this.result = {
duration: 0,
success: true
};
this.slaves = {};
}
(0, _createClass3.default)(Master, [{
key: 'parseSlaveLine',
value: function parseSlaveLine(slave, line) {
var input = JSON.parse(line);
switch (input.command) {
case _command_types2.default.READY:
this.giveSlaveWork(slave);
break;
case _command_types2.default.EVENT:
this.eventBroadcaster.emit(input.name, input.data);
if (input.name === 'test-case-finished') {
this.parseTestCaseResult(input.data.result);
}
break;
default:
throw new Error('Unexpected message from slave: ' + line);
}
}
}, {
key: 'startSlave',
value: function startSlave(id, total) {
var _this = this;
var slaveProcess = _child_process2.default.spawn(slaveCommand, [], {
env: _lodash2.default.assign({}, process.env, {
CUCUMBER_PARALLEL: 'true',
CUCUMBER_TOTAL_SLAVES: total,
CUCUMBER_SLAVE_ID: id
}),
stdio: ['pipe', 'pipe', process.stderr]
});
var rl = _readline2.default.createInterface({ input: slaveProcess.stdout });
var slave = { process: slaveProcess };
this.slaves[id] = slave;
rl.on('line', function (line) {
_this.parseSlaveLine(slave, line);
});
rl.on('close', function () {
slave.closed = true;
_this.onSlaveClose();
});
slave.process.stdin.write(JSON.stringify({
command: _command_types2.default.INITIALIZE,
filterStacktraces: this.options.filterStacktraces,
supportCodePaths: this.supportCodePaths,
supportCodeRequiredModules: this.supportCodeRequiredModules,
worldParameters: this.options.worldParameters
}) + '\n');
}
}, {
key: 'onSlaveClose',
value: function onSlaveClose() {
if (_lodash2.default.every(this.slaves, 'closed')) {
this.eventBroadcaster.emit('test-run-finished', { result: this.result });
this.onFinish(this.result.success);
}
}
}, {
key: 'parseTestCaseResult',
value: function parseTestCaseResult(testCaseResult) {
this.testCasesCompleted += 1;
if (testCaseResult.duration) {
this.result.duration += testCaseResult.duration;
}
if (this.shouldCauseFailure(testCaseResult.status)) {
this.result.success = false;
}
}
}, {
key: 'run',
value: function run(numberOfSlaves, done) {
var _this2 = this;
this.eventBroadcaster.emit('test-run-started');
_lodash2.default.times(numberOfSlaves, function (id) {
return _this2.startSlave(id, numberOfSlaves);
});
this.onFinish = done;
}
}, {
key: 'giveSlaveWork',
value: function giveSlaveWork(slave) {
if (this.nextTestCaseIndex === this.testCases.length) {
slave.process.stdin.write(JSON.stringify({ command: _command_types2.default.FINALIZE }) + '\n');
return;
}
var testCase = this.testCases[this.nextTestCaseIndex];
this.nextTestCaseIndex += 1;
var skip = this.options.dryRun || this.options.failFast && !this.result.success;
slave.process.stdin.write(JSON.stringify({ command: _command_types2.default.RUN, skip: skip, testCase: testCase }) + '\n');
}
}, {
key: 'shouldCauseFailure',
value: function shouldCauseFailure(status) {
return _lodash2.default.includes([_status2.default.AMBIGUOUS, _status2.default.FAILED, _status2.default.UNDEFINED], status) || status === _status2.default.PENDING && this.options.strict;
}
}]);
return Master;
}();
exports.default = Master;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/runtime/parallel/master.js"],"names":["slaveCommand","resolve","__dirname","Master","eventBroadcaster","options","supportCodePaths","supportCodeRequiredModules","testCases","nextTestCaseIndex","testCasesCompleted","result","duration","success","slaves","slave","line","input","JSON","parse","command","READY","giveSlaveWork","EVENT","emit","name","data","parseTestCaseResult","Error","id","total","slaveProcess","spawn","env","assign","process","CUCUMBER_PARALLEL","CUCUMBER_TOTAL_SLAVES","CUCUMBER_SLAVE_ID","stdio","stderr","rl","createInterface","stdout","on","parseSlaveLine","closed","onSlaveClose","stdin","write","stringify","INITIALIZE","filterStacktraces","worldParameters","every","onFinish","testCaseResult","shouldCauseFailure","status","numberOfSlaves","done","times","startSlave","length","FINALIZE","testCase","skip","dryRun","failFast","RUN","includes","AMBIGUOUS","FAILED","UNDEFINED","PENDING","strict"],"mappings":"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","file":"master.js","sourcesContent":["import _ from 'lodash'\nimport childProcess from 'child_process'\nimport commandTypes from './command_types'\nimport path from 'path'\nimport readline from 'readline'\nimport Status from '../../status'\n\nconst slaveCommand = path.resolve(\n  __dirname,\n  '..',\n  '..',\n  '..',\n  'bin',\n  'run_slave'\n)\n\nexport default class Master {\n  // options - {dryRun, failFast, filterStacktraces, strict}\n  constructor({\n    eventBroadcaster,\n    options,\n    supportCodePaths,\n    supportCodeRequiredModules,\n    testCases,\n  }) {\n    this.eventBroadcaster = eventBroadcaster\n    this.options = options || {}\n    this.supportCodePaths = supportCodePaths\n    this.supportCodeRequiredModules = supportCodeRequiredModules\n    this.testCases = testCases || []\n    this.nextTestCaseIndex = 0\n    this.testCasesCompleted = 0\n    this.result = {\n      duration: 0,\n      success: true,\n    }\n    this.slaves = {}\n  }\n\n  parseSlaveLine(slave, line) {\n    const input = JSON.parse(line)\n    switch (input.command) {\n      case commandTypes.READY:\n        this.giveSlaveWork(slave)\n        break\n      case commandTypes.EVENT:\n        this.eventBroadcaster.emit(input.name, input.data)\n        if (input.name === 'test-case-finished') {\n          this.parseTestCaseResult(input.data.result)\n        }\n        break\n      default:\n        throw new Error(`Unexpected message from slave: ${line}`)\n    }\n  }\n\n  startSlave(id, total) {\n    const slaveProcess = childProcess.spawn(slaveCommand, [], {\n      env: _.assign({}, process.env, {\n        CUCUMBER_PARALLEL: 'true',\n        CUCUMBER_TOTAL_SLAVES: total,\n        CUCUMBER_SLAVE_ID: id,\n      }),\n      stdio: ['pipe', 'pipe', process.stderr],\n    })\n    const rl = readline.createInterface({ input: slaveProcess.stdout })\n    const slave = { process: slaveProcess }\n    this.slaves[id] = slave\n    rl.on('line', line => {\n      this.parseSlaveLine(slave, line)\n    })\n    rl.on('close', () => {\n      slave.closed = true\n      this.onSlaveClose()\n    })\n    slave.process.stdin.write(\n      JSON.stringify({\n        command: commandTypes.INITIALIZE,\n        filterStacktraces: this.options.filterStacktraces,\n        supportCodePaths: this.supportCodePaths,\n        supportCodeRequiredModules: this.supportCodeRequiredModules,\n        worldParameters: this.options.worldParameters,\n      }) + '\\n'\n    )\n  }\n\n  onSlaveClose() {\n    if (_.every(this.slaves, 'closed')) {\n      this.eventBroadcaster.emit('test-run-finished', { result: this.result })\n      this.onFinish(this.result.success)\n    }\n  }\n\n  parseTestCaseResult(testCaseResult) {\n    this.testCasesCompleted += 1\n    if (testCaseResult.duration) {\n      this.result.duration += testCaseResult.duration\n    }\n    if (this.shouldCauseFailure(testCaseResult.status)) {\n      this.result.success = false\n    }\n  }\n\n  run(numberOfSlaves, done) {\n    this.eventBroadcaster.emit('test-run-started')\n    _.times(numberOfSlaves, id => this.startSlave(id, numberOfSlaves))\n    this.onFinish = done\n  }\n\n  giveSlaveWork(slave) {\n    if (this.nextTestCaseIndex === this.testCases.length) {\n      slave.process.stdin.write(\n        JSON.stringify({ command: commandTypes.FINALIZE }) + '\\n'\n      )\n      return\n    }\n    const testCase = this.testCases[this.nextTestCaseIndex]\n    this.nextTestCaseIndex += 1\n    const skip =\n      this.options.dryRun || (this.options.failFast && !this.result.success)\n    slave.process.stdin.write(\n      JSON.stringify({ command: commandTypes.RUN, skip, testCase }) + '\\n'\n    )\n  }\n\n  shouldCauseFailure(status) {\n    return (\n      _.includes([Status.AMBIGUOUS, Status.FAILED, Status.UNDEFINED], status) ||\n      (status === Status.PENDING && this.options.strict)\n    )\n  }\n}\n"]}