testcafe
Version:
Automated browser testing for the modern web development stack.
297 lines • 57.3 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 debug_1 = __importDefault(require("debug"));
const pretty_hrtime_1 = __importDefault(require("pretty-hrtime"));
const compiler_1 = __importDefault(require("../compiler"));
const connection_1 = __importDefault(require("../browser/connection"));
const browser_set_1 = __importDefault(require("./browser-set"));
const runtime_1 = require("../errors/runtime");
const types_1 = require("../errors/types");
const tested_app_1 = __importDefault(require("./tested-app"));
const parse_file_list_1 = __importDefault(require("../utils/parse-file-list"));
const load_1 = __importDefault(require("../custom-client-scripts/load"));
const string_1 = require("../utils/string");
const warning_log_1 = __importDefault(require("../notifications/warning-log"));
const warning_message_1 = __importDefault(require("../notifications/warning-message"));
const guard_time_execution_1 = __importDefault(require("../utils/guard-time-execution"));
const async_filter_1 = __importDefault(require("../utils/async-filter"));
const wrap_test_function_1 = __importDefault(require("../api/wrap-test-function"));
const type_assertions_1 = require("../errors/runtime/type-assertions");
const testcafe_hammerhead_1 = require("testcafe-hammerhead");
const assert_type_1 = __importDefault(require("../api/request-hooks/assert-type"));
const option_names_1 = __importDefault(require("../configuration/option-names"));
const status_1 = __importDefault(require("../browser/connection/gateway/status"));
const DEBUG_SCOPE = 'testcafe:bootstrapper';
function isPromiseError(value) {
return value.error !== void 0;
}
class Bootstrapper {
constructor({ browserConnectionGateway, messageBus, configuration }) {
this.browserConnectionGateway = browserConnectionGateway;
this.concurrency = 1;
this.sources = [];
this.browsers = [];
this.reporters = [];
this.filter = void 0;
this.appCommand = void 0;
this.appInitDelay = void 0;
this.tsConfigPath = void 0;
this.clientScripts = [];
this.disableMultipleWindows = false;
this.compilerOptions = void 0;
this.debugLogger = (0, debug_1.default)(DEBUG_SCOPE);
this.warningLog = new warning_log_1.default(null, warning_log_1.default.createAddWarningCallback(messageBus));
this.messageBus = messageBus;
this.configuration = configuration;
this.TESTS_COMPILATION_UPPERBOUND = 60;
}
static _getBrowserName(browser) {
if (browser instanceof connection_1.default)
return browser.browserInfo.browserName;
return browser.browserName;
}
static _splitBrowserInfo(browserInfo) {
const remotes = [];
const automated = [];
browserInfo.forEach(browser => {
if (browser instanceof connection_1.default)
remotes.push(browser);
else
automated.push(browser);
});
return { remotes, automated };
}
_createAutomatedConnections(browserInfo) {
if (!browserInfo)
return [];
return browserInfo.map(browser => (0, lodash_1.times)(this.concurrency, () => {
const options = {
disableMultipleWindows: this.disableMultipleWindows,
developmentMode: this.configuration.getOption(option_names_1.default.developmentMode),
nativeAutomation: !this.configuration.getOption(option_names_1.default.disableNativeAutomation),
};
const connection = new connection_1.default(this.browserConnectionGateway, Object.assign({}, browser), false, options, this.messageBus);
connection.initialize();
return connection;
}));
}
_getBrowserSetOptions() {
return {
concurrency: this.concurrency,
browserInitTimeout: this.browserInitTimeout,
warningLog: this.warningLog,
};
}
async _setupProxy() {
if (this.browserConnectionGateway.status === status_1.default.initialized)
return;
await this.configuration.calculateHostname({ nativeAutomation: !this.configuration.getOption(option_names_1.default.disableNativeAutomation) });
this.browserConnectionGateway.initialize(this.configuration.startOptions);
}
_hasNotSupportedBrowserInNativeAutomation(browserInfos) {
return browserInfos.some(browserInfo => {
return !browserInfo.provider.supportNativeAutomation();
});
}
getBrowsersWithUserProfileEnabled(browserInfos) {
return browserInfos.filter(browserInfo => { var _a; return (_a = browserInfo.browserOption) === null || _a === void 0 ? void 0 : _a.userProfile; });
}
_disableNativeAutomationIfNecessary(remotes, automated) {
// NOTE: CDP API allows connecting only for the local browser. So, the 'remote' browser cannot be run in the 'nativeAutomation' mode.
// However, sometimes in tests or TestCafe Studio Recorder, we use the 'remote' browser connection as a local one.
const containsNotAutomatedRemotes = remotes.some(remote => !remote.isNativeAutomationEnabled());
if (remotes.length && containsNotAutomatedRemotes || this._hasNotSupportedBrowserInNativeAutomation(automated))
this.configuration.mergeOptions({ disableNativeAutomation: true });
}
async _getBrowserConnections(browserInfo) {
const { automated, remotes } = Bootstrapper._splitBrowserInfo(browserInfo);
if (remotes && remotes.length % this.concurrency)
throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotDivideRemotesCountByConcurrency);
this._disableNativeAutomationIfNecessary(remotes, automated);
this._validateUserProfileOptionInNativeAutomation(automated);
await this._setupProxy();
let browserConnections = this._createAutomatedConnections(automated);
remotes.forEach(remoteConnection => {
remoteConnection.messageBus = this.messageBus;
remoteConnection.initMessageBus();
});
browserConnections = browserConnections.concat((0, lodash_1.chunk)(remotes, this.concurrency));
return browser_set_1.default.from(browserConnections, this._getBrowserSetOptions());
}
_validateUserProfileOptionInNativeAutomation(automated) {
const isNativeAutomation = !this.configuration.getOption(option_names_1.default.disableNativeAutomation);
const browsersWithUserProfile = this.getBrowsersWithUserProfileEnabled(automated);
if (isNativeAutomation && browsersWithUserProfile.length) {
const browsers = browsersWithUserProfile.map(b => b.alias).join(', ');
throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.setUserProfileInNativeAutomation, browsers);
}
}
async _filterTests(tests, predicate) {
return (0, async_filter_1.default)(tests, test => {
const testFixture = test.fixture;
return predicate(test.name, testFixture.name, testFixture.path, test.meta, testFixture.meta);
});
}
async _compileTests({ sourceList, compilerOptions }) {
const baseUrl = this.configuration.getOption(option_names_1.default.baseUrl);
const esm = this.configuration.getOption(option_names_1.default.esm);
const compiler = new compiler_1.default(sourceList, compilerOptions, { baseUrl, esm });
return compiler.getTests();
}
_assertGlobalHooks() {
var _a, _b, _c, _d;
if (!this.hooks)
return;
if ((_a = this.hooks.fixture) === null || _a === void 0 ? void 0 : _a.before)
(0, type_assertions_1.assertType)(type_assertions_1.is.function, 'globalBefore', 'The fixture.globalBefore hook', this.hooks.fixture.before);
if ((_b = this.hooks.fixture) === null || _b === void 0 ? void 0 : _b.after)
(0, type_assertions_1.assertType)(type_assertions_1.is.function, 'globalAfter', 'The fixture.globalAfter hook', this.hooks.fixture.after);
if ((_c = this.hooks.test) === null || _c === void 0 ? void 0 : _c.before)
(0, type_assertions_1.assertType)(type_assertions_1.is.function, 'globalBefore', 'The test.globalBefore hook', this.hooks.test.before);
if ((_d = this.hooks.test) === null || _d === void 0 ? void 0 : _d.after)
(0, type_assertions_1.assertType)(type_assertions_1.is.function, 'globalAfter', 'The test.globalAfter hook', this.hooks.test.after);
if (this.hooks.request)
(0, assert_type_1.default)((0, lodash_1.flattenDeep)((0, lodash_1.castArray)(this.hooks.request)));
}
_setGlobalHooksToTests(tests) {
var _a, _b, _c, _d;
if (!this.hooks)
return;
this._assertGlobalHooks();
const fixtureBefore = ((_a = this.hooks.fixture) === null || _a === void 0 ? void 0 : _a.before) || null;
const fixtureAfter = ((_b = this.hooks.fixture) === null || _b === void 0 ? void 0 : _b.after) || null;
const testBefore = ((_c = this.hooks.test) === null || _c === void 0 ? void 0 : _c.before) ? (0, wrap_test_function_1.default)(this.hooks.test.before) : null;
const testAfter = ((_d = this.hooks.test) === null || _d === void 0 ? void 0 : _d.after) ? (0, wrap_test_function_1.default)(this.hooks.test.after) : null;
const request = this.hooks.request || [];
tests.forEach(item => {
if (item.fixture) {
item.fixture.globalBeforeFn = item.fixture.globalBeforeFn || fixtureBefore;
item.fixture.globalAfterFn = item.fixture.globalAfterFn || fixtureAfter;
}
item.globalBeforeFn = testBefore;
item.globalAfterFn = testAfter;
item.requestHooks = (0, lodash_1.union)((0, lodash_1.flattenDeep)((0, lodash_1.castArray)(request)), item.requestHooks);
});
}
async _getTests(id) {
const cwd = process.cwd();
const sourceList = await (0, parse_file_list_1.default)(this.sources, cwd);
if (!sourceList.length)
throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.testFilesNotFound, cwd, (0, string_1.getConcatenatedValuesString)(this.sources, '\n', ''));
let tests = await (0, guard_time_execution_1.default)(async () => await this._compileTests({ sourceList, compilerOptions: this.compilerOptions, runnableConfigurationId: id }), elapsedTime => {
this.debugLogger(`tests compilation took ${(0, pretty_hrtime_1.default)(elapsedTime)}`);
const [elapsedSeconds] = elapsedTime;
if (elapsedSeconds > this.TESTS_COMPILATION_UPPERBOUND)
this.warningLog.addWarning(warning_message_1.default.testsCompilationTakesTooLong, (0, pretty_hrtime_1.default)(elapsedTime));
});
const testsWithOnlyFlag = tests.filter(test => test.only);
if (testsWithOnlyFlag.length)
tests = testsWithOnlyFlag;
if (!tests.length)
throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.noTestsToRun);
if (this.filter)
tests = await this._filterTests(tests, this.filter);
if (!tests.length)
throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.noTestsToRunDueFiltering);
this._setGlobalHooksToTests(tests);
return tests;
}
async _startTestedApp() {
if (!this.appCommand)
return void 0;
const testedApp = new tested_app_1.default();
await testedApp.start(this.appCommand, this.appInitDelay);
return testedApp;
}
async _canUseParallelBootstrapping(browserInfo) {
const isLocalPromises = browserInfo.map(browser => browser.provider.isLocalBrowser(void 0, Bootstrapper._getBrowserName(browser)));
const isLocalBrowsers = await Promise.all(isLocalPromises);
return isLocalBrowsers.every(result => result);
}
async _bootstrapSequence(browserInfo, id) {
const tests = await this._getTests(id);
const testedApp = await this._startTestedApp();
const browserSet = await this._getBrowserConnections(browserInfo);
return { tests, testedApp, browserSet };
}
_wrapBootstrappingPromise(promise) {
return promise
.then(result => ({ error: void 0, result }))
.catch(error => ({ result: void 0, error }));
}
async _getBootstrappingError(browserSetStatus, testsStatus, testedAppStatus) {
if (!isPromiseError(browserSetStatus))
await browserSetStatus.result.dispose();
if (!isPromiseError(browserSetStatus) && !isPromiseError(testedAppStatus) && testedAppStatus.result)
await testedAppStatus.result.kill();
if (isPromiseError(testsStatus))
return testsStatus.error;
if (isPromiseError(testedAppStatus))
return testedAppStatus.error;
if (isPromiseError(browserSetStatus))
return browserSetStatus.error;
return new Error('Unexpected call');
}
_getBootstrappingPromises(arg) {
const result = {};
for (const k in arg)
result[k] = this._wrapBootstrappingPromise(arg[k]);
return result;
}
async _assertNativeAutomationForLegacyTests(resources) {
var _a;
const isNativeAutomation = !this.configuration.getOption(option_names_1.default.disableNativeAutomation);
if (!isNativeAutomation)
return;
const hasLegacyTests = resources.tests.some(test => test.isLegacy);
if (!hasLegacyTests)
return;
await resources.browserSet.dispose();
await ((_a = resources.testedApp) === null || _a === void 0 ? void 0 : _a.kill());
throw new runtime_1.GeneralError(types_1.RUNTIME_ERRORS.cannotRunLegacyTestsInNativeAutomationMode);
}
async _bootstrapParallel(browserInfo, id) {
const bootstrappingPromises = {
browserSet: this._getBrowserConnections(browserInfo),
tests: this._getTests(id),
app: this._startTestedApp(),
};
const bootstrappingResultPromises = this._getBootstrappingPromises(bootstrappingPromises);
const bootstrappingResults = await Promise.all([
bootstrappingResultPromises.browserSet,
bootstrappingResultPromises.tests,
bootstrappingResultPromises.app,
]);
const [browserSetResults, testResults, appResults] = bootstrappingResults;
if (isPromiseError(browserSetResults) || isPromiseError(testResults) || isPromiseError(appResults))
throw await this._getBootstrappingError(...bootstrappingResults);
const resources = {
browserSet: browserSetResults.result,
tests: testResults.result,
testedApp: appResults.result,
};
await this._assertNativeAutomationForLegacyTests(resources);
return resources;
}
// API
async createRunnableConfiguration() {
const id = (0, testcafe_hammerhead_1.generateUniqueId)();
const commonClientScripts = await (0, load_1.default)(this.clientScripts);
if (await this._canUseParallelBootstrapping(this.browsers))
return Object.assign(Object.assign({}, await this._bootstrapParallel(this.browsers, id)), { commonClientScripts, id });
return Object.assign(Object.assign({}, await this._bootstrapSequence(this.browsers, id)), { commonClientScripts, id });
}
restoreMessageBusListeners() {
const connections = this.browserConnectionGateway.getConnections();
Object.values(connections).forEach(connection => {
connection.assignTestRunStartEventListener();
});
}
}
exports.default = Bootstrapper;
module.exports = exports.default;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bootstrapper.js","sourceRoot":"","sources":["../../src/runner/bootstrapper.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    chunk,\n    times,\n    union,\n    castArray,\n    flattenDeep as flatten,\n} from 'lodash';\n\nimport debug from 'debug';\nimport prettyTime from 'pretty-hrtime';\nimport Compiler from '../compiler';\nimport BrowserConnection, { BrowserInfo } from '../browser/connection';\nimport BrowserSet from './browser-set';\nimport { GeneralError } from '../errors/runtime';\nimport { RUNTIME_ERRORS } from '../errors/types';\nimport TestedApp from './tested-app';\nimport parseFileList from '../utils/parse-file-list';\nimport loadClientScripts from '../custom-client-scripts/load';\nimport { getConcatenatedValuesString } from '../utils/string';\nimport { ReporterSource } from '../reporter/interfaces';\nimport ClientScript from '../custom-client-scripts/client-script';\nimport ClientScriptInit from '../custom-client-scripts/client-script-init';\nimport BrowserConnectionGateway from '../browser/connection/gateway';\nimport { CompilerArguments } from '../compiler/interfaces';\nimport Test from '../api/structure/test';\nimport { BootstrapperInit, BrowserSetOptions } from './interfaces';\nimport WarningLog from '../notifications/warning-log';\nimport WARNING_MESSAGES from '../notifications/warning-message';\nimport guardTimeExecution from '../utils/guard-time-execution';\nimport asyncFilter from '../utils/async-filter';\nimport Fixture from '../api/structure/fixture';\nimport MessageBus from '../utils/message-bus';\nimport wrapTestFunction from '../api/wrap-test-function';\nimport { assertType, is } from '../errors/runtime/type-assertions';\nimport { generateUniqueId } from 'testcafe-hammerhead';\nimport assertRequestHookType from '../api/request-hooks/assert-type';\nimport OPTION_NAMES from '../configuration/option-names';\nimport TestCafeConfiguration from '../configuration/testcafe-configuration';\nimport BrowserConnectionGatewayStatus from '../browser/connection/gateway/status';\n\nconst DEBUG_SCOPE = 'testcafe:bootstrapper';\n\ntype TestSource = unknown;\n\ntype BrowserInfoSource = BrowserInfo | BrowserConnection;\n\ninterface PromiseSuccess<T> {\n    result: T;\n}\n\ninterface PromiseError<E extends Error = Error> {\n    error: E;\n}\n\ninterface BasicRuntimeResources {\n    browserSet: BrowserSet;\n    tests: Test[];\n    testedApp?: TestedApp;\n}\n\ninterface RunnableConfiguration extends BasicRuntimeResources {\n    commonClientScripts: ClientScript[];\n    id: string;\n}\n\ntype PromiseResult<T, E extends Error = Error> = PromiseSuccess<T> | PromiseError<E>;\n\nfunction isPromiseError<T, E extends Error = Error> (value: PromiseResult<T, E>): value is PromiseError<E> {\n    return (value as PromiseError<E>).error !== void 0;\n}\n\ninterface SeparatedBrowserInfo {\n    remotes: BrowserConnection[];\n    automated: BrowserInfo[];\n}\n\ntype PromiseCollection<T> = {\n    [K in keyof T]: Promise<T[K]>\n}\n\ntype ResultCollection<T> = { [P in keyof T]: PromiseResult<T[P]> };\n\nexport default class Bootstrapper {\n    private readonly browserConnectionGateway: BrowserConnectionGateway;\n    public concurrency: number;\n    public sources: TestSource[];\n    public browsers: BrowserInfoSource[];\n    public reporters: ReporterSource[];\n    public filter?: FilterFunction;\n    public appCommand?: string;\n    public appInitDelay?: number;\n    public tsConfigPath?: string;\n    public clientScripts: ClientScriptInit[];\n    public disableMultipleWindows: boolean;\n    public compilerOptions?: CompilerOptions;\n    public browserInitTimeout?: number;\n    public hooks?: GlobalHooks;\n    public configuration: TestCafeConfiguration;\n    private readonly debugLogger: debug.Debugger;\n    private readonly warningLog: WarningLog;\n    private readonly messageBus: MessageBus;\n\n    private readonly TESTS_COMPILATION_UPPERBOUND: number;\n\n    public constructor ({ browserConnectionGateway, messageBus, configuration }: BootstrapperInit) {\n        this.browserConnectionGateway = browserConnectionGateway;\n        this.concurrency              = 1;\n        this.sources                  = [];\n        this.browsers                 = [];\n        this.reporters                = [];\n        this.filter                   = void 0;\n        this.appCommand               = void 0;\n        this.appInitDelay             = void 0;\n        this.tsConfigPath             = void 0;\n        this.clientScripts            = [];\n        this.disableMultipleWindows   = false;\n        this.compilerOptions          = void 0;\n        this.debugLogger              = debug(DEBUG_SCOPE);\n        this.warningLog               = new WarningLog(null, WarningLog.createAddWarningCallback(messageBus));\n        this.messageBus               = messageBus;\n        this.configuration            = configuration;\n\n        this.TESTS_COMPILATION_UPPERBOUND = 60;\n    }\n\n    private static _getBrowserName (browser: BrowserInfoSource): string {\n        if (browser instanceof BrowserConnection)\n            return browser.browserInfo.browserName;\n\n        return browser.browserName;\n    }\n\n    private static _splitBrowserInfo (browserInfo: BrowserInfoSource[]): SeparatedBrowserInfo {\n        const remotes: BrowserConnection[] = [];\n        const automated: BrowserInfo[]     = [];\n\n        browserInfo.forEach(browser => {\n            if (browser instanceof BrowserConnection)\n                remotes.push(browser);\n            else\n                automated.push(browser);\n        });\n\n        return { remotes, automated };\n    }\n\n    private _createAutomatedConnections (browserInfo: BrowserInfo[]): BrowserConnection[][] {\n        if (!browserInfo)\n            return [];\n\n        return browserInfo.map(browser => times(this.concurrency, () => {\n            const options = {\n                disableMultipleWindows: this.disableMultipleWindows,\n                developmentMode:        this.configuration.getOption(OPTION_NAMES.developmentMode) as boolean,\n                nativeAutomation:       !this.configuration.getOption(OPTION_NAMES.disableNativeAutomation),\n            };\n\n            const connection = new BrowserConnection(this.browserConnectionGateway, { ...browser }, false, options, this.messageBus);\n\n            connection.initialize();\n\n            return connection;\n        }));\n    }\n\n    private _getBrowserSetOptions (): BrowserSetOptions {\n        return {\n            concurrency:        this.concurrency,\n            browserInitTimeout: this.browserInitTimeout,\n            warningLog:         this.warningLog,\n        };\n    }\n\n    private async _setupProxy (): Promise<void> {\n        if (this.browserConnectionGateway.status === BrowserConnectionGatewayStatus.initialized)\n            return;\n\n        await this.configuration.calculateHostname({ nativeAutomation: !this.configuration.getOption(OPTION_NAMES.disableNativeAutomation) });\n\n        this.browserConnectionGateway.initialize(this.configuration.startOptions);\n    }\n\n    private _hasNotSupportedBrowserInNativeAutomation (browserInfos: BrowserInfo[]): boolean {\n        return browserInfos.some(browserInfo => {\n            return !browserInfo.provider.supportNativeAutomation();\n        });\n    }\n\n    private getBrowsersWithUserProfileEnabled (browserInfos: BrowserInfo[]): BrowserInfo[] {\n        return browserInfos.filter(browserInfo => (browserInfo.browserOption as any)?.userProfile);\n    }\n\n    private _disableNativeAutomationIfNecessary (remotes: BrowserConnection[], automated: BrowserInfo[]): void {\n        // NOTE: CDP API allows connecting only for the local browser. So, the 'remote' browser cannot be run in the 'nativeAutomation' mode.\n        // However, sometimes in tests or TestCafe Studio Recorder, we use the 'remote' browser connection as a local one.\n        const containsNotAutomatedRemotes = remotes.some(remote => !remote.isNativeAutomationEnabled());\n\n        if (remotes.length && containsNotAutomatedRemotes || this._hasNotSupportedBrowserInNativeAutomation(automated))\n            this.configuration.mergeOptions({ disableNativeAutomation: true });\n    }\n\n    private async _getBrowserConnections (browserInfo: BrowserInfoSource[]): Promise<BrowserSet> {\n        const { automated, remotes } = Bootstrapper._splitBrowserInfo(browserInfo);\n\n        if (remotes && remotes.length % this.concurrency)\n            throw new GeneralError(RUNTIME_ERRORS.cannotDivideRemotesCountByConcurrency);\n\n        this._disableNativeAutomationIfNecessary(remotes, automated);\n\n        this._validateUserProfileOptionInNativeAutomation(automated);\n\n        await this._setupProxy();\n\n        let browserConnections = this._createAutomatedConnections(automated);\n\n        remotes.forEach(remoteConnection => {\n            remoteConnection.messageBus = this.messageBus;\n\n            remoteConnection.initMessageBus();\n        });\n\n        browserConnections = browserConnections.concat(chunk(remotes, this.concurrency));\n\n        return BrowserSet.from(browserConnections, this._getBrowserSetOptions());\n    }\n\n    protected _validateUserProfileOptionInNativeAutomation (automated: BrowserInfo[]): void {\n        const isNativeAutomation      = !this.configuration.getOption(OPTION_NAMES.disableNativeAutomation);\n        const browsersWithUserProfile = this.getBrowsersWithUserProfileEnabled(automated);\n\n        if (isNativeAutomation && browsersWithUserProfile.length) {\n            const browsers = browsersWithUserProfile.map(b => b.alias).join(', ');\n\n            throw new GeneralError(RUNTIME_ERRORS.setUserProfileInNativeAutomation, browsers);\n        }\n    }\n\n    private async _filterTests (tests: Test[], predicate: FilterFunction): Promise<Test[]> {\n        return asyncFilter(tests, test => {\n            const testFixture = test.fixture as Fixture;\n\n            return predicate(\n                test.name as string,\n                testFixture.name as string,\n                testFixture.path,\n                test.meta,\n                testFixture.meta);\n        });\n    }\n\n    private async _compileTests ({ sourceList, compilerOptions }: CompilerArguments): Promise<Test[]> {\n        const baseUrl  = this.configuration.getOption(OPTION_NAMES.baseUrl) as string;\n        const esm      = this.configuration.getOption(OPTION_NAMES.esm);\n        const compiler = new Compiler(sourceList, compilerOptions, { baseUrl, esm });\n\n        return compiler.getTests();\n    }\n\n    private _assertGlobalHooks (): void {\n        if (!this.hooks)\n            return;\n\n        if (this.hooks.fixture?.before)\n            assertType(is.function, 'globalBefore', 'The fixture.globalBefore hook', this.hooks.fixture.before);\n\n        if (this.hooks.fixture?.after)\n            assertType(is.function, 'globalAfter', 'The fixture.globalAfter hook', this.hooks.fixture.after);\n\n        if (this.hooks.test?.before)\n            assertType(is.function, 'globalBefore', 'The test.globalBefore hook', this.hooks.test.before);\n\n        if (this.hooks.test?.after)\n            assertType(is.function, 'globalAfter', 'The test.globalAfter hook', this.hooks.test.after);\n\n        if (this.hooks.request)\n            assertRequestHookType(flatten(castArray(this.hooks.request)));\n    }\n\n    private _setGlobalHooksToTests (tests: Test[]): void {\n        if (!this.hooks)\n            return;\n\n        this._assertGlobalHooks();\n\n        const fixtureBefore = this.hooks.fixture?.before || null;\n        const fixtureAfter  = this.hooks.fixture?.after || null;\n        const testBefore    = this.hooks.test?.before ? wrapTestFunction(this.hooks.test.before) : null;\n        const testAfter     = this.hooks.test?.after ? wrapTestFunction(this.hooks.test.after) : null;\n        const request       = this.hooks.request || [];\n\n        tests.forEach(item => {\n            if (item.fixture) {\n                item.fixture.globalBeforeFn = item.fixture.globalBeforeFn || fixtureBefore;\n                item.fixture.globalAfterFn  = item.fixture.globalAfterFn || fixtureAfter;\n            }\n\n            item.globalBeforeFn = testBefore;\n            item.globalAfterFn  = testAfter;\n            item.requestHooks   = union(flatten(castArray(request)), item.requestHooks);\n        });\n    }\n\n    private async _getTests (id: string): Promise<Test[]> {\n        const cwd        = process.cwd();\n        const sourceList = await parseFileList(this.sources, cwd);\n\n        if (!sourceList.length)\n            throw new GeneralError(RUNTIME_ERRORS.testFilesNotFound, cwd, getConcatenatedValuesString(this.sources, '\\n', ''));\n\n        let tests = await guardTimeExecution(\n            async () => await this._compileTests({ sourceList, compilerOptions: this.compilerOptions, runnableConfigurationId: id }),\n            elapsedTime => {\n                this.debugLogger(`tests compilation took ${prettyTime(elapsedTime)}`);\n\n                const [ elapsedSeconds ] = elapsedTime;\n\n                if (elapsedSeconds > this.TESTS_COMPILATION_UPPERBOUND)\n                    this.warningLog.addWarning(WARNING_MESSAGES.testsCompilationTakesTooLong, prettyTime(elapsedTime));\n            }\n        );\n\n        const testsWithOnlyFlag = tests.filter(test => test.only);\n\n        i