Solidity 智能合约开发 - 玩转 Web3.py

Solidity 智能合约开发 - 玩转 Web3.py #

前言 #

在前文《Solidity 智能合约开发 - 基础》中,我们学习了 Solidity 的基本语法,并且了解了可以通过 BrownieHardHat 等框架进行调试。但在使用这些封装好的框架之前,我们可以通过 Web3.py 直接与我们本地的 Ganache 节点进行交互,以便更好了解其原理,也为我们后续更好使用框架打好基础。

本文以 Web3.py 为例,实现了基础的合约编译、部署至本地 Ganache 网络、与合约交互等功能。

可以点击这里访问本测试 Demo 代码仓库。

Web3.py #

Web3.py 是 Python 的一个开源库,它提供了一个简单的 API,可以让我们通过 Python 程序与以太坊网络进行交互。其 GitHub 地址为 ethereum/web3.py,可以访问其官方文档进行使用。

安装 #

我们可以通过 Python 包管理工具 pip 安装 Web3.py,如下:

pip3 install web3

pip_install_web3

使用 #

使用 import 导入所需方法即可使用

from web3 import Web3

w3 = Web3(Web3.HTTPProvider("HTTP://127.0.0.1:7545"))

Solidity 合约编译 #

合约源码 #

// SPDX-License-Identifier: MIT

pragma solidity ^0.6.0;

contract SimpleStorage {
    uint256 favoriteNumber;
    bool favoriteBool;

    struct People {
        uint256 favoriteNumber;
        string name;
    }

    People public person = People({favoriteNumber: 2, name: "Arthur"});

    People[] public people;

    mapping(string => uint256) public nameToFavoriteNumber;

    function store(uint256 _favoriteNumber) public returns (uint256) {
        favoriteNumber = _favoriteNumber;
        return favoriteNumber;
    }

    function retrieve() public view returns (uint256) {
        return favoriteNumber;
    }

    function addPerson(string memory _name, uint256 _favoriteNumber) public {
        people.push(People({favoriteNumber: _favoriteNumber, name: _name}));
        nameToFavoriteNumber[_name] = _favoriteNumber;
    }
}

这是一个简单的存储合约,通过一个 People 结构体对象来存储人名和他喜欢数字,通过一个数组来存储多个人的信息,并提供了添加、查找方法。

读取合约源文件 #

当我们通过 VSCode 或其他编辑器完成 Solidity 合约编写与语法检查后,需要读取合约源文件并存入变量,供后续编译使用。

import os

with open("./SimpleStorage.sol", "r") as file:
    simple_storage_file = file.read()

上述代码将 SimpleStorage.sol 文件内容读取到变量 simple_storage_file 中。

编译合约 #

安装 solcx #

合约编译需要预先安装 solcx 工具。

pip3 install py-solc-x

pip_install_solcx

导入 solcx #

使用 import 导入所需方法即可使用

from solcx import compile_standard, install_solc

编译 #

install_solc("0.6.0")
compiled_sol = compile_standard(
    {
        "language": "Solidity",
        "sources": {"SimpleStorage.sol": {"content": simple_storage_file}},
        "settings": {
            "outputSelection": {
                "*": {"*": ["abi", "metadata", "evm.bytecode", "evm.sourceMap"]}
            }
        },
    },
    solc_version="0.6.0",
)

上述代码我们安装了 0.6.0 版本的 Solidity 编译程序,使用 solcx 库中的compile_standard 方法对上文读取的合约源文件进行编译,并将编译结果存入变量 compiled_sol 中。

获取编译结果 #

编译成功后,使用以下代码将编译好的合约写入文件

import json

with open("compiled_code.json", "w") as file:
    json.dump(compiled_sol, file)

获取 bytecode 与 abi #

Solidity 合约的部署与交互需要 bytecode 与 abi 两个部分,我们可以通过通过以下代码将其写入对应变量供后续操作使用。

# get bytecode
bytecode = compiled_sol["contracts"]["SimpleStorage.sol"]["SimpleStorage"]["evm"][
    "bytecode"
]["object"]

# get abi
abi = compiled_sol["contracts"]["SimpleStorage.sol"]["SimpleStorage"]["abi"]

本地 Ganache 环境 #

智能合约的调试需要将合约部署到实际的链上,而部署到 Ethereum 主网络或 Rinkeby/Koven 等测试网等也不方便调试,因此,我们需要一个本地的区块链环境,Ganache 就给我们提供了一个这样的本地调试环境。Ganache 主要分为 GUI 和 CLI 两种安装方式。

Ganache GUI #

在自己的本地环境,如 Mac/Windows 等系统,我们可以选择带图形界面的 Ganache 客户端,安装与使用都十分便捷,在 Ganache 官网选择对应版本即可。

ganache_download

安装完成后选择 Quick Start 即可快速启动一条本地运行的区块链网络,并初始化了十个拥有 100 ETH 的账户,开发调试过程中可使用。

ganache_account

Ganache CLI 安装 #

如果您的系统不支持 GUI 安装,我们可以使用 CLI 安装,安装方式如下:

npm install --global yarn
yarn global add ganache-cli

ganache_cli_install

等待其安装完成后即可启动本地测试网络,与 Ganache GUI 一致,也包含初始化账户与余额。

ganache_cli_start

通过 web3 连接本地 Ganache 环境 #

web3 提供了库可以方便地连接到本地 Ganache 环境:

w3 = Web3(Web3.HTTPProvider("HTTP://127.0.0.1:7545"))
chain_id = 5777
my_address = "0x2F490e1eA91DF6d3cC856e7AC391a20b1eceD6A5"
private_key = "0fa88bf96b526a955a6126ae4cca0e72c9c82144ae9af37b497eb6afbe8a9711"

Solidity 合约部署 #

创建合约 #

我们可以通过 web3 库创建合约。

SimpleStorage = w3.eth.contract(abi=abi, bytecode=bytecode)

部署合约 #

部署合约分为三个主要步骤:

  1. 构造交易
  2. 签名交易
  3. 发送交易

构造交易 #

nonce = w3.eth.getTransactionCount(my_address)

transaction = SimpleStorage.constructor().buildTransaction(
    {
        "chainId": chain_id,
        "gasPrice": w3.eth.gas_price,
        "from": my_address,
        "nonce": nonce,
    }
)

签名交易 #

signed_txn = w3.eth.account.sign_transaction(transaction, private_key=private_key)

发送交易 #

tx_hash = w3.eth.send_raw_transaction(signed_txn.rawTransaction)
tx_receipt = w3.eth.wait_for_transaction_receipt(tx_hash)

与合约交互 #

与部署合约步骤类似,我们可以通过 web3 库与合约交互,也分为构造交易、签名交易和发送交易三个步骤。

构造交易 #

simple_storage = w3.eth.contract(address=tx_receipt.contractAddress, abi=abi)

store_transaction = simple_storage.functions.store(67).buildTransaction(
    {
        "chainId": chain_id,
        "gasPrice": w3.eth.gas_price,
        "from": my_address,
        "nonce": nonce + 1,
    }
)

签名交易 #

signed_store_txn = w3.eth.account.sign_transaction(
    store_transaction, private_key=private_key
)

发送交易 #

send_store_tx = w3.eth.send_raw_transaction(signed_store_txn.rawTransaction)
tx_receipt = w3.eth.wait_for_transaction_receipt(send_store_tx)

总结 #

以上就是我们通过 Web3.py 库与本地 Ganache 测试网络进行交互的步骤,在真正的生产项目开发中我们一般不会直接使用 Web3.py 这样的库,而是会使用 Brownie、HardHat 等进一步封装的库,但了解 Web3.py 或 Web3.js 等库的使用方法也非常重要。

参考资料 #

  1. Solidity 智能合约开发 - 基础
  2. ethereum/web3.py
  3. Solidity, Blockchain, and Smart Contract - Beginner to Expert Full Course | Python Edition