1 | ;
|
2 | var __importDefault = (this && this.__importDefault) || function (mod) {
|
3 | return (mod && mod.__esModule) ? mod : { "default": mod };
|
4 | };
|
5 | Object.defineProperty(exports, "__esModule", { value: true });
|
6 | const async_event_emitter_1 = __importDefault(require("../utils/async-event-emitter"));
|
7 | const testcafe_legacy_api_1 = require("testcafe-legacy-api");
|
8 | const test_run_1 = __importDefault(require("../test-run"));
|
9 | const session_controller_1 = __importDefault(require("../test-run/session-controller"));
|
10 | const QUARANTINE_THRESHOLD = 3;
|
11 | const DISCONNECT_THRESHOLD = 3;
|
12 | class Quarantine {
|
13 | constructor() {
|
14 | this.attempts = [];
|
15 | }
|
16 | getFailedAttempts() {
|
17 | return this.attempts.filter(errors => !!errors.length);
|
18 | }
|
19 | getPassedAttempts() {
|
20 | return this.attempts.filter(errors => errors.length === 0);
|
21 | }
|
22 | getNextAttemptNumber() {
|
23 | return this.attempts.length + 1;
|
24 | }
|
25 | isThresholdReached(extraErrors) {
|
26 | const { failedTimes, passedTimes } = this._getAttemptsResult(extraErrors);
|
27 | const failedThresholdReached = failedTimes >= QUARANTINE_THRESHOLD;
|
28 | const passedThresholdReached = passedTimes >= QUARANTINE_THRESHOLD;
|
29 | return failedThresholdReached || passedThresholdReached;
|
30 | }
|
31 | isFirstAttemptSuccessful(extraErrors) {
|
32 | const { failedTimes, passedTimes } = this._getAttemptsResult(extraErrors);
|
33 | return failedTimes === 0 && passedTimes > 0;
|
34 | }
|
35 | _getAttemptsResult(extraErrors) {
|
36 | let failedTimes = this.getFailedAttempts().length;
|
37 | let passedTimes = this.getPassedAttempts().length;
|
38 | if (extraErrors) {
|
39 | if (extraErrors.length)
|
40 | failedTimes += extraErrors.length;
|
41 | else
|
42 | passedTimes += 1;
|
43 | }
|
44 | return { failedTimes, passedTimes };
|
45 | }
|
46 | }
|
47 | class TestRunController extends async_event_emitter_1.default {
|
48 | constructor(test, index, proxy, screenshots, warningLog, fixtureHookController, opts) {
|
49 | super();
|
50 | this.test = test;
|
51 | this.index = index;
|
52 | this.opts = opts;
|
53 | this.proxy = proxy;
|
54 | this.screenshots = screenshots;
|
55 | this.warningLog = warningLog;
|
56 | this.fixtureHookController = fixtureHookController;
|
57 | this.TestRunCtor = TestRunController._getTestRunCtor(test, opts);
|
58 | this.testRun = null;
|
59 | this.done = false;
|
60 | this.quarantine = null;
|
61 | this.disconnectionCount = 0;
|
62 | if (this.opts.quarantineMode)
|
63 | this.quarantine = new Quarantine();
|
64 | }
|
65 | static _getTestRunCtor(test, opts) {
|
66 | if (opts.TestRunCtor)
|
67 | return opts.TestRunCtor;
|
68 | return test.isLegacy ? testcafe_legacy_api_1.TestRun : test_run_1.default;
|
69 | }
|
70 | async _createTestRun(connection) {
|
71 | const screenshotCapturer = this.screenshots.createCapturerFor(this.test, this.index, this.quarantine, connection, this.warningLog);
|
72 | const TestRunCtor = this.TestRunCtor;
|
73 | this.testRun = new TestRunCtor(this.test, connection, screenshotCapturer, this.warningLog, this.opts);
|
74 | if (this.testRun.addQuarantineInfo)
|
75 | this.testRun.addQuarantineInfo(this.quarantine);
|
76 | if (!this.quarantine || this._isFirstQuarantineAttempt()) {
|
77 | await this.emit('test-run-create', {
|
78 | testRun: this.testRun,
|
79 | legacy: TestRunCtor === testcafe_legacy_api_1.TestRun,
|
80 | test: this.test,
|
81 | index: this.index,
|
82 | quarantine: this.quarantine,
|
83 | });
|
84 | }
|
85 | return this.testRun;
|
86 | }
|
87 | async _endQuarantine() {
|
88 | if (this.quarantine.attempts.length > 1)
|
89 | this.testRun.unstable = this.quarantine.getPassedAttempts().length > 0;
|
90 | await this._emitTestRunDone();
|
91 | }
|
92 | _shouldKeepInQuarantine() {
|
93 | const errors = this.testRun.errs;
|
94 | const hasErrors = !!errors.length;
|
95 | const attempts = this.quarantine.attempts;
|
96 | const isFirstAttempt = this._isFirstQuarantineAttempt();
|
97 | attempts.push(errors);
|
98 | return isFirstAttempt ? hasErrors : !this.quarantine.isThresholdReached();
|
99 | }
|
100 | _isFirstQuarantineAttempt() {
|
101 | return this.quarantine && !this.quarantine.attempts.length;
|
102 | }
|
103 | async _keepInQuarantine() {
|
104 | await this._restartTest();
|
105 | }
|
106 | async _restartTest() {
|
107 | await this.emit('test-run-restart');
|
108 | }
|
109 | async _testRunDoneInQuarantineMode() {
|
110 | if (this._shouldKeepInQuarantine())
|
111 | await this._keepInQuarantine();
|
112 | else
|
113 | await this._endQuarantine();
|
114 | }
|
115 | async _testRunDone() {
|
116 | if (this.quarantine)
|
117 | await this._testRunDoneInQuarantineMode();
|
118 | else
|
119 | await this._emitTestRunDone();
|
120 | }
|
121 | async _emitActionStart(args) {
|
122 | await this.emit('test-action-start', args);
|
123 | }
|
124 | async _emitActionDone(args) {
|
125 | await this.emit('test-action-done', args);
|
126 | }
|
127 | async _emitTestRunDone() {
|
128 | // NOTE: we should report test run completion in order they were completed in browser.
|
129 | // To keep a sequence after fixture hook execution we use completion queue.
|
130 | await this.fixtureHookController.runFixtureAfterHookIfNecessary(this.testRun);
|
131 | this.done = true;
|
132 | await this.emit('test-run-done');
|
133 | }
|
134 | async _emitTestRunStart() {
|
135 | await this.emit('test-run-start');
|
136 | }
|
137 | async _testRunBeforeDone() {
|
138 | let raiseEvent = !this.quarantine;
|
139 | if (!raiseEvent) {
|
140 | const isSuccessfulQuarantineFirstAttempt = this._isFirstQuarantineAttempt() && !this.testRun.errs.length;
|
141 | const isAttemptsThresholdReached = this.quarantine.isThresholdReached(this.testRun.errs);
|
142 | raiseEvent = isSuccessfulQuarantineFirstAttempt || isAttemptsThresholdReached;
|
143 | }
|
144 | if (raiseEvent)
|
145 | await this.emit('test-run-before-done');
|
146 | }
|
147 | _testRunDisconnected(connection) {
|
148 | this.disconnectionCount++;
|
149 | const disconnectionThresholdExceedeed = this.disconnectionCount >= DISCONNECT_THRESHOLD;
|
150 | return connection
|
151 | .processDisconnection(disconnectionThresholdExceedeed)
|
152 | .then(() => {
|
153 | return this._restartTest();
|
154 | });
|
155 | }
|
156 | _assignTestRunEvents(testRun, connection) {
|
157 | testRun.on('action-start', async (args) => this._emitActionStart(Object.assign(args, { testRun })));
|
158 | testRun.on('action-done', async (args) => this._emitActionDone(Object.assign(args, { testRun })));
|
159 | testRun.once('start', async () => this._emitTestRunStart());
|
160 | testRun.once('ready', async () => {
|
161 | if (!this.quarantine || this._isFirstQuarantineAttempt())
|
162 | await this.emit('test-run-ready');
|
163 | });
|
164 | testRun.once('before-done', () => this._testRunBeforeDone());
|
165 | testRun.once('done', () => this._testRunDone());
|
166 | testRun.once('disconnected', () => this._testRunDisconnected(connection));
|
167 | }
|
168 | get blocked() {
|
169 | return this.fixtureHookController.isTestBlocked(this.test);
|
170 | }
|
171 | async start(connection) {
|
172 | const testRun = await this._createTestRun(connection);
|
173 | const hookOk = await this.fixtureHookController.runFixtureBeforeHookIfNecessary(testRun);
|
174 | if (this.test.skip || !hookOk) {
|
175 | await this.emit('test-run-start');
|
176 | await this._emitTestRunDone();
|
177 | return null;
|
178 | }
|
179 | this._assignTestRunEvents(testRun, connection);
|
180 | testRun.start();
|
181 | return session_controller_1.default.getSessionUrl(testRun, this.proxy);
|
182 | }
|
183 | }
|
184 | exports.default = TestRunController;
|
185 | module.exports = exports.default;
|
186 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-run-controller.js","sourceRoot":"","sources":["../../src/runner/test-run-controller.js"],"names":[],"mappings":";;;;;AAAA,uFAA6D;AAC7D,6DAA+D;AAC/D,2DAAkC;AAClC,wFAA+D;AAE/D,MAAM,oBAAoB,GAAG,CAAC,CAAC;AAC/B,MAAM,oBAAoB,GAAG,CAAC,CAAC;AAE/B,MAAM,UAAU;IACZ;QACI,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACvB,CAAC;IAED,iBAAiB;QACb,OAAO,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IAC3D,CAAC;IAED,iBAAiB;QACb,OAAO,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;IAC/D,CAAC;IAED,oBAAoB;QAChB,OAAO,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC;IACpC,CAAC;IAED,kBAAkB,CAAE,WAAW;QAC3B,MAAM,EAAE,WAAW,EAAE,WAAW,EAAE,GAAG,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC;QAE1E,MAAM,sBAAsB,GAAG,WAAW,IAAI,oBAAoB,CAAC;QACnE,MAAM,sBAAsB,GAAG,WAAW,IAAI,oBAAoB,CAAC;QAEnE,OAAO,sBAAsB,IAAI,sBAAsB,CAAC;IAC5D,CAAC;IAED,wBAAwB,CAAE,WAAW;QACjC,MAAM,EAAE,WAAW,EAAE,WAAW,EAAE,GAAG,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC;QAE1E,OAAO,WAAW,KAAK,CAAC,IAAI,WAAW,GAAG,CAAC,CAAC;IAChD,CAAC;IAED,kBAAkB,CAAE,WAAW;QAC3B,IAAI,WAAW,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC,MAAM,CAAC;QAClD,IAAI,WAAW,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC,MAAM,CAAC;QAElD,IAAI,WAAW,EAAE;YACb,IAAI,WAAW,CAAC,MAAM;gBAClB,WAAW,IAAI,WAAW,CAAC,MAAM,CAAC;;gBAElC,WAAW,IAAI,CAAC,CAAC;SACxB;QAED,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,CAAC;IACxC,CAAC;CACJ;AAED,MAAqB,iBAAkB,SAAQ,6BAAiB;IAC5D,YAAa,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,WAAW,EAAE,UAAU,EAAE,qBAAqB,EAAE,IAAI;QACjF,KAAK,EAAE,CAAC;QAER,IAAI,CAAC,IAAI,GAAI,IAAI,CAAC;QAClB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,IAAI,GAAI,IAAI,CAAC;QAElB,IAAI,CAAC,KAAK,GAAmB,KAAK,CAAC;QACnC,IAAI,CAAC,WAAW,GAAa,WAAW,CAAC;QACzC,IAAI,CAAC,UAAU,GAAc,UAAU,CAAC;QACxC,IAAI,CAAC,qBAAqB,GAAG,qBAAqB,CAAC;QAEnD,IAAI,CAAC,WAAW,GAAG,iBAAiB,CAAC,eAAe,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QAEjE,IAAI,CAAC,OAAO,GAAc,IAAI,CAAC;QAC/B,IAAI,CAAC,IAAI,GAAiB,KAAK,CAAC;QAChC,IAAI,CAAC,UAAU,GAAW,IAAI,CAAC;QAC/B,IAAI,CAAC,kBAAkB,GAAG,CAAC,CAAC;QAE5B,IAAI,IAAI,CAAC,IAAI,CAAC,cAAc;YACxB,IAAI,CAAC,UAAU,GAAG,IAAI,UAAU,EAAE,CAAC;IAC3C,CAAC;IAED,MAAM,CAAC,eAAe,CAAE,IAAI,EAAE,IAAI;QAC9B,IAAI,IAAI,CAAC,WAAW;YAChB,OAAO,IAAI,CAAC,WAAW,CAAC;QAE5B,OAAO,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,6BAAa,CAAC,CAAC,CAAC,kBAAO,CAAC;IACnD,CAAC;IAED,KAAK,CAAC,cAAc,CAAE,UAAU;QAC5B,MAAM,kBAAkB,GAAG,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,UAAU,EAAE,UAAU,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;QACnI,MAAM,WAAW,GAAU,IAAI,CAAC,WAAW,CAAC;QAE5C,IAAI,CAAC,OAAO,GAAG,IAAI,WAAW,CAAC,IAAI,CAAC,IAAI,EAAE,UAAU,EAAE,kBAAkB,EAAE,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;QAEtG,IAAI,IAAI,CAAC,OAAO,CAAC,iBAAiB;YAC9B,IAAI,CAAC,OAAO,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAEpD,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,yBAAyB,EAAE,EAAE;YACtD,MAAM,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE;gBAC/B,OAAO,EAAK,IAAI,CAAC,OAAO;gBACxB,MAAM,EAAM,WAAW,KAAK,6BAAa;gBACzC,IAAI,EAAQ,IAAI,CAAC,IAAI;gBACrB,KAAK,EAAO,IAAI,CAAC,KAAK;gBACtB,UAAU,EAAE,IAAI,CAAC,UAAU;aAC9B,CAAC,CAAC;SACN;QAED,OAAO,IAAI,CAAC,OAAO,CAAC;IACxB,CAAC;IAED,KAAK,CAAC,cAAc;QAChB,IAAI,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC;YACnC,IAAI,CAAC,OAAO,CAAC,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,iBAAiB,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC;QAE3E,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAClC,CAAC;IAED,uBAAuB;QACnB,MAAM,MAAM,GAAW,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC;QACzC,MAAM,SAAS,GAAQ,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC;QACvC,MAAM,QAAQ,GAAS,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC;QAChD,MAAM,cAAc,GAAG,IAAI,CAAC,yBAAyB,EAAE,CAAC;QAExD,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAEtB,OAAO,cAAc,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,kBAAkB,EAAE,CAAC;IAC9E,CAAC;IAED,yBAAyB;QACrB,OAAO,IAAI,CAAC,UAAU,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,MAAM,CAAC;IAC/D,CAAC;IAED,KAAK,CAAC,iBAAiB;QACnB,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;IAC9B,CAAC;IAED,KAAK,CAAC,YAAY;QACd,MAAM,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;IACxC,CAAC;IAED,KAAK,CAAC,4BAA4B;QAC9B,IAAI,IAAI,CAAC,uBAAuB,EAAE;YAC9B,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;;YAE/B,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC;IACpC,CAAC;IAED,KAAK,CAAC,YAAY;QACd,IAAI,IAAI,CAAC,UAAU;YACf,MAAM,IAAI,CAAC,4BAA4B,EAAE,CAAC;;YAE1C,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;IACtC,CAAC;IAED,KAAK,CAAC,gBAAgB,CAAE,IAAI;QACxB,MAAM,IAAI,CAAC,IAAI,CAAC,mBAAmB,EAAE,IAAI,CAAC,CAAC;IAC/C,CAAC;IAED,KAAK,CAAC,eAAe,CAAE,IAAI;QACvB,MAAM,IAAI,CAAC,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,CAAC;IAC9C,CAAC;IAED,KAAK,CAAC,gBAAgB;QAClB,sFAAsF;QACtF,2EAA2E;QAC3E,MAAM,IAAI,CAAC,qBAAqB,CAAC,8BAA8B,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAE9E,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QAEjB,MAAM,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;IACrC,CAAC;IAED,KAAK,CAAC,iBAAiB;QACnB,MAAM,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;IACtC,CAAC;IAED,KAAK,CAAC,kBAAkB;QACpB,IAAI,UAAU,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC;QAElC,IAAI,CAAC,UAAU,EAAE;YACb,MAAM,kCAAkC,GAAG,IAAI,CAAC,yBAAyB,EAAE,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC;YACzG,MAAM,0BAA0B,GAAW,IAAI,CAAC,UAAU,CAAC,kBAAkB,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YAEjG,UAAU,GAAG,kCAAkC,IAAI,0BAA0B,CAAC;SACjF;QAED,IAAI,UAAU;YACV,MAAM,IAAI,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC;IAChD,CAAC;IAED,oBAAoB,CAAE,UAAU;QAC5B,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAE1B,MAAM,+BAA+B,GAAG,IAAI,CAAC,kBAAkB,IAAI,oBAAoB,CAAC;QAExF,OAAO,UAAU;aACZ,oBAAoB,CAAC,+BAA+B,CAAC;aACrD,IAAI,CAAC,GAAG,EAAE;YACP,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC;QAC/B,CAAC,CAAC,CAAC;IACX,CAAC;IAED,oBAAoB,CAAE,OAAO,EAAE,UAAU;QACrC,OAAO,CAAC,EAAE,CAAC,cAAc,EAAE,KAAK,EAAC,IAAI,EAAC,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC,CAAC;QAClG,OAAO,CAAC,EAAE,CAAC,aAAa,EAAE,KAAK,EAAC,IAAI,EAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC,CAAC;QAEhG,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,IAAI,EAAE,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC;QAC5D,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,IAAI,EAAE;YAC7B,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,yBAAyB,EAAE;gBACpD,MAAM,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAC1C,CAAC,CAAC,CAAC;QACH,OAAO,CAAC,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,CAAC;QAC7D,OAAO,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC;QAChD,OAAO,CAAC,IAAI,CAAC,cAAc,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,UAAU,CAAC,CAAC,CAAC;IAC9E,CAAC;IAED,IAAI,OAAO;QACP,OAAO,IAAI,CAAC,qBAAqB,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC/D,CAAC;IAED,KAAK,CAAC,KAAK,CAAE,UAAU;QACnB,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC;QAEtD,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,qBAAqB,CAAC,+BAA+B,CAAC,OAAO,CAAC,CAAC;QAEzF,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,MAAM,EAAE;YAC3B,MAAM,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;YAClC,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC9B,OAAO,IAAI,CAAC;SACf;QAED,IAAI,CAAC,oBAAoB,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;QAE/C,OAAO,CAAC,KAAK,EAAE,CAAC;QAEhB,OAAO,4BAAiB,CAAC,aAAa,CAAC,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;IAChE,CAAC;CACJ;AApLD,oCAoLC","sourcesContent":["import AsyncEventEmitter from '../utils/async-event-emitter';\nimport { TestRun as LegacyTestRun } from 'testcafe-legacy-api';\nimport TestRun from '../test-run';\nimport SessionController from '../test-run/session-controller';\n\nconst QUARANTINE_THRESHOLD = 3;\nconst DISCONNECT_THRESHOLD = 3;\n\nclass Quarantine {\n    constructor () {\n        this.attempts = [];\n    }\n\n    getFailedAttempts () {\n        return this.attempts.filter(errors => !!errors.length);\n    }\n\n    getPassedAttempts () {\n        return this.attempts.filter(errors => errors.length === 0);\n    }\n\n    getNextAttemptNumber () {\n        return this.attempts.length + 1;\n    }\n\n    isThresholdReached (extraErrors) {\n        const { failedTimes, passedTimes } = this._getAttemptsResult(extraErrors);\n\n        const failedThresholdReached = failedTimes >= QUARANTINE_THRESHOLD;\n        const passedThresholdReached = passedTimes >= QUARANTINE_THRESHOLD;\n\n        return failedThresholdReached || passedThresholdReached;\n    }\n\n    isFirstAttemptSuccessful (extraErrors) {\n        const { failedTimes, passedTimes } = this._getAttemptsResult(extraErrors);\n\n        return failedTimes === 0 && passedTimes > 0;\n    }\n\n    _getAttemptsResult (extraErrors) {\n        let failedTimes = this.getFailedAttempts().length;\n        let passedTimes = this.getPassedAttempts().length;\n\n        if (extraErrors) {\n            if (extraErrors.length)\n                failedTimes += extraErrors.length;\n            else\n                passedTimes += 1;\n        }\n\n        return { failedTimes, passedTimes };\n    }\n}\n\nexport default class TestRunController extends AsyncEventEmitter {\n    constructor (test, index, proxy, screenshots, warningLog, fixtureHookController, opts) {\n        super();\n\n        this.test  = test;\n        this.index = index;\n        this.opts  = opts;\n\n        this.proxy                 = proxy;\n        this.screenshots           = screenshots;\n        this.warningLog            = warningLog;\n        this.fixtureHookController = fixtureHookController;\n\n        this.TestRunCtor = TestRunController._getTestRunCtor(test, opts);\n\n        this.testRun            = null;\n        this.done               = false;\n        this.quarantine         = null;\n        this.disconnectionCount = 0;\n\n        if (this.opts.quarantineMode)\n            this.quarantine = new Quarantine();\n    }\n\n    static _getTestRunCtor (test, opts) {\n        if (opts.TestRunCtor)\n            return opts.TestRunCtor;\n\n        return test.isLegacy ? LegacyTestRun : TestRun;\n    }\n\n    async _createTestRun (connection) {\n        const screenshotCapturer = this.screenshots.createCapturerFor(this.test, this.index, this.quarantine, connection, this.warningLog);\n        const TestRunCtor        = this.TestRunCtor;\n\n        this.testRun = new TestRunCtor(this.test, connection, screenshotCapturer, this.warningLog, this.opts);\n\n        if (this.testRun.addQuarantineInfo)\n            this.testRun.addQuarantineInfo(this.quarantine);\n\n        if (!this.quarantine || this._isFirstQuarantineAttempt()) {\n            await this.emit('test-run-create', {\n                testRun:    this.testRun,\n                legacy:     TestRunCtor === LegacyTestRun,\n                test:       this.test,\n                index:      this.index,\n                quarantine: this.quarantine,\n            });\n        }\n\n        return this.testRun;\n    }\n\n    async _endQuarantine () {\n        if (this.quarantine.attempts.length > 1)\n            this.testRun.unstable = this.quarantine.getPassedAttempts().length > 0;\n\n        await this._emitTestRunDone();\n    }\n\n    _shouldKeepInQuarantine () {\n        const errors         = this.testRun.errs;\n        const hasErrors      = !!errors.length;\n        const attempts       = this.quarantine.attempts;\n        const isFirstAttempt = this._isFirstQuarantineAttempt();\n\n        attempts.push(errors);\n\n        return isFirstAttempt ? hasErrors : !this.quarantine.isThresholdReached();\n    }\n\n    _isFirstQuarantineAttempt () {\n        return this.quarantine && !this.quarantine.attempts.length;\n    }\n\n    async _keepInQuarantine () {\n        await this._restartTest();\n    }\n\n    async _restartTest () {\n        await this.emit('test-run-restart');\n    }\n\n    async _testRunDoneInQuarantineMode () {\n        if (this._shouldKeepInQuarantine())\n            await this._keepInQuarantine();\n        else\n            await this._endQuarantine();\n    }\n\n    async _testRunDone () {\n        if (this.quarantine)\n            await this._testRunDoneInQuarantineMode();\n        else\n            await this._emitTestRunDone();\n    }\n\n    async _emitActionStart (args) {\n        await this.emit('test-action-start', args);\n    }\n\n    async _emitActionDone (args) {\n        await this.emit('test-action-done', args);\n    }\n\n    async _emitTestRunDone () {\n        // NOTE: we should report test run completion in order they were completed in browser.\n        // To keep a sequence after fixture hook execution we use completion queue.\n        await this.fixtureHookController.runFixtureAfterHookIfNecessary(this.testRun);\n\n        this.done = true;\n\n        await this.emit('test-run-done');\n    }\n\n    async _emitTestRunStart () {\n        await this.emit('test-run-start');\n    }\n\n    async _testRunBeforeDone () {\n        let raiseEvent = !this.quarantine;\n\n        if (!raiseEvent) {\n            const isSuccessfulQuarantineFirstAttempt = this._isFirstQuarantineAttempt() && !this.testRun.errs.length;\n            const isAttemptsThresholdReached         = this.quarantine.isThresholdReached(this.testRun.errs);\n\n            raiseEvent = isSuccessfulQuarantineFirstAttempt || isAttemptsThresholdReached;\n        }\n\n        if (raiseEvent)\n            await this.emit('test-run-before-done');\n    }\n\n    _testRunDisconnected (connection) {\n        this.disconnectionCount++;\n\n        const disconnectionThresholdExceedeed = this.disconnectionCount >= DISCONNECT_THRESHOLD;\n\n        return connection\n            .processDisconnection(disconnectionThresholdExceedeed)\n            .then(() => {\n                return this._restartTest();\n            });\n    }\n\n    _assignTestRunEvents (testRun, connection) {\n        testRun.on('action-start', async args => this._emitActionStart(Object.assign(args, { testRun })));\n        testRun.on('action-done', async args => this._emitActionDone(Object.assign(args, { testRun })));\n\n        testRun.once('start', async () => this._emitTestRunStart());\n        testRun.once('ready', async () => {\n            if (!this.quarantine || this._isFirstQuarantineAttempt())\n                await this.emit('test-run-ready');\n        });\n        testRun.once('before-done', () => this._testRunBeforeDone());\n        testRun.once('done', () => this._testRunDone());\n        testRun.once('disconnected', () => this._testRunDisconnected(connection));\n    }\n\n    get blocked () {\n        return this.fixtureHookController.isTestBlocked(this.test);\n    }\n\n    async start (connection) {\n        const testRun = await this._createTestRun(connection);\n\n        const hookOk = await this.fixtureHookController.runFixtureBeforeHookIfNecessary(testRun);\n\n        if (this.test.skip || !hookOk) {\n            await this.emit('test-run-start');\n            await this._emitTestRunDone();\n            return null;\n        }\n\n        this._assignTestRunEvents(testRun, connection);\n\n        testRun.start();\n\n        return SessionController.getSessionUrl(testRun, this.proxy);\n    }\n}\n"]} |
\ | No newline at end of file |