Skip to content

Commit 3b2ef8d

Browse files
committedDec 13, 2024·
lighthouse feature
1 parent 7439b04 commit 3b2ef8d

20 files changed

+17651
-33
lines changed
 

‎lighthouse-report/authPage.html

+2,630
Large diffs are not rendered by default.

‎lighthouse-report/categoriesPage.html

+2,630
Large diffs are not rendered by default.

‎lighthouse-report/dashboardPage.html

+2,630
Large diffs are not rendered by default.

‎lighthouse-report/homePage.html

+2,630
Large diffs are not rendered by default.

‎lighthouse-report/ordersPage.html

+2,630
Large diffs are not rendered by default.

‎lighthouse-report/productsPage.html

+2,630
Large diffs are not rendered by default.

‎package-lock.json

+1,561-21
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎package.json

+1
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,7 @@
4747
"allure-playwright": "^3.0.6",
4848
"eslint": "^8",
4949
"eslint-config-next": "15.0.3",
50+
"lighthouse": "^12.3.0",
5051
"postcss": "^8",
5152
"tailwindcss": "^3.4.1",
5253
"typescript": "^5"

‎playwright.config.ts

+12
Original file line numberDiff line numberDiff line change
@@ -30,8 +30,18 @@ export default defineConfig({
3030
name: 'logout',
3131
testMatch: /logout\.spec\.ts/,
3232
},
33+
// lighthouse
34+
{
35+
name: 'lighthouse',
36+
testMatch: /lighthouse\.spec\.ts/,
37+
use: {
38+
...devices['Desktop Chrome'],
39+
},
40+
dependencies: ['setup'],
41+
},
3342
{
3443
name: 'chromium',
44+
testIgnore: /lighthouse\.spec\.ts/, // 忽略 Lighthouse 测试
3545
use: {
3646
...devices['Desktop Chrome'],
3747
storageState: 'tests/.auth/admin.json',
@@ -40,6 +50,7 @@ export default defineConfig({
4050
},
4151
{
4252
name: 'firefox',
53+
testIgnore: /lighthouse\.spec\.ts/, // 忽略 Lighthouse 测试
4354
use: {
4455
...devices['Desktop Firefox'],
4556
storageState: 'tests/.auth/admin.json',
@@ -49,6 +60,7 @@ export default defineConfig({
4960

5061
{
5162
name: 'webkit',
63+
testIgnore: /lighthouse\.spec\.ts/, // 忽略 Lighthouse 测试
5264
use: {
5365
...devices['Desktop Safari'],
5466
storageState: 'tests/.auth/admin.json',

‎tests/.auth/admin.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,10 @@
22
"cookies": [
33
{
44
"name": "sb-binfgbgsoetsbeapeoks-auth-token",
5-
"value": "base64-eyJhY2Nlc3NfdG9rZW4iOiJleUpoYkdjaU9pSklVekkxTmlJc0ltdHBaQ0k2SWpsa1RXazVhSFJXWjFsNFRXMXpkSEVpTENKMGVYQWlPaUpLVjFRaWZRLmV5SnBjM01pT2lKb2RIUndjem92TDJKcGJtWm5ZbWR6YjJWMGMySmxZWEJsYjJ0ekxuTjFjR0ZpWVhObExtTnZMMkYxZEdndmRqRWlMQ0p6ZFdJaU9pSmhNREF6WXpBeE55MHlNV1F6TFRReFpqWXRPRE0yTkMwMVpUbGxaV0psWWprek5XUWlMQ0poZFdRaU9pSmhkWFJvWlc1MGFXTmhkR1ZrSWl3aVpYaHdJam94TnpNME1UQTFOalV6TENKcFlYUWlPakUzTXpReE1ESXdOVE1zSW1WdFlXbHNJam9pZEdWemRFQmhaRzFwYmk1amIyMGlMQ0p3YUc5dVpTSTZJaUlzSW1Gd2NGOXRaWFJoWkdGMFlTSTZleUp3Y205MmFXUmxjaUk2SW1WdFlXbHNJaXdpY0hKdmRtbGtaWEp6SWpwYkltVnRZV2xzSWwxOUxDSjFjMlZ5WDIxbGRHRmtZWFJoSWpwN0ltVnRZV2xzSWpvaWRHVnpkRUJoWkcxcGJpNWpiMjBpTENKbGJXRnBiRjkyWlhKcFptbGxaQ0k2Wm1Gc2MyVXNJbkJvYjI1bFgzWmxjbWxtYVdWa0lqcG1ZV3h6WlN3aWMzVmlJam9pWVRBd00yTXdNVGN0TWpGa015MDBNV1kyTFRnek5qUXROV1U1WldWaVpXSTVNelZrSW4wc0luSnZiR1VpT2lKaGRYUm9aVzUwYVdOaGRHVmtJaXdpWVdGc0lqb2lZV0ZzTVNJc0ltRnRjaUk2VzNzaWJXVjBhRzlrSWpvaWNHRnpjM2R2Y21RaUxDSjBhVzFsYzNSaGJYQWlPakUzTXpReE1ESXdOVE45WFN3aWMyVnpjMmx2Ymw5cFpDSTZJak0zWldSbU9URXhMVEEzWXprdE5EazFNeTFoTkRVekxUWmhNak5sTjJJeE1XVTVaU0lzSW1selgyRnViMjU1Ylc5MWN5STZabUZzYzJWOS5pSXZXTlFabjhsclhLeHN5NFUzZ1NOU2U5OURSZkVjTTVDWVFVSlZkdXdJIiwidG9rZW5fdHlwZSI6ImJlYXJlciIsImV4cGlyZXNfaW4iOjM2MDAsImV4cGlyZXNfYXQiOjE3MzQxMDU2NTMsInJlZnJlc2hfdG9rZW4iOiJkUlBXLTl2Yk14alNGSURCUVZLWmF3IiwidXNlciI6eyJpZCI6ImEwMDNjMDE3LTIxZDMtNDFmNi04MzY0LTVlOWVlYmViOTM1ZCIsImF1ZCI6ImF1dGhlbnRpY2F0ZWQiLCJyb2xlIjoiYXV0aGVudGljYXRlZCIsImVtYWlsIjoidGVzdEBhZG1pbi5jb20iLCJlbWFpbF9jb25maXJtZWRfYXQiOiIyMDI0LTEyLTA2VDA4OjUwOjM0LjUzMDU2NVoiLCJwaG9uZSI6IiIsImNvbmZpcm1lZF9hdCI6IjIwMjQtMTItMDZUMDg6NTA6MzQuNTMwNTY1WiIsImxhc3Rfc2lnbl9pbl9hdCI6IjIwMjQtMTItMTNUMTU6MDA6NTMuMjcxNDk0MTc0WiIsImFwcF9tZXRhZGF0YSI6eyJwcm92aWRlciI6ImVtYWlsIiwicHJvdmlkZXJzIjpbImVtYWlsIl19LCJ1c2VyX21ldGFkYXRhIjp7ImVtYWlsIjoidGVzdEBhZG1pbi5jb20iLCJlbWFpbF92ZXJpZmllZCI6ZmFsc2UsInBob25lX3ZlcmlmaWVkIjpmYWxzZSwic3ViIjoiYTAwM2MwMTctMjFkMy00MWY2LTgzNjQtNWU5ZWViZWI5MzVkIn0sImlkZW50aXRpZXMiOlt7ImlkZW50aXR5X2lkIjoiMGUyZGIzZTYtMWIzMi00MWI4LTlmNWMtMGZhMjdlOTExMWE2IiwiaWQiOiJhMDAzYzAxNy0yMWQzLTQxZjYtODM2NC01ZTllZWJlYjkzNWQiLCJ1c2VyX2lkIjoiYTAwM2MwMTctMjFkMy00MWY2LTgzNjQtNWU5ZWViZWI5MzVkIiwiaWRlbnRpdHlfZGF0YSI6eyJlbWFpbCI6InRlc3RAYWRtaW4uY29tIiwiZW1haWxfdmVyaWZpZWQiOmZhbHNlLCJwaG9uZV92ZXJpZmllZCI6ZmFsc2UsInN1YiI6ImEwMDNjMDE3LTIxZDMtNDFmNi04MzY0LTVlOWVlYmViOTM1ZCJ9LCJwcm92aWRlciI6ImVtYWlsIiwibGFzdF9zaWduX2luX2F0IjoiMjAyNC0xMi0wNlQwODo1MDozNC41MjQ2NVoiLCJjcmVhdGVkX2F0IjoiMjAyNC0xMi0wNlQwODo1MDozNC41MjQ3MDhaIiwidXBkYXRlZF9hdCI6IjIwMjQtMTItMDZUMDg6NTA6MzQuNTI0NzA4WiIsImVtYWlsIjoidGVzdEBhZG1pbi5jb20ifV0sImNyZWF0ZWRfYXQiOiIyMDI0LTEyLTA2VDA4OjUwOjM0LjQ4OTA2OFoiLCJ1cGRhdGVkX2F0IjoiMjAyNC0xMi0xM1QxNTowMDo1My4yNzM5ODVaIiwiaXNfYW5vbnltb3VzIjpmYWxzZX19",
5+
"value": "base64-eyJhY2Nlc3NfdG9rZW4iOiJleUpoYkdjaU9pSklVekkxTmlJc0ltdHBaQ0k2SWpsa1RXazVhSFJXWjFsNFRXMXpkSEVpTENKMGVYQWlPaUpLVjFRaWZRLmV5SnBjM01pT2lKb2RIUndjem92TDJKcGJtWm5ZbWR6YjJWMGMySmxZWEJsYjJ0ekxuTjFjR0ZpWVhObExtTnZMMkYxZEdndmRqRWlMQ0p6ZFdJaU9pSmhNREF6WXpBeE55MHlNV1F6TFRReFpqWXRPRE0yTkMwMVpUbGxaV0psWWprek5XUWlMQ0poZFdRaU9pSmhkWFJvWlc1MGFXTmhkR1ZrSWl3aVpYaHdJam94TnpNME1USXhOekF3TENKcFlYUWlPakUzTXpReE1UZ3hNREFzSW1WdFlXbHNJam9pZEdWemRFQmhaRzFwYmk1amIyMGlMQ0p3YUc5dVpTSTZJaUlzSW1Gd2NGOXRaWFJoWkdGMFlTSTZleUp3Y205MmFXUmxjaUk2SW1WdFlXbHNJaXdpY0hKdmRtbGtaWEp6SWpwYkltVnRZV2xzSWwxOUxDSjFjMlZ5WDIxbGRHRmtZWFJoSWpwN0ltVnRZV2xzSWpvaWRHVnpkRUJoWkcxcGJpNWpiMjBpTENKbGJXRnBiRjkyWlhKcFptbGxaQ0k2Wm1Gc2MyVXNJbkJvYjI1bFgzWmxjbWxtYVdWa0lqcG1ZV3h6WlN3aWMzVmlJam9pWVRBd00yTXdNVGN0TWpGa015MDBNV1kyTFRnek5qUXROV1U1WldWaVpXSTVNelZrSW4wc0luSnZiR1VpT2lKaGRYUm9aVzUwYVdOaGRHVmtJaXdpWVdGc0lqb2lZV0ZzTVNJc0ltRnRjaUk2VzNzaWJXVjBhRzlrSWpvaWNHRnpjM2R2Y21RaUxDSjBhVzFsYzNSaGJYQWlPakUzTXpReE1UZ3hNREI5WFN3aWMyVnpjMmx2Ymw5cFpDSTZJakprTW1aallXSmxMVEF6T1dVdE5EQTFOaTFoTVdRNExXVXhZMkUyTVRZMFl6TTJNeUlzSW1selgyRnViMjU1Ylc5MWN5STZabUZzYzJWOS4xRWppRnFjZDl6QmltVzJqLUtoWmVVWGtjOWtXVU9TbnBvdFNvWm1hUlFRIiwidG9rZW5fdHlwZSI6ImJlYXJlciIsImV4cGlyZXNfaW4iOjM2MDAsImV4cGlyZXNfYXQiOjE3MzQxMjE3MDAsInJlZnJlc2hfdG9rZW4iOiItLWtfbkgyeWY3STNGSVJqRWVCMnJBIiwidXNlciI6eyJpZCI6ImEwMDNjMDE3LTIxZDMtNDFmNi04MzY0LTVlOWVlYmViOTM1ZCIsImF1ZCI6ImF1dGhlbnRpY2F0ZWQiLCJyb2xlIjoiYXV0aGVudGljYXRlZCIsImVtYWlsIjoidGVzdEBhZG1pbi5jb20iLCJlbWFpbF9jb25maXJtZWRfYXQiOiIyMDI0LTEyLTA2VDA4OjUwOjM0LjUzMDU2NVoiLCJwaG9uZSI6IiIsImNvbmZpcm1lZF9hdCI6IjIwMjQtMTItMDZUMDg6NTA6MzQuNTMwNTY1WiIsImxhc3Rfc2lnbl9pbl9hdCI6IjIwMjQtMTItMTNUMTk6Mjg6MjAuNTY4MjQ5OTI3WiIsImFwcF9tZXRhZGF0YSI6eyJwcm92aWRlciI6ImVtYWlsIiwicHJvdmlkZXJzIjpbImVtYWlsIl19LCJ1c2VyX21ldGFkYXRhIjp7ImVtYWlsIjoidGVzdEBhZG1pbi5jb20iLCJlbWFpbF92ZXJpZmllZCI6ZmFsc2UsInBob25lX3ZlcmlmaWVkIjpmYWxzZSwic3ViIjoiYTAwM2MwMTctMjFkMy00MWY2LTgzNjQtNWU5ZWViZWI5MzVkIn0sImlkZW50aXRpZXMiOlt7ImlkZW50aXR5X2lkIjoiMGUyZGIzZTYtMWIzMi00MWI4LTlmNWMtMGZhMjdlOTExMWE2IiwiaWQiOiJhMDAzYzAxNy0yMWQzLTQxZjYtODM2NC01ZTllZWJlYjkzNWQiLCJ1c2VyX2lkIjoiYTAwM2MwMTctMjFkMy00MWY2LTgzNjQtNWU5ZWViZWI5MzVkIiwiaWRlbnRpdHlfZGF0YSI6eyJlbWFpbCI6InRlc3RAYWRtaW4uY29tIiwiZW1haWxfdmVyaWZpZWQiOmZhbHNlLCJwaG9uZV92ZXJpZmllZCI6ZmFsc2UsInN1YiI6ImEwMDNjMDE3LTIxZDMtNDFmNi04MzY0LTVlOWVlYmViOTM1ZCJ9LCJwcm92aWRlciI6ImVtYWlsIiwibGFzdF9zaWduX2luX2F0IjoiMjAyNC0xMi0wNlQwODo1MDozNC41MjQ2NVoiLCJjcmVhdGVkX2F0IjoiMjAyNC0xMi0wNlQwODo1MDozNC41MjQ3MDhaIiwidXBkYXRlZF9hdCI6IjIwMjQtMTItMDZUMDg6NTA6MzQuNTI0NzA4WiIsImVtYWlsIjoidGVzdEBhZG1pbi5jb20ifV0sImNyZWF0ZWRfYXQiOiIyMDI0LTEyLTA2VDA4OjUwOjM0LjQ4OTA2OFoiLCJ1cGRhdGVkX2F0IjoiMjAyNC0xMi0xM1QxOToyODoyMC41Njk5NDJaIiwiaXNfYW5vbnltb3VzIjpmYWxzZX19",
66
"domain": "localhost",
77
"path": "/",
8-
"expires": 1768662053.739505,
8+
"expires": 1768678100.844763,
99
"httpOnly": false,
1010
"secure": false,
1111
"sameSite": "Lax"

‎tests/e2e/adminLayout.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import test from '../fixtures/pageFixtrue'
1+
import { test } from '../fixtures/pageFixtrue'
22

33
test.describe('admin layout common functions', () => {
44
test('should toggle light theme', async ({ adminLayout }) => {

‎tests/e2e/authPage.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import test from '../fixtures/pageFixtrue'
1+
import { test } from '../fixtures/pageFixtrue'
22

33
test.describe('Auth Page', () => {
44
// 表单验证错误信息验证

‎tests/e2e/categoriesPage.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import test from '../fixtures/pageFixtrue'
1+
import { test } from '../fixtures/pageFixtrue'
22
import { faker } from '@faker-js/faker'
33

44
test.describe('Categories Page', () => {

‎tests/e2e/dashboardPage.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import test from '../fixtures/pageFixtrue'
1+
import { test } from '../fixtures/pageFixtrue'
22

33
test.describe('Dashboard Page', () => {
44
test('ordersc chart check', async ({ dashboardPage }) => {

‎tests/e2e/homePage.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import test from '../fixtures/pageFixtrue'
1+
import { test } from '../fixtures/pageFixtrue'
22

33
test.describe('Home page', () => {
44
test('home page header show', async ({ homePage }) => {

‎tests/e2e/lighthouse.spec.ts

+224
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,224 @@
1+
import { test } from '../fixtures/lighthouseFixture'
2+
3+
test.describe.serial('Lighthouse', () => {
4+
test('check home page performance', async ({
5+
browserName,
6+
playwright,
7+
lighthouse,
8+
}) => {
9+
test.skip(browserName !== 'chromium', 'Run only on chromium')
10+
11+
const browser = await playwright['chromium'].launch({
12+
args: ['--remote-debugging-port=9222'],
13+
})
14+
15+
try {
16+
const page = await browser.newPage()
17+
await page.goto('/')
18+
19+
await lighthouse(page, 'homePage', {
20+
thresholds: {
21+
performance: 0.9,
22+
'best-practices': 0.9,
23+
seo: 0.9,
24+
},
25+
})
26+
27+
await page.close()
28+
} finally {
29+
await browser.close()
30+
}
31+
})
32+
test('check auth page performance', async ({
33+
browserName,
34+
playwright,
35+
lighthouse,
36+
}) => {
37+
test.skip(browserName !== 'chromium', 'Run only on chromium')
38+
39+
const browser = await playwright['chromium'].launch({
40+
args: ['--remote-debugging-port=9222'],
41+
})
42+
43+
try {
44+
const page = await browser.newPage()
45+
await page.goto('/auth')
46+
47+
await lighthouse(page, 'authPage', {
48+
thresholds: {
49+
performance: 0.9,
50+
'best-practices': 0.9,
51+
seo: 0.9,
52+
},
53+
})
54+
55+
await page.close()
56+
} finally {
57+
await browser.close()
58+
}
59+
})
60+
test('check dashboard page performance', async ({
61+
browserName,
62+
playwright,
63+
lighthouse,
64+
}) => {
65+
test.skip(browserName !== 'chromium', 'Run only on chromium')
66+
67+
const browser = await playwright['chromium'].launch({
68+
args: ['--remote-debugging-port=9222'],
69+
})
70+
71+
try {
72+
const page = await browser.newPage({
73+
storageState: 'tests/.auth/admin.json',
74+
})
75+
await page.goto('/admin/dashboard')
76+
77+
// 读取并解析 cookies
78+
const cookies = await page.context().cookies()
79+
const cookieString = cookies
80+
.map((cookie) => `${cookie.name}=${cookie.value}`)
81+
.join('; ')
82+
83+
// 将 cookies 传递给 Lighthouse
84+
await lighthouse(page, 'dashboardPage', {
85+
thresholds: {
86+
performance: 0.9,
87+
'best-practices': 0.9,
88+
seo: 0.9,
89+
},
90+
// 这里可以添加其他选项,例如 cookies
91+
extraHeaders: {
92+
Cookie: cookieString,
93+
},
94+
})
95+
96+
await page.close()
97+
} finally {
98+
await browser.close()
99+
}
100+
})
101+
test('check orders page performance', async ({
102+
browserName,
103+
playwright,
104+
lighthouse,
105+
}) => {
106+
test.skip(browserName !== 'chromium', 'Run only on chromium')
107+
108+
const browser = await playwright['chromium'].launch({
109+
args: ['--remote-debugging-port=9222'],
110+
})
111+
112+
try {
113+
const page = await browser.newPage({
114+
storageState: 'tests/.auth/admin.json',
115+
})
116+
await page.goto('/admin/orders')
117+
118+
// 读取并解析 cookies
119+
const cookies = await page.context().cookies()
120+
const cookieString = cookies
121+
.map((cookie) => `${cookie.name}=${cookie.value}`)
122+
.join('; ')
123+
124+
// 将 cookies 传递给 Lighthouse
125+
await lighthouse(page, 'ordersPage', {
126+
thresholds: {
127+
performance: 0.9,
128+
'best-practices': 0.9,
129+
seo: 0.9,
130+
},
131+
// 这里可以添加其他选项,例如 cookies
132+
extraHeaders: {
133+
Cookie: cookieString,
134+
},
135+
})
136+
137+
await page.close()
138+
} finally {
139+
await browser.close()
140+
}
141+
})
142+
test('check products page performance', async ({
143+
browserName,
144+
playwright,
145+
lighthouse,
146+
}) => {
147+
test.skip(browserName !== 'chromium', 'Run only on chromium')
148+
149+
const browser = await playwright['chromium'].launch({
150+
args: ['--remote-debugging-port=9222'],
151+
})
152+
153+
try {
154+
const page = await browser.newPage({
155+
storageState: 'tests/.auth/admin.json',
156+
})
157+
await page.goto('/admin/products')
158+
159+
// 读取并解析 cookies
160+
const cookies = await page.context().cookies()
161+
const cookieString = cookies
162+
.map((cookie) => `${cookie.name}=${cookie.value}`)
163+
.join('; ')
164+
165+
// 将 cookies 传递给 Lighthouse
166+
await lighthouse(page, 'productsPage', {
167+
thresholds: {
168+
performance: 0.9,
169+
'best-practices': 0.9,
170+
seo: 0.9,
171+
},
172+
// 这里可以添加其他选项,例如 cookies
173+
extraHeaders: {
174+
Cookie: cookieString,
175+
},
176+
})
177+
178+
await page.close()
179+
} finally {
180+
await browser.close()
181+
}
182+
})
183+
test('check categories page performance', async ({
184+
browserName,
185+
playwright,
186+
lighthouse,
187+
}) => {
188+
test.skip(browserName !== 'chromium', 'Run only on chromium')
189+
190+
const browser = await playwright['chromium'].launch({
191+
args: ['--remote-debugging-port=9222'],
192+
})
193+
194+
try {
195+
const page = await browser.newPage({
196+
storageState: 'tests/.auth/admin.json',
197+
})
198+
await page.goto('/admin/categories')
199+
200+
// 读取并解析 cookies
201+
const cookies = await page.context().cookies()
202+
const cookieString = cookies
203+
.map((cookie) => `${cookie.name}=${cookie.value}`)
204+
.join('; ')
205+
206+
// 将 cookies 传递给 Lighthouse
207+
await lighthouse(page, 'categoriesPage', {
208+
thresholds: {
209+
performance: 0.9,
210+
'best-practices': 0.9,
211+
seo: 0.9,
212+
},
213+
// 这里可以添加其他选项,例如 cookies
214+
extraHeaders: {
215+
Cookie: cookieString,
216+
},
217+
})
218+
219+
await page.close()
220+
} finally {
221+
await browser.close()
222+
}
223+
})
224+
})

‎tests/e2e/ordersPage.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import test from '../fixtures/pageFixtrue'
1+
import { test } from '../fixtures/pageFixtrue'
22

33
test.describe('Orders Page', () => {
44
test('navigateToDashboard', async ({ ordersPage }) => {

‎tests/e2e/productsPage.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import test from '../fixtures/pageFixtrue'
1+
import { test } from '../fixtures/pageFixtrue'
22
import { faker } from '@faker-js/faker'
33

44
test.describe('Products Page', () => {

‎tests/fixtures/lighthouseFixture.ts

+63
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
import { Page, test as base } from '@playwright/test'
2+
import fs from 'fs'
3+
4+
type LighthouseCategories = Partial<{
5+
performance: number
6+
accessibility: number
7+
'best-practices': number
8+
seo: number
9+
pwa: number
10+
}>
11+
12+
const defaultThresholds: LighthouseCategories = {
13+
performance: 0.9,
14+
accessibility: 0.9,
15+
'best-practices': 0.9,
16+
seo: 0.9,
17+
pwa: 0.9,
18+
}
19+
20+
export const test = base.extend<{
21+
lighthouse: (
22+
page: Page,
23+
reportFileName: string,
24+
options?: {
25+
thresholds?: LighthouseCategories
26+
extraHeaders?: Record<string, string>
27+
}
28+
) => Promise<void>
29+
}>({
30+
lighthouse: async ({}, use) => {
31+
async function loadLighthouse(
32+
page: Page,
33+
reportFileName: string,
34+
options?: {
35+
thresholds?: LighthouseCategories
36+
extraHeaders?: Record<string, string>
37+
}
38+
) {
39+
const thresholds: LighthouseCategories =
40+
options?.thresholds || defaultThresholds
41+
const categories = Object.keys(thresholds)
42+
43+
const { default: lighthouse } = await import('lighthouse')
44+
45+
const runnerResult = await lighthouse(page.url(), {
46+
port: 9222,
47+
logLevel: 'error',
48+
onlyCategories: categories,
49+
output: 'html',
50+
extraHeaders: options?.extraHeaders,
51+
})
52+
53+
if (!runnerResult) throw new Error('Lighthouse failed to run')
54+
const reportHtml = Array.isArray(runnerResult.report)
55+
? runnerResult.report.join('') // 将数组连接成字符串
56+
: runnerResult.report // 如果是字符串,直接使用
57+
58+
fs.writeFileSync(`lighthouse-report/${reportFileName}.html`, reportHtml)
59+
}
60+
61+
await use(loadLighthouse)
62+
},
63+
})

‎tests/fixtures/pageFixtrue.ts

+1-3
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ import { ProductsPage } from '../pages/productsPage'
77
import { CategoriesPage } from '../pages/categoriesPage'
88
import { AdminLayout } from '../pages/adminLayout'
99

10-
const test = baseTest.extend<{
10+
export const test = baseTest.extend<{
1111
homePage: HomePage
1212
authPage: AuthPage
1313
dashboardPage: DashboardPage
@@ -38,5 +38,3 @@ const test = baseTest.extend<{
3838
await use(new CategoriesPage(page))
3939
},
4040
})
41-
42-
export default test

0 commit comments

Comments
 (0)
Please sign in to comment.