-
Notifications
You must be signed in to change notification settings - Fork 326
/
Copy pathplaywright.config.ts
183 lines (172 loc) · 5.88 KB
/
playwright.config.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
/** @file Playwright browser testing configuration. */
/**
* Note that running Playwright in CI poses a number of issues:
* - `backdrop-filter: blur` is disabled, due to issues with Chromium's `--disable-gpu` flag
* (see below).
* - System validation dialogs are not reliable between computers, as they may have different
* default fonts.
*/
import { defineConfig } from '@playwright/test'
import net from 'node:net'
import path from 'node:path'
import url from 'node:url'
import invariant from 'tiny-invariant'
const DEBUG = process.env.DEBUG_TEST === 'true'
const isCI = process.env.CI === 'true'
const isProd = process.env.PROD === 'true'
const TIMEOUT_MS = DEBUG ? 100_000_000 : 25_000
// We tend to use less CPU on CI to reduce the number of failures due to timeouts.
// Instead of using workers on CI, we use shards to run tests in parallel.
const WORKERS = isCI ? 2 : '35%'
const dirName = path.dirname(url.fileURLToPath(import.meta.url))
async function findFreePortInRange(min: number, max: number) {
const range = max - min + 1
invariant(range > 0, 'Minimum port must be less than maximum port.')
for (let i = 0; i < range; i++) {
const portToCheck = min + i
if (await checkAvailablePort(portToCheck)) return portToCheck
}
throw new Error('Failed to find a free port.')
}
function checkAvailablePort(port: number) {
return new Promise((resolve, reject) => {
const server = net.createServer()
server
.unref()
.on('error', (e: any) =>
'EADDRINUSE' === e.code ? reject('Port is already in use.') : reject(e),
)
.listen({ host: '0.0.0.0', port }, () => server.close(() => resolve(true)))
})
}
const portsFromEnv = {
projectView: parseInt(process.env.PLAYWRIGHT_PORT_PV ?? '', 10),
dashboard: parseInt(process.env.PLAYWRIGHT_PORT ?? '', 10),
}
const ports = {
projectView:
Number.isFinite(portsFromEnv.projectView) ?
portsFromEnv.projectView
: await findFreePortInRange(5300, 5999),
dashboard:
Number.isFinite(portsFromEnv.dashboard) ?
portsFromEnv.dashboard
: await findFreePortInRange(4300, 4999),
}
console.log(`Selected playwright servers' ports: ${ports.projectView} and ${ports.dashboard}`)
// Make sure to set the env to actual port that is being used. This is necessary for workers to
// pick up the same configuration.
process.env.PLAYWRIGHT_PORT = `${ports.dashboard}`
process.env.PLAYWRIGHT_PORT_PV = `${ports.projectView}`
export default defineConfig({
fullyParallel: true,
...(WORKERS ? { workers: WORKERS } : {}),
forbidOnly: isCI,
reporter: isCI ? [['list'], ['blob']] : [['html']],
retries: isCI ? 3 : 0,
use: {
headless: !DEBUG,
actionTimeout: 5000,
trace: 'retain-on-failure',
...(DEBUG ?
{}
: {
launchOptions: {
ignoreDefaultArgs: ['--headless'],
args: [
// Much closer to headful Chromium than classic headless.
'--headless=new',
// Required for `backdrop-filter: blur` to work.
'--use-angle=swiftshader',
// FIXME: `--disable-gpu` disables `backdrop-filter: blur`, which is not handled by
// the software (CPU) compositor. This SHOULD be fixed eventually, but this flag
// MUST stay as CI does not have a GPU.
'--disable-gpu',
// Fully disable GPU process.
'--disable-software-rasterizer',
// Disable text subpixel antialiasing.
'--font-render-hinting=none',
'--disable-skia-runtime-opts',
'--disable-system-font-check',
'--disable-font-subpixel-positioning',
'--disable-lcd-text',
],
},
}),
},
projects: [
// Setup project
{
name: 'Setup Dashboard',
testDir: './integration-test/dashboard',
testMatch: /.*\.setup\.ts/,
timeout: TIMEOUT_MS,
use: {
baseURL: `http://localhost:${ports.dashboard}`,
actionTimeout: TIMEOUT_MS,
offline: false,
},
},
{
name: 'Dashboard',
testDir: './integration-test/dashboard',
testMatch: /.*\.spec\.ts/,
dependencies: ['Setup Dashboard'],
expect: {
toHaveScreenshot: { threshold: 0 },
timeout: TIMEOUT_MS,
},
timeout: TIMEOUT_MS,
use: {
baseURL: `http://localhost:${ports.dashboard}`,
actionTimeout: TIMEOUT_MS,
offline: false,
storageState: path.join(dirName, './playwright/.auth/user.json'),
},
},
{
name: 'Setup Tests for Project View',
testMatch: /integration-test\/project-view\/setup\.ts/,
},
{
name: 'Project View',
dependencies: ['Setup Tests for Project View'],
testDir: './integration-test/project-view',
timeout: 60000,
repeatEach: 3,
retries: 0,
expect: {
timeout: 5000,
toHaveScreenshot: { threshold: 0 },
},
use: {
viewport: { width: 1920, height: 1750 },
baseURL: `http://localhost:${ports.projectView}`,
},
},
],
webServer: [
{
env: {
INTEGRATION_TEST: 'true',
ENSO_IDE_PROJECT_MANAGER_URL: 'ws://__HOSTNAME__:30536',
},
command: `corepack pnpm build && corepack pnpm exec vite preview --port ${ports.projectView} --strictPort`,
// Build from scratch apparently can take a while on CI machines.
timeout: 480 * 1000,
port: ports.projectView,
// We use our special, mocked version of server, thus do not want to re-use user's one.
reuseExistingServer: false,
},
{
env: { NODE_ENV: 'test' },
command:
isCI || isProd ?
`corepack pnpm exec vite -c vite.test.config.ts build && vite -c vite.test.config.ts preview --port ${ports.dashboard} --strictPort`
: `corepack pnpm exec vite -c vite.test.config.ts --port ${ports.dashboard}`,
timeout: 480 * 1000,
port: ports.dashboard,
reuseExistingServer: false,
},
],
})