網(wǎng)站色哦優(yōu)化8888電商如何從零做起
如果不是只讀取數(shù)據(jù)的合約函數(shù),需要異步的執(zhí)行,因此并不能直接獲取到合約函數(shù)的返回值,需要等到交易執(zhí)行完畢,得到確認(rèn)后才能獲取到合約函數(shù)的返回值。而且合約函數(shù)返回值一般是通過(guò)事件日志獲取到的。
這里給出一個(gè)例子來(lái)展示我是如何獲取合約函數(shù)返回值的。
我使用的以太坊版本為:github.com/ethereum/go-ethereum v1.13.0
solidity合約:
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;contract StoreString {event ItemSetStr(bytes32 indexed hash,string key,string value);mapping (string => string) public itemstr;function setItemstr(string memory _key, string memory _value) external returns(bytes32){itemstr[_key] = _value;bytes32 hash=sha256(abi.encodePacked(_key,_value));emit ItemSetStr(hash,_key,_value);return hash;}
}
這是一個(gè)存儲(chǔ)鍵值對(duì)的合約,利用函數(shù)setItemstr
向map中存儲(chǔ)鍵值對(duì),并通過(guò)key,value計(jì)算哈希值,要求返回哈希值。通過(guò)事件ItemSetStr
將哈希值(返回值)、key、value記錄下來(lái)。
將合約的abi編碼通過(guò)abigen工具生成go代碼后。
以下是測(cè)試獲取合約函數(shù)返回值的代碼:
func TestSetStoreString(t *testing.T) {url="" //節(jié)點(diǎn)鏈接client, err := ethclient.Dial(url)if err != nil {log.Fatal(err)}//合約地址contractAddress := common.HexToAddress("0xd9Ed5E352F84E182eB499ae1b5F9935C06d78Ddb")privateKeyStr := "" //私鑰字符串auth := InitAuth(privateKeyStr, client, nil)//得到合約實(shí)例storeString, err := NewStoreString(contractAddress, client)if err != nil {log.Fatal("NewStoreString: ", err)}tx, err := storeString.SetItemstr(auth, "key10", "9900")if err != nil {log.Fatal("SetItemstr: ", err)}//等待交易確認(rèn),獲取到交易的收據(jù)receipt, err := bind.WaitMined(context.Background(), client, tx)if err != nil {log.Fatal("WaitMined: ", err)}//如果交易成功if receipt.Status == 1 {//解析該交易收據(jù)里包含的日志eventItemSetStr, err := storeString.ParseItemSetStr(*receipt.Logs[0])if err != nil {log.Fatal("ParseItemSetStr: ", err)}fmt.Println("交易成功!")fmt.Println("txHash:", tx.Hash().Hex())fmt.Println("返回值:", eventItemSetStr.Hash)fmt.Println("日志Key:", eventItemSetStr.Key)fmt.Println("日志Value:", eventItemSetStr.Value)} else {t.Error("交易失敗")}}func InitAuth(privateKeyStr string, client *ethclient.Client, value *big.Int) *bind.TransactOpts {privateKeyECDSA, err := crypto.HexToECDSA(privateKeyStr)if err != nil {log.Fatal("crypto.HexToECDSA: ", err)}chainID, err := client.ChainID(context.Background())if err != nil {log.Fatal("ChainID: ", err)}auth, err := bind.NewKeyedTransactorWithChainID(privateKeyECDSA, chainID)if err != nil {log.Fatal("NewKeyedTransactorWithChainID: ", err)}//設(shè)置交易參數(shù),例如gas費(fèi)等,這里全部由區(qū)塊鏈系統(tǒng)評(píng)估決定auth.GasFeeCap = big.NewInt(1).Mul(big.NewInt(10), big.NewInt(1000000000)) // maxFee Per Gas:100gwei//獲取平均小費(fèi)gasTipCap, err := client.SuggestGasTipCap(context.Background())if err != nil {log.Fatal("SuggestGasTipCap: ", err)}auth.GasTipCap = gasTipCap //priorityfee Per Gas:auth.GasLimit = uint64(3000000)//還需要設(shè)置鎖定金額auth.Value = valuereturn auth
}
返回值并不能立馬獲得,需要等待交易成功上鏈。再?gòu)氖录罩局薪馕龀龇祷刂?。因此這個(gè)過(guò)程中用到了兩個(gè)重要代碼:
bind包下的WaitMined
:等待tx交易的確認(rèn),并獲取其收據(jù),該收據(jù)中必然包含我們想要的日志
func WaitMined(ctx context.Context, b DeployBackend, tx *types.Transaction) (*types.Receipt, error)
合約代碼自動(dòng)生成的go文件中的ParseItemSetStr
,傳入日志即可解析出信息。
func (_StoreString *StoreStringFilterer) ParseItemSetStr(log types.Log) (*StoreStringItemSetStr, error)