UNPKG

32.2 kBJavaScriptView Raw
1"use strict";
2var __importDefault = (this && this.__importDefault) || function (mod) {
3 return (mod && mod.__esModule) ? mod : { "default": mod };
4};
5Object.defineProperty(exports, "__esModule", { value: true });
6const lodash_1 = require("lodash");
7const is_stream_1 = require("is-stream");
8const plugin_host_1 = __importDefault(require("./plugin-host"));
9const command_report_item_1 = require("./command-report-item");
10const error_list_1 = __importDefault(require("../errors/error-list"));
11class 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}
198exports.default = Reporter;
199module.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