testcafe
Version:
Automated browser testing for the modern web development stack.
203 lines • 37 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 lodash_1 = require("lodash");
const async_event_emitter_1 = __importDefault(require("../utils/async-event-emitter"));
const test_run_controller_1 = __importDefault(require("./test-run-controller"));
const session_controller_1 = __importDefault(require("../test-run/session-controller"));
const browser_job_result_1 = __importDefault(require("./browser-job-result"));
const test_run_hook_controller_1 = __importDefault(require("./test-run-hook-controller"));
var BrowserJobStatus;
(function (BrowserJobStatus) {
BrowserJobStatus[BrowserJobStatus["initialized"] = 0] = "initialized";
BrowserJobStatus[BrowserJobStatus["starting"] = 1] = "starting";
BrowserJobStatus[BrowserJobStatus["started"] = 2] = "started";
})(BrowserJobStatus || (BrowserJobStatus = {}));
class BrowserJob extends async_event_emitter_1.default {
constructor({ tests, browserConnections, proxy, screenshots, warningLog, fixtureHookController, opts, messageBus, }) {
var _a;
super();
this._status = BrowserJobStatus.initialized;
this._startTime = new Date();
this._total = 0;
this._passed = 0;
this._opts = opts;
this._proxy = proxy;
this.browserConnections = browserConnections;
this._screenshots = screenshots;
this.warningLog = warningLog;
this.fixtureHookController = fixtureHookController;
this._result = null;
this._messageBus = messageBus;
this._testRunHook = new test_run_hook_controller_1.default(tests, (_a = opts.hooks) === null || _a === void 0 ? void 0 : _a.testRun);
this._testRunControllerQueue = tests.map((test, index) => this._createTestRunController(test, index));
this._disableConcurrencyQueue = {};
this._completionQueue = [];
this._reportsPending = [];
this._connectionErrorListener = (error) => this._setResult(browser_job_result_1.default.errored, error);
this._resolveWaitingLastTestInFixture = null;
this.browserConnections.map(bc => bc.once('error', this._connectionErrorListener));
}
_createTestRunController(test, index) {
const testRunController = new test_run_controller_1.default({
test,
index: index + 1,
proxy: this._proxy,
screenshots: this._screenshots,
warningLog: this.warningLog,
fixtureHookController: this.fixtureHookController,
opts: this._opts,
messageBus: this._messageBus,
testRunHook: this._testRunHook,
});
testRunController.on('test-run-create', async (testRunInfo) => {
await this.emit('test-run-create', testRunInfo);
});
testRunController.on('test-run-ready', async () => {
await this.emit('test-run-ready', testRunController);
});
testRunController.on('test-run-restart', async () => this._onTestRunRestart(testRunController));
testRunController.on('test-run-before-done', async () => {
await this.emit('test-run-before-done', testRunController);
});
testRunController.on('test-run-done', async () => this._onTestRunDone(testRunController));
testRunController.on('test-action-done', async (args) => {
await this.emit('test-action-done', args);
});
return testRunController;
}
async _setResult(status, data) {
if (this._result)
return;
this._result = { status, data };
this.browserConnections.forEach(bc => bc.removeListener('error', this._connectionErrorListener));
await Promise.all(this.browserConnections.map(bc => bc.reportJobResult(this._result.status, this._result.data)));
}
_addToCompletionQueue(testRunInfo) {
this._completionQueue.push(testRunInfo);
}
_removeFromCompletionQueue(testRunInfo) {
(0, lodash_1.pull)(this._completionQueue, testRunInfo);
}
async _onTestRunRestart(testRunController) {
const conectionId = testRunController.testRun.browserConnection.id;
this._removeFromCompletionQueue(testRunController);
this._getTestControllerQueue(conectionId).unshift(testRunController);
await this.emit('test-run-restart', testRunController);
}
async _onTestRunDone(testRunController) {
testRunController.testRun.finishTime = new Date();
this._total++;
if (!testRunController.testRun.errs.length)
this._passed++;
while (this._completionQueue.length && this._completionQueue[0].done) {
testRunController = this._completionQueue.shift();
await this.emit('test-run-done', testRunController.testRun);
(0, lodash_1.pull)(this._reportsPending, testRunController);
if (!this._reportsPending.length && this._resolveWaitingLastTestInFixture) {
this._resolveWaitingLastTestInFixture();
this._resolveWaitingLastTestInFixture = null;
}
}
if (!this._completionQueue.length && !this.hasQueuedTestRuns) {
if (!this._opts.live)
session_controller_1.default.closeSession(testRunController.testRun);
this
._setResult(browser_job_result_1.default.done, { total: this._total, passed: this._passed })
.then(() => this.emit('done'));
}
}
async _isNextTestRunAvailable(testRunController) {
// NOTE: event task start is currently executing,
// so test run is temporary blocked
if (this._status === BrowserJobStatus.starting)
return false;
// NOTE: before hook for test run fixture is currently
// executing, so test run is temporary blocked
const isBlocked = testRunController.blocked;
const isConcurrency = this._opts.concurrency > 1;
const hasIncompleteTestRuns = this._completionQueue.some(controller => !controller.done);
const needWaitLastTestInFixture = this._reportsPending.some(controller => controller.test.fixture !== testRunController.test.fixture);
if (isBlocked || (hasIncompleteTestRuns || needWaitLastTestInFixture) && !isConcurrency) {
const disablePageReloads = testRunController.test.disablePageReloads ||
this._opts.disablePageReloads && testRunController.test.disablePageReloads !== false;
if (!needWaitLastTestInFixture || !disablePageReloads)
return false;
// NOTE: if we have `disablePageReloads` enabled and the next test is from next
// fixture, then we need to wait until all reporters finished to prevent
// redirecting to the `idle` page
await new Promise(resolve => {
this._resolveWaitingLastTestInFixture = resolve;
});
}
return true;
}
_getTestControllerQueue(connectionId) {
var _a;
if ((_a = this._disableConcurrencyQueue[connectionId]) === null || _a === void 0 ? void 0 : _a.length)
return this._disableConcurrencyQueue[connectionId];
return this._testRunControllerQueue;
}
_getNextFixtureFirstTestIndex(fixtureId) {
const nextFixtureFirstTestIndex = this._testRunControllerQueue.findIndex(testRun => { var _a; return ((_a = testRun.test.fixture) === null || _a === void 0 ? void 0 : _a.id) !== fixtureId; });
return nextFixtureFirstTestIndex === -1 ? this._testRunControllerQueue.length : nextFixtureFirstTestIndex;
}
_updateTestControllerQueues({ test }, connectionId) {
var _a, _b;
if (!test.disableConcurrency || ((_a = this._disableConcurrencyQueue[connectionId]) === null || _a === void 0 ? void 0 : _a.length))
return;
this._disableConcurrencyQueue[connectionId] = this._testRunControllerQueue.splice(0, this._getNextFixtureFirstTestIndex((_b = test.fixture) === null || _b === void 0 ? void 0 : _b.id));
}
// API
get hasQueuedTestRuns() {
if (this._testRunControllerQueue.length)
return true;
for (const connectionId in this._disableConcurrencyQueue) {
if (this._disableConcurrencyQueue[connectionId].length)
return true;
}
return false;
}
get currentTestRun() {
return this._completionQueue.length ? this._completionQueue[0].testRun : null;
}
async popNextTestRunInfo(connection) {
while (this.hasQueuedTestRuns) {
const currentQueue = this._getTestControllerQueue(connection.id);
const testRunController = currentQueue[0];
if (!testRunController)
break;
const isNextTestRunAvailable = await this._isNextTestRunAvailable(testRunController);
if (!isNextTestRunAvailable)
break;
this._reportsPending.push(testRunController);
currentQueue.shift();
this._updateTestControllerQueues(testRunController, connection.id);
this._addToCompletionQueue(testRunController);
if (this._status === BrowserJobStatus.initialized) {
this._status = BrowserJobStatus.starting;
this._startTime = new Date();
await this.emit('start', this._startTime);
this._status = BrowserJobStatus.started;
}
const testRunUrl = await testRunController.start(connection, this._startTime);
if (testRunUrl) {
return {
testRunId: testRunController.testRun.id,
url: testRunUrl,
};
}
}
return null;
}
abort() {
this.clearListeners();
this._setResult(browser_job_result_1.default.aborted);
this.browserConnections.map(bc => bc.removeJob(this));
}
}
exports.default = BrowserJob;
module.exports = exports.default;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"browser-job.js","sourceRoot":"","sources":["../../src/runner/browser-job.ts"],"names":[],"mappings":"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","sourcesContent":["import { pull as remove } from 'lodash';\nimport AsyncEventEmitter from '../utils/async-event-emitter';\nimport TestRunController from './test-run-controller';\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 BrowserJobResult from './browser-job-result';\nimport { BrowserJobInit } from './interfaces';\nimport MessageBus from '../utils/message-bus';\nimport TestRunHookController from './test-run-hook-controller';\nimport TestRun from '../test-run';\n// @ts-ignore\nimport { TestRun as LegacyTestRun } from 'testcafe-legacy-api';\nimport { NextTestRunInfo } from '../shared/types';\n\ninterface BrowserJobResultInfo {\n    status: BrowserJobResult;\n    data?: any;\n}\n\nenum BrowserJobStatus { initialized, starting, started }\n\nexport default class BrowserJob extends AsyncEventEmitter {\n    private _status: BrowserJobStatus;\n    private _startTime: Date;\n    private _total: number;\n    private _passed: number;\n    private readonly _opts: Dictionary<OptionValue>;\n    private readonly _proxy: Proxy;\n    public readonly browserConnections: BrowserConnection[];\n    private readonly _screenshots: Screenshots;\n    public readonly warningLog: WarningLog;\n    public readonly fixtureHookController: FixtureHookController;\n    private _result: BrowserJobResultInfo | null;\n    private _testRunControllerQueue: TestRunController[];\n    private readonly _reportsPending: TestRunController[];\n    private readonly _connectionErrorListener: (error: Error) => void;\n    private readonly _completionQueue: TestRunController[];\n    private _resolveWaitingLastTestInFixture: Function | null;\n    private readonly _messageBus: MessageBus;\n    private readonly _testRunHook: TestRunHookController;\n    private readonly _disableConcurrencyQueue: Dictionary<TestRunController[]>;\n\n    public constructor ({\n        tests,\n        browserConnections,\n        proxy,\n        screenshots,\n        warningLog,\n        fixtureHookController,\n        opts,\n        messageBus,\n    }: BrowserJobInit) {\n        super();\n\n        this._status = BrowserJobStatus.initialized;\n        this._startTime = new Date();\n\n        this._total                = 0;\n        this._passed               = 0;\n        this._opts                 = opts;\n        this._proxy                = proxy;\n        this.browserConnections    = browserConnections;\n        this._screenshots          = screenshots;\n        this.warningLog            = warningLog;\n        this.fixtureHookController = fixtureHookController;\n        this._result               = null;\n        this._messageBus           = messageBus;\n        this._testRunHook          = new TestRunHookController(tests, (opts.hooks as GlobalHooks)?.testRun);\n\n        this._testRunControllerQueue = tests.map((test, index) => this._createTestRunController(test, index));\n\n        this._disableConcurrencyQueue = {};\n        this._completionQueue         = [];\n        this._reportsPending          = [];\n\n        this._connectionErrorListener = (error: Error) => this._setResult(BrowserJobResult.errored, error);\n\n        this._resolveWaitingLastTestInFixture = null;\n\n        this.browserConnections.map(bc => bc.once('error', this._connectionErrorListener));\n    }\n\n    private _createTestRunController (test: Test, index: number): TestRunController {\n        const testRunController = new TestRunController({\n            test,\n            index:                 index + 1,\n            proxy:                 this._proxy,\n            screenshots:           this._screenshots,\n            warningLog:            this.warningLog,\n            fixtureHookController: this.fixtureHookController,\n            opts:                  this._opts,\n            messageBus:            this._messageBus,\n            testRunHook:           this._testRunHook,\n        });\n\n        testRunController.on('test-run-create', async testRunInfo => {\n            await this.emit('test-run-create', testRunInfo);\n        });\n        testRunController.on('test-run-ready', async () => {\n            await this.emit('test-run-ready', testRunController);\n        });\n        testRunController.on('test-run-restart', async () => this._onTestRunRestart(testRunController));\n        testRunController.on('test-run-before-done', async () => {\n            await this.emit('test-run-before-done', testRunController);\n        });\n        testRunController.on('test-run-done', async () => this._onTestRunDone(testRunController));\n\n        testRunController.on('test-action-done', async args => {\n            await this.emit('test-action-done', args);\n        });\n\n        return testRunController;\n    }\n\n    private async _setResult (status: BrowserJobResult, data?: any): Promise<void> {\n        if (this._result)\n            return;\n\n        this._result = { status, data };\n\n        this.browserConnections.forEach(bc => bc.removeListener('error', this._connectionErrorListener));\n\n        await Promise.all(this.browserConnections.map(bc => bc.reportJobResult((this._result as BrowserJobResultInfo).status, (this._result as BrowserJobResultInfo).data)));\n    }\n\n    private _addToCompletionQueue (testRunInfo: TestRunController): void {\n        this._completionQueue.push(testRunInfo);\n    }\n\n    private _removeFromCompletionQueue (testRunInfo: TestRunController): void {\n        remove(this._completionQueue, testRunInfo);\n    }\n\n    private async _onTestRunRestart (testRunController: TestRunController): Promise<void> {\n        const conectionId = testRunController.testRun.browserConnection.id;\n\n        this._removeFromCompletionQueue(testRunController);\n        this._getTestControllerQueue(conectionId).unshift(testRunController);\n\n        await this.emit('test-run-restart', testRunController);\n    }\n\n    private async _onTestRunDone (testRunController: TestRunController): Promise<void> {\n        testRunController.testRun.finishTime = new Date();\n\n        this._total++;\n\n        if (!testRunController.testRun.errs.length)\n            this._passed++;\n\n        while (this._completionQueue.length && this._completionQueue[0].done) {\n            testRunController = this._completionQueue.shift() as TestRunController;\n\n            await this.emit('test-run-done', testRunController.testRun);\n\n            remove(this._reportsPending, testRunController);\n\n            if (!this._reportsPending.length && this._resolveWaitingLastTestInFixture) {\n                this._resolveWaitingLastTestInFixture();\n\n                this._resolveWaitingLastTestInFixture = null;\n            }\n        }\n\n        if (!this._completionQueue.length && !this.hasQueuedTestRuns) {\n            if (!this._opts.live)\n                SessionController.closeSession(testRunController.testRun);\n\n            this\n                ._setResult(BrowserJobResult.done, { total: this._total, passed: this._passed })\n                .then(() => this.emit('done'));\n        }\n    }\n\n    private async _isNextTestRunAvailable (testRunController: TestRunController): Promise<boolean> {\n        // NOTE: event task start is currently executing,\n        // so test run is temporary blocked\n        if (this._status === BrowserJobStatus.starting)\n            return false;\n\n        // NOTE: before hook for test run fixture is currently\n        // executing, so test run is temporary blocked\n        const isBlocked                 = testRunController.blocked;\n        const isConcurrency             = this._opts.concurrency as number > 1;\n        const hasIncompleteTestRuns     = this._completionQueue.some(controller => !controller.done);\n        const needWaitLastTestInFixture = this._reportsPending.some(controller => controller.test.fixture !== testRunController.test.fixture);\n\n        if (isBlocked || (hasIncompleteTestRuns || needWaitLastTestInFixture) && !isConcurrency) {\n            const disablePageReloads = testRunController.test.disablePageReloads ||\n                this._opts.disablePageReloads && testRunController.test.disablePageReloads !== false;\n\n            if (!needWaitLastTestInFixture || !disablePageReloads)\n                return false;\n\n            // NOTE: if we have `disablePageReloads` enabled and the next test is from next\n            // fixture, then we need to wait until all reporters finished to prevent\n            // redirecting to the `idle` page\n            await new Promise(resolve => {\n                this._resolveWaitingLastTestInFixture = resolve;\n            });\n        }\n\n        return true;\n    }\n\n    private _getTestControllerQueue (connectionId: string): TestRunController[] {\n        if (this._disableConcurrencyQueue[connectionId]?.length)\n            return this._disableConcurrencyQueue[connectionId];\n\n        return this._testRunControllerQueue;\n    }\n\n    private _getNextFixtureFirstTestIndex ( fixtureId: string ): number {\n        const nextFixtureFirstTestIndex = this._testRunControllerQueue.findIndex(testRun => testRun.test.fixture?.id !== fixtureId);\n\n        return nextFixtureFirstTestIndex === -1 ? this._testRunControllerQueue.length : nextFixtureFirstTestIndex;\n    }\n\n    private _updateTestControllerQueues ({ test }: TestRunController, connectionId: string): void {\n        if (!test.disableConcurrency || this._disableConcurrencyQueue[connectionId]?.length)\n            return;\n\n        this._disableConcurrencyQueue[connectionId] = this._testRunControllerQueue.splice(0, this._getNextFixtureFirstTestIndex(test.fixture?.id as string));\n    }\n\n    // API\n    public get hasQueuedTestRuns (): boolean {\n        if (this._testRunControllerQueue.length)\n            return true;\n\n        for (const connectionId in this._disableConcurrencyQueue) {\n            if (this._disableConcurrencyQueue[connectionId].length)\n                return true;\n        }\n\n        return false;\n    }\n\n    public get currentTestRun (): LegacyTestRun | TestRun | null {\n        return this._completionQueue.length ? this._completionQueue[0].testRun : null;\n    }\n\n    public async popNextTestRunInfo (connection: BrowserConnection): Promise<NextTestRunInfo | null> {\n        while (this.hasQueuedTestRuns) {\n            const currentQueue      = this._getTestControllerQueue(connection.id);\n            const testRunController = currentQueue[0];\n\n            if (!testRunController)\n                break;\n\n            const isNextTestRunAvailable = await this._isNextTestRunAvailable(testRunController);\n\n            if (!isNextTestRunAvailable)\n                break;\n\n            this._reportsPending.push(testRunController);\n            currentQueue.shift();\n            this._updateTestControllerQueues(testRunController, connection.id);\n            this._addToCompletionQueue(testRunController);\n\n            if (this._status === BrowserJobStatus.initialized) {\n                this._status = BrowserJobStatus.starting;\n                this._startTime   = new Date();\n\n                await this.emit('start', this._startTime);\n\n                this._status = BrowserJobStatus.started;\n            }\n\n            const testRunUrl = await testRunController.start(connection, this._startTime);\n\n            if (testRunUrl) {\n                return {\n                    testRunId: testRunController.testRun.id,\n                    url:       testRunUrl,\n                };\n            }\n        }\n\n        return null;\n    }\n\n    public abort (): void {\n        this.clearListeners();\n        this._setResult(BrowserJobResult.aborted);\n        this.browserConnections.map(bc => bc.removeJob(this));\n    }\n}\n"]}