英文原文地址:https://rahulsethuram.medium.com/the-new-solidity-dev-stack-buidler-ethers-waffle-typescript-tutorial-f07917de48ae
原文发表日期:2020-02-09
【译文省略了很多原作者抒情性文字,直接进入主题。】
本文相关代码:Starter Kit repo
Hardhat (替代Truffle)
Hardhat 称自己为“task runner for Ethereum smart contract developers”。 在实践中,Hardhat 将帮助你使用模板启动 Solidity 项目,并提供测试及部署智能合约所需的所有脚手架。 之前,使用 Truffle 初始化 truffle init
,编译 truffle compile
,测试 truffle test
和部署 truffle migrate
功能来推动Solidity项目。
Hardhat的杀手级功能是堆栈跟踪信息,使得在调试 Solidity 时,可以使用回退(revert)和console.log()。
Ethers.js (替代Web3.js)
Ethers.js 是一个Javascript SDK,用于与以太坊区块链进行交互。 我从使用Solidity开发以来,一直使用Web3.js。 当我第一次尝试Ethers.js时,我对它如此简单及API的出色程度感到震惊。 我推荐曾经使用Web3.js的任何人尝试一下Ethers.js。 它拥有针对钱包,帐户和合约的所有必需功能,并且还具有一些简洁的实用程序,例如ABICoder,HDNode,BigNumber,以及用于十六进制字符串,以太单位转换和以太坊地址的各种格式化实用工具。
Waffle (替代Truffle 测试工具)
Ethereum Waffle 是以太坊智能合约的轻量级测试运行器。 Waffle内置了一些非常不错的测试工具函数,例如用于以太坊地址,哈希和BigNumbers的Chai匹配器,Waffle使用原生Typescript,与Ethers.js配合非常好。
译者注:Chai 是一个断言库,使用链式结构进行断言。
Typescript 无处不在
Typescript 最近很火,这是有原因的。 对我而言,Typescript 的最大的改变是 IDE的集成,它提供所有类属性,对象键,函数参数等的自动补全功能。熟悉Typescript之后,我再也不会回过头来编写原始Javascript了。
上面提到的所有工具都可以与Typescript一起很好地工作,并且一旦完成所有设置,开发的体验很梦幻。
项目启动(Project setup)
现在开始真正有趣的实践! 在一个空文件夹中,运行以下命令初始化一个npm项目:
npm init
初始化过程中,需要多项目有一个简单的设置,因为我们只是演练,可以随意填。
安装 Hardhat:
$ npm install --save-dev hardhat
译者注,如果npm 安装慢,本文的npm 命令都可以用cnpm替换
进行Hardhat项目引导:
$ npx hardhat
选择"Create an empty hardhat.config.js"选项,意思是常见一个新的而不是参考一个样例。
$ npx buidler
888 d8b 888 888
888 Y8P 888 888
888 888 888
88888b. 888 888 888 .d88888 888 .d88b. 888d888
888 "88b 888 888 888 d88" 888 888 d8P Y8b 888P"
888 888 888 888 888 888 888 888 88888888 888
888 d88P Y88b 888 888 Y88b 888 888 Y8b. 888
88888P" "Y88888 888 "Y88888 888 "Y8888 888
👷 Welcome to Buidler v1.3.8 👷
? What do you want to do? …
Create a sample project
❯ Create an empty hardhat.config.js
Quit
创建一些目录来保存项目文件,命令如下:
$ mkdir contracts test scripts
设置 Typescript
安装Typescript需要的依赖项:
$ npm install --save-dev ts-node typescript @types/node @types/mocha
在根目录中创建tsconfig文件:
{
"compilerOptions": {
"target": "es5",
"module": "commonjs",
"strict": true,
"esModuleInterop": true,
"outDir": "dist"
},
"include": ["./scripts", "./test"],
"files": [
"./hardhat.config.ts"
]
}
重命名hardhat配置文件,修改后缀并使其类型安全:
mv hardhat.config.js hardhat.config.ts
// hardhat.config.ts
import { HardhatUserConfig } from "hardhat/types";
const config: HardhatUserConfig = {};
export default config;
创建和编译合约
现在可以开始编写代码:
在 contracts/ 目录创建一个非常简单的 Counter.sol 合约文件(当前使用的最新Solidity 版本是 0.6.8):
pragma solidity ^0.6.8;
import "hardhat/console.sol";
contract Counter {
uint256 count = 0;
event CountedTo(uint256 number);
function getCount() public view returns (uint256) {
return count;
}
function countUp() public returns (uint256) {
console.log("countUp: count =", count);
uint256 newCount = count + 1;
require(newCount > count, "Uint256 overflow");
count = newCount;
emit CountedTo(count);
return count;
}
function countDown() public returns (uint256) {
console.log("countDown: count =", count);
uint256 newCount = count - 1;
require(newCount < count, "Uint256 underflow");
count = newCount;
emit CountedTo(count);
return count;
}
}
在hardhat.config.ts中通过修改solidity.compilers.$.version 来设置Solidity版本:
译者注,此处可能是旧版本才需要修改,新版本config内结构也不尽相同
hardhat 集成了编译任务,因此编译是小菜一碟:
> npx builder compile
Compiling...
Compiled 1 contract successfully
hardhat使用AMAZING 来对Solidity进行版本控制。 切换版本很容易,hardhat会根据需要自动下载并安装Solidity版本,您所需要做的就是在配置中进行更改。 hardhat团队提供了很多选项进行设置!你可以通过将config文件中solidity key赋值为数组(array),而很简单的编译出不同solidity版本的项目。
使用Ethers和Waffle配置测试环境
现在,创建一个测试环境,安装 Ethers, Waffle, 以及 Buidler 插件:
$npm install --save-dev ethers @nomiclabs/buidler-waffle ethereum-waffle
在tsconfig.json添加需要的类型定义:
{
"compilerOptions": {
"target": "es5",
"module": "commonjs",
"strict": true,
"esModuleInterop": true,
"outDir": "dist",
"resolveJsonModule": true
},
"include": [
"./scripts",
"./test"
],
"files": [
"./hardhat.config.ts"
]
}
设置 hardhat.config.ts 以便使用 waffle 插件
import { HardhatUserConfig } from "hardhat/types";
import "@nomiclabs/hardhat-waffle";
const config: HardhatUserConfig = {
solidity: {
compilers: [{ version: "0.6.8", settings: {} }],
},
};
export default config;
设置 TypeChain
TypeChain 是一款非常酷的工具,可为智能合约提供完整的类型接口。 设置完成后,我们可以在Typescript中获得合约函数的类型提示!
可以通过我构建的Typechain插件使用,先安装:
$ npm install --save-dev hardhat-typechain typechain ts-generator @typechain/ethers-v5
Builder配置文件中添加 typechain以配置插件:
import { HardhatUserConfig } from "hardhat/types";
import "@nomiclabs/hardhat-waffle";
import "hardhat-typechain";
const config: BuidlerConfig = {
solidity: {
compilers: [{ version: "0.6.8", settings: {} }],
},
};
export default config;
编译
& npx hardhat compile
现在,进入目录 typechain/
,你可以看到一些生成的文件,其中有一个是 Counter.d.ts
。这是主要的合同文件。
编写和运行合约测试
编写测试大多遵循Waffle语法,但有一个主要区别:ethers.provider
对象是从 hardhat
库而不是 ethereum-waffle
库导入的。
现在让我们编写一个测试。 在 test/ 目录中创建一个名为counter.ts的文件:
import { ethers } from "hardhat";
import chai from "chai";
import { solidity } from "ethereum-waffle";
import { Counter } from "../typechain/Counter";
chai.use(solidity);
const { expect } = chai;
describe("Counter", () => {
let counter: Counter;
beforeEach(async () => {
// 1
const signers = await ethers.getSigners();
// 2
const counterFactory = await ethers.getContractFactory(
"Counter",
signers[0]
);
counter = (await counterFactory.deploy()) as Counter;
await counter.deployed();
const initialCount = await counter.getCount();
// 3
expect(initialCount).to.eq(0);
expect(counter.address).to.properAddress;
});
// 4
describe("count up", async () => {
it("should count up", async () => {
await counter.countUp();
let count = await counter.getCount();
expect(count).to.eq(1);
});
});
describe("count down", async () => {
// 5
it("should fail", async () => {
// this test will fail
await counter.countDown();
});
it("should count down", async () => {
await counter.countUp();
await counter.countDown();
const count = await counter.getCount();
expect(count).to.eq(0);
});
});
});
代码中几处标记处解释如下:
- 从Ethers获取一写预先存款的签名器。
- 使用从 1 获取的签名器部署合约。 导入 Counter 类型,并将其作为 beforeEach 中部署的变量的类型。
- Waffle有一些有用的Chai匹配器可用于编写合约测试,例如BigNumber匹配器和以太坊地址匹配器。 在这里查看所有内容。
- 简单计数测试,确保计数器正常工作。
- 此测试将失败,值得关注,等下会看到 Hardhat 的真正魔力。
让我们运行测试。
首先,我们将Hardhat配置为使用其hardhat
网络,该网络提供了所有Solidity调试魔法:
import { HardhatUserConfig } from "hardhat/types";
import "[@nomiclabs/hardhat-waffle](http://twitter.com/nomiclabs/hardhat-waffle)";
import "hardhat-typechain";
const config: HardhatUserConfig = {
defaultNetwork: "hardhat",
solidity: {
compilers: [{ version: "0.6.8", settings: {} }],
},
};
export default config;
现在运行测试:
$ npx hardhat test
注意在结果中有不寻常的内容:
2 passing (1s)
1 failing
1) Counter
count down
should fail:
Error: VM Exception while processing transaction: revert Uint256 underflow
at Counter.countDown (contracts/Counter.sol:29)
它打印了Solidity 的输出以及堆栈信息,显示了触发回退的行号!!! 😱👻💀
逐行注释合约以查看触发了哪个还原(revert)去猜测变量值的日子已经一去不复返了。
译者注:这里原作者稍微有点夸张,其实现在其他工具链也会给出 revert 原因。
部署合约
经过测试后,开发周期的最后一步是部署合约。
第一步是将网络配置添加到hardhat.config.ts文件。 为此,本案例我们将使用rinkeby,但您可以类似地添加任何网络(如:mainnet),配置很简单:
import { HardhatUserConfig } from "hardhat/types";
import "[@nomiclabs/hardhat-waffle](http://twitter.com/nomiclabs/hardhat-waffle)";
import "hardhat-typechain";
const config: HardhatUserConfig = {
defaultNetwork: "hardhat",
solidity: {
compilers: [{ version: "0.6.8", settings: {} }],
},
networks: {
hardhat: {},
rinkeby: {
url: `[https://rinkeby.infura.io/v3/${INFURA_API_KEY}`],
accounts: [RINKEBY_PRIVATE_KEY],
},
},
};
export default config;
我将Infura用作以太坊节点,不过你可以使用任何远程以太坊节点。 如果你之前没有使用过 Infura,请从Infura获取API密钥。
现在,我们在 scripts 文件夹内创建一个名为deploy.ts的部署脚本:
import { ethers } from "hardhat";
async function main() {
const factory = await ethers.getContract("Counter");
// If we had constructor arguments, they would be passed into deploy()
let contract = await factory.deploy();
// The address the Contract WILL have once mined
console.log(contract.address);
// The transaction that was sent to the network to deploy the Contract
console.log(contract.deployTransaction.hash);
// The contract is NOT deployed yet; we must wait until it is mined
await contract.deployed();
}
main()
.then(() => process.exit(0))
.catch(error => {
console.error(error);
process.exit(1);
});
部署非常简单,现在运行这个脚本,我们在控制台可以看到地址及交易hash.
$ npx hardhat run --network rinkeby scripts/deploy.ts
All contracts have already been compiled, skipping compilation.
0x01FF454Dd078dC7f3cd0905601d093b17E7B9CD7
0x2ae1444920ed76420fb69c9f2fc914c20956efc2ae05c94ab1ea53f224aa0930
我们可以前往 Etherscan 查看交易详情。
这基本就是全部了,本文一步步进行创建项目测试、部署环境,他们都是类型安全的并且使用一些很酷的工具。
封装一下
为了使一切保持干净漂亮,让我们编写一些顺手的NPM脚本。 将以下内容添加到您的package.json中:
"scripts": {
"build": "npm run compile",
"compile": "npx buidler compile",
"test": "npx buidler test"
}
build
用于执行合约编译并生成TypeChain绑定
test
脚本运行合约测试。
福利: 在Etherscan上验证
Hardhat有一个超级方便的插件,可用于在Etherscan上验证合约,此任务其实比看起来要复杂。 Hardhat的工具可以为帮我们处理合约组合,当我们导入了其他合约,例如使用了OpenZeppelin等库的合约会非常方便。
安装插件:
$ npm install --save-dev @nomiclabs/hardhat-etherscan
接下来,我们在hardhat.config.ts中添加所需的配置,前往Etherscan并从您的帐户页面获取API密钥:
import { HardhatUserConfig } from "hardhat/types";
import "[@nomiclabs/hardhat-waffle](http://twitter.com/nomiclabs/hardhat-waffle)";
import "hardhat-typechain";
const config: HardhatUserConfig = {
defaultNetwork: "hardhat",
solidity: {
compilers: [{ version: "0.6.8", settings: {} }],
},
networks: {
hardhat: {},
rinkeby: {
url: `[https://rinkeby.infura.io/v3/${INFURA_API_KEY}`](https://rinkeby.infura.io/v3/$%7BINFURA_API_KEY%7D%60),
accounts: [RINKEBY_PRIVATE_KEY],
},
},
etherscan: {
// Your API key for Etherscan
// Obtain one at [https://etherscan.io/](https://etherscan.io/)
apiKey: ETHERSCAN_API_KEY,
},
};
export default config;
希望我们可以顺手保存上一步的部署地址,因为这样就可以简单地运行插件提供的内置命令来验证合约:
$ npx buidler verify-contract --contract-name Counter --address 0xF0E6Ea29799E85fc1A97B7b78382fd034A6d7864
All contracts have already been compiled, skipping compilation.
Successfully submitted contract at 0xF0E6Ea29799E85fc1A97B7b78382fd034A6d7864 for verification on etherscan. Waiting for verification result...
Successfully verified contract on etherscan
非常简单! 现在,在Etherscan上查看合约地址,可以查看到完整的合约源代码,并在网页上读写合约。