"use strict"; var __rest = (this && this.__rest) || function (s, e) { var t = {}; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0) t[p] = s[p]; if (s != null && typeof Object.getOwnPropertySymbols === "function") for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) { if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i])) t[p[i]] = s[p[i]]; } return t; }; 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 is_stream_1 = require("is-stream"); const plugin_host_1 = __importDefault(require("./plugin-host")); const plugin_methods_1 = __importDefault(require("./plugin-methods")); const format_command_1 = __importDefault(require("./command/format-command")); const runtime_1 = require("../errors/runtime"); const reporter_1 = require("../utils/reporter"); const resolve_path_relatively_cwd_1 = __importDefault(require("../utils/resolve-path-relatively-cwd")); const make_dir_1 = __importDefault(require("make-dir")); const path_1 = __importDefault(require("path")); const fs_1 = __importDefault(require("fs")); const debug_1 = __importDefault(require("debug")); const debugLog = (0, debug_1.default)('testcafe:reporter'); class Reporter { constructor(plugin, messageBus, outStream, name) { this.plugin = new plugin_host_1.default(plugin, outStream, name); this.messageBus = messageBus; this.disposed = false; this.taskInfo = null; this.outStream = outStream; this._assignMessageBusEventHandlers(); } static _isSpecialStream(stream) { return stream.isTTY || stream === process.stdout || stream === process.stderr; } static _createPendingPromise() { let resolver = null; const promise = new Promise(resolve => { resolver = resolve; }); promise.resolve = resolver; return promise; } async init() { await this.dispatchToPlugin({ method: plugin_methods_1.default.init, initialObject: null, args: [{}], }); } async dispatchToPlugin({ method, initialObject, args = [] }) { try { // @ts-ignore await this.plugin[method](...args); } catch (originalError) { const uncaughtError = new runtime_1.ReporterPluginError({ name: this.plugin.name, method, originalError, }); debugLog('Plugin error: %O', uncaughtError); debugLog('Plugin error: initialObject: %O', initialObject); if (initialObject) await initialObject.emit('error', uncaughtError); else throw uncaughtError; } } _assignMessageBusEventHandlers() { const messageBus = this.messageBus; messageBus.on('warning-add', async (e) => await this._onWarningAddHandler(e)); messageBus.once('start', async (task) => await this._onceTaskStartHandler(task)); messageBus.on('test-run-start', async (testRun) => await this._onTaskTestRunStartHandler(testRun)); messageBus.on('test-run-done', async (testRun) => await this._onTaskTestRunDoneHandler(testRun)); messageBus.on('test-action-start', async (e) => await this._onTaskTestActionStart(e)); messageBus.on('test-action-done', async (e) => await this._onTaskTestActionDone(e)); messageBus.once('done', async () => await this._onceTaskDoneHandler()); } async dispose() { var _a; if (this.disposed) return Promise.resolve(); this.disposed = true; if (!(0, lodash_1.isFunction)((_a = this === null || this === void 0 ? void 0 : this.outStream) === null || _a === void 0 ? void 0 : _a.once) || Reporter._isSpecialStream(this.outStream) || !(0, is_stream_1.writable)(this.outStream)) return Promise.resolve(); const streamFinishedPromise = new Promise(resolve => { this.outStream.once('finish', resolve); this.outStream.once('error', resolve); }); this.outStream.end(); return streamFinishedPromise; } static async _ensureOutStream(outStream) { if (typeof outStream !== 'string') return outStream; const fullReporterOutputPath = (0, resolve_path_relatively_cwd_1.default)(outStream); await (0, make_dir_1.default)(path_1.default.dirname(fullReporterOutputPath)); return fs_1.default.createWriteStream(fullReporterOutputPath); } static _addDefaultReporter(reporters) { reporters.push({ name: 'spec', output: process.stdout, }); } static async getReporterPlugins(reporters = []) { if (!reporters.length) Reporter._addDefaultReporter(reporters); return Promise.all(reporters.map(async ({ name, output, options }) => { const pluginFactory = (0, reporter_1.getPluginFactory)(name); const processedName = (0, reporter_1.processReporterName)(name); const outStream = output ? await Reporter._ensureOutStream(output) : void 0; return { plugin: pluginFactory(options), name: processedName, outStream, }; })); } async _onWarningAddHandler({ message, testRun, actionId }) { await this.dispatchToPlugin({ method: plugin_methods_1.default.reportWarnings, initialObject: this.messageBus, args: [ { message, testRunId: testRun === null || testRun === void 0 ? void 0 : testRun.id, actionId, }, ], }); } //Task static _createTestItem(test, runsPerTest) { return { fixture: test.fixture, test: test, testRunIds: [], screenshotPath: null, screenshots: [], videos: [], quarantine: null, errs: [], warnings: [], unstable: false, startTime: null, testRunInfo: null, pendingRuns: runsPerTest, pendingStarts: runsPerTest, pendingTestRunDonePromise: Reporter._createPendingPromise(), pendingTestRunStartPromise: Reporter._createPendingPromise(), browsers: [], }; } static _createTestQueue(task) { const runsPerTest = task.browserConnectionGroups.length; return task.tests.map(test => Reporter._createTestItem(test, runsPerTest)); } static _createTestRunInfo(reportItem) { return { errs: (0, lodash_1.sortBy)(reportItem.errs, ['userAgent', 'code']), warnings: reportItem.warnings, durationMs: +new Date() - reportItem.startTime, unstable: reportItem.unstable, screenshotPath: reportItem.screenshotPath, screenshots: reportItem.screenshots, videos: reportItem.videos, quarantine: reportItem.quarantine, skipped: reportItem.test.skip, browsers: reportItem.browsers, testId: reportItem.test.id, fixture: { id: reportItem.fixture.id, name: reportItem.fixture.name, path: reportItem.fixture.path, meta: reportItem.fixture.meta, }, }; } _getTestItemForTestRun(taskInfo, testRun) { return (0, lodash_1.find)(taskInfo.testQueue, i => i.test === testRun.test); } async _shiftTestQueue() { if (!this.taskInfo) return; let currentFixture = null; let nextReportItem = null; let testItem = null; const testQueue = this.taskInfo.testQueue; while (testQueue.length && testQueue[0].testRunInfo) { testItem = testQueue.shift(); currentFixture = testItem.fixture; // NOTE: here we assume that tests are sorted by fixture. // Therefore, if the next report item has a different // fixture, we can report this fixture start. nextReportItem = testQueue[0]; await this.dispatchToPlugin({ method: plugin_methods_1.default.reportTestDone, initialObject: this.taskInfo.task, args: [ testItem.test.name, testItem.testRunInfo, testItem.test.meta, ], }); if (!nextReportItem || nextReportItem.fixture === currentFixture) continue; await this.dispatchToPlugin({ method: plugin_methods_1.default.reportFixtureStart, initialObject: this.taskInfo.task, args: [ nextReportItem.fixture.name, nextReportItem.fixture.path, nextReportItem.fixture.meta, ], }); } } async _resolveTestItem(taskInfo, testItem, testRun) { if (!taskInfo.task) return; if (taskInfo.task.screenshots.hasCapturedFor(testRun.test)) { testItem.screenshotPath = taskInfo.task.screenshots.getPathFor(testRun.test); testItem.screenshots = taskInfo.task.screenshots.getScreenshotsInfo(testRun.test); } if (taskInfo.task.videos) testItem.videos = taskInfo.task.videos.getTestVideos(testItem.test.id); if (testRun.quarantine) { const testItemQuarantine = testRun.quarantine.attempts.reduce((result, { errors }, index) => { const passed = !errors.length; const quarantineAttempt = index + 1; result[quarantineAttempt] = { passed }; return result; }, {}); Object.assign(testItem.quarantine, testItemQuarantine); } if (!testItem.testRunInfo) { testItem.testRunInfo = Reporter._createTestRunInfo(testItem); if (testItem.test.skip) taskInfo.skipped++; else if (testItem.errs.length) taskInfo.failed++; else taskInfo.passed++; } await this._shiftTestQueue(); testItem.pendingTestRunDonePromise.resolve(); } _prepareReportTestActionEventArgs({ command, duration, result, testRun, err }) { const args = {}; if (err) args.err = err; if (typeof duration === 'number') args.duration = duration; const testFixture = testRun.test.fixture; return Object.assign(args, { testRunId: testRun.id, test: { id: testRun.test.id, name: testRun.test.name, phase: testRun.phase, }, fixture: { name: testFixture.name, id: testFixture.id, }, command: (0, format_command_1.default)(command, result), browser: testRun.browser, }); } async _onceTaskStartHandler(task) { this.taskInfo = { task: task, passed: 0, failed: 0, skipped: 0, testCount: task.tests.filter(test => !test.skip).length, testQueue: Reporter._createTestQueue(task), stopOnFirstFail: task.opts.stopOnFirstFail, pendingTaskDonePromise: Reporter._createPendingPromise(), }; const startTime = task.startTime; const browserConnectionsInfo = [] .concat(...task.browserConnectionGroups) .map(connection => connection.connectionInfo); const first = this.taskInfo.testQueue[0]; const taskProperties = { configuration: task.opts, dashboardUrl: task.opts.dashboardUrl, }; await this.dispatchToPlugin({ method: plugin_methods_1.default.reportTaskStart, initialObject: task, args: [ startTime, browserConnectionsInfo, this.taskInfo.testCount, task.testStructure, taskProperties, ], }); if (first) { await this.dispatchToPlugin({ method: plugin_methods_1.default.reportFixtureStart, initialObject: task, args: [ first.fixture.name, first.fixture.path, first.fixture.meta, ], }); } } async _onTaskTestRunStartHandler(testRun) { if (!this.taskInfo) return void 0; const testItem = this._getTestItemForTestRun(this.taskInfo, testRun); testItem.testRunIds.push(testRun.id); if (!testItem.startTime) testItem.startTime = +new Date(); testItem.pendingStarts--; if (!testItem.pendingStarts) { // @ts-ignore if (this.plugin.reportTestStart) { const testStartInfo = { testRunIds: testItem.testRunIds, testId: testItem.test.id, startTime: new Date(testItem.startTime), skipped: testItem.test.skip, }; await this.dispatchToPlugin({ method: plugin_methods_1.default.reportTestStart, initialObject: this.taskInfo.task, args: [ testItem.test.name, testItem.test.meta, testStartInfo, ], }); } testItem.pendingTestRunStartPromise.resolve(); } return testItem.pendingTestRunStartPromise; } async _onTaskTestRunDoneHandler(testRun) { if (!this.taskInfo) return; const reportItem = this._getTestItemForTestRun(this.taskInfo, testRun); const isTestRunStoppedTaskExecution = !!testRun.errs.length && this.taskInfo.stopOnFirstFail; const browser = Object.assign({ testRunId: testRun.id }, testRun.browser); reportItem.browsers.push(browser); reportItem.pendingRuns = isTestRunStoppedTaskExecution ? 0 : reportItem.pendingRuns - 1; reportItem.unstable = reportItem.unstable || testRun.unstable; reportItem.errs = reportItem.errs.concat(testRun.errs); reportItem.warnings = testRun.warningLog ? (0, lodash_1.union)(reportItem.warnings, testRun.warningLog.messages) : []; if (testRun.quarantine) { reportItem.quarantine = reportItem.quarantine || {}; const reportItemQuarantine = testRun.quarantine.attempts.reduce((result, { errors, testRunId }) => { const passed = !errors.length; result[testRunId] = { passed, errors }; browser.quarantineAttemptsTestRunIds = browser.quarantineAttemptsTestRunIds || []; browser.quarantineAttemptsTestRunIds.push(testRunId); return result; }, {}); Object.assign(reportItem.quarantine, reportItemQuarantine); } if (!reportItem.pendingRuns) await this._resolveTestItem(this.taskInfo, reportItem, testRun); await reportItem.pendingTestRunDonePromise; } async _onTaskTestActionStart(_a) { var { apiActionName } = _a, restArgs = __rest(_a, ["apiActionName"]); if (!this.taskInfo) return; // @ts-ignore if (this.plugin.reportTestActionStart) { restArgs = this._prepareReportTestActionEventArgs(restArgs); await this.dispatchToPlugin({ method: plugin_methods_1.default.reportTestActionStart, initialObject: this.taskInfo.task, args: [ apiActionName, restArgs, ], }); } } async _onTaskTestActionDone(_a) { var { apiActionName } = _a, restArgs = __rest(_a, ["apiActionName"]); if (!this.taskInfo) return; // @ts-ignore if (this.plugin.reportTestActionDone) { restArgs = this._prepareReportTestActionEventArgs(restArgs); await this.dispatchToPlugin({ method: plugin_methods_1.default.reportTestActionDone, initialObject: this.taskInfo.task, args: [ apiActionName, restArgs, ], }); } } async _onceTaskDoneHandler() { var _a; if (!this.taskInfo) return; const endTime = new Date(); const result = { passedCount: this.taskInfo.passed, failedCount: this.taskInfo.failed, skippedCount: this.taskInfo.skipped, }; await this.dispatchToPlugin({ method: plugin_methods_1.default.reportTaskDone, initialObject: this.taskInfo.task, args: [ endTime, this.taskInfo.passed, (_a = this.taskInfo.task) === null || _a === void 0 ? void 0 : _a.warningLog.messages, result, ], }); this.taskInfo.pendingTaskDonePromise.resolve(); } } exports.default = Reporter; module.exports = exports.default; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/reporter/index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    find,\n    sortBy,\n    union,\n    isFunction,\n} from 'lodash';\n\nimport { writable as isWritableStream } from 'is-stream';\nimport ReporterPluginHost from './plugin-host';\nimport ReporterPluginMethod from './plugin-methods';\nimport formatCommand from './command/format-command';\nimport { ReporterPluginError } from '../errors/runtime';\nimport Task from '../runner/task';\nimport { Writable as WritableStream, Writable } from 'stream';\nimport { WriteStream } from 'tty';\nimport TestRun from '../test-run';\nimport Test from '../api/structure/test';\nimport Fixture from '../api/structure/fixture';\nimport TestRunErrorFormattableAdapter from '../errors/test-run/formattable-adapter';\nimport { CommandBase } from '../test-run/commands/base';\n\nimport {\n    ReporterPlugin,\n    ReporterPluginSource,\n    ReporterSource,\n} from './interfaces';\n\nimport { getPluginFactory, processReporterName } from '../utils/reporter';\nimport resolvePathRelativelyCwd from '../utils/resolve-path-relatively-cwd';\nimport makeDir from 'make-dir';\nimport path from 'path';\nimport fs from 'fs';\nimport MessageBus from '../utils/message-bus';\nimport BrowserConnection from '../browser/connection';\nimport { Dictionary } from '../configuration/interfaces';\nimport debug from 'debug';\n\ninterface PendingPromise {\n    resolve: Function | null;\n    then: Function;\n}\n\ninterface TaskInfo {\n    task: Task | null;\n    passed: number;\n    failed: number;\n    skipped: number;\n    testCount: number;\n    testQueue: TestInfo[];\n    readonly stopOnFirstFail: boolean;\n    readonly pendingTaskDonePromise: PendingPromise;\n}\n\ninterface TestInfo {\n    fixture: Fixture;\n    test: Test;\n    testRunIds: string[];\n    screenshotPath: null | string;\n    screenshots: unknown[];\n    videos: unknown[];\n    quarantine: null | Record<string, object>;\n    errs: TestRunErrorFormattableAdapter[];\n    warnings: string[];\n    unstable: boolean;\n    startTime: null | number;\n    testRunInfo: null | TestRunInfo;\n    pendingRuns: number;\n    pendingStarts: number;\n    pendingTestRunDonePromise: PendingPromise;\n    pendingTestRunStartPromise: PendingPromise;\n    browsers: BrowserRunInfo[];\n}\n\ninterface FixtureInfo {\n    id: string;\n    name: string | null;\n    path: string;\n    meta: Dictionary<string>;\n}\n\ninterface BrowserRunInfo extends Browser {\n    testRunId: string;\n    quarantineAttemptsTestRunIds?: string[];\n}\n\ninterface TestRunInfo {\n    errs: TestRunErrorFormattableAdapter[];\n    warnings: string[];\n    durationMs: number;\n    unstable: boolean;\n    screenshotPath: string;\n    screenshots: unknown;\n    videos: unknown;\n    quarantine: unknown;\n    skipped: boolean;\n    browsers: unknown[];\n    testId: string;\n    fixture: FixtureInfo;\n}\n\ninterface PluginMethodArguments {\n    initialObject: Task | MessageBus | null;\n    method: string;\n    args: unknown[];\n}\n\ninterface ReportTestActionEventArguments {\n    command: CommandBase;\n    duration: number;\n    result: unknown;\n    testRun: TestRun;\n    err: TestRunErrorFormattableAdapter;\n}\n\ninterface ReportTaskActionEventArguments {\n    apiActionName: string;\n    restArgs: object;\n}\n\ninterface ReportWarningEventArguments {\n    message: string;\n    testRun?: TestRun;\n    actionId?: string;\n}\n\nconst debugLog = debug('testcafe:reporter');\n\nexport default class Reporter {\n    public readonly plugin: ReporterPluginHost;\n    public readonly messageBus: MessageBus;\n    public disposed: boolean;\n    public taskInfo: TaskInfo | null;\n    public readonly outStream: Writable;\n\n    public constructor (plugin: ReporterPlugin, messageBus: MessageBus, outStream: Writable, name: string) {\n        this.plugin     = new ReporterPluginHost(plugin, outStream, name);\n        this.messageBus = messageBus;\n\n        this.disposed  = false;\n        this.taskInfo  = null;\n        this.outStream = outStream;\n\n        this._assignMessageBusEventHandlers();\n    }\n\n    private static _isSpecialStream (stream: Writable): boolean {\n        return (stream as WriteStream).isTTY || stream === process.stdout || stream === process.stderr;\n    }\n\n    private static _createPendingPromise (): PendingPromise {\n        let resolver = null;\n\n        const promise = new Promise(resolve => {\n            resolver = resolve;\n        }) as unknown as PendingPromise;\n\n        promise.resolve = resolver;\n\n        return promise;\n    }\n\n    public async init (): Promise<void> {\n        await this.dispatchToPlugin({\n            method:        ReporterPluginMethod.init,\n            initialObject: null,\n            args:          [{}],\n        });\n    }\n\n    public async dispatchToPlugin ({ method, initialObject, args = [] }: PluginMethodArguments): Promise<void> {\n        try {\n            // @ts-ignore\n            await this.plugin[method](...args);\n        }\n        catch (originalError) {\n            const uncaughtError = new ReporterPluginError({\n                name: this.plugin.name,\n                method,\n                originalError,\n            });\n\n            debugLog('Plugin error: %O', uncaughtError);\n            debugLog('Plugin error: initialObject: %O', initialObject);\n\n            if (initialObject)\n                await initialObject.emit('error', uncaughtError);\n            else\n                throw uncaughtError;\n        }\n    }\n\n    private _assignMessageBusEventHandlers (): void {\n        const messageBus = this.messageBus;\n\n        messageBus.on('warning-add', async e => await this._onWarningAddHandler(e));\n\n        messageBus.once('start', async (task: Task) => await this._onceTaskStartHandler(task));\n\n        messageBus.on('test-run-start', async testRun => await this._onTaskTestRunStartHandler(testRun));\n\n        messageBus.on('test-run-done', async testRun => await this._onTaskTestRunDoneHandler(testRun));\n\n        messageBus.on('test-action-start', async e => await this._onTaskTestActionStart(e));\n\n        messageBus.on('test-action-done', async e => await this._onTaskTestActionDone(e));\n\n        messageBus.once('done', async () => await this._onceTaskDoneHandler());\n    }\n\n    public async dispose (): Promise<unknown> {\n        if (this.disposed)\n            return Promise.resolve();\n\n        this.disposed = true;\n\n        if (!isFunction(this?.outStream?.once)\n            || Reporter._isSpecialStream(this.outStream)\n            || !isWritableStream(this.outStream))\n            return Promise.resolve();\n\n        const streamFinishedPromise = new Promise(resolve => {\n            this.outStream.once('finish', resolve);\n            this.outStream.once('error', resolve);\n        });\n\n        this.outStream.end();\n\n        return streamFinishedPromise;\n    }\n\n    private static async _ensureOutStream (outStream: string | WritableStream): Promise<WritableStream> {\n        if (typeof outStream !== 'string')\n            return outStream;\n\n        const fullReporterOutputPath = resolvePathRelativelyCwd(outStream);\n\n        await makeDir(path.dirname(fullReporterOutputPath));\n\n        return fs.createWriteStream(fullReporterOutputPath);\n    }\n\n    private static _addDefaultReporter (reporters: ReporterSource[]): void {\n        reporters.push({\n            name:   'spec',\n            output: process.stdout,\n        });\n    }\n\n    public static async getReporterPlugins (reporters: ReporterSource[] = []): Promise<ReporterPluginSource[]> {\n        if (!reporters.length)\n            Reporter._addDefaultReporter(reporters);\n\n        return Promise.all(reporters.map(async ({ name, output, options }) => {\n            const pluginFactory = getPluginFactory(name);\n            const processedName = processReporterName(name);\n            const outStream     = output ? await Reporter._ensureOutStream(output) : void 0;\n\n            return {\n                plugin: pluginFactory(options),\n                name:   processedName,\n                outStream,\n            };\n        }));\n    }\n\n    private async _onWarningAddHandler ({ message, testRun, actionId }: ReportWarningEventArguments): Promise<void> {\n        await this.dispatchToPlugin({\n            method:        ReporterPluginMethod.reportWarnings as string,\n            initialObject: this.messageBus,\n            args:          [\n                {\n                    message,\n                    testRunId: testRun?.id,\n                    actionId,\n                },\n            ],\n        });\n    }\n\n    //Task\n    private static _createTestItem (test: Test, runsPerTest: number): TestInfo {\n        return {\n            fixture:                    test.fixture as Fixture,\n            test:                       test,\n            testRunIds:                 [],\n            screenshotPath:             null,\n            screenshots:                [],\n            videos:                     [],\n            quarantine:                 null,\n            errs:                       [],\n            warnings:                   [],\n            unstable:                   false,\n            startTime:                  null,\n            testRunInfo:                null,\n            pendingRuns:                runsPerTest,\n            pendingStarts:              runsPerTest,\n            pendingTestRunDonePromise:  Reporter._createPendingPromise(),\n            pendingTestRunStartPromise: Reporter._createPendingPromise(),\n            browsers:                   [],\n        };\n    }\n\n    private static _createTestQueue (task: Task): TestInfo[] {\n        const runsPerTest = task.browserConnectionGroups.length;\n\n        return task.tests.map(test => Reporter._createTestItem(test, runsPerTest));\n    }\n\n    private static _createTestRunInfo (reportItem: TestInfo): TestRunInfo {\n        return {\n            errs:           sortBy(reportItem.errs, ['userAgent', 'code']),\n            warnings:       reportItem.warnings,\n            durationMs:     +new Date() - (reportItem.startTime as number), //eslint-disable-line  @typescript-eslint/no-extra-parens\n            unstable:       reportItem.unstable,\n            screenshotPath: reportItem.screenshotPath as string,\n            screenshots:    reportItem.screenshots,\n            videos:         reportItem.videos,\n            quarantine:     reportItem.quarantine,\n            skipped:        reportItem.test.skip,\n            browsers:       reportItem.browsers,\n            testId:         reportItem.test.id,\n            fixture:        {\n                id:   reportItem.fixture.id,\n                name: reportItem.fixture.name,\n                path: reportItem.fixture.path,\n                meta: reportItem.fixture.meta,\n            },\n        };\n    }\n\n    private _getTestItemForTestRun (taskInfo: TaskInfo, testRun: TestRun): TestInfo | undefined {\n        return find(taskInfo.testQueue, i => i.test === testRun.test);\n    }\n\n    private async _shiftTestQueue (): Promise<void> {\n        if (!this.taskInfo)\n            return;\n\n        let currentFixture = null;\n        let nextReportItem = null;\n        let testItem       = null;\n        const testQueue    = this.taskInfo.testQueue;\n\n        while (testQueue.length && testQueue[0].testRunInfo) {\n            testItem       = testQueue.shift() as TestInfo;\n            currentFixture = testItem.fixture;\n\n            // NOTE: here we assume that tests are sorted by fixture.\n            // Therefore, if the next report item has a different\n            // fixture, we can report this fixture start.\n            nextReportItem = testQueue[0];\n\n            await this.dispatchToPlugin({\n                method:        ReporterPluginMethod.reportTestDone,\n                initialObject: this.taskInfo.task,\n                args:          [\n                    testItem.test.name,\n                    testItem.testRunInfo,\n                    testItem.test.meta,\n                ],\n            });\n\n            if (!nextReportItem || nextReportItem.fixture === currentFixture)\n                continue;\n\n            await this.dispatchToPlugin({\n                method:        ReporterPluginMethod.reportFixtureStart,\n                initialObject: this.taskInfo.task,\n                args:          [\n                    nextReportItem.fixture.name,\n                    nextReportItem.fixture.path,\n                    nextReportItem.fixture.meta,\n                ],\n            });\n        }\n    }\n\n    private async _resolveTestItem (taskInfo: TaskInfo, testItem: TestInfo, testRun: TestRun): Promise<void> {\n        if (!taskInfo.task)\n            return;\n\n        if (taskInfo.task.screenshots.hasCapturedFor(testRun.test)) {\n            testItem.screenshotPath = taskInfo.task.screenshots.getPathFor(testRun.test);\n            testItem.screenshots    = taskInfo.task.screenshots.getScreenshotsInfo(testRun.test);\n        }\n\n        if (taskInfo.task.videos)\n            testItem.videos = taskInfo.task.videos.getTestVideos(testItem.test.id);\n\n        if (testRun.quarantine) {\n            const testItemQuarantine = testRun.quarantine.attempts.reduce((result: Record<string, object>, { errors }, index: number) => {\n                const passed            = !errors.length;\n                const quarantineAttempt = index + 1;\n\n                result[quarantineAttempt] = { passed };\n\n                return result;\n            }, { });\n\n            Object.assign(testItem.quarantine as object, testItemQuarantine);\n        }\n\n        if (!testItem.testRunInfo) {\n            testItem.testRunInfo = Reporter._createTestRunInfo(testItem);\n\n            if (testItem.test.skip)\n                taskInfo.skipped++;\n            else if (testItem.errs.length)\n                taskInfo.failed++;\n            else\n                taskInfo.passed++;\n        }\n\n        await this._shiftTestQueue();\n\n        (testItem.pendingTestRunDonePromise.resolve as Function)();\n    }\n\n    private _prepareReportTestActionEventArgs ({ command, duration, result, testRun, err }: ReportTestActionEventArguments): any {\n        const args: any = {};\n\n        if (err)\n            args.err = err;\n\n        if (typeof duration === 'number')\n            args.duration = duration;\n\n        const testFixture = testRun.test.fixture as Fixture;\n\n        return Object.assign(args, {\n            testRunId: testRun.id,\n            test:      {\n                id:    testRun.test.id,\n                name:  testRun.test.name,\n                phase: testRun.phase,\n            },\n            fixture: {\n                name: testFixture.name,\n                id:   testFixture.id,\n            },\n            command: formatCommand(command, result),\n            browser: testRun.browser,\n        });\n    }\n\n\n    private async _onceTaskStartHandler (task: Task): Promise<void> {\n        this.taskInfo = {\n            task:                   task,\n            passed:                 0,\n            failed:                 0,\n            skipped:                0,\n            testCount:              task.tests.filter(test => !test.skip).length,\n            testQueue:              Reporter._createTestQueue(task),\n            stopOnFirstFail:        task.opts.stopOnFirstFail as boolean,\n            pendingTaskDonePromise: Reporter._createPendingPromise(),\n        };\n\n        const startTime              = task.startTime;\n        const browserConnectionsInfo = ([] as BrowserConnection[])\n            .concat(...task.browserConnectionGroups)\n            .map(connection => connection.connectionInfo);\n        const first                  = this.taskInfo.testQueue[0];\n\n        const taskProperties = {\n            configuration: task.opts,\n            dashboardUrl:  task.opts.dashboardUrl,\n        };\n\n        await this.dispatchToPlugin({\n            method:        ReporterPluginMethod.reportTaskStart,\n            initialObject: task,\n            args:          [\n                startTime,\n                browserConnectionsInfo,\n                this.taskInfo.testCount,\n                task.testStructure,\n                taskProperties,\n            ],\n        });\n\n        if (first) {\n            await this.dispatchToPlugin({\n                method:        ReporterPluginMethod.reportFixtureStart,\n                initialObject: task,\n                args:          [\n                    first.fixture.name,\n                    first.fixture.path,\n                    first.fixture.meta,\n                ],\n            });\n        }\n    }\n\n    private async _onTaskTestRunStartHandler (testRun: TestRun): Promise<unknown> {\n        if (!this.taskInfo)\n            return void 0;\n\n        const testItem = this._getTestItemForTestRun(this.taskInfo, testRun) as TestInfo;\n\n        testItem.testRunIds.push(testRun.id);\n\n        if (!testItem.startTime)\n            testItem.startTime = +new Date();\n\n        testItem.pendingStarts--;\n\n        if (!testItem.pendingStarts) {\n            // @ts-ignore\n            if (this.plugin.reportTestStart) {\n                const testStartInfo = {\n                    testRunIds: testItem.testRunIds,\n                    testId:     testItem.test.id,\n                    startTime:  new Date(testItem.startTime),\n                    skipped:    testItem.test.skip,\n                };\n\n                await this.dispatchToPlugin({\n                    method:        ReporterPluginMethod.reportTestStart as string,\n                    initialObject: this.taskInfo.task,\n                    args:          [\n                        testItem.test.name,\n                        testItem.test.meta,\n                        testStartInfo,\n                    ],\n                });\n            }\n\n            (testItem.pendingTestRunStartPromise.resolve as Function)();\n        }\n\n        return testItem.pendingTestRunStartPromise;\n    }\n\n    private async _onTaskTestRunDoneHandler (testRun: TestRun): Promise<void> {\n        if (!this.taskInfo)\n            return;\n\n        const reportItem                    = this._getTestItemForTestRun(this.taskInfo, testRun) as TestInfo;\n        const isTestRunStoppedTaskExecution = !!testRun.errs.length && this.taskInfo.stopOnFirstFail;\n        const browser: BrowserRunInfo       = Object.assign({ testRunId: testRun.id }, testRun.browser);\n\n        reportItem.browsers.push(browser);\n\n        reportItem.pendingRuns = isTestRunStoppedTaskExecution ? 0 : reportItem.pendingRuns - 1;\n        reportItem.unstable    = reportItem.unstable || testRun.unstable;\n        reportItem.errs        = reportItem.errs.concat(testRun.errs);\n        reportItem.warnings    = testRun.warningLog ? union(reportItem.warnings, testRun.warningLog.messages) : [];\n\n        if (testRun.quarantine) {\n            reportItem.quarantine = reportItem.quarantine || {};\n\n            const reportItemQuarantine = testRun.quarantine.attempts.reduce((result: Record<string, object>, { errors, testRunId }) => {\n                const passed = !errors.length;\n\n                result[testRunId]                    = { passed, errors };\n                browser.quarantineAttemptsTestRunIds = browser.quarantineAttemptsTestRunIds || [];\n\n                browser.quarantineAttemptsTestRunIds.push(testRunId);\n\n                return result;\n            }, {});\n\n            Object.assign(reportItem.quarantine, reportItemQuarantine);\n        }\n\n        if (!reportItem.pendingRuns)\n            await this._resolveTestItem(this.taskInfo, reportItem, testRun);\n\n        await reportItem.pendingTestRunDonePromise;\n    }\n\n    private async _onTaskTestActionStart ({ apiActionName, ...restArgs }: ReportTaskActionEventArguments): Promise<void> {\n        if (!this.taskInfo)\n            return;\n\n        // @ts-ignore\n        if (this.plugin.reportTestActionStart) {\n            restArgs = this._prepareReportTestActionEventArgs(restArgs as unknown as ReportTestActionEventArguments);\n\n            await this.dispatchToPlugin({\n                method:        ReporterPluginMethod.reportTestActionStart as string,\n                initialObject: this.taskInfo.task,\n                args:          [\n                    apiActionName,\n                    restArgs,\n                ],\n            });\n        }\n    }\n\n    private async _onTaskTestActionDone ({ apiActionName, ...restArgs }: ReportTaskActionEventArguments): Promise<void> {\n        if (!this.taskInfo)\n            return;\n\n        // @ts-ignore\n        if (this.plugin.reportTestActionDone) {\n            restArgs = this._prepareReportTestActionEventArgs(restArgs as unknown as ReportTestActionEventArguments);\n\n            await this.dispatchToPlugin({\n                method:        ReporterPluginMethod.reportTestActionDone as string,\n                initialObject: this.taskInfo.task,\n                args:          [\n                    apiActionName,\n                    restArgs,\n                ],\n            });\n        }\n    }\n\n    private async _onceTaskDoneHandler (): Promise<void> {\n        if (!this.taskInfo)\n            return;\n\n        const endTime = new Date();\n\n        const result = {\n            passedCount:  this.taskInfo.passed,\n            failedCount:  this.taskInfo.failed,\n            skippedCount: this.taskInfo.skipped,\n        };\n\n        await this.dispatchToPlugin({\n            method:        ReporterPluginMethod.reportTaskDone,\n            initialObject: this.taskInfo.task,\n            args:          [\n                endTime,\n                this.taskInfo.passed,\n                this.taskInfo.task?.warningLog.messages,\n                result,\n            ],\n        });\n\n        (this.taskInfo.pendingTaskDonePromise.resolve as Function)();\n    }\n}\n"]}