@browserstack/testcafe
Version:
Automated browser testing for the modern web development stack.
186 lines • 30.4 kB
JavaScript
"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const async_event_emitter_1 = __importDefault(require("../utils/async-event-emitter"));
//@ts-ignore
const testcafe_legacy_api_1 = require("testcafe-legacy-api");
const test_run_1 = __importDefault(require("../test-run"));
const session_controller_1 = __importDefault(require("../test-run/session-controller"));
const QUARANTINE_THRESHOLD = 3;
const DISCONNECT_THRESHOLD = 3;
class Quarantine {
constructor() {
this.attempts = [];
}
getFailedAttempts() {
return this.attempts.filter(errors => !!errors.length);
}
getPassedAttempts() {
return this.attempts.filter(errors => errors.length === 0);
}
getNextAttemptNumber() {
return this.attempts.length + 1;
}
isThresholdReached(extraErrors) {
const { failedTimes, passedTimes } = this._getAttemptsResult(extraErrors);
const failedThresholdReached = failedTimes >= QUARANTINE_THRESHOLD;
const passedThresholdReached = passedTimes >= QUARANTINE_THRESHOLD;
return failedThresholdReached || passedThresholdReached;
}
isFirstAttemptSuccessful(extraErrors) {
const { failedTimes, passedTimes } = this._getAttemptsResult(extraErrors);
return failedTimes === 0 && passedTimes > 0;
}
_getAttemptsResult(extraErrors) {
let failedTimes = this.getFailedAttempts().length;
let passedTimes = this.getPassedAttempts().length;
if (extraErrors) {
if (extraErrors.length)
failedTimes += extraErrors.length;
else
passedTimes += 1;
}
return { failedTimes, passedTimes };
}
}
class TestRunController extends async_event_emitter_1.default {
constructor(test, index, proxy, screenshots, warningLog, fixtureHookController, opts) {
super();
this.test = test;
this.index = index;
this._opts = opts;
this._proxy = proxy;
this._screenshots = screenshots;
this._warningLog = warningLog;
this._fixtureHookController = fixtureHookController;
this._testRunCtor = TestRunController._getTestRunCtor(test, opts);
this.testRun = null;
this.done = false;
this._quarantine = this._opts.quarantineMode ? new Quarantine() : null;
this._disconnectionCount = 0;
}
static _getTestRunCtor(test, opts) {
if (opts.TestRunCtor)
return opts.TestRunCtor;
return test.isLegacy ? testcafe_legacy_api_1.TestRun : test_run_1.default;
}
async _createTestRun(connection) {
const screenshotCapturer = this._screenshots.createCapturerFor(this.test, this.index, this._quarantine, connection, this._warningLog);
const TestRunCtor = this._testRunCtor;
this.testRun = new TestRunCtor(this.test, connection, screenshotCapturer, this._warningLog, this._opts);
this._screenshots.addTestRun(this.test, this.testRun);
if (this.testRun.addQuarantineInfo)
this.testRun.addQuarantineInfo(this._quarantine);
if (!this._quarantine || this._isFirstQuarantineAttempt()) {
await this.emit('test-run-create', {
testRun: this.testRun,
legacy: TestRunCtor === testcafe_legacy_api_1.TestRun,
test: this.test,
index: this.index,
quarantine: this._quarantine,
});
}
return this.testRun;
}
async _endQuarantine() {
if (this._quarantine.attempts.length > 1)
this.testRun.unstable = this._quarantine.getPassedAttempts().length > 0;
await this._emitTestRunDone();
}
_shouldKeepInQuarantine() {
const errors = this.testRun.errs;
const hasErrors = !!errors.length;
const attempts = this._quarantine.attempts;
const isFirstAttempt = this._isFirstQuarantineAttempt();
attempts.push(errors);
return isFirstAttempt ? hasErrors : !this._quarantine.isThresholdReached();
}
_isFirstQuarantineAttempt() {
return !!this._quarantine && !this._quarantine.attempts.length;
}
async _keepInQuarantine() {
await this._restartTest();
}
async _restartTest() {
await this.emit('test-run-restart');
}
async _testRunDoneInQuarantineMode() {
if (this._shouldKeepInQuarantine())
await this._keepInQuarantine();
else
await this._endQuarantine();
}
async _testRunDone() {
if (this._quarantine)
await this._testRunDoneInQuarantineMode();
else
await this._emitTestRunDone();
}
async _emitActionStart(args) {
await this.emit('test-action-start', args);
}
async _emitActionDone(args) {
await this.emit('test-action-done', args);
}
async _emitTestRunDone() {
// NOTE: we should report test run completion in order they were completed in browser.
// To keep a sequence after fixture hook execution we use completion queue.
await this._fixtureHookController.runFixtureAfterHookIfNecessary(this.testRun);
this.done = true;
await this.emit('test-run-done');
}
async _emitTestRunStart() {
await this.emit('test-run-start');
}
async _testRunBeforeDone() {
let raiseEvent = !this._quarantine;
if (!raiseEvent) {
const isSuccessfulQuarantineFirstAttempt = this._isFirstQuarantineAttempt() && !this.testRun.errs.length;
const isAttemptsThresholdReached = this._quarantine.isThresholdReached(this.testRun.errs);
raiseEvent = isSuccessfulQuarantineFirstAttempt || isAttemptsThresholdReached;
}
if (raiseEvent)
await this.emit('test-run-before-done');
}
_testRunDisconnected(connection) {
this._disconnectionCount++;
const disconnectionThresholdExceedeed = this._disconnectionCount >= DISCONNECT_THRESHOLD;
return connection
.processDisconnection(disconnectionThresholdExceedeed)
.then(() => {
return this._restartTest();
});
}
_assignTestRunEvents(testRun, connection) {
testRun.on('action-start', async (args) => this._emitActionStart(Object.assign(args, { testRun })));
testRun.on('action-done', async (args) => this._emitActionDone(Object.assign(args, { testRun })));
testRun.once('start', async () => this._emitTestRunStart());
testRun.once('ready', async () => {
if (!this._quarantine || this._isFirstQuarantineAttempt())
await this.emit('test-run-ready');
});
testRun.once('before-done', () => this._testRunBeforeDone());
testRun.once('done', () => this._testRunDone());
testRun.once('disconnected', () => this._testRunDisconnected(connection));
}
get blocked() {
return this._fixtureHookController.isTestBlocked(this.test);
}
async start(connection) {
const testRun = await this._createTestRun(connection);
const hookOk = await this._fixtureHookController.runFixtureBeforeHookIfNecessary(testRun);
if (this.test.skip || !hookOk) {
await this.emit('test-run-start');
await this._emitTestRunDone();
return null;
}
this._assignTestRunEvents(testRun, connection);
testRun.start();
return session_controller_1.default.getSessionUrl(testRun, this._proxy);
}
}
exports.default = TestRunController;
module.exports = exports.default;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-run-controller.js","sourceRoot":"","sources":["../../src/runner/test-run-controller.ts"],"names":[],"mappings":"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","sourcesContent":["import AsyncEventEmitter from '../utils/async-event-emitter';\n//@ts-ignore\nimport { TestRun as LegacyTestRun } from 'testcafe-legacy-api';\nimport TestRun from '../test-run';\nimport SessionController from '../test-run/session-controller';\nimport BrowserConnection from '../browser/connection';\nimport { Proxy } from 'testcafe-hammerhead';\nimport Test from '../api/structure/test';\nimport Screenshots from '../screenshots';\nimport WarningLog from '../notifications/warning-log';\nimport FixtureHookController from './fixture-hook-controller';\nimport { Dictionary } from '../configuration/interfaces';\nimport { ActionEventArg } from './interfaces';\nimport TestRunErrorFormattableAdapter from '../errors/test-run/formattable-adapter';\n\nconst QUARANTINE_THRESHOLD = 3;\nconst DISCONNECT_THRESHOLD = 3;\n\ninterface AttemptResult {\n    failedTimes: number;\n    passedTimes: number;\n}\n\nclass Quarantine {\n    public attempts: TestRunErrorFormattableAdapter[][];\n\n    public constructor () {\n        this.attempts = [];\n    }\n\n    public getFailedAttempts (): TestRunErrorFormattableAdapter[][] {\n        return this.attempts.filter(errors => !!errors.length);\n    }\n\n    public getPassedAttempts (): TestRunErrorFormattableAdapter[][] {\n        return this.attempts.filter(errors => errors.length === 0);\n    }\n\n    public getNextAttemptNumber (): number {\n        return this.attempts.length + 1;\n    }\n\n    public isThresholdReached (extraErrors?: TestRunErrorFormattableAdapter[]): boolean {\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    public isFirstAttemptSuccessful (extraErrors: TestRunErrorFormattableAdapter[]): boolean {\n        const { failedTimes, passedTimes } = this._getAttemptsResult(extraErrors);\n\n        return failedTimes === 0 && passedTimes > 0;\n    }\n\n    private _getAttemptsResult (extraErrors?: TestRunErrorFormattableAdapter[]): AttemptResult {\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    private readonly _quarantine: null | Quarantine;\n    private _disconnectionCount: number;\n    private readonly _proxy: Proxy;\n    public readonly index: number;\n    public test: Test;\n    private readonly _opts: Dictionary<OptionValue>;\n    private _screenshots: Screenshots;\n    private readonly _warningLog: WarningLog;\n    private readonly _fixtureHookController: FixtureHookController;\n    private readonly _testRunCtor: LegacyTestRun['constructor'] | TestRun['constructor'];\n    public testRun: null | LegacyTestRun | TestRun;\n    public done: boolean;\n\n    public constructor (test: Test, index: number, proxy: Proxy, screenshots: Screenshots, warningLog: WarningLog, fixtureHookController: FixtureHookController, opts: Dictionary<OptionValue>) {\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         = this._opts.quarantineMode ? new Quarantine() : null;\n        this._disconnectionCount = 0;\n    }\n\n    private static _getTestRunCtor (test: Test, opts: Dictionary<OptionValue>): LegacyTestRun | TestRun {\n        if (opts.TestRunCtor)\n            return opts.TestRunCtor;\n\n        return (test as LegacyTestRun).isLegacy ? LegacyTestRun : TestRun;\n    }\n\n    private async _createTestRun (connection: BrowserConnection): Promise<TestRun | LegacyTestRun> {\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        this._screenshots.addTestRun(this.test, this.testRun);\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    private async _endQuarantine (): Promise<void> {\n        if ((this._quarantine as Quarantine).attempts.length > 1)\n            this.testRun.unstable = (this._quarantine as Quarantine).getPassedAttempts().length > 0;\n\n        await this._emitTestRunDone();\n    }\n\n    private _shouldKeepInQuarantine (): boolean {\n        const errors         = this.testRun.errs;\n        const hasErrors      = !!errors.length;\n        const attempts       = (this._quarantine as Quarantine).attempts;\n        const isFirstAttempt = this._isFirstQuarantineAttempt();\n\n        attempts.push(errors);\n\n        return isFirstAttempt ? hasErrors : !(this._quarantine as Quarantine).isThresholdReached();\n    }\n\n    private _isFirstQuarantineAttempt (): boolean {\n        return !!this._quarantine && !this._quarantine.attempts.length;\n    }\n\n    private async _keepInQuarantine (): Promise<void> {\n        await this._restartTest();\n    }\n\n    private async _restartTest (): Promise<void> {\n        await this.emit('test-run-restart');\n    }\n\n    private async _testRunDoneInQuarantineMode (): Promise<void> {\n        if (this._shouldKeepInQuarantine())\n            await this._keepInQuarantine();\n        else\n            await this._endQuarantine();\n    }\n\n    private async _testRunDone (): Promise<void> {\n        if (this._quarantine)\n            await this._testRunDoneInQuarantineMode();\n        else\n            await this._emitTestRunDone();\n    }\n\n    private async _emitActionStart (args: ActionEventArg): Promise<void> {\n        await this.emit('test-action-start', args);\n    }\n\n    private async _emitActionDone (args: ActionEventArg): Promise<void> {\n        await this.emit('test-action-done', args);\n    }\n\n    private async _emitTestRunDone (): Promise<void> {\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    private async _emitTestRunStart (): Promise<void> {\n        await this.emit('test-run-start');\n    }\n\n    private async _testRunBeforeDone (): Promise<void> {\n        let raiseEvent = !this._quarantine;\n\n        if (!raiseEvent) {\n            const isSuccessfulQuarantineFirstAttempt = this._isFirstQuarantineAttempt() && !this.testRun.errs.length;\n            const isAttemptsThresholdReached         = (this._quarantine as 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    private _testRunDisconnected (connection: BrowserConnection): Promise<void> {\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    private _assignTestRunEvents (testRun: TestRun | LegacyTestRun, connection: BrowserConnection): void {\n        testRun.on('action-start', async (args: ActionEventArg) => this._emitActionStart(Object.assign(args, { testRun })));\n        testRun.on('action-done', async (args: ActionEventArg) => 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    public get blocked (): boolean {\n        return this._fixtureHookController.isTestBlocked(this.test);\n    }\n\n    public async start (connection: BrowserConnection): Promise<string | null> {\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\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"]}