1 | ;
|
2 | var __importDefault = (this && this.__importDefault) || function (mod) {
|
3 | return (mod && mod.__esModule) ? mod : { "default": mod };
|
4 | };
|
5 | Object.defineProperty(exports, "__esModule", { value: true });
|
6 | const lodash_1 = require("lodash");
|
7 | const is_stream_1 = require("is-stream");
|
8 | const plugin_host_1 = __importDefault(require("./plugin-host"));
|
9 | const command_report_item_1 = require("./command-report-item");
|
10 | const error_list_1 = __importDefault(require("../errors/error-list"));
|
11 | class Reporter {
|
12 | constructor(plugin, task, outStream) {
|
13 | this.plugin = new plugin_host_1.default(plugin, outStream);
|
14 | this.task = task;
|
15 | this.disposed = false;
|
16 | this.passed = 0;
|
17 | this.failed = 0;
|
18 | this.skipped = 0;
|
19 | this.testCount = task.tests.filter(test => !test.skip).length;
|
20 | this.reportQueue = Reporter._createReportQueue(task);
|
21 | this.stopOnFirstFail = task.opts.stopOnFirstFail;
|
22 | this.outStream = outStream;
|
23 | this.pendingTaskDonePromise = Reporter._createPendingPromise();
|
24 | this._assignTaskEventHandlers();
|
25 | }
|
26 | static _isSpecialStream(stream) {
|
27 | return stream.isTTY || stream === process.stdout || stream === process.stderr;
|
28 | }
|
29 | static _createPendingPromise() {
|
30 | let resolver = null;
|
31 | const promise = new Promise(resolve => {
|
32 | resolver = resolve;
|
33 | });
|
34 | promise.resolve = resolver;
|
35 | return promise;
|
36 | }
|
37 | static _createReportItem(test, runsPerTest) {
|
38 | return {
|
39 | fixture: test.fixture,
|
40 | test: test,
|
41 | screenshotPath: null,
|
42 | screenshots: [],
|
43 | quarantine: null,
|
44 | errs: [],
|
45 | warnings: [],
|
46 | unstable: false,
|
47 | startTime: null,
|
48 | testRunInfo: null,
|
49 | pendingRuns: runsPerTest,
|
50 | pendingStarts: runsPerTest,
|
51 | pendingTestRunDonePromise: Reporter._createPendingPromise(),
|
52 | pendingTestRunStartPromise: Reporter._createPendingPromise()
|
53 | };
|
54 | }
|
55 | static _createReportQueue(task) {
|
56 | const runsPerTest = task.browserConnectionGroups.length;
|
57 | return task.tests.map(test => Reporter._createReportItem(test, runsPerTest));
|
58 | }
|
59 | static _createTestRunInfo(reportItem) {
|
60 | return {
|
61 | errs: lodash_1.sortBy(reportItem.errs, ['userAgent', 'code']),
|
62 | warnings: reportItem.warnings,
|
63 | durationMs: new Date() - reportItem.startTime,
|
64 | unstable: reportItem.unstable,
|
65 | screenshotPath: reportItem.screenshotPath,
|
66 | screenshots: reportItem.screenshots,
|
67 | quarantine: reportItem.quarantine,
|
68 | skipped: reportItem.test.skip
|
69 | };
|
70 | }
|
71 | _getReportItemForTestRun(testRun) {
|
72 | return lodash_1.find(this.reportQueue, i => i.test === testRun.test);
|
73 | }
|
74 | async _shiftReportQueue(reportItem) {
|
75 | let currentFixture = null;
|
76 | let nextReportItem = null;
|
77 | while (this.reportQueue.length && this.reportQueue[0].testRunInfo) {
|
78 | reportItem = this.reportQueue.shift();
|
79 | currentFixture = reportItem.fixture;
|
80 | await this.plugin.reportTestDone(reportItem.test.name, reportItem.testRunInfo, reportItem.test.meta);
|
81 | // NOTE: here we assume that tests are sorted by fixture.
|
82 | // Therefore, if the next report item has a different
|
83 | // fixture, we can report this fixture start.
|
84 | nextReportItem = this.reportQueue[0];
|
85 | if (nextReportItem && nextReportItem.fixture !== currentFixture)
|
86 | await this.plugin.reportFixtureStart(nextReportItem.fixture.name, nextReportItem.fixture.path, nextReportItem.fixture.meta);
|
87 | }
|
88 | }
|
89 | async _resolveReportItem(reportItem, testRun) {
|
90 | if (this.task.screenshots.hasCapturedFor(testRun.test)) {
|
91 | reportItem.screenshotPath = this.task.screenshots.getPathFor(testRun.test);
|
92 | reportItem.screenshots = this.task.screenshots.getScreenshotsInfo(testRun.test);
|
93 | }
|
94 | if (testRun.quarantine) {
|
95 | reportItem.quarantine = testRun.quarantine.attempts.reduce((result, errors, index) => {
|
96 | const passed = !errors.length;
|
97 | const quarantineAttempt = index + 1;
|
98 | result[quarantineAttempt] = { passed };
|
99 | return result;
|
100 | }, {});
|
101 | }
|
102 | if (!reportItem.testRunInfo) {
|
103 | reportItem.testRunInfo = Reporter._createTestRunInfo(reportItem);
|
104 | if (reportItem.test.skip)
|
105 | this.skipped++;
|
106 | else if (reportItem.errs.length)
|
107 | this.failed++;
|
108 | else
|
109 | this.passed++;
|
110 | }
|
111 | await this._shiftReportQueue(reportItem);
|
112 | reportItem.pendingTestRunDonePromise.resolve();
|
113 | }
|
114 | _prepareReportTestActionEventArgs({ command, testRun, errors }) {
|
115 | const args = {};
|
116 | if (errors) {
|
117 | errors = errors instanceof error_list_1.default ? errors.items : [errors];
|
118 | args.errors = errors;
|
119 | }
|
120 | return Object.assign(args, {
|
121 | test: {
|
122 | name: testRun.test.name,
|
123 | phase: testRun.phase
|
124 | },
|
125 | command: new command_report_item_1.CommandReportItem(command),
|
126 | browser: testRun.controller.browser,
|
127 | });
|
128 | }
|
129 | _assignTaskEventHandlers() {
|
130 | const task = this.task;
|
131 | task.once('start', async () => {
|
132 | const startTime = new Date();
|
133 | const userAgents = task.browserConnectionGroups.map(group => group[0].userAgent);
|
134 | const first = this.reportQueue[0];
|
135 | await this.plugin.reportTaskStart(startTime, userAgents, this.testCount);
|
136 | await this.plugin.reportFixtureStart(first.fixture.name, first.fixture.path, first.fixture.meta);
|
137 | });
|
138 | task.on('test-run-start', async (testRun) => {
|
139 | const reportItem = this._getReportItemForTestRun(testRun);
|
140 | if (!reportItem.startTime)
|
141 | reportItem.startTime = new Date();
|
142 | reportItem.pendingStarts--;
|
143 | if (!reportItem.pendingStarts) {
|
144 | if (this.plugin.reportTestStart)
|
145 | await this.plugin.reportTestStart(reportItem.test.name, reportItem.test.meta);
|
146 | reportItem.pendingTestRunStartPromise.resolve();
|
147 | }
|
148 | return reportItem.pendingTestRunStartPromise;
|
149 | });
|
150 | task.on('test-run-done', async (testRun) => {
|
151 | const reportItem = this._getReportItemForTestRun(testRun);
|
152 | const isTestRunStoppedTaskExecution = !!testRun.errs.length && this.stopOnFirstFail;
|
153 | reportItem.pendingRuns = isTestRunStoppedTaskExecution ? 0 : reportItem.pendingRuns - 1;
|
154 | reportItem.unstable = reportItem.unstable || testRun.unstable;
|
155 | reportItem.errs = reportItem.errs.concat(testRun.errs);
|
156 | reportItem.warnings = testRun.warningLog ? lodash_1.union(reportItem.warnings, testRun.warningLog.messages) : [];
|
157 | if (!reportItem.pendingRuns)
|
158 | await this._resolveReportItem(reportItem, testRun);
|
159 | await reportItem.pendingTestRunDonePromise;
|
160 | });
|
161 | task.on('test-action-start', async ({ apiActionName, command, testRun }) => {
|
162 | if (this.plugin.reportTestActionStart) {
|
163 | const args = this._prepareReportTestActionEventArgs({ command, testRun });
|
164 | await this.plugin.reportTestActionStart(apiActionName, args);
|
165 | }
|
166 | });
|
167 | task.on('test-action-done', async ({ apiActionName, command, testRun, errors }) => {
|
168 | if (this.plugin.reportTestActionDone) {
|
169 | const args = this._prepareReportTestActionEventArgs({ command, testRun, errors });
|
170 | await this.plugin.reportTestActionDone(apiActionName, args);
|
171 | }
|
172 | });
|
173 | task.once('done', async () => {
|
174 | const endTime = new Date();
|
175 | const result = {
|
176 | passedCount: this.passed,
|
177 | failedCount: this.failed,
|
178 | skippedCount: this.skipped
|
179 | };
|
180 | await this.plugin.reportTaskDone(endTime, this.passed, task.warningLog.messages, result);
|
181 | this.pendingTaskDonePromise.resolve();
|
182 | });
|
183 | }
|
184 | async dispose() {
|
185 | if (this.disposed)
|
186 | return Promise.resolve();
|
187 | this.disposed = true;
|
188 | if (!this.outStream || Reporter._isSpecialStream(this.outStream) || !is_stream_1.writable(this.outStream))
|
189 | return Promise.resolve();
|
190 | const streamFinishedPromise = new Promise(resolve => {
|
191 | this.outStream.once('finish', resolve);
|
192 | this.outStream.once('error', resolve);
|
193 | });
|
194 | this.outStream.end();
|
195 | return streamFinishedPromise;
|
196 | }
|
197 | }
|
198 | exports.default = Reporter;
|
199 | module.exports = exports.default;
|
200 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/reporter/index.js"],"names":[],"mappings":"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","sourcesContent":["import { find, sortBy, union } from 'lodash';\nimport { writable as isWritableStream } from 'is-stream';\nimport ReporterPluginHost from './plugin-host';\nimport { CommandReportItem } from './command-report-item';\nimport TestCafeErrorList from '../errors/error-list';\n\nexport default class Reporter {\n    constructor (plugin, task, outStream) {\n        this.plugin = new ReporterPluginHost(plugin, outStream);\n        this.task   = task;\n\n        this.disposed        = false;\n        this.passed          = 0;\n        this.failed          = 0;\n        this.skipped         = 0;\n        this.testCount       = task.tests.filter(test => !test.skip).length;\n        this.reportQueue     = Reporter._createReportQueue(task);\n        this.stopOnFirstFail = task.opts.stopOnFirstFail;\n        this.outStream       = outStream;\n\n        this.pendingTaskDonePromise = Reporter._createPendingPromise();\n\n        this._assignTaskEventHandlers();\n    }\n\n    static _isSpecialStream (stream) {\n        return stream.isTTY || stream === process.stdout || stream === process.stderr;\n    }\n\n    static _createPendingPromise () {\n        let resolver = null;\n\n        const promise = new Promise(resolve => {\n            resolver = resolve;\n        });\n\n        promise.resolve = resolver;\n\n        return promise;\n    }\n\n    static _createReportItem (test, runsPerTest) {\n        return {\n            fixture:                    test.fixture,\n            test:                       test,\n            screenshotPath:             null,\n            screenshots:                [],\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        };\n    }\n\n    static _createReportQueue (task) {\n        const runsPerTest = task.browserConnectionGroups.length;\n\n        return task.tests.map(test => Reporter._createReportItem(test, runsPerTest));\n    }\n\n    static _createTestRunInfo (reportItem) {\n        return {\n            errs:           sortBy(reportItem.errs, ['userAgent', 'code']),\n            warnings:       reportItem.warnings,\n            durationMs:     new Date() - reportItem.startTime,\n            unstable:       reportItem.unstable,\n            screenshotPath: reportItem.screenshotPath,\n            screenshots:    reportItem.screenshots,\n            quarantine:     reportItem.quarantine,\n            skipped:        reportItem.test.skip\n        };\n    }\n\n    _getReportItemForTestRun (testRun) {\n        return find(this.reportQueue, i => i.test === testRun.test);\n    }\n\n    async _shiftReportQueue (reportItem) {\n        let currentFixture = null;\n        let nextReportItem = null;\n\n        while (this.reportQueue.length && this.reportQueue[0].testRunInfo) {\n            reportItem     = this.reportQueue.shift();\n            currentFixture = reportItem.fixture;\n\n            await this.plugin.reportTestDone(reportItem.test.name, reportItem.testRunInfo, reportItem.test.meta);\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 = this.reportQueue[0];\n\n            if (nextReportItem && nextReportItem.fixture !== currentFixture)\n                await this.plugin.reportFixtureStart(nextReportItem.fixture.name, nextReportItem.fixture.path, nextReportItem.fixture.meta);\n        }\n    }\n\n    async _resolveReportItem (reportItem, testRun) {\n        if (this.task.screenshots.hasCapturedFor(testRun.test)) {\n            reportItem.screenshotPath = this.task.screenshots.getPathFor(testRun.test);\n            reportItem.screenshots    = this.task.screenshots.getScreenshotsInfo(testRun.test);\n        }\n\n        if (testRun.quarantine) {\n            reportItem.quarantine = testRun.quarantine.attempts.reduce((result, errors, index) => {\n                const passed            = !errors.length;\n                const quarantineAttempt = index + 1;\n\n                result[quarantineAttempt] = { passed };\n\n                return result;\n            }, {});\n        }\n\n        if (!reportItem.testRunInfo) {\n            reportItem.testRunInfo = Reporter._createTestRunInfo(reportItem);\n\n            if (reportItem.test.skip)\n                this.skipped++;\n            else if (reportItem.errs.length)\n                this.failed++;\n            else\n                this.passed++;\n        }\n\n        await this._shiftReportQueue(reportItem);\n\n        reportItem.pendingTestRunDonePromise.resolve();\n    }\n\n    _prepareReportTestActionEventArgs ({ command, testRun, errors }) {\n        const args = {};\n\n        if (errors) {\n            errors = errors instanceof TestCafeErrorList ? errors.items : [errors];\n\n            args.errors = errors;\n        }\n\n        return Object.assign(args, {\n            test: {\n                name:  testRun.test.name,\n                phase: testRun.phase\n            },\n            command: new CommandReportItem(command),\n            browser: testRun.controller.browser,\n        });\n    }\n\n    _assignTaskEventHandlers () {\n        const task = this.task;\n\n        task.once('start', async () => {\n            const startTime  = new Date();\n            const userAgents = task.browserConnectionGroups.map(group => group[0].userAgent);\n            const first      = this.reportQueue[0];\n\n            await this.plugin.reportTaskStart(startTime, userAgents, this.testCount);\n            await this.plugin.reportFixtureStart(first.fixture.name, first.fixture.path, first.fixture.meta);\n        });\n\n        task.on('test-run-start', async testRun => {\n            const reportItem = this._getReportItemForTestRun(testRun);\n\n            if (!reportItem.startTime)\n                reportItem.startTime = new Date();\n\n            reportItem.pendingStarts--;\n\n            if (!reportItem.pendingStarts) {\n                if (this.plugin.reportTestStart)\n                    await this.plugin.reportTestStart(reportItem.test.name, reportItem.test.meta);\n\n                reportItem.pendingTestRunStartPromise.resolve();\n            }\n\n            return reportItem.pendingTestRunStartPromise;\n        });\n\n        task.on('test-run-done', async testRun => {\n            const reportItem                    = this._getReportItemForTestRun(testRun);\n            const isTestRunStoppedTaskExecution = !!testRun.errs.length && this.stopOnFirstFail;\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 (!reportItem.pendingRuns)\n                await this._resolveReportItem(reportItem, testRun);\n\n            await reportItem.pendingTestRunDonePromise;\n        });\n\n        task.on('test-action-start', async ({ apiActionName, command, testRun }) => {\n            if (this.plugin.reportTestActionStart) {\n                const args = this._prepareReportTestActionEventArgs({ command, testRun });\n\n                await this.plugin.reportTestActionStart(apiActionName, args);\n            }\n        });\n\n        task.on('test-action-done', async ({ apiActionName, command, testRun, errors }) => {\n            if (this.plugin.reportTestActionDone) {\n                const args = this._prepareReportTestActionEventArgs({ command, testRun, errors });\n\n                await this.plugin.reportTestActionDone(apiActionName, args);\n            }\n        });\n\n        task.once('done', async () => {\n            const endTime = new Date();\n\n            const result = {\n                passedCount:  this.passed,\n                failedCount:  this.failed,\n                skippedCount: this.skipped\n            };\n\n            await this.plugin.reportTaskDone(endTime, this.passed, task.warningLog.messages, result);\n\n            this.pendingTaskDonePromise.resolve();\n        });\n    }\n\n    async dispose () {\n        if (this.disposed)\n            return Promise.resolve();\n\n        this.disposed = true;\n\n        if (!this.outStream || Reporter._isSpecialStream(this.outStream) || !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"]} |
\ | No newline at end of file |