Devahoy Logo
PublishedAt

Playwright

ลอง Playwright เทสแอพ React + Vite แบบง่ายๆ

ลอง Playwright เทสแอพ React + Vite แบบง่ายๆ

หลังจากวันก่อน ได้ลองใช้งาน Playwright ไป วันนี้ก็เลยลองเอาตัว Playwright มาลองทำ testing ง่ายๆ ด้วยการจำลองเว็บ โดยใช้ default เว็บ ของ React + Vite (เว็บที่เป็น counter)

ลองใช้งาน Playwright ในการทำ e2e testingสวัสดีครับ บทความนี้มาลองเล่น Playwright เพื่อเอามาทำ automate e2e testing ครับ Playwright เป็น e2e testing framework (end to end) ที่พัฒนาโดย Microsoft ตัว API มีความคล้ายกับ Puppeteer (แน่นอนแหละ main contributor ของ puppeteer ย้ายมา playwright) Fast and reliable end-to-end testingDEVAHOYChai Phonbopit

สร้างโปรเจ็คไว้เทส

เริ่มแรก ผมสร้างเว็บ เพื่อจะเอาไว้เทสก่อน ซึ่งใช้ starter ของ React + Vite เลย

Terminal window
npm create vite@latest my-react-app -- --template react-ts

จากนั้น ก็ install

Terminal window
cd my-react-app
npm install

ผมมีการเพิ่ม Input นิดหน่อย เพื่อให้เวลา Input และกด Set Title ให้มันไปเปลี่ยนค่า React + Vite เป็นค่าที่เราใส่ ไฟล์ src/App.tsx ผมปรับแต่งนิดหน่อย แบบนี้

1
import { useState } from 'react'
2
import reactLogo from './assets/react.svg'
3
import viteLogo from '/vite.svg'
4
import './App.css'
5
6
function App() {
7
const [count, setCount] = useState(0)
8
const [title, setTitle] = useState('Vite + React')
9
10
const handleSubmit = (e: React.FormEvent<HTMLFormElement>) => {
11
e.preventDefault()
12
const formData = new FormData(e.currentTarget)
13
setTitle(formData.get('title') as string)
14
}
15
16
return (
17
<div className="App">
18
<div>
19
<a href="https://vitejs.dev" target="_blank">
20
<img src={viteLogo} className="logo" alt="Vite logo" />
21
</a>
22
<a href="https://reactjs.org" target="_blank">
23
<img src={reactLogo} className="logo react" alt="React logo" />
24
</a>
25
</div>
26
<h1>{title}</h1>
27
<div className="card">
28
<button onClick={() => setCount((count) => count + 1)}>
29
count is {count}
30
</button>
31
<p>
32
Edit <code>src/App.tsx</code> and save to test HMR
33
</p>
34
</div>
35
36
<form onSubmit={handleSubmit}>
37
<input type="text" name="title" className="input-title" />
38
<button type="submit">Set Title</button>
39
</form>
40
</div>
41
)
42
}
43
44
export default App

เมื่อลองรัน dev server จะเป็น url http://localhost:5173 และหน้าตาก็ประมาณนี้

React vite playwright

ซึ่งถ้าเราดูตัวเว็บ React สิ่งที่มีคือ

  • มี document title ชื่อ Vite + React + TS
  • มีปุ่ม count ( count is 0 เริ่มต้น) เมื่อเรากด ค่ามันก็จะเพิ่มทีละ 1
  • มี input เพื่อ set ค่า title ( h1)

ทีนี้ ส่วนของเทส ผมก็จะตั้งเทสเคส ไว้ 3 ตัว คร่าวๆ คือ

  1. เทสว่า เว็บมี document title หรือไม่
  2. เทสว่า เมื่อกด count ค่าจะเพิ่มตามจำนวนคลิ๊กมั้ย
  3. เทสว่า เมื่อพิมพ์ input และกด Set Title ค่า heading 1 จะเปลี่ยนมั้ย

สร้างโปรเจ็ค Playwright

Terminal window
npm init playwright@latest

เช็ค document title

1
import { test, expect } from '@playwright/test'
2
3
const BASE_URL = 'http://localhost:5173/'
4
5
test('has document title', async ({ page }) => {
6
await page.goto(BASE_URL)
7
await expect(page).toHaveTitle('Vite + React + TS')
8
9
await page.screenshot({ path: `screenshots/example-has-title.png` })
10
})

เทสนี้ไม่มีอะไรมาก

  1. สั่งให้เปิดเว็บด้วย page.goto()
  2. เช็คว่า page มี document title มั้ย ด้วย .toHaveTitle()
  3. ลอง screenshot ถ่ายรูปไว้ด้วย ด้วยคำสั่ง page.screenshot

เทส click count button

1
test('click count button', async ({ page }) => {
2
await page.goto(BASE_URL)
3
4
const counter = page.locator('.card button')
5
counter.click()
6
7
await expect(counter).toHaveText('count is 1')
8
})
  1. ก็เปิดเว็บ React ด้วย goto() เหมือนเดิม
  2. จากนั้น ใช้ page.locator เพื่อหา element ด้วย selector .card button แบบเดียวกับ css / js
  3. เมื่อได้ element ก็ทำการกด click 1 ที element.click()
  4. สุดท้าย expect ว่า element ต้องมี text เป็น count is 1

เรื่องของ locator เราสามารถใช้ ได้หลายวิธีเช่น หา button ในเพจนี้

  • page.locator('.card button')
  • page.getByRole('button') - มีโอกาสได้ หลาย elements
  • page.getByText('count is 0') - ใช้ได้เฉพาะเริ่มต้นครั้งแรก

ส่วนตัวผมชอบใช้ selector มากกว่า เพราะถ้าเคสนี้ใช้ getByRole() หรือ getByText() เวลา expect ก็จะหา element นั้นไม่เจอ เพราะค่า มันเปลี่ยนแล้ว

1
const counter = await page.getByRole('button', { name: 'count is 0' })
2
counter.click()
3
4
// ถูก
5
await expect(page.locator('.card button')).toHaveText('count is 1')
6
7
// ผิด
8
await expect(counter).toHaveText('count is 1')
Locators | Playwright
Introductionplaywright.dev

อ่าน Locators เพิ่มเติม

เทส set heading title

1
test('set heading title', async ({ page }) => {
2
await page.goto(BASE_URL)
3
4
const input = page.locator('input')
5
await input.fill('Hello World')
6
7
const button = page.locator('button[type=submit]')
8
button.click()
9
10
await expect(page.locator('h1')).toHaveText('Hello World')
11
12
await page.screenshot({ path: `screenshots/example-set-title.png` })
13
})

เทสสุดท้าย เริ่มจาก

  1. page.goto เพื่อเปิดเว็บ
  2. หา element ด้วย page.locator เนื่องจากเว็บมี input ที่เดียว
  3. ทำการกรอกค่าด้วย input.fill()
  4. หา element button ที่เป็น type=submit จากนั้นกด click
  5. expect หา h1 ต้องมีค่า Hello World
  6. สุดท้าย screenshots ไว้ซักหน่อย

ลองรันเทส playwright ดูผลลัพธ์

Terminal window
npx playwright test
Running 9 tests using 5 workers
9 passed (4.7s)
To open last HTML report run:
npx playwright show-report

สุดท้าย ไฟล์ test ของผม react-vite.spec.ts เผื่ออยากลองเอาไปรัน

react-vite.spec.ts
1
import { test, expect } from '@playwright/test'
2
3
const BASE_URL = 'http://localhost:5173/'
4
5
test.describe('React Vite', () => {
6
test('has document title', async ({ page }) => {
7
await page.goto(BASE_URL)
8
await expect(page).toHaveTitle('Vite + React + TS')
9
10
await page.screenshot({ path: `screenshots/example-has-title.png` })
11
})
12
13
test('click count button', async ({ page }) => {
14
await page.goto(BASE_URL)
15
16
// const counter = page.locator('.card button')
17
const counter = await page.getByText('count is 0')
18
counter.click()
19
20
await expect(page.locator('.card button')).toHaveText('count is 1')
21
})
22
23
test('set heading title', async ({ page }) => {
24
await page.goto(BASE_URL)
25
26
const input = page.locator('input')
27
await input.fill('Hello World')
28
29
const button = page.locator('button[type=submit]')
30
button.click()
31
32
await expect(page.locator('h1')).toHaveText('Hello World')
33
34
await page.screenshot({ path: `screenshots/example-set-title.png` })
35
})
36
})

สรุป

วันนี้ก็ลองเทส Playwright ด้วยการใช้ Selector ง่ายๆ มีการ fill input และก็ take screenshot ไป หวังว่าบทความนี้ะเป็นไอเดีย และแนวทางให้ใครหลายๆ คนที่ลองใช้ Playwright ลองไปเล่นกันดูนะครับ

Happy Coding ❤️

Authors
avatar

Chai Phonbopit

เป็น Web Dev ในบริษัทแห่งหนึ่ง ทำงานมา 10 ปีกว่าๆ ด้วยภาษาและเทคโนโลยี เช่น JavaScript, Node.js, React, Vue และปัจจุบันกำลังสนใจในเรื่องของ Blockchain และ Crypto กำลังหัดเรียนภาษา Rust

Related Posts