区块链技术文章

编写 Solidity 测试脚本

2018-07-04  本文已影响45人  唯夜

编写 Solidity 测试脚本

与 JavaScript 编写的测试脚本一样,基本特性也一直,支持净室环境,可以访问任意不说过的合约。
Truffle的可靠性测试框架是基于以下想法构建的:

例子

让我们看一个例子,在还没有太深入之前。下面是 truffle unbox metacoin 提供的实度测试示例:

import "truffle/Assert.sol";
import "truffle/DeployedAddresses.sol";
import "../contracts/MetaCoin.sol";

contract TestMetacoin {
  function testInitialBalanceUsingDeployedContract() {
    MetaCoin meta = MetaCoin(DeployedAddresses.MetaCoin());

    uint expected = 10000;

    Assert.equal(meta.getBalance(tx.origin), expected, "Owner should have 10000 MetaCoin initially");
  }

  function testInitialBalanceWithNewMetaCoin() {
    MetaCoin meta = new MetaCoin();

    uint expected = 10000;

    Assert.equal(meta.getBalance(tx.origin), expected, "Owner should have 10000 MetaCoin initially");
  }
}

你将会看到如下输出:

$ truffle test
Compiling ConvertLib.sol...
Compiling MetaCoin.sol...
Compiling truffle/Assert.sol
Compiling truffle/DeployedAddresses.sol
Compiling ../test/TestMetacoin.sol...

  TestMetacoin
    ✓ testInitialBalanceUsingDeployedContract (61ms)
    ✓ testInitialBalanceWithNewMetaCoin (69ms)

  2 passing (3s)

测试结构

为了更好的了解发生了什么,我们深入讨论下细节。

ASSERTIONS

你的断言像 Assert.equal() 是由 truffle/Assert.sol 提供给你的。这是默认的断言库,
但是只要库通过触发正确的断言事件与 Truffle 的测试运行器松散集成,就可以包含您自己的断言库。
您可以在 Assert.sol 中找到所有可用的断言函数。

DEPLOYED ADDRESSES 已发布账户

已部署合约的地址(例如,作为迁移的一部分部署的合同)可以通过 truffle/DeployedAddresses.sol 获得。
这是由Truffle提供的,并在运行每个套件之前重新编译和重新链接,
以向您的测试提供Truffle的洁净房间环境。本库以如下形式为你的所有已部署合约提供功能:

DeployedAddresses.<contract name>();

这将返回一个地址,可以用来访问合约。请参阅上面的示例测试。

为了使用已部署的契约,您必须将契约代码导入到您的测试套件中。注意 import"../contracts/ MetaCoin.sol”;
在这个例子中。这个导入是相对于存在于./test目录中的测试契约的,为了找到MetaCoin契约,它会在测试目录之外。
然后它使用该契约将地址转换为MetaCoin类型。

测试合约名

所有合约的测试必须以 test 开头,使用大写的 T 。这样的命名方式就将测试合约和普通合约区分开来了,
让测试运行器知道那个合约代表测试套件。

测试方法名

跟测试合约类似,所有的测试方法,都必须以小写单词 test 开头。每个测试方法都会被当作一个独立的交易,
根据在测试文件中定义的顺序执行。truffle/Assert.sol 提供的断言函数,测试运行程序评估触发事件以确定测试的结果。
断言函数返回一个布尔值,表示断言成功或失败,你可以使用它从测试早期返回,以防止执行错误(例如,Ganache或Truffle开发的错误将暴露)。

hooks 使用的前后

在如下例子中为你提供了许多测试 hooks 。这些 hooks 是 beforeAll 、 beforeEach 、 afterEach ,
它们是 Mocha 在 Javascript 测试中提供的相同的 hooks 。您可以使用这些 hooks 在每个测试之前和之后,
或者在每个套件运行之前和之后执行安装和拆卸操作。和测试方法一样,每个测试 hooks 都会是一个独立的交易。
您可以通过创建许多带有不同后缀的 hooks 来绕过这个限制,如下面的示例所示:

import "truffle/Assert.sol";

contract TestHooks {
  uint someValue;

  function beforeEach() {
    someValue = 5;
  }

  function beforeEachAgain() {
    someValue += 1;
  }

  function testSomeValueIsSix() {
    uint expected = 6;

    Assert.equal(someValue, expected, "someValue should have been 6");
  }
}

这个测试合约还表明,您的测试函数和hook函数都共享相同的合约状态。您可以在测试之前设置合约数据,
在测试期间使用该数据,并在测试之后重置它,为下一个测试做准备。注意,就像Javascript测试一样,
下一个测试函数将从运行的前一个测试函数的状态继续。

高级特性

Solidity 有一些新特新可以让你使用特殊的用力来测试。

异常测试

你可以很容易地测试你的合同应不应该抛出一个异常。( 例如:require()/ assert()/revert() 语句;之前的版本的 throw)。
这个话题是特约作家西蒙·德·拉·鲁维埃(Simon de la Rouviere)在他的 Truffle Solidity 测试教程中首次提出的。
本教程不推荐的关键字 throw 大量使用异常。从Solidity v0.4.13开始取而代之的是 revert()、require()和assert()。

测试以太币交易

你也可以测试你的合约是如何接收以太币的,并且用 JavaScriptSolidity 交互。为了达到这目的,你需要在
Solidity 测试脚本中写一个名为 initialBalance 的方法,并且返回 uint
这可以直接写为函数或公共变量,如下所示。当你的测试合约被部署到网络上时,Truffle将从你的测试账户发送到你的测试合约。
你的测试合约中就可以使用这个以太币来进行测试。注意,initialBalance是可选的,不是必需的。

import "truffle/Assert.sol";
import "truffle/DeployedAddresses.sol";
import "../contracts/MyContract.sol";

contract TestContract {
  // Truffle will send the TestContract one Ether after deploying the contract.
  uint public initialBalance = 1 ether;

  function testInitialBalanceUsingDeployedContract() {
    MyContract myContract = MyContract(DeployedAddresses.MyContract());

    // perform an action which sends value to myContract, then assert.
    myContract.send(...);
  }

  function () {
    // This will NOT be executed when Ether is sent. \o/
  }
}

请注意,Truffle以不执行回退函数的方式发送到测试合约,因此您仍然可以在固态测试中对高级测试用例使用回退函数。

上一篇 下一篇

猜你喜欢

热点阅读