如何从 Go 调用以太坊合约?

Posted

技术标签:

【中文标题】如何从 Go 调用以太坊合约?【英文标题】:How to call an Ethereum contract from Go? 【发布时间】:2017-07-20 02:42:15 【问题描述】:

我尝试关注go-ethereum guide for this,但它却是seems out of date。我尝试在 Go 中四处寻找 web3 库,但一切都试图指向“使用 geth”。

从 Go 中,如何调用以太坊区块链中的合约?如果我以 testnet 或 testrpc 为目标,会有什么不同?

【问题讨论】:

您可能会发现这很有帮助 ethereum.stackexchange.com/questions/3514/… 。取决于你想用 Go 和 Ethereum 做什么样的事情。 【参考方案1】:

下面是一个简单示例,说明如何在 Go 中与智能合约进行交互。假设您安装了 solcabigen

solc --abi Store.sol > Store.abi
solc --bin Store.sol > Store.bin
abigen --bin=Store.bin --abi=Store.abi --pkg=store --out=Store.go

Store.sol

pragma solidity ^0.4.24;

contract Store 
  event ItemSet(bytes32 key, bytes32 value);

  string public version;
  mapping (bytes32 => bytes32) public items;

  constructor(string _version) public 
    version = _version;
  

  function setItem(bytes32 key, bytes32 value) external 
    items[key] = value;
    emit ItemSet(key, value);
  

main.go

package main

import (
    "context"
    "crypto/ecdsa"
    "fmt"
    "log"
    "math/big"

    "github.com/ethereum/go-ethereum/accounts/abi/bind"
    "github.com/ethereum/go-ethereum/common"
    "github.com/ethereum/go-ethereum/crypto"
    "github.com/ethereum/go-ethereum/ethclient"

    store "./contracts" // for demo
)

func main() 
    client, err := ethclient.Dial("https://rinkeby.infura.io")
    if err != nil 
        log.Fatal(err)
    

    privateKey, err := crypto.HexToECDSA("fad9c8855b740a0b7ed4c221dbad0f33a83a49cad6b3fe8d5817ac83d38b6a19")
    if err != nil 
        log.Fatal(err)
    

    publicKey := privateKey.Public()
    publicKeyECDSA, ok := publicKey.(*ecdsa.PublicKey)
    if !ok 
        log.Fatal("error casting public key to ECDSA")
    

    fromAddress := crypto.PubkeyToAddress(*publicKeyECDSA)
    nonce, err := client.PendingNonceAt(context.Background(), fromAddress)
    if err != nil 
        log.Fatal(err)
    

    gasPrice, err := client.SuggestGasPrice(context.Background())
    if err != nil 
        log.Fatal(err)
    

    auth := bind.NewKeyedTransactor(privateKey)
    auth.Nonce = big.NewInt(int64(nonce))
    auth.Value = big.NewInt(0)     // in wei
    auth.GasLimit = uint64(300000) // in units
    auth.GasPrice = gasPrice

    address := common.HexToAddress("0x147B8eb97fD247D06C4006D269c90C1908Fb5D54")
    instance, err := store.NewStore(address, client)
    if err != nil 
        log.Fatal(err)
    

    key := [32]byte
    value := [32]byte
    copy(key[:], []byte("foo"))
    copy(value[:], []byte("bar"))

    tx, err := instance.SetItem(auth, key, value)
    if err != nil 
        log.Fatal(err)
    

    fmt.Printf("tx sent: %s", tx.Hash().Hex()) // tx sent: 0x8d490e535678e9a24360e955d75b27ad307bdfb97a1dca51d0f3035dcee3e870

    result, err := instance.Items(&bind.CallOpts, key)
    if err != nil 
        log.Fatal(err)
    

    fmt.Println(string(result[:])) // "bar"

如需更多示例,请查看Ethereum Development with Go book。

【讨论】:

是否可以在不构建 ABI 生成步骤的情况下访问智能合约。比如我们如何在 web3.js 或 web3.py 中访问合约实例? @SushantKumar 是的,您不必使用 abi 帮助程序包,您可以发送原始事务来读取或写入数据,但您需要手动构造编码参数

以上是关于如何从 Go 调用以太坊合约?的主要内容,如果未能解决你的问题,请参考以下文章

以太坊智能合约虚拟机(EVM)原理与实现

用Go来做以太坊开发⑤事件日志

创建以太坊合约(去以太坊)

rpc接口调用以太坊智能合约

下面的以太坊合约函数调用数据是如何构造的?

以太坊的ABI编码