在以太坊及众多兼容链上,智能合约是构建去中心化应用(DApps)的核心,ERC(Ethereum Request for Comments)标准为这些合约提供了统一的接口和行为规范,确保了代币的互操作性和安全性,ERC777作为对ERC20标准的增强,引入了许多先进特性,如可扩展性、内置发送钩子和操作员模式等,使其在某些场景下更具优势,本文将详细介绍如何在以太坊网络上部署一个ERC777标准的智能合约。

ERC777标准简介:为何选择它

在深入部署之前,我们先简要了解ERC777相较于ERC20的优势:

  1. 可扩展性与可替代性:ERC777代币可以是可替代的(如ERC20),也可以是不可替代的(如ERC721),或者两者的混合,这为更复杂的代币设计提供了可能。
  2. 内置发送钩子(Send Hooks):当代币被发送(包括transfersend)时,合约可以触发特定的回调函数(tokensToSend),这使得接收方合约能够自动处理收到的代币,无需额外的交易步骤,极大地提升了用户体验和合约间的交互效率。
  3. 操作员(Operators)模式:代币持有者可以批准其他地址(操作员)来管理他们的代币,而无需每次操作都进行授权,操作员可以代表持有者发送或接收代币,这在需要批量管理或授权第三方服务(如交易所、钱包)的场景下非常有用。
  4. 改进的事件:ERC777提供了更丰富的事件,如SentMintedBurned等,使得代币流转的追踪和监听更加方便。
  5. 向后兼容ERC20:ERC777合约实现了ERC20的所有接口,因此ERC20的应用程序也可以与ERC777代币交互,但反之则不然,可以通过“ERC777代币作为ERC20代理”的方式实现部分兼容。

部署ERC777智能合约的前期准备

在开始部署之前,你需要准备以下几样东西:

  1. 一个以太坊钱包:如MetaMask,并确保钱包中有足够的ETH用于支付 gas 费。
  2. 测试网ETH(可选但推荐):如果你想在以太坊测试网(如Goerli、Sepolia)上进行测试和调试,需要从对应的 Faucet 获取测试ETH。
  3. 开发环境
    • Node.js 和 npm/yarn:用于运行JavaScript/TypeScript代码。
    • Truffle 或 Hardhat:流行的以太坊开发框架,用于编译、部署和测试智能合约。
    • Solidity 编译器:确保安装了与合约代码兼容的Solidity版本。
  4. ERC777合约代码

    你可以编写自己的ERC777合约,但更常见的是使用经过审计的标准实现,OpenZeppelin提供了广泛使用的、安全的ERC777合约实现。

使用OpenZeppelin库部署ERC777合约

OpenZeppelin是一个提供安全、社区审计的智能合约标准库的开源项目,使用其ERC777实现可以大大降低开发风险。

步骤1:初始化项目

如果你还没有项目,可以创建一个新的目录并初始化:

mkdir my-erc777-project
cd my-erc777-project
npm init -y

步骤2:安装依赖

安装OpenZeppelin合约库和开发框架(以Hardhat为例):

npm install @openzeppelin/contracts hardhat

然后初始化Hardhat项目:

npx hardhat init

按照提示选择配置(Create a basic sample project)。

步骤3:编写ERC777合约

contracts目录下,创建一个新的合约文件,例如MyERC777Token.sol

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/token/ERC777/ERC777.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
/**MyERC777Token
 * @dev 一个简单的ERC777代币合约,继承自OpenZeppelin的ERC777和Ownable。
 */
contract MyERC777Token is ERC777, Ownable {
    constructor(
        string memory name,
        string memory symbol,
        address[] memory defaultOperators
    ) ERC777(name, symbol, defaultOperators) {
        // 通常在构造函数中铸造初始代币给自己或指定地址
        // _mint(address to, uint256 amount, bytes memory userData, bytes memory operatorData)
        _mint(msg.sender, 1000000 * 10**decimals(), "Initial mint", "0x");
    }
    // 你可以在这里添加自定义的逻辑,例如额外的铸造、销毁函数等
    // 但要注意,ERC777本身没有标准的mint/burn函数,需要由合约实现者提供
    // 或者
随机配图
使用ERC777PresetMinterPauser(如果需要预铸造者和暂停功能) }

注意:上面的例子是一个基础的ERC777合约,OpenZeppelin还提供了ERC777PresetMinterPauser预设合约,它已经包含了铸造者角色和暂停功能,更适合直接使用或在此基础上修改,你可以参考OpenZeppelin文档。

步骤4:配置部署脚本

scripts目录下,创建一个部署脚本,例如deploy.js

async function main() {
  // 获取合约工厂
  const MyERC777Token = await ethers.getContractFactory("MyERC777Token");
  // 合约构造参数
  const name = "My Awesome ERC777 Token";
  const symbol = "MAET7";
  const defaultOperators = []; // 可以传入默认操作员地址数组,或为空
  // 部署合约
  const myERC777Token = await MyERC777Token.deploy(name, symbol, defaultOperators);
  await myERC777Token.deployed();
  console.log("MyERC777Token deployed to:", myERC777Token.address);
}
main()
  .then(() => process.exit(0))
  .catch((error) => {
    console.error(error);
    process.exit(1);
  });

步骤5:配置网络(如果部署到测试网或主网)

hardhat.config.js中,你需要配置你要部署的网络信息,包括RPC URL和私钥(或使用HD Wallet Provider,如MetaMask)。

添加Goerli测试网配置:

require("@nomicfoundation/hardhat-toolbox");
require('dotenv').config(); // 使用dotenv管理环境变量
const GOERLI_RPC_URL = process.env.GOERLI_RPC_URL;
const PRIVATE_KEY = process.env.PRIVATE_KEY;
/** @type import('hardhat/config').HardhatUserConfig */
module.exports = {
  solidity: "0.8.17",
  networks: {
    goerli: {
      url: GOERLI_RPC_URL,
      accounts: [PRIVATE_KEY],
      chainId: 5,
    },
    // 可以添加其他网络,如主网、Sepolia等
  },
  etherscan: {
    apiKey: process.env.ETHERSCAN_API_KEY, // 用于验证合约
  },
};

创建.env文件存储敏感信息:

GOERLI_RPC_URL=https://goerli.infura.io/v3/YOUR_INFURA_PROJECT_ID
PRIVATE_KEY=你的钱包私钥(不要泄露!)
ETHERSCAN_API_KEY=你的Etherscan API Key

步骤6:编译和部署合约

在终端中运行以下命令编译合约:

npx hardhat compile

然后部署合约(以Goerli测试网为例):

npx hardhat run scripts/deploy.js --network goerli

如果部署成功,你会在终端看到合约的地址,类似:MyERC777Token deployed to: 0x1234567890123456789012345678901234567890

验证合约(可选但推荐)

为了增加合约的可信度和透明度,你可以将合约代码验证到以太坊浏览器(如Etherscan)上。

  1. 确保你在.env文件中配置了ETHERSCAN_API_KEY
  2. 在部署成功后,Hardhat会提示你如何验证,或者你可以手动运行:
npx hardhat verify --network goerli "0x你的合约地址" "参数1" "参数2" "参数3"

对于MyERC777Token,参数就是部署时的name, symbol, defaultOperators

与部署的ERC777合约交互

部署完成后,你可以:

  1. 在以太坊浏览器中查看合约:输入合约地址,可以看到合约代码、事件、代币余额等。
  2. 使用钱包测试:将合约添加到MetaMask(通过添加自定义代币