creevey 0.10.0-beta.0 → 0.10.0-beta.10

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.
Files changed (159) hide show
  1. package/dist/client/addon/components/Panel.js +2 -2
  2. package/dist/client/addon/components/Panel.js.map +1 -1
  3. package/dist/client/addon/controller.js +4 -5
  4. package/dist/client/addon/controller.js.map +1 -1
  5. package/dist/client/addon/withCreevey.js +18 -34
  6. package/dist/client/addon/withCreevey.js.map +1 -1
  7. package/dist/client/shared/components/ImagesView/SwapView.js +12 -0
  8. package/dist/client/shared/components/ImagesView/SwapView.js.map +1 -1
  9. package/dist/client/shared/components/PageHeader/ImagePreview.js +1 -0
  10. package/dist/client/shared/components/PageHeader/ImagePreview.js.map +1 -1
  11. package/dist/client/shared/components/ResultsPage.js +23 -5
  12. package/dist/client/shared/components/ResultsPage.js.map +1 -1
  13. package/dist/client/web/CreeveyApp.js +22 -6
  14. package/dist/client/web/CreeveyApp.js.map +1 -1
  15. package/dist/client/web/CreeveyContext.d.ts +5 -0
  16. package/dist/client/web/CreeveyContext.js +3 -0
  17. package/dist/client/web/CreeveyContext.js.map +1 -1
  18. package/dist/client/web/CreeveyView/SideBar/Search.js +2 -2
  19. package/dist/client/web/CreeveyView/SideBar/Search.js.map +1 -1
  20. package/dist/client/web/CreeveyView/SideBar/SideBar.js +1 -0
  21. package/dist/client/web/CreeveyView/SideBar/SideBar.js.map +1 -1
  22. package/dist/client/web/CreeveyView/SideBar/SideBarFooter.js +49 -6
  23. package/dist/client/web/CreeveyView/SideBar/SideBarFooter.js.map +1 -1
  24. package/dist/client/web/CreeveyView/SideBar/SuiteLink.js +1 -3
  25. package/dist/client/web/CreeveyView/SideBar/SuiteLink.js.map +1 -1
  26. package/dist/client/web/CreeveyView/SideBar/TestLink.js +1 -3
  27. package/dist/client/web/CreeveyView/SideBar/TestLink.js.map +1 -1
  28. package/dist/client/web/KeyboardEventsContext.d.ts +1 -8
  29. package/dist/client/web/KeyboardEventsContext.js +62 -57
  30. package/dist/client/web/KeyboardEventsContext.js.map +1 -1
  31. package/dist/client/web/assets/{index-DkmZfG9C.js → index-BE9CL5_G.js} +94 -94
  32. package/dist/client/web/index.html +1 -1
  33. package/dist/creevey.js +13 -5
  34. package/dist/creevey.js.map +1 -1
  35. package/dist/index.js +1 -5
  36. package/dist/index.js.map +1 -1
  37. package/dist/playwright.d.ts +2 -0
  38. package/dist/playwright.js +6 -0
  39. package/dist/playwright.js.map +1 -0
  40. package/dist/selenium.d.ts +2 -0
  41. package/dist/selenium.js +6 -0
  42. package/dist/selenium.js.map +1 -0
  43. package/dist/server/config.d.ts +1 -1
  44. package/dist/server/config.js +12 -5
  45. package/dist/server/config.js.map +1 -1
  46. package/dist/server/docker.js +2 -2
  47. package/dist/server/docker.js.map +1 -1
  48. package/dist/server/index.js +40 -4
  49. package/dist/server/index.js.map +1 -1
  50. package/dist/server/logger.d.ts +2 -1
  51. package/dist/server/logger.js +7 -3
  52. package/dist/server/logger.js.map +1 -1
  53. package/dist/server/master/api.js +1 -1
  54. package/dist/server/master/api.js.map +1 -1
  55. package/dist/server/master/pool.d.ts +3 -3
  56. package/dist/server/master/pool.js +10 -63
  57. package/dist/server/master/pool.js.map +1 -1
  58. package/dist/server/master/queue.d.ts +13 -0
  59. package/dist/server/master/queue.js +64 -0
  60. package/dist/server/master/queue.js.map +1 -0
  61. package/dist/server/master/runner.d.ts +1 -0
  62. package/dist/server/master/runner.js +4 -1
  63. package/dist/server/master/runner.js.map +1 -1
  64. package/dist/server/master/server.js +1 -1
  65. package/dist/server/master/server.js.map +1 -1
  66. package/dist/server/master/start.js +4 -4
  67. package/dist/server/master/start.js.map +1 -1
  68. package/dist/server/playwright/docker-file.js +12 -1
  69. package/dist/server/playwright/docker-file.js.map +1 -1
  70. package/dist/server/playwright/docker.d.ts +1 -1
  71. package/dist/server/playwright/docker.js +1 -6
  72. package/dist/server/playwright/docker.js.map +1 -1
  73. package/dist/server/playwright/internal.d.ts +2 -2
  74. package/dist/server/playwright/internal.js +56 -44
  75. package/dist/server/playwright/internal.js.map +1 -1
  76. package/dist/server/playwright/webdriver.d.ts +2 -1
  77. package/dist/server/playwright/webdriver.js +4 -1
  78. package/dist/server/playwright/webdriver.js.map +1 -1
  79. package/dist/server/providers/browser.js +2 -1
  80. package/dist/server/providers/browser.js.map +1 -1
  81. package/dist/server/providers/hybrid.js +1 -1
  82. package/dist/server/providers/hybrid.js.map +1 -1
  83. package/dist/server/reporter.js +4 -4
  84. package/dist/server/reporter.js.map +1 -1
  85. package/dist/server/selenium/internal.d.ts +3 -3
  86. package/dist/server/selenium/internal.js +126 -89
  87. package/dist/server/selenium/internal.js.map +1 -1
  88. package/dist/server/selenium/selenoid.js +2 -2
  89. package/dist/server/selenium/selenoid.js.map +1 -1
  90. package/dist/server/selenium/webdriver.d.ts +2 -1
  91. package/dist/server/selenium/webdriver.js +8 -1
  92. package/dist/server/selenium/webdriver.js.map +1 -1
  93. package/dist/server/telemetry.js +7 -3
  94. package/dist/server/telemetry.js.map +1 -1
  95. package/dist/server/utils.d.ts +2 -1
  96. package/dist/server/utils.js +13 -3
  97. package/dist/server/utils.js.map +1 -1
  98. package/dist/server/webdriver.d.ts +4 -4
  99. package/dist/server/webdriver.js +10 -9
  100. package/dist/server/webdriver.js.map +1 -1
  101. package/dist/server/worker/chai-image.d.ts +1 -2
  102. package/dist/server/worker/chai-image.js +4 -3
  103. package/dist/server/worker/chai-image.js.map +1 -1
  104. package/dist/server/worker/start.js +27 -39
  105. package/dist/server/worker/start.js.map +1 -1
  106. package/dist/types.d.ts +32 -21
  107. package/dist/types.js +13 -1
  108. package/dist/types.js.map +1 -1
  109. package/package.json +45 -38
  110. package/playwright.d.ts +2 -0
  111. package/selenium.d.ts +2 -0
  112. package/src/client/addon/components/Panel.tsx +2 -2
  113. package/src/client/addon/controller.ts +13 -6
  114. package/src/client/addon/withCreevey.ts +25 -13
  115. package/src/client/shared/components/ImagesView/SwapView.tsx +18 -0
  116. package/src/client/shared/components/PageHeader/ImagePreview.tsx +1 -0
  117. package/src/client/shared/components/ResultsPage.tsx +28 -7
  118. package/src/client/web/CreeveyApp.tsx +25 -7
  119. package/src/client/web/CreeveyContext.tsx +9 -0
  120. package/src/client/web/CreeveyView/SideBar/Search.tsx +3 -3
  121. package/src/client/web/CreeveyView/SideBar/SideBar.tsx +1 -0
  122. package/src/client/web/CreeveyView/SideBar/SideBarFooter.tsx +37 -6
  123. package/src/client/web/CreeveyView/SideBar/SuiteLink.tsx +3 -5
  124. package/src/client/web/CreeveyView/SideBar/TestLink.tsx +2 -4
  125. package/src/client/web/KeyboardEventsContext.tsx +61 -73
  126. package/src/creevey.ts +13 -6
  127. package/src/index.ts +0 -2
  128. package/src/playwright.ts +1 -0
  129. package/src/selenium.ts +1 -0
  130. package/src/server/config.ts +19 -7
  131. package/src/server/docker.ts +2 -2
  132. package/src/server/index.ts +45 -4
  133. package/src/server/logger.ts +6 -2
  134. package/src/server/master/api.ts +1 -1
  135. package/src/server/master/pool.ts +18 -56
  136. package/src/server/master/queue.ts +64 -0
  137. package/src/server/master/runner.ts +4 -1
  138. package/src/server/master/server.ts +1 -1
  139. package/src/server/master/start.ts +7 -4
  140. package/src/server/playwright/docker-file.ts +14 -1
  141. package/src/server/playwright/docker.ts +2 -12
  142. package/src/server/playwright/internal.ts +76 -49
  143. package/src/server/playwright/webdriver.ts +7 -2
  144. package/src/server/providers/browser.ts +2 -1
  145. package/src/server/providers/hybrid.ts +1 -1
  146. package/src/server/reporter.ts +4 -3
  147. package/src/server/selenium/internal.ts +147 -93
  148. package/src/server/selenium/selenoid.ts +2 -2
  149. package/src/server/selenium/webdriver.ts +12 -2
  150. package/src/server/telemetry.ts +7 -3
  151. package/src/server/utils.ts +14 -4
  152. package/src/server/webdriver.ts +13 -15
  153. package/src/server/worker/chai-image.ts +4 -4
  154. package/src/server/worker/start.ts +29 -48
  155. package/src/types.ts +35 -23
  156. package/types/playwright-context.d.ts +7 -0
  157. package/types/selenium-context.d.ts +7 -0
  158. package/.yarnrc.yml +0 -1
  159. package/chromatic.config.json +0 -5
@@ -0,0 +1 @@
1
+ export { SeleniumWebdriver } from './server/selenium/webdriver.js';
@@ -5,12 +5,13 @@ import { loadStories as browserStoriesProvider } from './providers/browser.js';
5
5
  import { Config, BrowserConfig, BrowserConfigObject, Options, isDefined } from '../types.js';
6
6
  import { configExt, loadThroughTSX } from './utils.js';
7
7
  import { CreeveyReporter, TeamcityReporter } from './reporter.js';
8
- import { SeleniumWebdriver } from './selenium/webdriver.js';
8
+ import { logger } from './logger.js';
9
9
 
10
10
  export const defaultBrowser = 'chrome';
11
11
 
12
- export const defaultConfig: Omit<Config, 'gridUrl' | 'testsDir' | 'tsConfig'> = {
12
+ export const defaultConfig: Omit<Config, 'gridUrl' | 'testsDir' | 'tsConfig' | 'webdriver'> = {
13
13
  disableTelemetry: false,
14
+ useWorkerQueue: false,
14
15
  useDocker: true,
15
16
  dockerImage: 'aerokube/selenoid:latest-release',
16
17
  dockerImagePlatform: '',
@@ -21,11 +22,10 @@ export const defaultConfig: Omit<Config, 'gridUrl' | 'testsDir' | 'tsConfig'> =
21
22
  reportDir: path.resolve('report'),
22
23
  reporter: process.env.TEAMCITY_VERSION ? TeamcityReporter : CreeveyReporter,
23
24
  storiesProvider: browserStoriesProvider,
24
- webdriver: SeleniumWebdriver,
25
25
  maxRetries: 0,
26
26
  testTimeout: 30000,
27
- diffOptions: { threshold: 0.05, includeAA: false },
28
- odiffOptions: { threshold: 0.05, antialiasing: true },
27
+ diffOptions: { threshold: 0.1, includeAA: false },
28
+ odiffOptions: { threshold: 0.1, antialiasing: true },
29
29
  browsers: { [defaultBrowser]: true },
30
30
  hooks: {},
31
31
  testsRegex: /\.creevey\.(t|j)s$/,
@@ -62,11 +62,23 @@ export async function readConfig(options: Options): Promise<Config> {
62
62
  const userConfig: typeof defaultConfig & Partial<Pick<Config, 'gridUrl' | 'storiesProvider'>> = { ...defaultConfig };
63
63
 
64
64
  if (isDefined(configPath)) {
65
- const configModule = await loadThroughTSX<{ default: Partial<Config> } | Partial<Config>>((load) => {
65
+ const configModule = await loadThroughTSX<
66
+ { default: { default: Partial<Config> } | Partial<Config> } | Partial<Config>
67
+ >((load) => {
66
68
  const configFileUrl = pathToFileURL(configPath).toString();
67
69
  return load(configFileUrl);
68
70
  });
69
- const configData = 'default' in configModule ? configModule.default : configModule;
71
+ let configData = 'default' in configModule ? configModule.default : configModule;
72
+ // NOTE In node > 18 with commonjs project and esm config with tsconfig moduleResolution nodeNext there is additional 'default'
73
+ configData = 'default' in configData ? configData.default : configData;
74
+
75
+ if (!configData.webdriver) {
76
+ const { SeleniumWebdriver } = await import('./selenium/webdriver.js');
77
+ logger().warn(
78
+ "Creevey supports `Selenium` and `Playwright` webdrivers. For backward compatibility `Selenium` is used by default, but it might changed in the future. Please explicitly specify one of webdrivers in your Creevey's config",
79
+ );
80
+ configData.webdriver = SeleniumWebdriver;
81
+ }
70
82
 
71
83
  Object.assign(userConfig, configData);
72
84
  }
@@ -21,7 +21,7 @@ export async function pullImages(
21
21
  if (auth) args.authconfig = auth;
22
22
  if (platform) args.platform = platform;
23
23
 
24
- logger.info('Pull docker images');
24
+ logger().info('Pull docker images');
25
25
  // TODO Replace with `import from`
26
26
  const { default: yoctoSpinner } = await import('yocto-spinner');
27
27
  for (const image of images) {
@@ -61,7 +61,7 @@ export async function buildImage(imageName: string, dockerfile: string): Promise
61
61
  const images = await docker.listImages({ filters: { label: [`creevey=${imageName}`] } });
62
62
 
63
63
  if (images.at(0)) {
64
- logger.info(`Image ${imageName} already exists`);
64
+ logger().info(`Image ${imageName} already exists`);
65
65
  return;
66
66
  }
67
67
 
@@ -1,4 +1,5 @@
1
1
  import cluster from 'cluster';
2
+ import path from 'path';
2
3
  import { readConfig, defaultBrowser } from './config.js';
3
4
  import { Options, Config, BrowserConfigObject, isWorkerMessage } from '../types.js';
4
5
  import { logger } from './logger.js';
@@ -6,11 +7,13 @@ import { SeleniumWebdriver } from './selenium/webdriver.js';
6
7
  import { LOCALHOST_REGEXP } from './webdriver.js';
7
8
  import { isInsideDocker } from './utils.js';
8
9
  import { sendWorkerMessage } from './messages.js';
10
+ import { playwrightDockerFile } from './playwright/docker-file.js';
11
+ import { buildImage } from './docker.js';
12
+ import { mkdir, writeFile } from 'fs/promises';
9
13
 
10
14
  async function startWebdriverServer(browser: string, config: Config, options: Options): Promise<string | undefined> {
11
15
  if (config.webdriver === SeleniumWebdriver) {
12
16
  if (cluster.isPrimary) {
13
- // TODO Get random free port
14
17
  const { startSelenoidContainer, startSelenoidStandalone } = await import('./selenium/selenoid.js');
15
18
  const gridUrl = 'http://localhost:4444/wd/hub';
16
19
  if (config.useDocker) {
@@ -22,17 +25,36 @@ async function startWebdriverServer(browser: string, config: Config, options: Op
22
25
  }
23
26
  // TODO Worker might want to use docker image of browser or start standalone selenium
24
27
  } else {
28
+ if (config.gridUrl) return undefined;
29
+
25
30
  // TODO start standalone playwright server (useDocker == false)
31
+ const {
32
+ default: { version },
33
+ } = await import('playwright-core/package.json', { with: { type: 'json' } });
26
34
 
27
35
  if (cluster.isWorker) {
28
36
  // TODO Re-use dockerImage
29
37
 
38
+ // TODO Use https://hub.docker.com/r/playwright/chrome
39
+ // NOTE It will be possible to use `chrome` browserName
30
40
  const { startPlaywrightContainer } = await import('./playwright/docker.js');
31
41
  const { browserName } = config.browsers[browser] as BrowserConfigObject;
32
- const host = await startPlaywrightContainer(browserName, options.debug);
42
+
43
+ const imageName = `creevey/${browserName}:v${version}`;
44
+ const host = await startPlaywrightContainer(imageName, options.debug);
33
45
 
34
46
  return host;
35
47
  } else {
48
+ const browsers = [...new Set(Object.values(config.browsers).map((c) => (c as BrowserConfigObject).browserName))];
49
+ await Promise.all(
50
+ browsers.map(async (browserName) => {
51
+ const imageName = `creevey/${browserName}:v${version}`;
52
+ const dockerfile = playwrightDockerFile(browserName, version);
53
+
54
+ await buildImage(imageName, dockerfile);
55
+ }),
56
+ );
57
+
36
58
  const { default: getPort } = await import('get-port');
37
59
 
38
60
  cluster.on('message', (worker, message: unknown) => {
@@ -59,6 +81,10 @@ export default async function (options: Options): Promise<void> {
59
81
  const { browser = defaultBrowser, update, ui, port } = options;
60
82
  let gridUrl = cluster.isPrimary ? config.gridUrl : options.gridUrl;
61
83
 
84
+ // TODO Add package.json with `"type": "commonjs"` as workaround for esm packages to load `data.js`
85
+ await mkdir(config.reportDir, { recursive: true });
86
+ await writeFile(path.join(config.reportDir, 'package.json'), '{"type": "commonjs"}');
87
+
62
88
  // NOTE: We don't need docker nor selenoid for update option
63
89
  if (
64
90
  !(gridUrl || (Object.values(config.browsers) as BrowserConfigObject[]).every(({ gridUrl }) => gridUrl)) &&
@@ -73,14 +99,29 @@ export default async function (options: Options): Promise<void> {
73
99
  return;
74
100
  }
75
101
  case cluster.isPrimary: {
76
- logger.info('Starting Master Process');
102
+ if (config.webdriver === SeleniumWebdriver) {
103
+ try {
104
+ await import('selenium-webdriver');
105
+ } catch {
106
+ logger().error('Failed to start Creevey, missing required dependency: "selenium-webdriver"');
107
+ process.exit(-1);
108
+ }
109
+ } else {
110
+ try {
111
+ await import('playwright-core');
112
+ } catch {
113
+ logger().error('Failed to start Creevey, missing required dependency: "playwright-core"');
114
+ process.exit(-1);
115
+ }
116
+ }
117
+ logger().info('Starting Master Process');
77
118
 
78
119
  const resolveApi = (await import('./master/server.js')).start(config.reportDir, port, ui);
79
120
 
80
121
  return (await import('./master/start.js')).start(gridUrl, config, options, resolveApi);
81
122
  }
82
123
  default: {
83
- logger.info(`Starting Worker for ${browser}`);
124
+ logger().info(`Starting Worker for ${browser}`);
84
125
 
85
126
  // NOTE: We assume that we pass `gridUrl` to worker CLI options
86
127
  // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
@@ -10,12 +10,16 @@ export const colors = {
10
10
  ERROR: chalk.red,
11
11
  };
12
12
 
13
+ let rootName = 'Creevey';
14
+
13
15
  prefix.reg(Logger);
14
16
  prefix.apply(Logger, {
15
- format(level, name = 'Creevey') {
17
+ format(level, name = rootName) {
16
18
  const levelColor = colors[level.toUpperCase() as keyof typeof colors];
17
19
  return `[${name}:${chalk.gray(process.pid)}] ${levelColor(level)} =>`;
18
20
  },
19
21
  });
20
22
 
21
- export const logger = Logger.getLogger('Creevey');
23
+ export const setRootName = (newName: string) => (rootName = newName);
24
+
25
+ export const logger = () => Logger.getLogger(rootName);
@@ -26,7 +26,7 @@ export default function creeveyApi(runner: Runner): CreeveyApi {
26
26
 
27
27
  handleMessage(ws: WebSocket, message: WebSocket.Data) {
28
28
  if (typeof message != 'string') {
29
- logger.info('unhandled message', message);
29
+ logger().info('unhandled message', message);
30
30
  return;
31
31
  }
32
32
 
@@ -1,18 +1,9 @@
1
- import cluster, { Worker as ClusterWorker } from 'cluster';
1
+ import { Worker as ClusterWorker } from 'cluster';
2
2
  import { EventEmitter } from 'events';
3
- import {
4
- Worker,
5
- Config,
6
- TestResult,
7
- BrowserConfigObject,
8
- WorkerMessage,
9
- TestStatus,
10
- isWorkerMessage,
11
- } from '../../types.js';
12
- import { sendTestMessage, sendShutdownMessage, subscribeOnWorker } from '../messages.js';
13
- import { isShuttingDown } from '../utils.js';
14
-
15
- const FORK_RETRIES = 5;
3
+ import { Worker, Config, TestResult, BrowserConfigObject, TestStatus } from '../../types.js';
4
+ import { sendTestMessage, subscribeOnWorker } from '../messages.js';
5
+ import { gracefullyKill, isShuttingDown } from '../utils.js';
6
+ import { WorkerQueue } from './queue.js';
16
7
 
17
8
  interface WorkerTest {
18
9
  id: string;
@@ -32,6 +23,7 @@ export default class Pool extends EventEmitter {
32
23
  return this.workers.length !== this.freeWorkers.length;
33
24
  }
34
25
  constructor(
26
+ public scheduler: WorkerQueue,
35
27
  config: Config,
36
28
  private browser: string,
37
29
  gridUrl?: string,
@@ -46,10 +38,11 @@ export default class Pool extends EventEmitter {
46
38
 
47
39
  async init(): Promise<void> {
48
40
  const poolSize = Math.max(1, this.config.limit ?? 1);
49
- // TODO Init queue for workers to smooth browser starting load
50
- this.workers = (await Promise.all(Array.from({ length: poolSize }).map(() => this.forkWorker()))).filter(
51
- (workerOrError): workerOrError is Worker => workerOrError instanceof ClusterWorker,
52
- );
41
+ this.workers = (
42
+ await Promise.all(
43
+ Array.from({ length: poolSize }).map(() => this.scheduler.forkWorker(this.browser, this.gridUrl)),
44
+ )
45
+ ).filter((workerOrError): workerOrError is Worker => workerOrError instanceof ClusterWorker);
53
46
  if (this.workers.length != poolSize)
54
47
  throw new Error(`Can't instantiate workers for ${this.browser} due many errors`);
55
48
  this.workers.forEach((worker) => {
@@ -66,7 +59,7 @@ export default class Pool extends EventEmitter {
66
59
  return true;
67
60
  }
68
61
 
69
- stop(): void {
62
+ stop() {
70
63
  if (!this.isRunning) {
71
64
  this.emit('stop');
72
65
  return;
@@ -76,7 +69,7 @@ export default class Pool extends EventEmitter {
76
69
  this.queue = [];
77
70
  }
78
71
 
79
- process(): void {
72
+ process() {
80
73
  const worker = this.getFreeWorker();
81
74
  const test = this.queue.at(0);
82
75
 
@@ -99,7 +92,9 @@ export default class Pool extends EventEmitter {
99
92
 
100
93
  sendTestMessage(worker, { type: 'start', payload: test });
101
94
 
102
- this.process();
95
+ setImmediate(() => {
96
+ this.process();
97
+ });
103
98
  }
104
99
 
105
100
  private sendStatus(message: { id: string; status: TestStatus; result?: TestResult }): void {
@@ -120,34 +115,12 @@ export default class Pool extends EventEmitter {
120
115
  return this.aliveWorkers.filter((worker) => !worker.isRunning);
121
116
  }
122
117
 
123
- private async forkWorker(retry = 0): Promise<Worker | { error: string }> {
124
- cluster.setupPrimary({
125
- args: ['--browser', this.browser, ...(this.gridUrl ? ['--gridUrl', this.gridUrl] : []), ...process.argv.slice(2)],
126
- });
127
- const worker = cluster.fork();
128
- const message = await new Promise((resolve: (value: WorkerMessage) => void) => {
129
- const readyHandler = (message: unknown): void => {
130
- if (!isWorkerMessage(message) || message.type == 'port') return;
131
- worker.off('message', readyHandler);
132
- resolve(message);
133
- };
134
- worker.on('message', readyHandler);
135
- });
136
-
137
- if (message.type != 'error') return worker;
138
-
139
- this.gracefullyKill(worker);
140
-
141
- if (retry == FORK_RETRIES) return message.payload;
142
- return this.forkWorker(retry + 1);
143
- }
144
-
145
118
  private exitHandler(worker: Worker): void {
146
119
  // eslint-disable-next-line @typescript-eslint/no-misused-promises
147
120
  worker.once('exit', async () => {
148
121
  if (isShuttingDown.current) return;
149
122
 
150
- const workerOrError = await this.forkWorker();
123
+ const workerOrError = await this.scheduler.forkWorker(this.browser, this.gridUrl);
151
124
 
152
125
  if (!(workerOrError instanceof ClusterWorker))
153
126
  throw new Error(`Can't instantiate worker for ${this.browser} due many errors`);
@@ -158,17 +131,6 @@ export default class Pool extends EventEmitter {
158
131
  });
159
132
  }
160
133
 
161
- private gracefullyKill(worker: Worker): void {
162
- worker.isShuttingDown = true;
163
- const timeout = setTimeout(() => {
164
- worker.kill();
165
- }, 10000);
166
- worker.on('exit', () => {
167
- clearTimeout(timeout);
168
- });
169
- sendShutdownMessage(worker);
170
- }
171
-
172
134
  private shouldRetry(test: WorkerTest): boolean {
173
135
  return test.retries < this.maxRetries && !this.forcedStop;
174
136
  }
@@ -200,7 +162,7 @@ export default class Pool extends EventEmitter {
200
162
  });
201
163
 
202
164
  if (message.payload.subtype == 'unknown') {
203
- this.gracefullyKill(worker);
165
+ gracefullyKill(worker);
204
166
  }
205
167
 
206
168
  this.handleTestResult(worker, test, { status: 'failed', error: message.payload.error });
@@ -0,0 +1,64 @@
1
+ import cluster from 'cluster';
2
+ import { isWorkerMessage, Worker, WorkerMessage } from '../../types.js';
3
+ import { gracefullyKill, isShuttingDown } from '../utils.js';
4
+
5
+ const FORK_RETRIES = 5;
6
+
7
+ type MaybeWorker = Worker | { error: string };
8
+
9
+ export class WorkerQueue {
10
+ private isProcessing = false;
11
+ private queue: { browser: string; gridUrl?: string; retry: number; resolve: (mw: MaybeWorker) => void }[] = [];
12
+
13
+ // TODO Add concurrency
14
+ constructor(private useQueue: boolean) {}
15
+
16
+ async forkWorker(browser: string, gridUrl?: string, retry = 0): Promise<MaybeWorker> {
17
+ return new Promise<MaybeWorker>((resolve) => {
18
+ this.queue.push({ browser, gridUrl, retry, resolve });
19
+
20
+ void this.process();
21
+ });
22
+ }
23
+
24
+ private async process() {
25
+ if (this.useQueue && this.isProcessing) return;
26
+
27
+ const { browser, gridUrl, retry, resolve } = this.queue.pop() ?? {};
28
+
29
+ if (browser == undefined || retry == undefined || resolve == undefined) return;
30
+
31
+ if (isShuttingDown.current) {
32
+ resolve({ error: 'Master process is shutting down' });
33
+ return;
34
+ }
35
+
36
+ this.isProcessing = true;
37
+
38
+ cluster.setupPrimary({
39
+ args: ['--browser', browser, ...(gridUrl ? ['--gridUrl', gridUrl] : []), ...process.argv.slice(2)],
40
+ });
41
+ const worker = cluster.fork();
42
+ const message = await new Promise((resolve: (value: WorkerMessage) => void) => {
43
+ const readyHandler = (message: unknown): void => {
44
+ if (!isWorkerMessage(message) || message.type == 'port') return;
45
+ worker.off('message', readyHandler);
46
+ resolve(message);
47
+ };
48
+ worker.on('message', readyHandler);
49
+ });
50
+
51
+ if (message.type == 'error') {
52
+ gracefullyKill(worker);
53
+
54
+ if (retry == FORK_RETRIES) resolve(message.payload);
55
+ else this.queue.push({ browser, gridUrl, retry: retry + 1, resolve });
56
+ } else {
57
+ resolve(worker);
58
+ }
59
+
60
+ this.isProcessing = false;
61
+
62
+ setImmediate(() => void this.process());
63
+ }
64
+ }
@@ -13,12 +13,14 @@ import {
13
13
  TestMeta,
14
14
  } from '../../types.js';
15
15
  import Pool from './pool.js';
16
+ import { WorkerQueue } from './queue.js';
16
17
 
17
18
  export default class Runner extends EventEmitter {
18
19
  private failFast: boolean;
19
20
  private screenDir: string;
20
21
  private reportDir: string;
21
22
  private browsers: string[];
23
+ private scheduler: WorkerQueue;
22
24
  private pools: Record<string, Pool> = {};
23
25
  tests: Partial<Record<string, ServerTest>> = {};
24
26
  public get isRunning(): boolean {
@@ -30,9 +32,10 @@ export default class Runner extends EventEmitter {
30
32
  this.failFast = config.failFast;
31
33
  this.screenDir = config.screenDir;
32
34
  this.reportDir = config.reportDir;
35
+ this.scheduler = new WorkerQueue(config.useWorkerQueue);
33
36
  this.browsers = Object.keys(config.browsers);
34
37
  this.browsers
35
- .map((browser) => (this.pools[browser] = new Pool(config, browser, gridUrl)))
38
+ .map((browser) => (this.pools[browser] = new Pool(this.scheduler, config, browser, gridUrl)))
36
39
  .map((pool) => pool.on('test', this.handlePoolMessage));
37
40
  }
38
41
 
@@ -96,7 +96,7 @@ export function start(reportDir: string, port: number, ui: boolean): (api: Creev
96
96
  app.use(mount('/report', serve(reportDir)));
97
97
 
98
98
  wss.on('error', (error) => {
99
- logger.error(error);
99
+ logger().error(error);
100
100
  });
101
101
 
102
102
  server.listen(port);
@@ -42,7 +42,10 @@ function outputUnnecessaryImages(imagesDir: string, images: Set<string>): void {
42
42
  .map((imagePath) => path.posix.relative(imagesDir, imagePath))
43
43
  .filter((imagePath) => !images.has(imagePath));
44
44
  if (unnecessaryImages.length > 0) {
45
- logger.warn('We found unnecessary screenshot images, those can be safely removed:\n', unnecessaryImages.join('\n'));
45
+ logger().warn(
46
+ 'We found unnecessary screenshot images, those can be safely removed:\n',
47
+ unnecessaryImages.join('\n'),
48
+ );
46
49
  }
47
50
  }
48
51
 
@@ -81,10 +84,10 @@ export async function start(
81
84
 
82
85
  if (options.ui) {
83
86
  resolveApi(creeveyApi(runner));
84
- logger.info(`Started on http://localhost:${options.port}`);
87
+ logger().info(`Started on http://localhost:${options.port}`);
85
88
  } else {
86
89
  if (Object.values(runner.status.tests).filter((test) => test && !test.skip).length == 0) {
87
- logger.warn("Don't have any tests to run");
90
+ logger().warn("Don't have any tests to run");
88
91
 
89
92
  void shutdownWorkers().then(() => process.exit());
90
93
  return;
@@ -101,7 +104,7 @@ export async function start(
101
104
  void sendScreenshotsCount(config, options, runner.status)
102
105
  .catch((reason: unknown) => {
103
106
  const error = reason instanceof Error ? (reason.stack ?? reason.message) : (reason as string);
104
- logger.warn(`Can't send telemetry: ${error}`);
107
+ logger().warn(`Can't send telemetry: ${error}`);
105
108
  })
106
109
  .finally(() => {
107
110
  void shutdownWorkers().then(() => process.exit());
@@ -1,9 +1,17 @@
1
1
  import semver from 'semver';
2
+ import { exec } from 'shelljs';
2
3
 
3
4
  // TODO Support custom docker images
4
5
  export function playwrightDockerFile(browser: string, version: string): string {
5
6
  const sv = semver.coerce(version);
6
7
 
8
+ let npmRegistry;
9
+ try {
10
+ npmRegistry = exec('npm config get registry', { silent: true }).stdout.trim();
11
+ } catch {
12
+ /* noop */
13
+ }
14
+
7
15
  return `
8
16
  FROM mcr.microsoft.com/playwright:v${sv?.format() ?? version}
9
17
 
@@ -11,7 +19,12 @@ WORKDIR /creevey
11
19
 
12
20
  RUN echo "{ \\"type\\": \\"module\\" }" > package.json && \\
13
21
  echo "import { ${browser} as browser } from 'playwright-core';" >> index.js && \\
14
- echo "const ws = await browser.launchServer({ port: 4444, wsPath: 'creevey' })" >> index.js && \\
22
+ echo "const ws = await browser.launchServer({ port: 4444, wsPath: 'creevey' })" >> index.js && \\${
23
+ npmRegistry
24
+ ? `
25
+ echo "registry=${npmRegistry}" > .npmrc && \\`
26
+ : ''
27
+ }
15
28
  npm i playwright-core${sv ? `@${sv.format()}` : ''}
16
29
 
17
30
  EXPOSE 4444
@@ -1,19 +1,9 @@
1
- import { buildImage, runImage } from '../docker';
1
+ import { runImage } from '../docker';
2
2
  import { emitWorkerMessage, subscribeOn } from '../messages';
3
3
  import { isInsideDocker } from '../utils';
4
4
  import { LOCALHOST_REGEXP } from '../webdriver';
5
- import { playwrightDockerFile } from './docker-file';
6
-
7
- export async function startPlaywrightContainer(browserName: string, debug: boolean): Promise<string> {
8
- const {
9
- default: { version },
10
- } = await import('playwright-core/package.json', { with: { type: 'json' } });
11
-
12
- const imageName = `creevey/${browserName}:v${version}`;
13
- const dockerfile = playwrightDockerFile(browserName, version);
14
-
15
- await buildImage(imageName, dockerfile);
16
5
 
6
+ export async function startPlaywrightContainer(imageName: string, debug: boolean): Promise<string> {
17
7
  const port = await new Promise<number>((resolve) => {
18
8
  subscribeOn('worker', (message) => {
19
9
  if (message.type == 'port') {