Tất tần tật về RSpec cơ bản
Vài lời phi lộ 😙
Thỉnh thoảng, chúng ta có thể thấy một số bình luận rằng "RSpec khá khó". Đúng là RSpec có cú pháp đặc biệt một chút và tính năng khá nhiều, nên cảm giác ban đầu có thể là "khó hiểu". (Nhớ lại, lúc đầu khi mình thấy cú pháp của nó, cảm giác cũng hơi "bối rối" một chút 😄)
Tuy nhiên, không chỉ riêng RSpec, bất kì framework nào, khi bạn đã quen thuộc, việc viết code trở nên trôi chảy hơn. Thực sự mình viết test code với RSpec và cảm thấy rất tiện lợi.
Trong bài viết này, mình sẽ tổng hợp những cú pháp và tính năng của RSpec mà theo mình, chỉ cần nắm vững những điều này, chúng ta sẽ ổn!
Cụ thể như sau:
- Hiểu vai trò của
describe
/it
/expect
describe
lồng nhau- Cách sử dụng
context
- Cách sử dụng
before
- Cách sử dụng
let
/let!
/subject
- Cách sử dụng
shared_examples
- Cách sử dụng
shared_context
- Biết lựa chọn giữa
pending
vàskip
Đối tượng độc giả
- Những ai mới bắt đầu với RSpec và cảm thấy hơi "e dè".
- Những ai đã sử dụng RSpec một vài lần nhưng vẫn cảm thấy "chưa thấm".
- Những ai đã từng làm việc với dự án liên quan đến Ruby và bị "đẩy" vào viết test bằng RSpec.
Phiên bản RSpec và Ruby được áp dụng
- RSpec 3.1.0
- Ruby 2.1.3 Lưu ý, trong bài viết này, Rails sẽ không được đề cập. Chúng ta sẽ tập trung vào "chương trình Ruby thuần".
Về việc thiết lập RSpec
Mình đã làm viết một bài hướng dẫn cách thiết lập môi trường RSpec (không dùng Rails). Những ai mới bắt đầu sử dụng RSpec có thể xem blog này để tham khảo.
Cách setup môi trường Rspec3 không cần dùng Rails - 2023
Môi trường thực thi trong video này như sau:
- ruby 2.6.3
- rspec 3.8.0
- bundler 2.0.2
- rubygems 3.0.3
Bắt đầu bước đầu tiên nào
Hiểu vai trò của describe
/ it
/ expect
Một test RSpec đơn giản sẽ có dạng như sau:
Nhân tiện, khi viết hướng dẫn, mình nhận ra rằng chúng ta nên viết
describe
ở ngoài cùng làRSpec.describe
.
RSpec.describe 'Phép toán cơ bản' do
it '1 + 1 sẽ bằng 2' do
expect(1 + 1).to eq 2
end
end
describe
(hay RSpec.describe
) dùng để nhóm các test lại với nhau. Ở đây, chúng ta đang thông báo rằng "Đây là nhóm test cho phép toán cơ bản". Nếu dịch describe
ra tiếng Việt, nó có nghĩa là "mô tả" hoặc "giải thích".
it
giúp tổ chức các test thành các đơn vị gọi là example. Nếu tất cả các expectation (so sánh) trong một example đều thành công, example đó sẽ được xem là đã "thành công". Nếu một expectation bị thất bại, example đó sẽ "thất bại".
expect
là một method dùng để thực hiện so sánh. Ở đây, chúng ta đang so sánh xem 1 + 1
có bằng 2 không bằng cách sử dụng to eq
.
Một lưu ý quan trọng là khi sử dụng describe
, bạn nên tránh viết phức tạp. Mục tiêu của describe
chỉ là nhóm các test lại với nhau dưới một chủ đề cụ thể.
Trong quá trình làm việc, bạn sẽ gặp phải nhiều trường hợp khác nhau cần so sánh, nhưng chỉ cần nắm vững cú pháp này, việc viết test với RSpec sẽ trở nên dễ dàng hơn rất nhiều.
Tóm tắt:
describe
dùng để nhóm các test lại với nhau.it
dùng để định nghĩa một test đơn lẻ.expect
dùng để so sánh giữa kết quả thực tế và kết quả mong đợi.
Cú pháp khác
Trong RSpec, có một số cú pháp khác bạn có thể sử dụng:
describe
có thể được thay thế bằngcontext
it
có thể được thay thế bằngspecify
expect
cũng có thể viết dưới dạng khác nhưis_expected.to
hoặcshould
Tuy nhiên, trong bài viết này, chúng ta sẽ tập trung vào cú pháp cơ bản và phổ biến nhất.
Làm sao để describe
lồng nhau?
Nếu bạn muốn tổ chức các test một cách cụ thể hơn, bạn có thể sử dụng describe
lồng nhau. Điều này giúp bạn nhóm các test dưới một chủ đề con.
RSpec.describe 'Phép toán' do
describe 'cơ bản' do
it '1 + 1 sẽ bằng 2' do
expect(1 + 1).to eq 2
end
end
describe 'nâng cao' do
it '1 + 2 không bằng 2' do
expect(1 + 2).not_to eq 2
end
end
end
Ở đây, chúng ta có một nhóm test cho "Phép toán", và bên trong đó lại chia thành hai nhóm con là "cơ bản" và "nâng cao". Bằng cách này, bạn có thể tổ chức các test một cách hợp lý hơn.
Tóm tắt:
describe
có thể lồng nhau để nhóm các test dưới một chủ đề con.
Trong phần này, chúng ta đã tìm hiểu về cú pháp cơ bản của RSpec và cách tổ chức các test. Trong các phần tiếp theo, chúng ta sẽ đi sâu vào các tính năng và kỹ thuật khác của RSpec.
Sử dụng context và before để hiệu quả hơn
Hãy xem xét cách sử dụng context
và before
thông qua một ví dụ thực tế.
Ở đây, chúng ta sẽ kiểm tra một class như sau:
class User
def initialize(name:, age:)
@name = name
@age = age
end
def greet
if @age <= 12
"ぼくは#{@name}だよ。"
else
"僕は#{@name}です。"
end
end
end
Dựa trên kiến thức đã học từ "Bước đầu tiên", chúng ta có thể viết bài kiểm tra như sau:
RSpec.describe User do
describe '#greet' do
it 'Trường hợp dưới 12 tuổi, trả lời bằng chữ Hiragana' do
user = User.new(name: 'たろう', age: 12)
expect(user.greet).to eq 'ぼくはたろうだよ。'
end
it 'Trường hợp từ 13 tuổi trở lên, trả lời bằng Kanji' do
user = User.new(name: 'たろう', age: 13)
expect(user.greet).to eq '僕はたろうです。'
end
end
end
Chú ý, "greet" trong tiếng Nhật có nghĩa là "chào hỏi".
Dù không có vấn đề gì với cách viết trên, nhưng hãy thử refactor bài kiểm tra theo phong cách RSpec hơn một chút.
Chú thích: Điểm khác biệt so với "Bước đầu tiên"
Trong describe
, bạn cũng có thể truyền class thay vì chuỗi, như describe User
.
Ngoài ra, khi bạn muốn nói rằng "chúng ta đang kiểm tra phương thức greet của instance", bạn thường viết như describe '#greet'
.
Nhóm theo điều kiện với context
Trong RSpec, ngoài describe
, chúng ta cũng có context
để nhóm các bài kiểm tra.
Cả hai đều giống nhau về chức năng, nhưng context
thường được sử dụng khi phân loại theo điều kiện.
"Context" trong tiếng Anh có nghĩa là "ngữ cảnh" hoặc "tình hình".
Ở đây, chúng ta đã chia thành hai nhóm điều kiện: "Dưới 12 tuổi" và "Từ 13 tuổi trở lên".
RSpec.describe User do
describe '#greet' do
context 'Trường hợp dưới 12 tuổi' do
it 'Trả lời bằng chữ Hiragana' do
user = User.new(name: 'たろう', age: 12)
expect(user.greet).to eq 'ぼくはたろうだよ。'
end
end
context 'Trường hợp từ 13 tuổi trở lên' do
it 'Trả lời bằng Kanji' do
user = User.new(name: 'たろう', age: 13)
expect(user.greet).to eq '僕はたろうです。'
end
end
end
end
Giống như describe
, khi nhóm đúng cách bằng context
, người đọc mã kiểm tra sẽ dễ dàng hiểu điều kiện đang được kiểm tra trong mỗi block context
.
Sử dụng before để chuẩn bị công việc chung
Phần được bao quanh bởi before do ... end
sẽ được gọi trước mỗi lần thực thi example.
Trong block before
, chúng ta thường thiết lập dữ liệu hoặc thực hiện các công việc chung trước khi kiểm tra.
Dù hơi cồng kềnh, nhưng trong mã mẫu, name: 'たろう'
đã được lặp lại. Hãy thử viết lại theo phong cách DRY (mình đã có một bài viết về chủ đề này bạn có thể tham khảo).
RSpec.describe User do
describe '#greet' do
before do
@params = { name: 'たろう' }
end
context 'Trường hợp dưới 12 tuổi' do
it 'Trả lời bằng chữ Hiragana' do
user = User.new(**@params.merge(age: 12))
expect(user.greet).to eq 'ぼくはたろうだよ。'
end
end
context 'Trường hợp từ 13 tuổi trở lên' do
it 'Trả lời bằng Kanji' do
user = User.new(**@params.merge(age: 13))
expect(user.greet).to eq '僕はたろうです。'
end
end
end
end
Như bạn có thể thấy trong ví dụ trên, thay vì sử dụng biến cục bộ, chúng ta đã đặt dữ liệu vào biến instance. Điều này là do block before
và block it
có phạm vi biến khác nhau.
(Double splat operator **
trước @params
được sử dụng để chuyển đổi đối tượng hash thành tham số từ khóa.)
Sử dụng before trong describe hoặc context lồng nhau
Bạn có thể thiết lập before
cho mỗi describe
hoặc context
.
Nếu describe
hoặc context
lồng nhau, before
sẽ được gọi theo thứ tự quan hệ cha con.
Hãy thay đổi ví dụ trước như sau:
RSpec.describe User do
describe '#greet' do
before do
@name = 'たろう'
end
context 'Trường hợp dưới 12 tuổi' do
before do
@age = 12
end
it 'Trả lời bằng chữ Hiragana' do
user = User.new(name: @name, age: @age)
expect(user.greet).to eq 'ぼくはたろうだよ。'
end
end
context 'Trường hợp từ 13 tuổi trở lên' do
before do
@age = 13
end
it 'Trả lời bằng Kanji' do
user = User.new(name: @name, age: @age)
expect(user.greet).to eq '僕はたろうです。'
end
end
end
end
Ở đây, chúng ta đã đặt @name
ở mức describe
và @age
ở mức context
.
Khi bạn cần chia sẻ dữ liệu giữa nhiều example hoặc context, before
là một công cụ hữu ích.
Ứng dụng (Kỹ thuật nâng cao)
Đến nay, với những kiến thức bạn đã học, chúng ta hoàn toàn có thể viết test bằng RSpec một cách bình thường.
Tuy nhiên, những người làm chủ RSpec xung quanh bạn có thể đang sử dụng một số kỹ thuật như sau. Vậy nên, chúng ta hãy tìm hiểu thêm một chút về những nội dung sau đây.
Sử dụng let thay vì biến instance
Trong ví dụ code trước đó, chúng ta sử dụng biến instance @params. Nhưng, trong RSpec, chúng ta có thể thay thế biến instance này bằng chức năng let.
Hãy xem ví dụ code đã sử dụng let:
RSpec.describe User do
describe '#greet' do
let(:params) { { name: 'たろう' } }
context 'Khi dưới 12 tuổi' do
before do
params.merge!(age: 12)
end
it 'Trả lời bằng chữ Hiragana' do
user = User.new(**params)
expect(user.greet).to eq 'ぼくはたろうだよ。'
end
end
context 'Khi trên 13 tuổi' do
before do
params.merge!(age: 13)
end
it 'Trả lời bằng chữ Kanji' do
user = User.new(**params)
expect(user.greet).to eq '僕はたろうです。'
end
end
end
end
Khi viết let(:foo) { ... }, chúng ta có thể tham chiếu đến giá trị trong { ... } như là foo. Đây chính là cách sử dụng cơ bản của let.
Nhưng, trong ví dụ trên, { { name: 'たろう' } } có hai cặp dấu {}, điều này có thể gây nhầm lẫn. Dấu {} bên ngoài là block của Ruby, trong khi dấu {} bên trong là một hash.
Nếu bạn cảm thấy khó hiểu, bạn có thể tham khảo cách viết sau:
# Đây là mã có cùng ý nghĩa với let(:params) { { name: 'たろう' } }
let(:params) do
hash = {}
hash[:name] = 'たろう'
hash
end
Chuyển user
thành let
Không chỉ với biến instance, việc thay biến cục bộ bằng let
cũng là một cách tiếp cận.
Hãy thử thay user
bằng let
.
RSpec.describe User do
describe '#greet' do
let(:user) { User.new(**params) }
let(:params) { { name: 'たろう' } }
context 'Trường hợp dưới 12 tuổi' do
before do
params.merge!(age: 12)
end
it 'Phản hồi bằng Hiragana' do
expect(user.greet).to eq 'ぼくはたろうだよ。'
end
end
context 'Trường hợp từ 13 tuổi trở lên' do
before do
params.merge!(age: 13)
end
it 'Phản hồi bằng Kanji' do
expect(user.greet).to eq '僕はたろうです。'
end
end
end
end
Phần code user = User.new(**params)
bị lặp đi lặp lại đã được tối ưu và sử dụng chung.
Tận dụng lợi ích của let
để thay thế age
bằng let
Trong đoạn code trên, việc sử dụng params.merge!(age: 12)
trong block before
không thực sự tốt.
Vậy nên, ta nên thay đổi phần này bằng let
để code trở nên gọn gàng hơn.
RSpec.describe User do
describe '#greet' do
let(:user) { User.new(**params) }
let(:params) { { name: 'たろう', age: age } }
context 'Trường hợp dưới 12 tuổi' do
let(:age) { 12 }
it 'Phản hồi bằng Hiragana' do
expect(user.greet).to eq 'ぼくはたろうだよ。'
end
end
context 'Trường hợp từ 13 tuổi trở lên' do
let(:age) { 13 }
it 'Phản hồi bằng Kanji' do
expect(user.greet).to eq '僕はたろうです。'
end
end
end
end
Giờ đây, lợi ích của let
được hiển thị rõ ràng.
let
có tính chất đánh giá chậm - nghĩa là nó sẽ không được gọi cho đến khi thực sự cần thiết.
Theo thứ tự trong ví dụ code trên:
expect(user.greet).to
được gọi => user là gì?let(:user) { User.new(**params) }
được gọi => params là gì?let(:params) { { name: 'たろう', age: age } }
được gọi => age là gì?let(:age) { 12 }
(hoặc 13) được gọi- Kết quả cuối cùng, ta đã gọi
expect(User.new(name: 'たろう', age: 12).greet).to
.
Nếu chúng ta cố gắng thực hiện điều này bằng "before + biến instance" sẽ phức tạp hơn nhiều. Thực tế, mình cũng không nghĩ ra cách viết ngay lúc này.
Như vậy, nếu tận dụng đúng cách, let
có thể giúp chúng ta viết code test hiệu quả hơn.
Sử dụng subject
để tổng hợp đối tượng test vào một chỗ
Khi đối tượng cần test (hoặc kết quả thực thi của một phương thức) rõ ràng chỉ có một, ta có thể sử dụng chức năng subject
để làm cho mã test trở nên DRY.
Chẳng hạn, trong ví dụ code ở trên, cả hai ví dụ đều test kết quả thực thi của user.greet
. Do đó, hãy thử nâng user.greet
lên làm subject
và xóa nó khỏi các ví dụ.
RSpec.describe User do
describe '#greet' do
let(:user) { User.new(**params) }
let(:params) { { name: 'たろう', age: age } }
subject { user.greet }
context 'Trường hợp dưới 12 tuổi' do
let(:age) { 12 }
it { is_expected.to eq 'ぼくはたろうだよ。' }
end
context 'Trường hợp từ 13 tuổi trở lên' do
let(:age) { 13 }
it { is_expected.to eq '僕はたろうです。' }
end
end
end
Khi khai báo subject { user.greet }
, phần code trước đây là expect(user.greet).to eq 'ぼくはたろうだよ。'
giờ đã được thay đổi thành is_expected.to eq 'ぼくはたろうだよ。'
.
Thêm vào đó, chúng ta cũng có thể bỏ qua chuỗi truyền vào it
.
RSpec.describe User do
describe '#greet' do
let(:user) { User.new(**params) }
let(:params) { { name: 'たろう', age: age } }
subject { user.greet }
context 'Trường hợp dưới 12 tuổi' do
let(:age) { 12 }
it { is_expected.to eq 'ぼくはたろうだよ。' }
end
context 'Trường hợp từ 13 tuổi trở lên' do
let(:age) { 13 }
it { is_expected.to eq '僕はたろうです。' }
end
end
end
Kết quả mong đợi là 'ぼくはたろうだよ。' thì có thể được diễn đạt một cách tự nhiên trong tiếng Anh như sau: 'it is expected to equal 'ぼくはたろうだよ。''.
Nếu như bạn có ý muốn đề cập đến "subject" bằng tiếng Việt là "Chủ thể" hay "Đối tượng", thì cách diễn đạt này không phù hợp. Tuy vậy, nếu ta suy nghĩ cách này user.greet { is_expected.to eq 'ぼくはたろうだよ。' }
, thì cũng có thể hiểu là "subject = 主題 của test case này.
Chỉ với hai dòng lệnh, chúng ta đã mô tả rõ ràng cách test và kết quả mong đợi của nó.
Hoàn thiện code test sau khi tái cấu trúc
Đến thời điểm này, chúng ta đã tách params
và user
. Tuy nhiên, việc tách ra không mang lại lợi ích gì nên hãy inline nội dung của params
.
RSpec.describe User do
describe '#greet' do
let(:user) { User.new(name: 'たろう', age: age) }
subject { user.greet }
context 'Khi dưới 12 tuổi' do
let(:age) { 12 }
it { is_expected.to eq 'ぼくはたろうだよ。' }
end
context 'Khi từ 13 tuổi trở lên' do
let(:age) { 13 }
it { is_expected.to eq '僕はたろうです。' }
end
end
end
Với điều này, code test đã hoàn thiện.
(Arg của User.new không phải là hash object nữa mà đã chuyển thành keyword arg, nên ** không cần thiết nữa)
Lưu ý: Hãy tránh viết code test quá phức tạp
Cho đến giờ, chúng ta đã giới thiệu nhiều kỹ thuật khác nhau, nhưng không cần phải sử dụng let
hay subject
một cách quá mức.
Nếu bạn cố gắng viết code test DRY hoàn toàn và sử dụng các chức năng này quá mức, code có thể trở nên quá phức tạp, khiến việc đọc code test trở nên khó khăn. Vì vậy, với code test, hãy coi trọng tính "dễ đọc" hơn là "DRY". Khác với code ứng dụng, một chút trùng lặp trong code test là có thể chấp nhận được.
Điều này cũng áp dụng cho các kỹ thuật khác như shared_examples
hay shared_context
.
Viết chuỗi cho it
hoặc context
bằng tiếng Anh hay tiếng Nhật?
Đến nay, chúng ta đã sử dụng tiếng Nhật nhiều cho người Nhật, nhưng khi bạn viết chuỗi cho it
hoặc context
bằng tiếng Anh trong RSpec, nó sẽ trở nên giống như tài liệu test tiếng Anh.
Ví dụ như sau:
RSpec.describe User do
describe '#greet' do
let(:user) { User.new(**params) }
let(:params) { { name: 'たろう', age: age } }
context 'Khi 12 tuổi hoặc dưới đó' do
let(:age) { 12 }
it 'chào bằng hiragana' do
expect(user.greet).to eq 'ぼくはたろうだよ。'
end
end
context 'Khi 13 tuổi trở lên' do
let(:age) { 13 }
it 'chào bằng kanji' do
expect(user.greet).to eq '僕はたろうです。'
end
end
end
end
Khi sử dụng context
, bạn có thể chuyển chuỗi mô tả như 'khi ~', 'với ~', 'không có ~', để rõ ràng việc bạn đang phân nhóm theo điều kiện.
Và sau it
, hãy kết nối động từ diễn đạt "hành vi" của phương thức bạn đang test và mô tả bằng câu tiếng Anh.
Tuy nhiên, việc viết câu tiếng Anh như bạn nghĩ không phải lúc nào cũng dễ dàng.
Nếu chỉ có người Nhật đọc code test, thì việc viết bằng tiếng Nhật không phải là vấn đề. Thay vào đó, sử dụng thời gian đó để viết code test sẽ hiệu quả hơn. Và, viết tiếng Anh một cách cố gắng có thể tạo ra ngữ pháp lộn xộn mà người khác không thể hiểu.
Vì vậy, nếu bạn không giỏi tiếng Anh, hãy viết bằng tiếng Nhật mà không cần ép buộc.
RSpec.describe User do
describe '#greet' do
let(:user) { User.new(**params) }
let(:params) { { name: 'たろう', age: age } }
context 'Khi dưới 12 tuổi' do
let(:age) { 12 }
it 'Trả lời bằng hiragana' do
expect(user.greet).to eq 'ぼくはたろうだよ。'
end
end
context 'Khi từ 13 tuổi trở lên' do
let(:age) { 13 }
it 'Trả lời bằng kanji' do
expect(user.greet).to eq '僕はたろうです。'
end
end
end
end
Bí danh của it
là example
và specify
Cho đến nay, chúng ta đã viết test dưới dạng it 'xxx'
, nhưng trong RSpec, ngoài it
, còn có example
và specify
cùng một chức năng. Ba phương thức này đều có mối quan hệ tương tự, vì vậy code test sau sẽ có cùng ý nghĩa:
it '1 + 1 sẽ là 2' do
expect(1 + 1).to eq 2
end
specify '1 + 1 sẽ là 2' do
expect(1 + 1
).to eq 2
end
example '1 + 1 sẽ là 2' do
expect(1 + 1).to eq 2
end
Tuy nhiên, việc sử dụng specify
hoặc example
không phổ biến bằng it
, vì vậy chúng ta sẽ tiếp tục sử dụng it
cho những ví dụ sau.
Sử dụng "should" thay vì "expect"
Trong RSpec, có cả 2 cách viết "should" và "expect". Tuy nhiên, từ RSpec 3 trở đi, "expect" đã trở thành cách viết tiêu chuẩn. Cách viết "should" cũng vẫn được hỗ trợ, nhưng nó không được khuyến nghị.
Vì vậy, trừ khi bạn có lý do đặc biệt, bạn nên sử dụng cách viết "expect".
Dưới đây là bản dịch của đoạn nội dung bạn cung cấp:
Chức năng cao cấp của RSpec
Nếu bạn đã sử dụng thành thạo let
và subject
, bạn có thể coi như đã vượt qua cấp độ cơ bản. Tuy nhiên, thêm vào đó, việc biết thêm vài kỹ thuật khác có thể hữu ích trong một số trường hợp.
Tái sử dụng example: shared_examples và it_behaves_like
Hãy thử bổ sung thêm một số mẫu test vào đoạn mã test mà chúng ta đã viết phía trên. Không chỉ là tuổi 12 và 13, chúng ta cũng sẽ thử viết lời chào cho trẻ con (0 tuổi) và người lớn hơn nữa (100 tuổi).
RSpec.describe User do
describe '#greet' do
let(:user) { User.new(name: 'たろう', age: age) }
subject { user.greet }
context 'Trường hợp 0 tuổi' do
let(:age) { 0 }
it { is_expected.to eq 'ぼくはたろうだよ。' }
end
context 'Trường hợp 12 tuổi' do
let(:age) { 12 }
it { is_expected.to eq 'ぼくはたろうだよ。' }
end
context 'Trường hợp 13 tuổi' do
let(:age) { 13 }
it { is_expected.to eq '僕はたろうです。' }
end
context 'Trường hợp 100 tuổi' do
let(:age) { 100 }
it { is_expected.to eq '僕はたろうです。' }
end
end
end
Như bạn có thể thấy, chúng ta có một số example bị lặp lại. Trong trường hợp này, chúng ta có thể sử dụng shared_examples
và it_behaves_like
để tái sử dụng example.
RSpec.describe User do
describe '#greet' do
let(:user) { User.new(name: 'たろう', age: age) }
subject { user.greet }
shared_examples 'Lời chào của trẻ con' do
it { is_expected.to eq 'ぼくはたろうだよ。' }
end
context 'Trường hợp 0 tuổi' do
let(:age) { 0 }
it_behaves_like 'Lời chào của trẻ con'
end
context 'Trường hợp 12 tuổi' do
let(:age) { 12 }
it_behaves_like 'Lời chào của trẻ con'
end
shared_examples 'Lời chào của người lớn' do
it { is_expected.to eq '僕はたろうです。' }
end
context 'Trường hợp 13 tuổi' do
let(:age) { 13 }
it_behaves_like 'Lời chào của người lớn'
end
context 'Trường hợp 100 tuổi' do
let(:age) { 100 }
it_behaves_like 'Lời chào của người lớn'
end
end
end
Bạn có thể tưởng tượng shared_examples 'foo' do ... end
để định nghĩa example muốn tái sử dụng và it_behaves_like 'foo'
để gọi example đã định nghĩa đó.
Tiếp theo là shared_context
và include_context
Chúng ta sẽ thêm một phương thức mới, child?
, vào lớp User.
class User
def initialize(name:, age:)
@name = name
@age = age
end
def greet
if child?
"ぼくは#{@name}だよ。"
else
"僕は#{@name}です。"
end
end
def child?
@age <= 12
end
end
Không chỉ viết mã test cho phương thức greet
, chúng ta cũng nên viết cho phương thức child?
.
RSpec.describe User do
describe '#greet' do
let(:user) { User.new(name: 'たろう', age: age) }
subject { user.greet }
context 'Trường hợp dưới 12 tuổi' do
let(:age) { 12 }
it { is_expected.to eq 'ぼくはたろうだよ。' }
end
context 'Trường hợp từ 13 tuổi trở lên' do
let(:age) { 13 }
it { is_expected.to eq '僕はたろうです。' }
end
end
describe '#child?' do
let(:user) { User.new(name: 'たろう', age: age) }
subject { user.child? }
context 'Trường hợp dưới 12 tuổi' do
let(:age) { 12 }
it { is_expected.to eq true }
end
context 'Trường hợp từ 13 tuổi trở lên' do
let(:age) { 13 }
it { is_expected.to eq false }
end
end
end
Khi viết mã test, chúng ta có thể thấy rằng cả hai bài test đều sử dụng những context
giống nhau. Trong trường hợp này, chúng ta có thể sử dụng shared_context
và include_context
để tái sử dụng context
.
RSpec.describe User do
let(:user) { User.new(name: 'たろう', age: age) }
shared_context 'Trường hợp 12 tuổi' do
let(:age
) { 12 }
end
shared_context 'Trường hợp 13 tuổi' do
let(:age) { 13 }
end
describe '#greet' do
subject { user.greet }
include_context 'Trường hợp 12 tuổi'
it { is_expected.to eq 'ぼくはたろうだよ。' }
include_context 'Trường hợp 13 tuổi'
it { is_expected.to eq '僕はたろうです。' }
end
describe '#child?' do
subject { user.child? }
include_context 'Trường hợp 12 tuổi'
it { is_expected.to eq true }
include_context 'Trường hợp 13 tuổi'
it { is_expected.to eq false }
end
end
Tương tự như với shared_examples
, bạn có thể sử dụng shared_context 'foo' do ... end
để định nghĩa context bạn muốn tái sử dụng và sử dụng include_context 'foo'
để gọi lại context đã định nghĩa.
Lưu ý: Khi viết code test, ta nên ưu tiên độ dễ đọc hơn là việc áp dụng DRY (Don't Repeat Yourself).
Mặc dù đã nói trước đó nhưng muốn nhắc lại là, nếu sử dụng quá nhiều shared_examples
và shared_context
, có thể gây rối rắm cho người đọc sau này hoặc cho chính bạn khi phải đọc lại code.
Khi viết code test, việc ưu tiên độ dễ đọc và chấp nhận một số trùng lặp sẽ mang lại kết quả tốt hơn so với việc áp dụng quá mức nguyên tắc DRY.
Tham khảo: Cũng có cách viết là be_truthy / be_falsey
Thay vì viết it { is_expected.to eq true }
và it { is_expected.to eq false }
, chúng ta có thể viết lại là it { is_expected.to be_truthy }
và it { is_expected.to be_falsey }
.
Chủ đề này liên quan đến matcher nên trong bài viết này, chúng ta sẽ không đi sâu vào chi tiết mà sẽ sử dụng eq true
và eq false
.
Dựa vào những kỹ thuật này, chúng ta có thể viết mã test hiệu quả hơn và giảm lượng mã bị lặp lại.
let
có đánh giá trễ và let!
được thực thi trước
let
thật sự là một tính năng tiện lợi khi bạn biết cách sử dụng nó. Tuy nhiên, tính chất "đánh giá trễ" của nó đôi khi khiến cho việc debug các test thất bại trở nên khó khăn.
Ví dụ, khi bạn viết test cho một model Blog trong Rails như sau, test sẽ thất bại. (Xin lỗi, chỉ ở đây mình sử dụng test của Rails)
RSpec.describe Blog do
let(:blog) { Blog.create(title: 'Kiểm thử với với RSpec', content: 'Sẽ viết sau') }
it 'Có thể truy xuất blog' do
expect(Blog.first).to eq blog
end
end
Bạn có biết lí do test này thất bại không?
Hãy chú ý đến phần expect(Blog.first).to eq blog
. Khi gọi Blog.first
, let(:blog)
vẫn chưa được thực thi, nghĩa là bản ghi chưa được lưu vào cơ sở dữ liệu. Do đó, Blog.first
sẽ trả về nil
và test sẽ thất bại. (Chỉ trong khoảnh khắc so sánh nil
và blog
, bản ghi mới được lưu vào cơ sở dữ liệu).
Một cách để tránh vấn đề này là gọi let(:blog)
một cách rõ ràng trong block before
.
Khi làm như vậy, bản ghi sẽ được lưu vào cơ sở dữ liệu trước khi example
được thực thi.
RSpec.describe Blog do
let(:blog) { Blog.create(title: 'Kiểm thử với với RSpec', content: 'Sẽ viết sau') }
before do
blog # Lưu bản ghi vào cơ sở dữ liệu ở đây
end
it 'Có thể truy xuất blog' do
expect(Blog.first).to eq blog
end
end
Tuy nhiên, thay vì viết như trên, chúng ta có thể sử dụng let!
. Khi dùng let!
, giá trị được định nghĩa sẽ được tạo ra trước khi example
thực thi.
RSpec.describe Blog do
let!(:blog) { Blog.create(title: 'Kiểm thử với với RSpec', content: 'Sẽ viết sau') }
it 'Có thể truy xuất blog' do
expect(Blog.first).to eq blog
end
end
Vì vậy, khi đánh giá trễ của let
gây ra lỗi trong test, sử dụng let!
sẽ rất tiện lợi.
Đánh dấu test không thực thi với: pending
Khi một test cần phải pass nhưng vì một lý do nào đó mà nó không thể pass, hãy dùng pending
để đánh dấu nó.
RSpec.describe 'Lớp nhạy cảm' do
it 'Test nhạy cảm' do
expect(1 + 2).to eq 3
pending 'Phần tiếp theo có gì đó sai, sẽ sửa sau'
# Một expectation không pass (được thực thi)
expect(foo).to eq bar
end
end
Khác với pending
thông thường, code vẫn tiếp tục chạy. Nếu test thất bại, nó sẽ được đánh dấu là "pending". Nếu test pass, RSpec sẽ phản ứng với thông báo "Sao lại pass!?" và test sẽ bị coi là thất bại.
Dù có vẻ khá kỳ cục, nhưng tính năng này có thể giúp tránh được tình huống "lạc lõng" khi một test bất ngờ pass.
Dừng thực thi test mà không cần lý do: skip
Nếu bạn thực sự muốn dừng việc thực thi test tại một điểm nào đó, hãy sử dụng skip
.
RSpec.describe 'Không muốn thực thi vì lý do nào đó' do
it 'Test không muốn thực thi' do
expect(1 + 2).to eq 3
skip 'Dừng thực thi tại đây'
# Code từ đây trở đi sẽ không được thực thi
expect(foo).to eq bar
end
end
Khác với pending
, skip
sẽ dừng việc thực thi và đánh dấu test là "pending". Cá nhân mình nghĩ rằng việc sử dụng skip
không phổ biến bằng pending
.
Dừng thực thi toàn bộ example
một cách nhanh chóng: xit
Khi bạn muốn dừng thực thi toàn bộ example
một cách nhanh chóng, hãy thay it
bằng xit
và example
đó sẽ không được thực thi (được xem như "pending").
RSpec.describe 'Không muốn thực thi vì lý do nào đó' do
xit 'Test không muốn thực thi' do
expect(1 + 2).to eq 3
expect(foo).to eq bar
end
end
Không chỉ xit
, chúng ta cũng có thể sử dụng xdescribe
và xcontext
để ngừng thực thi một nhóm example
cùng một lúc.
Điều này rất hữu ích khi bạn muốn tạm thời dừng thực thi một số example
mà không muốn xoá chúng.
Sử dụng focus
để chỉ thực thi một số example
Khi bạn muốn thực thi chỉ một số example
trong một file test lớn, bạn có thể sử dụng focus
để tập trung vào những example
đó.
Đầu tiên, hãy thêm cấu hình sau vào RSpec.configure
:
RSpec.configure do |config|
config.filter_run_when_matching :focus
end
Tiếp theo, bạn thêm focus: true
vào example
, describe
hoặc context
mà bạn muốn thực thi.
RSpec.describe 'Lớp với nhiều example' do
it 'Test thông thường', focus: true do
expect(1 + 2).to eq 3
end
it 'Test khác' do
expect(4 + 5).to eq 9
end
end
Khi thực thi test, chỉ example
có focus: true
sẽ được thực thi.
Lưu ý rằng sau khi hoàn thành việc debug, bạn nên xoá focus: true
để đảm bảo tất cả các example
đều được thực thi.
Sử dụng tag
để phân loại và thực thi test
Trong RSpec, bạn có thể gán các tag
cho example
và thực thi test dựa trên tag
.
Ví dụ:
RSpec.describe 'Lớp với tag' do
it 'Test dành cho trình duyệt Chrome', browser: :chrome do
# Mã cho test Chrome
end
it 'Test dành cho trình duyệt Firefox', browser: :firefox do
# Mã cho test Firefox
end
end
Khi bạn muốn thực thi chỉ những test dành cho Chrome, bạn chỉ cần chạy:
rspec --tag browser:chrome
Sử dụng tag
giúp bạn có thể dễ dàng quản lý và thực thi những test cụ thể trong một file test lớn.
Bỏ qua toàn bộ nhóm: xdescribe / xcontext
Không chỉ với "it", bạn cũng có thể thêm "x" vào trước "describe" và "context".
# Bỏ qua toàn bộ nhóm
xdescribe 'Phép toán cộng trừ nhân chia' do
it '1 + 1 sẽ bằng 2' do
expect(1 + 1).to eq 2
end
it '10 - 1 sẽ bằng 9' do
expect(10 - 1).to eq 9
end
end
# Bỏ qua toàn bộ nhóm
xcontext 'Trong trường hợp quản trị viên' do
it 'có thể chỉnh sửa thông tin nhân viên' do
# ...
end
it 'có thể xoá thông tin nhân viên' do
# ...
end
end
Viết test sau: it
không chứa nội dung
Khi bạn bỏ qua phần do ... end
trong it 'something' do ... end
, trường hợp đó sẽ được đánh dấu như một test đang chờ xử lý (pending).
Điều này rất tiện khi chúng ta muốn triển khai một phương thức và muốn liệt kê các trường hợp test cần xem xét ngay trong code.
Ví dụ, nếu chúng ta muốn thêm phương thức good_bye
vào class User
, trước khi triển khai, chúng ta có thể đặc tả như sau trong RSpec (có thể xem như một danh sách công việc trước khi thực hiện):
RSpec.describe User do
describe '#good_bye' do
context 'Khi dưới 12 tuổi' do
it 'nói lời tạm biệt bằng chữ Hiragana'
end
context 'Khi trên 13 tuổi' do
it 'nói lời tạm biệt bằng chữ Kanji'
end
end
end
Khi không còn test nào ở trạng thái pending, đó là lúc phương thức good_bye
đã được triển khai xong.
Tham khảo: Đây có phải là TDD? Việc viết test trước khi triển khai giống như phương pháp TDD (Phát triển Dựa trên Test), nhưng nói một cách chính xác thì chưa chắc đã đúng. Trước hết, các test này đang ở trạng thái pending nên không bị fail (không màu đỏ mà là màu vàng). Với TDD, bạn cần viết code test trước, nhưng ở đây chúng ta không quy định viết cái nào trước. Vì vậy, nó hơi khác với quy trình TDD thông thường.
Còn đối với mình, mình chỉ sử dụng TDD khi cảm thấy nó phù hợp. Khi viết code trong thực tế, mình thường tuân theo một trong ba hướng sau:
- Triển khai trước, viết test sau.
- Viết test trước rồi triển khai (TDD).
- Kiểm tra bằng tay sau khi triển khai và không viết code test. Theo tư duy chủ nghĩa TDD thuần túy có thể khiến hiệu suất phát triển giảm, vì vậy mình luôn cân nhắc lựa chọn phương pháp có hiệu quả nhất trong mỗi tình huống.
Tóm tắt
Trong bài viết này, chúng ta đã đi qua cơ bản về cú pháp RSpec và một số tính năng tiện ích thường dùng. Mặc dù nội dung bao gồm cả cho người mới học và người đã có kinh nghiệm, hy vọng rằng mọi người đọc cũng đã có cái nhìn tổng quan về RSpec.
RSpec có rất nhiều tính năng và một số cú pháp có thể trở nên phức tạp và khó hiểu. Tuy nhiên, bằng việc sử dụng đúng tính năng ở đúng nơi và tránh viết code test quá phức tạp, chúng ta có thể tận dụng những ưu điểm của RSpec và tạo ra "code test dễ đọc và dễ bảo trì".
Hy vọng rằng những ai mới tiếp xúc với RSpec hoặc đã sử dụng RSpec trước đây có thể áp dụng những kiến thức từ bài viết này.
Mình hy vọng bạn thích bài viết này và học thêm được điều gì đó mới.
Donate mình một ly cafe hoặc 1 cây bút bi để mình có thêm động lực cho ra nhiều bài viết hay và chất lượng hơn trong tương lai nhé. À mà nếu bạn có bất kỳ câu hỏi nào thì đừng ngại comment hoặc liên hệ mình qua: Zalo - 0374226770 hoặc Facebook. Mình xin cảm ơn.
Momo: NGUYỄN ANH TUẤN - 0374226770
TPBank: NGUYỄN ANH TUẤN - 0374226770 (hoặc 01681423001)
All Rights Reserved