概述: 用Go部署智能合约的教程。

部署智能合约

如果你还没看之前的章节,请先学习编译智能合约的章节因为这节内容,需要先了解如何将智能合约编译为Go文件。

假设你已经导入从abigen生成的新创建的Go包文件,并设置ethclient,加载您的私钥,下一步是创建一个有配置密匙的交易发送器(tansactor)。 首先从go-ethereum导入accounts/abi/bind包,然后调用传入私钥的NewKeyedTransactor。 然后设置通常的属性,如nonce,燃气价格,燃气上线限制和ETH值。

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

如果你还记得上个章节的内容, 我们创建了一个非常简单的“Store”合约,用于设置和存储键/值对。 生成的Go合约文件提供了部署方法。 部署方法名称始终以单词Deploy开头,后跟合约名称,在本例中为Store

deploy函数接受有密匙的事务处理器,ethclient,以及智能合约构造函数可能接受的任何输入参数。我们测试的智能合约接受一个版本号的字符串参数。 此函数将返回新部署的合约地址,事务对象,我们可以交互的合约实例,还有错误(如果有)。

  1. input := "1.0"
  2. address, tx, instance, err := store.DeployStore(auth, client, input)
  3. if err != nil {
  4. log.Fatal(err)
  5. }
  6. fmt.Println(address.Hex()) // 0x147B8eb97fD247D06C4006D269c90C1908Fb5D54
  7. fmt.Println(tx.Hash().Hex()) // 0xdae8ba5444eefdc99f4d45cd0c4f24056cba6a02cefbf78066ef9f4188ff7dc0
  8. _ = instance // will be using the instance in the 下个章节

就这么简单:)你可以用事务哈希来在Etherscan上查询合约的部署状态: https://rinkeby.etherscan.io/tx/0xdae8ba5444eefdc99f4d45cd0c4f24056cba6a02cefbf78066ef9f4188ff7dc0

完整代码

Commands

  1. solc --abi Store.sol
  2. solc --bin Store.sol
  3. abigen --bin=Store_sol_Store.bin --abi=Store_sol_Store.abi --pkg=store --out=Store.go

Store.sol

  1. pragma solidity ^0.4.24;
  2. contract Store {
  3. event ItemSet(bytes32 key, bytes32 value);
  4. string public version;
  5. mapping (bytes32 => bytes32) public items;
  6. constructor(string _version) public {
  7. version = _version;
  8. }
  9. function setItem(bytes32 key, bytes32 value) external {
  10. items[key] = value;
  11. emit ItemSet(key, value);
  12. }
  13. }

contract_deploy.go

  1. package main
  2. import (
  3. "context"
  4. "crypto/ecdsa"
  5. "fmt"
  6. "log"
  7. "math/big"
  8. "github.com/ethereum/go-ethereum/accounts/abi/bind"
  9. "github.com/ethereum/go-ethereum/crypto"
  10. "github.com/ethereum/go-ethereum/ethclient"
  11. store "./contracts" // for demo
  12. )
  13. func main() {
  14. client, err := ethclient.Dial("https://rinkeby.infura.io")
  15. if err != nil {
  16. log.Fatal(err)
  17. }
  18. privateKey, err := crypto.HexToECDSA("fad9c8855b740a0b7ed4c221dbad0f33a83a49cad6b3fe8d5817ac83d38b6a19")
  19. if err != nil {
  20. log.Fatal(err)
  21. }
  22. publicKey := privateKey.Public()
  23. publicKeyECDSA, ok := publicKey.(*ecdsa.PublicKey)
  24. if !ok {
  25. log.Fatal("cannot assert type: publicKey is not of type *ecdsa.PublicKey")
  26. }
  27. fromAddress := crypto.PubkeyToAddress(*publicKeyECDSA)
  28. nonce, err := client.PendingNonceAt(context.Background(), fromAddress)
  29. if err != nil {
  30. log.Fatal(err)
  31. }
  32. gasPrice, err := client.SuggestGasPrice(context.Background())
  33. if err != nil {
  34. log.Fatal(err)
  35. }
  36. auth := bind.NewKeyedTransactor(privateKey)
  37. auth.Nonce = big.NewInt(int64(nonce))
  38. auth.Value = big.NewInt(0) // in wei
  39. auth.GasLimit = uint64(300000) // in units
  40. auth.GasPrice = gasPrice
  41. input := "1.0"
  42. address, tx, instance, err := store.DeployStore(auth, client, input)
  43. if err != nil {
  44. log.Fatal(err)
  45. }
  46. fmt.Println(address.Hex()) // 0x147B8eb97fD247D06C4006D269c90C1908Fb5D54
  47. fmt.Println(tx.Hash().Hex()) // 0xdae8ba5444eefdc99f4d45cd0c4f24056cba6a02cefbf78066ef9f4188ff7dc0
  48. _ = instance
  49. }

solc version used for these examples

  1. $ solc --version
  2. 0.4.24+commit.e67f0147.Emscripten.clang