Hướng dẫn kiểm thử
Trong phần này, chúng tôi sẽ giới thiệu cách kiểm thử các hợp đồng thông minh. Các giao dịch trên blockchain không thể hoàn tác. Do đó, việc kiểm thử hợp đồng thông minh của bạn trước khi triển khai có vai trò rất quan trọng.
Kiểm thử bằng Truffle
Truffle cung cấp công cụ kiểm thử tự động. Công cụ này cho phép bạn viết các kiểm thử đơn giản và dễ quản lý theo hai cách khác nhau:
- Bằng
Javascript
vàTypeScript
, để thực hiện kiểm thử các hợp đồng từ bên ngoài, giống như ứng dụng. - Bằng
Solidity
, để thực hiện kiểm thử trước các hợp đồng, trong các tình huống không có hệ điều hành hoặc ứng dụng.
1. Bắt đầu
Chúng ta sẽ làm theo Hướng dẫn triển khai bằng Truffle để tạo và triển khai một hợp đồng. Tuy nhiên, trước khi triển khai, chúng ta sẽ thêm một hàm thiết lập setGreet
vào hợp đồng nhằm mục đích kiểm thử. Mã nguồn được đưa ra như sau.
LƯU Ý: Chúng tôi đã thực hiện một số sửa đổi với hợp đồng nhằm mục đích kiểm thử.
Dưới đây là mã nguồn của hợp đồng KlaytnGreeting.
pragma solidity 0.5.6;contract Mortal { /* Define variable owner of the type address */ address payable owner; /* This function is executed at initialization and sets the owner of the contract */ constructor () public { owner = msg.sender; } /* Function to recover the funds on the contract */ function kill() public payable { if (msg.sender == owner) selfdestruct(owner); }}contract KlaytnGreeter is Mortal { /* Define variable greeting of the type string */ string greeting; /* This runs when the contract is executed */ constructor (string memory _greeting) public { greeting = _greeting; } /* Main function */ function greet() public view returns (string memory) { return greeting; } /* Newly added function for testing. */ function setGreet(string memory _greeting) public { // only owner can change greeting message require(msg.sender == owner, "Only owner is allowed."); greeting = _greeting; }}
Chúng ta sẽ kiểm thử 1) hàm greet()
để xem nó có trả về thông báo "Hello, Klaytn" một cách chính xác không, 2) hàm setGreet()
để xem nó có thiết lập thông báo chào mừng mới một cách chính xác không và hoàn ngược khi tài khoản không phải là chủ sở hữu cố gắng cập nhật thông báo chào mừng không.
Đầu tiên, chúng ta sẽ cài đặt thư viện công cụ khẳng định Chai (hoặc bất kỳ thư viện công cụ khẳng định khác bạn sử dụng) cho các khẳng định chung và thư viện công cụ khẳng định truffle cho các khẳng định trong hợp đồng thông minh.
npm install --save-dev chai truffle-assertions
2. Viết kiểm thử bằng Solidity
Kiểm thử bằng Solidity có thể trực quan hơn một chút so với kiểm thử bằng JavaScript. Các hợp đồng kiểm thử Solidity được lưu trữ cùng với các kiểm thử JavaScript dưới dạng tập tin .sol.
Tạo một tập tin có tên là TestKlaytnGreeting.sol
trong thư mục test
. Bộ công cụ Truffle cung cấp các thư viện hỗ trợ kiểm thử, vì vậy chúng ta cần nhập các thư viện này. Hãy cùng xem ví dụ kiểm thử bằng Solidity dưới đây:
pragma solidity ^0.5.6;import "truffle/Assert.sol";import "truffle/DeployedAddresses.sol";import "../contracts/HashMarket.sol";
- Assert : Cho phép chúng ta truy cập vào các hàm kiểm thử khác nhau, như
Assert.equals()
,Assert.greaterThan()
, v.v. - DeployedAddresses : Mỗi khi bạn thay đổi hợp đồng của mình, bạn phải triển khai lại hợp đồng để có một địa chỉ mới. Bạn có thể lấy các địa chỉ hợp đồng đã triển khai thông qua thư viện này.
Bây giờ, hãy cùng viết mã kiểm thử.
pragma solidity ^0.5.6;import "truffle/Assert.sol";import "truffle/DeployedAddresses.sol";import "../contracts/KlaytnGreeter.sol";contract TestKlaytnGreeter { function testGreetingMessage() public { // DeployedAddresses.KlaytnGreeter() handles contract address. KlaytnGreeter greeter = KlaytnGreeter(DeployedAddresses.KlaytnGreeter()); string memory expectedGreet = "Hello Klaytn"; string memory greet = greeter.greet(); Assert.equal(greet, expectedGreet, "greeting message should match"); }}
Chạy mã kiểm thử Solidity của bạn.
$ truffle test# OutputUsing network 'development'.Compiling your contracts...===========================> Compiling ./test/TestKlaytnGreeter.sol TestKlaytnGreeter 1) testGreetingMessage Events emitted during test: --------------------------- --------------------------- 0 passing (5s) 1 failing 1) TestKlaytnGreeter testGreetingMessage: Error: greeting message should match (Tested: Hello, Klaytn, Against: Hello Klaytn) at result.logs.forEach.log (/Users/jieunkim/.nvm/versions/node/v10.16.0/lib/node_modules/truffle/build/webpack:/packages/core/lib/testing/soliditytest.js:71:1) at Array.forEach (<anonymous>) at processResult (/Users/jieunkim/.nvm/versions/node/v10.16.0/lib/node_modules/truffle/build/webpack:/packages/core/lib/testing/soliditytest.js:69:1) at process._tickCallback (internal/process/next_tick.js:68:7)
Đã xảy ra lỗi. Hãy kiểm tra thông báo lỗi,Error: greeting message should match (Tested: Hello, Klaytn, Against: Hello Klaytn)
. Tôi để ý thấy thiếu dấu ',(comma)'
ở bộ nhớ chuỗi expectedGreet = "Hello Klaytn".
Sửa mã và chạy thử lần nữa.
$ truffle test# OutputUsing network 'development'.Compiling your contracts...===========================> Compiling ./test/TestKlaytnGreeter.sol TestKlaytnGreeter ✓ testGreetingMessage (58ms) 1 passing (5s)
Xin chúc mừng! Sửa mã và chạy thử lần nữa. Bạn đã kiểm thử thành công.
3. Viết kiểm thử bằng JavaScript
Truffle sử dụng công cụ kiểm thử Mocha và thư viện công cụ khẳng định Chai để cung cấp công cụ vững chắc cho kiểm thử bằng JavaScript. Kiểm thử bằng JavaScript linh hoạt hơn và cho phép bạn viết các kiểm thử phức tạp hơn.
Hãy tạo một tập tin và đặt tên là 0_KlaytnGreeting.js
dưới thư mục test
.\
Mã kiểm thử là:
// Interacting directly with KlaytnGreeter contractconst KlaytnGreeter = artifacts.require("./KlaytnGreeter.sol");const truffleAssert = require('truffle-assertions');contract("KlaytnGreeter", async(accounts) => { // store the contract instance at a higher level // to enable access from all functions. var klaytnGreeterInstance; var owner = accounts[0]; var greetMsg = "Hello, Klaytn"; // This will run before each test proceed. before(async function() { // set contract instance into a variable klaytnGreeterInstance = await KlaytnGreeter.new(greetMsg, {from:owner}); }) it("#1 check Greeting message", async function() { // set the expected greeting message var expectedGreeting = greetMsg; var greet= await klaytnGreeterInstance.greet(); assert.equal(expectedGreeting, greet, "greeting message should match"); }) it("#2 update greeting message.", async function() { var newGreeting = "Hi, Klaytn"; await klaytnGreeterInstance.setGreet(newGreeting, { from:owner }); var greet = await klaytnGreeterInstance.greet(); assert.equal(newGreeting, greet, "greeting message should match"); }); it("#3 [Failure test] Only owner can change greeting.", async function() { var fakeOwner = accounts[1]; await truffleAssert.fails(klaytnGreeterInstance.setGreet(greetMsg, { from:fakeOwner })); });});
Nếu bạn không quen thuộc với kiểm thử đơn vị Mocha
, vui lòng tham khảo tài liệu Mocha.
-
Sử dụng
contract()
thay vìdescribe()
Về mặt cấu trúc, mã kiểm thử của Truffle không nên khác biệt nhiều so với mã kiểm thử thông thường của Mocha. Kiểm thử của bạn nên chứa mã giúp Mocha nhận ra đó là kiểm thử tự động. Khác biệt giữa kiểm thử Mocha và kiểm thử Truffle là việc sử dụng hàm contract().
LƯU Ý sử dụng hàm
contract()
và sử dụng mảngtài khoảns
để chỉ định các tài khoản Klaytn đang có sẵn. -
Tóm tắt hợp đồng trong các kiểm thử
Vì Truffle không có phương pháp nào để phát hiện hợp đồng nào bạn cần tương tác trong suốt quá trình kiểm thử, bạn phải nêu chi tiết hợp đ ồng một cách rõ ràng. Một trong các giải pháp là sử dụng phương pháp
artifacts.require()
. -
cú pháp
it
Cú pháp này thể hiện mỗi trường hợp kiểm thử cùng với mô tả. Mô tả sẽ được in ra trên bảng điều khiển khi chạy kiểm thử.
-
thư viện
truffle-assertion
Thư viện này cho phép bạn dễ dàng kiểm thử các trường hợp hoàn ngược hoặc các lỗi khác bằng cách sử dụng hàm
truffleAssert.reverts()
vàtruffleAssert.fails()
.
Kết quả phải như sau:
Using network 'development'.Compiling your contracts...===========================> Everything is up to date, there is nothing to compile. Contract: KlaytnGreeter ✓ #1 check Greeting message ✓ #2 update greeting message. (46ms) ✓ #3 [Failure test] Only owner can change greeting. 3 passing (158ms)
Xin chúc mừng! Bạn đã kiểm thử thành công.
4. Chỉ định kiểm thử
Bạn có thể lựa chọn tập tin kiểm thử cần thực thi.
truffle test ./test/0_KlaytnGreeting.js
Để biết thêm thông tin, vui lòng tham khảo kiểm thử Truffle và các lệnh Truffle.