creevey 0.10.0-beta.9 → 0.10.0-rc.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AUTHORS +2 -0
- package/CHANGELOG.md +281 -0
- package/README.md +19 -41
- package/dist/client/addon/components/Addon.js +18 -8
- package/dist/client/addon/components/Addon.js.map +1 -1
- package/dist/client/addon/components/Panel.js +2 -2
- package/dist/client/addon/components/Panel.js.map +1 -1
- package/dist/client/addon/components/TestSelect.js +2 -2
- package/dist/client/addon/components/TestSelect.js.map +1 -1
- package/dist/client/addon/components/Tools.js +19 -9
- package/dist/client/addon/components/Tools.js.map +1 -1
- package/dist/client/addon/controller.d.ts +1 -1
- package/dist/client/addon/controller.js +3 -3
- package/dist/client/addon/controller.js.map +1 -1
- package/dist/client/addon/decorator.d.ts +1 -1
- package/dist/client/addon/makeDecorator.d.ts +9 -0
- package/dist/client/addon/makeDecorator.js +48 -0
- package/dist/client/addon/makeDecorator.js.map +1 -0
- package/dist/client/addon/manager.js +38 -39
- package/dist/client/addon/manager.js.map +1 -1
- package/dist/client/addon/preset.d.ts +0 -1
- package/dist/client/addon/preset.js +3 -2
- package/dist/client/addon/preset.js.map +1 -1
- package/dist/client/addon/preview.d.ts +1 -1
- package/dist/client/addon/withCreevey.d.ts +5 -3
- package/dist/client/addon/withCreevey.js +5 -20
- package/dist/client/addon/withCreevey.js.map +1 -1
- package/dist/client/shared/components/ImagesView/BlendView.d.ts +2 -2
- package/dist/client/shared/components/ImagesView/BlendView.js +18 -8
- package/dist/client/shared/components/ImagesView/BlendView.js.map +1 -1
- package/dist/client/shared/components/ImagesView/ImagesView.js +1 -1
- package/dist/client/shared/components/ImagesView/ImagesView.js.map +1 -1
- package/dist/client/shared/components/ImagesView/SideBySideView.d.ts +2 -2
- package/dist/client/shared/components/ImagesView/SideBySideView.js +19 -9
- package/dist/client/shared/components/ImagesView/SideBySideView.js.map +1 -1
- package/dist/client/shared/components/ImagesView/SlideView.d.ts +2 -2
- package/dist/client/shared/components/ImagesView/SlideView.js +19 -9
- package/dist/client/shared/components/ImagesView/SlideView.js.map +1 -1
- package/dist/client/shared/components/ImagesView/SwapView.d.ts +2 -2
- package/dist/client/shared/components/ImagesView/SwapView.js +19 -9
- package/dist/client/shared/components/ImagesView/SwapView.js.map +1 -1
- package/dist/client/shared/components/ImagesView/common.d.ts +1 -1
- package/dist/client/shared/components/PageFooter/PageFooter.js +1 -1
- package/dist/client/shared/components/PageFooter/PageFooter.js.map +1 -1
- package/dist/client/shared/components/PageFooter/Paging.js +1 -1
- package/dist/client/shared/components/PageFooter/Paging.js.map +1 -1
- package/dist/client/shared/components/PageHeader/ImagePreview.d.ts +2 -2
- package/dist/client/shared/components/PageHeader/ImagePreview.js +1 -1
- package/dist/client/shared/components/PageHeader/ImagePreview.js.map +1 -1
- package/dist/client/shared/components/PageHeader/PageHeader.js +34 -13
- package/dist/client/shared/components/PageHeader/PageHeader.js.map +1 -1
- package/dist/client/shared/components/ResultsPage.d.ts +2 -2
- package/dist/client/shared/components/ResultsPage.js +22 -10
- package/dist/client/shared/components/ResultsPage.js.map +1 -1
- package/dist/client/shared/creeveyClientApi.js +18 -1
- package/dist/client/shared/creeveyClientApi.js.map +1 -1
- package/dist/client/shared/helpers.d.ts +1 -3
- package/dist/client/shared/helpers.js +4 -19
- package/dist/client/shared/helpers.js.map +1 -1
- package/dist/client/web/CreeveyApp.d.ts +1 -0
- package/dist/client/web/CreeveyApp.js +22 -9
- package/dist/client/web/CreeveyApp.js.map +1 -1
- package/dist/client/web/CreeveyContext.d.ts +1 -0
- package/dist/client/web/CreeveyContext.js +18 -7
- package/dist/client/web/CreeveyContext.js.map +1 -1
- package/dist/client/web/CreeveyLoader.d.ts +1 -1
- package/dist/client/web/CreeveyLoader.js +3 -3
- package/dist/client/web/CreeveyLoader.js.map +1 -1
- package/dist/client/web/CreeveyView/SideBar/Checkbox.d.ts +4 -4
- package/dist/client/web/CreeveyView/SideBar/Checkbox.js +36 -6
- package/dist/client/web/CreeveyView/SideBar/Checkbox.js.map +1 -1
- package/dist/client/web/CreeveyView/SideBar/Search.js +18 -8
- package/dist/client/web/CreeveyView/SideBar/Search.js.map +1 -1
- package/dist/client/web/CreeveyView/SideBar/SideBar.js +26 -12
- package/dist/client/web/CreeveyView/SideBar/SideBar.js.map +1 -1
- package/dist/client/web/CreeveyView/SideBar/SideBarFooter.js +28 -17
- package/dist/client/web/CreeveyView/SideBar/SideBarFooter.js.map +1 -1
- package/dist/client/web/CreeveyView/SideBar/SideBarHeader.js +32 -12
- package/dist/client/web/CreeveyView/SideBar/SideBarHeader.js.map +1 -1
- package/dist/client/web/CreeveyView/SideBar/SuiteLink.d.ts +6 -6
- package/dist/client/web/CreeveyView/SideBar/SuiteLink.js +20 -11
- package/dist/client/web/CreeveyView/SideBar/SuiteLink.js.map +1 -1
- package/dist/client/web/CreeveyView/SideBar/TestLink.js +20 -11
- package/dist/client/web/CreeveyView/SideBar/TestLink.js.map +1 -1
- package/dist/client/web/CreeveyView/SideBar/TestStatusIcon.d.ts +2 -2
- package/dist/client/web/CreeveyView/SideBar/TestStatusIcon.js +2 -2
- package/dist/client/web/CreeveyView/SideBar/TestStatusIcon.js.map +1 -1
- package/dist/client/web/CreeveyView/SideBar/TestsStatus.d.ts +2 -2
- package/dist/client/web/CreeveyView/SideBar/TestsStatus.js +3 -2
- package/dist/client/web/CreeveyView/SideBar/TestsStatus.js.map +1 -1
- package/dist/client/web/CreeveyView/SideBar/Toggle.js +1 -1
- package/dist/client/web/CreeveyView/SideBar/Toggle.js.map +1 -1
- package/dist/client/web/KeyboardEventsContext.js +17 -7
- package/dist/client/web/KeyboardEventsContext.js.map +1 -1
- package/dist/client/web/assets/index-CtSq3IhG.js +518 -0
- package/dist/client/web/index.html +1 -1
- package/dist/client/web/index.js +26 -11
- package/dist/client/web/index.js.map +1 -1
- package/dist/client/web/themes.d.ts +2 -0
- package/dist/client/web/themes.js +22 -0
- package/dist/client/web/themes.js.map +1 -0
- package/dist/creevey.d.ts +1 -1
- package/dist/creevey.js +122 -41
- package/dist/creevey.js.map +1 -1
- package/dist/index.d.ts +1 -0
- package/dist/playwright/generator.d.ts +25 -0
- package/dist/playwright/generator.js +243 -0
- package/dist/playwright/generator.js.map +1 -0
- package/dist/playwright/helpers.d.ts +2 -0
- package/dist/playwright/helpers.js +29 -0
- package/dist/playwright/helpers.js.map +1 -0
- package/dist/playwright/reporter.d.ts +83 -0
- package/dist/playwright/reporter.js +334 -0
- package/dist/playwright/reporter.js.map +1 -0
- package/dist/playwright/setup.d.ts +3 -0
- package/dist/playwright/setup.js +72 -0
- package/dist/playwright/setup.js.map +1 -0
- package/dist/playwright.d.ts +1 -0
- package/dist/playwright.js +3 -1
- package/dist/playwright.js.map +1 -1
- package/dist/server/compare.d.ts +18 -0
- package/dist/server/compare.js +182 -0
- package/dist/server/compare.js.map +1 -0
- package/dist/server/config.d.ts +3 -3
- package/dist/server/config.js +75 -8
- package/dist/server/config.js.map +1 -1
- package/dist/server/connection.d.ts +3 -0
- package/dist/server/connection.js +28 -0
- package/dist/server/connection.js.map +1 -0
- package/dist/server/docker.d.ts +1 -1
- package/dist/server/docker.js +54 -32
- package/dist/server/docker.js.map +1 -1
- package/dist/server/index.d.ts +2 -2
- package/dist/server/index.js +161 -64
- package/dist/server/index.js.map +1 -1
- package/dist/server/master/api.d.ts +11 -6
- package/dist/server/master/api.js +88 -25
- package/dist/server/master/api.js.map +1 -1
- package/dist/server/master/handlers/capture-handler.d.ts +5 -0
- package/dist/server/master/handlers/capture-handler.js +25 -0
- package/dist/server/master/handlers/capture-handler.js.map +1 -0
- package/dist/server/master/handlers/index.d.ts +4 -0
- package/dist/server/master/handlers/index.js +21 -0
- package/dist/server/master/handlers/index.js.map +1 -0
- package/dist/server/master/handlers/ping-handler.d.ts +2 -0
- package/dist/server/master/handlers/ping-handler.js +8 -0
- package/dist/server/master/handlers/ping-handler.js.map +1 -0
- package/dist/server/master/handlers/static-handler.d.ts +1 -0
- package/dist/server/master/handlers/static-handler.js +20 -0
- package/dist/server/master/handlers/static-handler.js.map +1 -0
- package/dist/server/master/handlers/stories-handler.d.ts +4 -0
- package/dist/server/master/handlers/stories-handler.js +24 -0
- package/dist/server/master/handlers/stories-handler.js.map +1 -0
- package/dist/server/master/master.js +7 -24
- package/dist/server/master/master.js.map +1 -1
- package/dist/server/master/pool.d.ts +1 -0
- package/dist/server/master/pool.js +5 -3
- package/dist/server/master/pool.js.map +1 -1
- package/dist/server/master/queue.d.ts +1 -1
- package/dist/server/master/queue.js +14 -6
- package/dist/server/master/queue.js.map +1 -1
- package/dist/server/master/runner.d.ts +6 -6
- package/dist/server/master/runner.js +98 -130
- package/dist/server/master/runner.js.map +1 -1
- package/dist/server/master/server.d.ts +1 -1
- package/dist/server/master/server.js +193 -88
- package/dist/server/master/server.js.map +1 -1
- package/dist/server/master/start.d.ts +1 -2
- package/dist/server/master/start.js +13 -29
- package/dist/server/master/start.js.map +1 -1
- package/dist/server/master/testsManager.d.ts +81 -0
- package/dist/server/master/testsManager.js +282 -0
- package/dist/server/master/testsManager.js.map +1 -0
- package/dist/server/playwright/docker-file.d.ts +1 -1
- package/dist/server/playwright/docker-file.js +17 -8
- package/dist/server/playwright/docker-file.js.map +1 -1
- package/dist/server/playwright/docker.d.ts +2 -1
- package/dist/server/playwright/docker.js +10 -2
- package/dist/server/playwright/docker.js.map +1 -1
- package/dist/server/playwright/index-source.mjs +16 -0
- package/dist/server/playwright/internal.d.ts +7 -7
- package/dist/server/playwright/internal.js +137 -79
- package/dist/server/playwright/internal.js.map +1 -1
- package/dist/server/playwright/webdriver.d.ts +3 -3
- package/dist/server/playwright/webdriver.js +0 -6
- package/dist/server/playwright/webdriver.js.map +1 -1
- package/dist/server/providers/browser.js +4 -3
- package/dist/server/providers/browser.js.map +1 -1
- package/dist/server/providers/hybrid.js +2 -2
- package/dist/server/providers/hybrid.js.map +1 -1
- package/dist/server/report.d.ts +10 -0
- package/dist/server/report.js +45 -0
- package/dist/server/report.js.map +1 -0
- package/dist/server/reporters/creevey.d.ts +7 -0
- package/dist/server/reporters/creevey.js +63 -0
- package/dist/server/reporters/creevey.js.map +1 -0
- package/dist/server/reporters/index.d.ts +2 -0
- package/dist/server/reporters/index.js +16 -0
- package/dist/server/reporters/index.js.map +1 -0
- package/dist/server/reporters/junit.d.ts +16 -0
- package/dist/server/reporters/junit.js +167 -0
- package/dist/server/reporters/junit.js.map +1 -0
- package/dist/server/reporters/teamcity.d.ts +7 -0
- package/dist/server/reporters/teamcity.js +60 -0
- package/dist/server/reporters/teamcity.js.map +1 -0
- package/dist/server/selenium/internal.d.ts +3 -3
- package/dist/server/selenium/internal.js +48 -34
- package/dist/server/selenium/internal.js.map +1 -1
- package/dist/server/selenium/selenoid.js +12 -6
- package/dist/server/selenium/selenoid.js.map +1 -1
- package/dist/server/selenium/webdriver.d.ts +3 -3
- package/dist/server/selenium/webdriver.js +4 -8
- package/dist/server/selenium/webdriver.js.map +1 -1
- package/dist/server/shutdown.d.ts +1 -0
- package/dist/server/shutdown.js +23 -0
- package/dist/server/shutdown.js.map +1 -0
- package/dist/server/stories.d.ts +0 -1
- package/dist/server/stories.js +0 -12
- package/dist/server/stories.js.map +1 -1
- package/dist/server/telemetry.js +3 -3
- package/dist/server/telemetry.js.map +1 -1
- package/dist/server/testsFiles/parser.js +45 -5
- package/dist/server/testsFiles/parser.js.map +1 -1
- package/dist/server/utils.d.ts +23 -0
- package/dist/server/utils.js +113 -13
- package/dist/server/utils.js.map +1 -1
- package/dist/server/webdriver.d.ts +1 -1
- package/dist/server/worker/context.d.ts +3 -0
- package/dist/server/worker/context.js +15 -0
- package/dist/server/worker/context.js.map +1 -0
- package/dist/server/worker/match-image.d.ts +8 -12
- package/dist/server/worker/match-image.js +11 -178
- package/dist/server/worker/match-image.js.map +1 -1
- package/dist/server/worker/start.d.ts +2 -2
- package/dist/server/worker/start.js +27 -63
- package/dist/server/worker/start.js.map +1 -1
- package/dist/shared/index.d.ts +1 -1
- package/dist/shared/index.js +9 -7
- package/dist/shared/index.js.map +1 -1
- package/dist/types.d.ts +84 -43
- package/dist/types.js +65 -1
- package/dist/types.js.map +1 -1
- package/docs/cli.md +80 -0
- package/docs/config.md +179 -165
- package/docs/examples/playwright-reporer/playwright.config.ts +37 -0
- package/docs/migration-0.9-to-0.10.md +144 -0
- package/docs/playwright-reporter.md +357 -0
- package/docs/storybook.md +60 -0
- package/docs/tests.md +50 -45
- package/package.json +78 -83
- package/playwright.config.mts +46 -0
- package/src/client/addon/components/Addon.tsx +1 -1
- package/src/client/addon/components/Panel.tsx +2 -2
- package/src/client/addon/components/TestSelect.tsx +2 -2
- package/src/client/addon/components/Tools.tsx +2 -2
- package/src/client/addon/controller.ts +4 -4
- package/src/client/addon/makeDecorator.ts +69 -0
- package/src/client/addon/manager.ts +38 -37
- package/src/client/addon/preset.ts +2 -1
- package/src/client/addon/withCreevey.ts +10 -18
- package/src/client/shared/components/ImagesView/BlendView.tsx +1 -1
- package/src/client/shared/components/ImagesView/ImagesView.tsx +1 -1
- package/src/client/shared/components/ImagesView/SideBySideView.tsx +2 -2
- package/src/client/shared/components/ImagesView/SlideView.tsx +2 -2
- package/src/client/shared/components/ImagesView/SwapView.tsx +2 -2
- package/src/client/shared/components/ImagesView/common.ts +1 -1
- package/src/client/shared/components/PageFooter/PageFooter.tsx +1 -1
- package/src/client/shared/components/PageFooter/Paging.tsx +1 -1
- package/src/client/shared/components/PageHeader/ImagePreview.tsx +1 -1
- package/src/client/shared/components/PageHeader/PageHeader.tsx +23 -7
- package/src/client/shared/components/ResultsPage.tsx +6 -4
- package/src/client/shared/creeveyClientApi.ts +19 -1
- package/src/client/shared/helpers.ts +4 -24
- package/src/client/web/CreeveyApp.tsx +5 -2
- package/src/client/web/CreeveyContext.tsx +2 -0
- package/src/client/web/CreeveyLoader.tsx +2 -2
- package/src/client/web/CreeveyView/SideBar/Checkbox.tsx +3 -3
- package/src/client/web/CreeveyView/SideBar/Search.tsx +1 -1
- package/src/client/web/CreeveyView/SideBar/SideBar.tsx +11 -6
- package/src/client/web/CreeveyView/SideBar/SideBarFooter.tsx +21 -19
- package/src/client/web/CreeveyView/SideBar/SideBarHeader.tsx +20 -5
- package/src/client/web/CreeveyView/SideBar/SuiteLink.tsx +10 -8
- package/src/client/web/CreeveyView/SideBar/TestLink.tsx +9 -7
- package/src/client/web/CreeveyView/SideBar/TestStatusIcon.tsx +2 -2
- package/src/client/web/CreeveyView/SideBar/TestsStatus.tsx +3 -2
- package/src/client/web/CreeveyView/SideBar/Toggle.tsx +1 -1
- package/src/client/web/index.tsx +10 -5
- package/src/client/web/themes.ts +24 -0
- package/src/creevey.ts +92 -38
- package/src/playwright/generator.ts +322 -0
- package/src/playwright/helpers.ts +31 -0
- package/src/playwright/reporter.ts +381 -0
- package/src/playwright/setup.ts +84 -0
- package/src/playwright.ts +1 -0
- package/src/server/compare.ts +260 -0
- package/src/server/config.ts +52 -9
- package/src/server/connection.ts +26 -0
- package/src/server/docker.ts +62 -34
- package/src/server/index.ts +161 -79
- package/src/server/master/api.ts +94 -28
- package/src/server/master/handlers/capture-handler.ts +20 -0
- package/src/server/master/handlers/index.ts +4 -0
- package/src/server/master/handlers/ping-handler.ts +6 -0
- package/src/server/master/handlers/static-handler.ts +16 -0
- package/src/server/master/handlers/stories-handler.ts +20 -0
- package/src/server/master/master.ts +10 -27
- package/src/server/master/pool.ts +7 -3
- package/src/server/master/queue.ts +21 -7
- package/src/server/master/runner.ts +123 -134
- package/src/server/master/server.ts +214 -101
- package/src/server/master/start.ts +19 -41
- package/src/server/master/testsManager.ts +316 -0
- package/src/server/playwright/docker-file.ts +20 -8
- package/src/server/playwright/docker.ts +16 -3
- package/src/server/playwright/index-source.mjs +16 -0
- package/src/server/playwright/internal.ts +169 -96
- package/src/server/playwright/webdriver.ts +4 -10
- package/src/server/providers/browser.ts +4 -3
- package/src/server/providers/hybrid.ts +2 -3
- package/src/server/report.ts +51 -0
- package/src/server/reporters/creevey.ts +71 -0
- package/src/server/reporters/index.ts +11 -0
- package/src/server/reporters/junit.ts +207 -0
- package/src/server/reporters/teamcity.ts +74 -0
- package/src/server/selenium/internal.ts +62 -45
- package/src/server/selenium/selenoid.ts +13 -6
- package/src/server/selenium/webdriver.ts +8 -12
- package/src/server/shutdown.ts +19 -0
- package/src/server/stories.ts +1 -12
- package/src/server/telemetry.ts +3 -3
- package/src/server/testsFiles/parser.ts +52 -4
- package/src/server/utils.ts +123 -14
- package/src/server/webdriver.ts +1 -1
- package/src/server/worker/context.ts +14 -0
- package/src/server/worker/match-image.ts +16 -248
- package/src/server/worker/start.ts +32 -75
- package/src/shared/index.ts +10 -8
- package/src/types.ts +91 -58
- package/types/global.d.ts +1 -0
- package/dist/client/web/assets/index-BE9CL5_G.js +0 -591
- package/dist/server/reporter.d.ts +0 -26
- package/dist/server/reporter.js +0 -108
- package/dist/server/reporter.js.map +0 -1
- package/dist/server/update.d.ts +0 -2
- package/dist/server/update.js +0 -53
- package/dist/server/update.js.map +0 -1
- package/src/server/reporter.ts +0 -139
- package/src/server/update.ts +0 -74
@@ -0,0 +1,381 @@
|
|
1
|
+
import fs from 'fs/promises';
|
2
|
+
import { createHash } from 'crypto';
|
3
|
+
import type { Reporter, FullConfig, Suite, TestCase, TestResult, TestStep } from '@playwright/test/reporter';
|
4
|
+
import {
|
5
|
+
type ServerTest,
|
6
|
+
type TestMeta,
|
7
|
+
type TestStatus,
|
8
|
+
type TestResult as CreeveyTestResult,
|
9
|
+
isDefined,
|
10
|
+
Images,
|
11
|
+
} from '../types.js';
|
12
|
+
import { TestsManager } from '../server/master/testsManager.js';
|
13
|
+
import { CreeveyApi } from '../server/master/api.js';
|
14
|
+
import { copyStatics } from '../server/utils.js';
|
15
|
+
import { start } from '../server/master/server.js';
|
16
|
+
import assert from 'assert';
|
17
|
+
|
18
|
+
/**
|
19
|
+
* Simple async queue to handle operations in sequence without returning promises
|
20
|
+
* from reporter methods that should be synchronous
|
21
|
+
*/
|
22
|
+
class AsyncQueue {
|
23
|
+
private queue: Promise<void>;
|
24
|
+
|
25
|
+
constructor() {
|
26
|
+
this.queue = Promise.resolve();
|
27
|
+
}
|
28
|
+
|
29
|
+
/**
|
30
|
+
* Add an async operation to the queue
|
31
|
+
* @param operation Async operation to execute
|
32
|
+
*/
|
33
|
+
enqueue(operation: () => Promise<void>): void {
|
34
|
+
this.queue = this.queue.then(operation).catch((error: unknown) => {
|
35
|
+
console.error(`Error in async queue: ${error instanceof Error ? error.message : String(error)}`);
|
36
|
+
});
|
37
|
+
}
|
38
|
+
|
39
|
+
/**
|
40
|
+
* Wait for all operations in the queue to complete
|
41
|
+
*/
|
42
|
+
async waitForCompletion(): Promise<void> {
|
43
|
+
await this.queue;
|
44
|
+
}
|
45
|
+
}
|
46
|
+
|
47
|
+
/**
|
48
|
+
* CreeveyPlaywrightReporter is a Playwright reporter that integrates with Creevey
|
49
|
+
* to provide visual testing capabilities and use Creevey's UI for reviewing and approving screenshots.
|
50
|
+
*/
|
51
|
+
class CreeveyPlaywrightReporter implements Reporter {
|
52
|
+
private testsManager: TestsManager | null = null;
|
53
|
+
private api: CreeveyApi | null = null;
|
54
|
+
private port: number;
|
55
|
+
private debug: boolean;
|
56
|
+
private testIdMap = new Map<string, string>(); // Maps Playwright test IDs to Creevey test IDs
|
57
|
+
private asyncQueue = new AsyncQueue();
|
58
|
+
|
59
|
+
/**
|
60
|
+
* Creates a new instance of the CreeveyPlaywrightReporter
|
61
|
+
* @param options Configuration options for the reporter
|
62
|
+
*/
|
63
|
+
constructor(options?: { port?: number; debug?: boolean }) {
|
64
|
+
this.port = options?.port ?? 3000;
|
65
|
+
this.debug = options?.debug ?? false;
|
66
|
+
}
|
67
|
+
|
68
|
+
/**
|
69
|
+
* Called when the test run starts
|
70
|
+
* @param config Playwright configuration
|
71
|
+
* @param suite Test suite information
|
72
|
+
*/
|
73
|
+
onBegin(config: FullConfig, suite: Suite): void {
|
74
|
+
this.logDebug('CreeveyPlaywrightReporter started');
|
75
|
+
|
76
|
+
const [snapshotDir, ...restSnapshotDirs] = [...new Set(config.projects.map((project) => project.snapshotDir))];
|
77
|
+
const [outputDir, ...restOutputDirs] = [...new Set(config.projects.map((project) => project.outputDir))];
|
78
|
+
|
79
|
+
assert(restSnapshotDirs.length === 0, 'Currently multiple snapshot directories are not supported');
|
80
|
+
assert(restOutputDirs.length === 0, 'Currently multiple output directories are not supported');
|
81
|
+
|
82
|
+
// Initialize TestsManager
|
83
|
+
this.testsManager = new TestsManager(snapshotDir, outputDir);
|
84
|
+
|
85
|
+
// Use the async queue to handle initialization without returning a promise
|
86
|
+
this.asyncQueue.enqueue(async () => {
|
87
|
+
assert(this.testsManager, 'TestsManager is not initialized');
|
88
|
+
|
89
|
+
try {
|
90
|
+
await fs.mkdir(outputDir, { recursive: true });
|
91
|
+
await copyStatics(outputDir);
|
92
|
+
} catch (error) {
|
93
|
+
this.logError(
|
94
|
+
`Failed to initialize report directory: ${error instanceof Error ? error.message : String(error)}`,
|
95
|
+
);
|
96
|
+
}
|
97
|
+
|
98
|
+
// Start server API
|
99
|
+
try {
|
100
|
+
const resolveApi = start(outputDir, this.port, true);
|
101
|
+
|
102
|
+
// Create and connect the API
|
103
|
+
this.api = new CreeveyApi(this.testsManager);
|
104
|
+
resolveApi(this.api);
|
105
|
+
|
106
|
+
const testsList = suite
|
107
|
+
.allTests()
|
108
|
+
.map((test) => {
|
109
|
+
const creeveyTest = this.mapToCreeveyTest(test);
|
110
|
+
if (!creeveyTest) return;
|
111
|
+
|
112
|
+
this.testIdMap.set(test.id, creeveyTest.id);
|
113
|
+
|
114
|
+
return creeveyTest;
|
115
|
+
})
|
116
|
+
.filter(isDefined);
|
117
|
+
|
118
|
+
const tests: Record<string, ServerTest> = {};
|
119
|
+
for (const test of testsList) {
|
120
|
+
tests[test.id] = test;
|
121
|
+
}
|
122
|
+
|
123
|
+
this.testsManager.updateTests(tests);
|
124
|
+
|
125
|
+
console.log(`Creevey report server started at http://localhost:${this.port}`);
|
126
|
+
} catch (error) {
|
127
|
+
this.logError(`Could not start Creevey server: ${error instanceof Error ? error.message : String(error)}`);
|
128
|
+
console.log('Screenshots will still be captured but UI will not be available');
|
129
|
+
}
|
130
|
+
});
|
131
|
+
}
|
132
|
+
|
133
|
+
/**
|
134
|
+
* Called when a test begins
|
135
|
+
* @param test Test case information
|
136
|
+
* @param result Test result (initially empty)
|
137
|
+
*/
|
138
|
+
onTestBegin(test: TestCase, _result: TestResult): void {
|
139
|
+
try {
|
140
|
+
assert(this.testsManager, 'TestsManager is not initialized');
|
141
|
+
|
142
|
+
const creeveyTestId = this.testIdMap.get(test.id);
|
143
|
+
|
144
|
+
if (creeveyTestId) {
|
145
|
+
// Update test status to running
|
146
|
+
this.testsManager.updateTestStatus(creeveyTestId, 'running');
|
147
|
+
|
148
|
+
this.logDebug(`Test started: ${test.title} (${creeveyTestId})`);
|
149
|
+
}
|
150
|
+
} catch (error) {
|
151
|
+
this.logError(`Error in onTestBegin: ${error instanceof Error ? error.message : String(error)}`);
|
152
|
+
}
|
153
|
+
}
|
154
|
+
|
155
|
+
/**
|
156
|
+
* Called when a test step begins
|
157
|
+
* @param test Test case information
|
158
|
+
* @param result Test result
|
159
|
+
* @param step Test step information
|
160
|
+
*/
|
161
|
+
onStepBegin(test: TestCase, _result: TestResult, step: TestStep): void {
|
162
|
+
/*
|
163
|
+
[Creevey Reporter] Step started: browserType.launch in test: 100 X 100 Vs 2000 X 100
|
164
|
+
[Creevey Reporter] Step started: browserType.launch in test: Side By Side
|
165
|
+
[Creevey Reporter] Step started: browserType.launch in test: Full
|
166
|
+
*/
|
167
|
+
if (this.debug) {
|
168
|
+
this.logDebug(`Step started: ${step.title} in test: ${test.title}`);
|
169
|
+
}
|
170
|
+
}
|
171
|
+
|
172
|
+
/**
|
173
|
+
* Called when a test step ends
|
174
|
+
* @param test Test case information
|
175
|
+
* @param result Test result
|
176
|
+
* @param step Test step information
|
177
|
+
*/
|
178
|
+
onStepEnd(_test: TestCase, _result: TestResult, step: TestStep): void {
|
179
|
+
try {
|
180
|
+
// If step has attachments, process them
|
181
|
+
if (step.attachments.length > 0) {
|
182
|
+
this.logDebug(`Processing ${step.attachments.length} attachments from step: ${step.title}`);
|
183
|
+
|
184
|
+
// We'll process attachments in onTestEnd for simplicity in this initial implementation
|
185
|
+
}
|
186
|
+
} catch (error) {
|
187
|
+
this.logError(`Error in onStepEnd: ${error instanceof Error ? error.message : String(error)}`);
|
188
|
+
}
|
189
|
+
}
|
190
|
+
|
191
|
+
/**
|
192
|
+
* Called when a test ends
|
193
|
+
* @param test Test case information
|
194
|
+
* @param result Test result
|
195
|
+
*/
|
196
|
+
onTestEnd(test: TestCase, result: TestResult): void {
|
197
|
+
const creeveyTestId = this.testIdMap.get(test.id);
|
198
|
+
|
199
|
+
// Use the async queue to handle result processing without returning a promise
|
200
|
+
this.asyncQueue.enqueue(async () => {
|
201
|
+
try {
|
202
|
+
// Process test results and screenshots
|
203
|
+
await this.processTestResult(test, result);
|
204
|
+
|
205
|
+
if (creeveyTestId) {
|
206
|
+
this.logDebug(`Test ended: ${test.title} (${creeveyTestId}) with status: ${result.status}`);
|
207
|
+
}
|
208
|
+
} catch (error) {
|
209
|
+
this.logError(`Error in onTestEnd: ${error instanceof Error ? error.message : String(error)}`);
|
210
|
+
}
|
211
|
+
});
|
212
|
+
}
|
213
|
+
|
214
|
+
/**
|
215
|
+
* Called when the test run ends
|
216
|
+
* @param result The overall test run result
|
217
|
+
*/
|
218
|
+
async onEnd(result: { status: 'passed' | 'failed' | 'timedout' | 'interrupted' }): Promise<void> {
|
219
|
+
// Use the async queue to handle final operations without returning a promise
|
220
|
+
this.asyncQueue.enqueue(async () => {
|
221
|
+
try {
|
222
|
+
assert(this.testsManager, 'TestsManager is not initialized');
|
223
|
+
|
224
|
+
// Save test data
|
225
|
+
await this.testsManager.saveTestData();
|
226
|
+
|
227
|
+
this.logDebug(`Test run ended with status: ${result.status}`);
|
228
|
+
// TODO: Tell how to run reporter `yarn creevey update ./report --ui`
|
229
|
+
} catch (error) {
|
230
|
+
this.logError(`Error during reporter cleanup: ${error instanceof Error ? error.message : String(error)}`);
|
231
|
+
}
|
232
|
+
});
|
233
|
+
|
234
|
+
// Wait for all previous operations to complete
|
235
|
+
await this.asyncQueue.waitForCompletion();
|
236
|
+
}
|
237
|
+
|
238
|
+
/**
|
239
|
+
* Maps a Playwright test to a Creevey test format
|
240
|
+
* @param test Playwright test case
|
241
|
+
* @returns Creevey test object or null if mapping fails
|
242
|
+
*/
|
243
|
+
private mapToCreeveyTest(test: TestCase): ServerTest | null {
|
244
|
+
try {
|
245
|
+
const storyName = test.title;
|
246
|
+
const storyTitle = test.parent.title;
|
247
|
+
const projectName = test.parent.project()?.name ?? 'chromium';
|
248
|
+
const testPath = [storyTitle, storyName, projectName];
|
249
|
+
const { description: storyId } = test.annotations.find((annotation) => annotation.type === 'storyId') ?? {};
|
250
|
+
|
251
|
+
// Generate a unique test ID
|
252
|
+
const testId = createHash('sha1').update(testPath.join('/')).digest('hex');
|
253
|
+
|
254
|
+
// Create the test metadata
|
255
|
+
const testMeta: TestMeta = {
|
256
|
+
id: testId,
|
257
|
+
storyPath: [...storyTitle.split('/').map((x) => x.trim()), storyName],
|
258
|
+
browser: projectName,
|
259
|
+
storyId: storyId ?? '',
|
260
|
+
};
|
261
|
+
|
262
|
+
// Create a stub ServerTest object
|
263
|
+
// This is missing the story and fn properties which would be used in a real Creevey test
|
264
|
+
// However, for our reporter purposes, we just need the metadata
|
265
|
+
const serverTest: ServerTest = {
|
266
|
+
...testMeta,
|
267
|
+
story: {
|
268
|
+
parameters: {},
|
269
|
+
initialArgs: {},
|
270
|
+
argTypes: {},
|
271
|
+
component: '',
|
272
|
+
componentId: '',
|
273
|
+
name: storyName,
|
274
|
+
tags: [],
|
275
|
+
title: storyTitle,
|
276
|
+
kind: storyTitle,
|
277
|
+
id: storyId ?? '',
|
278
|
+
story: storyName,
|
279
|
+
}, // Placeholder
|
280
|
+
fn: async () => {
|
281
|
+
/* Empty function as placeholder */
|
282
|
+
}, // Placeholder
|
283
|
+
};
|
284
|
+
|
285
|
+
return serverTest;
|
286
|
+
} catch (error) {
|
287
|
+
this.logError(`Error mapping test to Creevey format: ${error instanceof Error ? error.message : String(error)}`);
|
288
|
+
return null;
|
289
|
+
}
|
290
|
+
}
|
291
|
+
|
292
|
+
/**
|
293
|
+
* Process a test result and any attachments
|
294
|
+
* @param test Playwright test case
|
295
|
+
* @param result Playwright test result
|
296
|
+
*/
|
297
|
+
private async processTestResult(test: TestCase, result: TestResult): Promise<void> {
|
298
|
+
const creeveyTestId = this.testIdMap.get(test.id);
|
299
|
+
if (!creeveyTestId) {
|
300
|
+
this.logError(`No Creevey test ID found for test: ${test.title}`);
|
301
|
+
return Promise.resolve();
|
302
|
+
}
|
303
|
+
|
304
|
+
assert(this.testsManager, 'TestsManager is not initialized');
|
305
|
+
|
306
|
+
// Determine test status
|
307
|
+
let status: TestStatus;
|
308
|
+
switch (result.status) {
|
309
|
+
case 'passed':
|
310
|
+
status = 'success';
|
311
|
+
break;
|
312
|
+
case 'failed':
|
313
|
+
case 'timedOut':
|
314
|
+
status = 'failed';
|
315
|
+
break;
|
316
|
+
default:
|
317
|
+
status = 'unknown';
|
318
|
+
}
|
319
|
+
|
320
|
+
// Process attachments
|
321
|
+
const images: Record<string, Images> = {};
|
322
|
+
const attachmentPaths: string[] = [];
|
323
|
+
const projectName = test.parent.project()?.name ?? 'chromium';
|
324
|
+
|
325
|
+
for (const attachment of result.attachments) {
|
326
|
+
const { name, path: attachmentPath } = attachment;
|
327
|
+
|
328
|
+
if (!attachmentPath) continue;
|
329
|
+
|
330
|
+
attachmentPaths.push(attachmentPath);
|
331
|
+
|
332
|
+
switch (true) {
|
333
|
+
case name.includes('actual'): {
|
334
|
+
images[projectName] = { ...images[projectName], actual: name };
|
335
|
+
break;
|
336
|
+
}
|
337
|
+
case name.includes('expect'): {
|
338
|
+
images[projectName] = { ...images[projectName], expect: name };
|
339
|
+
break;
|
340
|
+
}
|
341
|
+
case name.includes('diff'): {
|
342
|
+
images[projectName] = { ...images[projectName], diff: name };
|
343
|
+
break;
|
344
|
+
}
|
345
|
+
}
|
346
|
+
}
|
347
|
+
|
348
|
+
// Update test status and result
|
349
|
+
const testResult: CreeveyTestResult = {
|
350
|
+
status: status === 'success' ? 'success' : 'failed',
|
351
|
+
retries: result.retry,
|
352
|
+
images,
|
353
|
+
error: result.error?.message ?? undefined,
|
354
|
+
duration: result.duration,
|
355
|
+
attachments: attachmentPaths,
|
356
|
+
browserName: projectName,
|
357
|
+
};
|
358
|
+
|
359
|
+
this.testsManager.updateTestStatus(creeveyTestId, status, testResult);
|
360
|
+
}
|
361
|
+
|
362
|
+
/**
|
363
|
+
* Logs a debug message if debug mode is enabled
|
364
|
+
* @param message Message to log
|
365
|
+
*/
|
366
|
+
private logDebug(message: string): void {
|
367
|
+
if (this.debug) {
|
368
|
+
console.log(`[Creevey Reporter] ${message}`);
|
369
|
+
}
|
370
|
+
}
|
371
|
+
|
372
|
+
/**
|
373
|
+
* Logs an error message
|
374
|
+
* @param message Error message to log
|
375
|
+
*/
|
376
|
+
private logError(message: string): void {
|
377
|
+
console.error(`[Creevey Reporter] ERROR: ${message}`);
|
378
|
+
}
|
379
|
+
}
|
380
|
+
|
381
|
+
export default CreeveyPlaywrightReporter;
|
@@ -0,0 +1,84 @@
|
|
1
|
+
import path from 'path';
|
2
|
+
import assert from 'assert';
|
3
|
+
import { mkdir, writeFile } from 'fs/promises';
|
4
|
+
import { chromium, firefox, webkit, Page, FullConfig } from '@playwright/test';
|
5
|
+
import { StoriesRaw } from '../types';
|
6
|
+
import { getCreeveyCache } from '../server/utils';
|
7
|
+
import { appendIframePath } from '../server/webdriver';
|
8
|
+
import { waitForStorybookReady } from './helpers';
|
9
|
+
|
10
|
+
// This function will fetch stories and cache them or an error if fetching fails.
|
11
|
+
// It's intended to be called once before tests that depend on stories are defined.
|
12
|
+
async function ensureStoriesFetched(page: Page, storybookUrl: string): Promise<StoriesRaw> {
|
13
|
+
try {
|
14
|
+
console.log(`Fetching stories from: ${storybookUrl}`);
|
15
|
+
|
16
|
+
await page.goto(appendIframePath(storybookUrl), { waitUntil: 'networkidle', timeout: 60000 });
|
17
|
+
await waitForStorybookReady(page);
|
18
|
+
|
19
|
+
// TODO: Inline `serializeRawStories` to serialize creevey skip parameters
|
20
|
+
const fetchedStories = await page.evaluate<StoriesRaw | undefined>(() => window.__STORYBOOK_PREVIEW__.extract());
|
21
|
+
|
22
|
+
if (!fetchedStories || Object.keys(fetchedStories).length === 0) {
|
23
|
+
throw new Error('No stories were found or cached from Storybook.');
|
24
|
+
}
|
25
|
+
console.log(`Successfully fetched and cached ${Object.keys(fetchedStories).length} stories.`);
|
26
|
+
|
27
|
+
return fetchedStories;
|
28
|
+
} catch (error: unknown) {
|
29
|
+
console.error('Error fetching stories');
|
30
|
+
throw error;
|
31
|
+
}
|
32
|
+
}
|
33
|
+
|
34
|
+
const browsers = {
|
35
|
+
chromium,
|
36
|
+
firefox,
|
37
|
+
webkit,
|
38
|
+
};
|
39
|
+
|
40
|
+
// TODO: Setup should generate test files for each story file (component)
|
41
|
+
// TODO: Add support for multiple storybook urls
|
42
|
+
async function globalSetup(config: FullConfig) {
|
43
|
+
const storybookUrl = config.webServer?.url;
|
44
|
+
const { defaultBrowserType = 'chromium', browserName = defaultBrowserType } = config.projects[0].use;
|
45
|
+
|
46
|
+
assert(storybookUrl, 'Storybook URL is required');
|
47
|
+
|
48
|
+
const cacheDir = await getCreeveyCache();
|
49
|
+
assert(cacheDir, 'Cache directory not found');
|
50
|
+
|
51
|
+
await mkdir(cacheDir, { recursive: true });
|
52
|
+
|
53
|
+
process.env.CREEVEY_CACHE_DIR = cacheDir;
|
54
|
+
|
55
|
+
const browser = await browsers[browserName].launch();
|
56
|
+
const context = await browser.newContext();
|
57
|
+
const page = await context.newPage();
|
58
|
+
|
59
|
+
if (process.env.PWDEBUG) {
|
60
|
+
await context.tracing.start({ name: 'storybook-setup' });
|
61
|
+
}
|
62
|
+
|
63
|
+
try {
|
64
|
+
const stories = await ensureStoriesFetched(page, storybookUrl);
|
65
|
+
|
66
|
+
await writeFile(path.join(cacheDir, 'stories.json'), JSON.stringify(stories, null, 2));
|
67
|
+
} catch (error) {
|
68
|
+
console.error('Error in globalSetup:', error);
|
69
|
+
|
70
|
+
if (process.env.PWDEBUG) {
|
71
|
+
const tracePath = path.join(cacheDir, 'storybook-setup-trace.zip');
|
72
|
+
|
73
|
+
console.log('Trace is saved to:', tracePath);
|
74
|
+
|
75
|
+
await context.tracing.stop({ path: tracePath });
|
76
|
+
}
|
77
|
+
|
78
|
+
throw error;
|
79
|
+
} finally {
|
80
|
+
await browser.close();
|
81
|
+
}
|
82
|
+
}
|
83
|
+
|
84
|
+
export default globalSetup;
|
package/src/playwright.ts
CHANGED