智能合約調(diào)用是實現(xiàn)一個 DApp 的關(guān)鍵,一個完整的 DApp 包括前端、后端、智能合約及區(qū)塊 鏈系統(tǒng),智能合約的調(diào)用是連接區(qū)塊鏈與前后端的關(guān)鍵。
從事鄭州服務(wù)器托管,服務(wù)器租用,云主機,網(wǎng)絡(luò)空間,空間域名,CDN,網(wǎng)絡(luò)代維等服務(wù)。
我們先來了解一下智能合約調(diào)用的基礎(chǔ)原理。智能合約運行在以太坊節(jié)點的 EVM 中。因此要 想調(diào)用合約必須要訪問某個節(jié)點。
以后端程序為例,后端服務(wù)若想連接節(jié)點有兩種可能,一種是雙 方在同一主機,此時后端連接節(jié)點可以采用 本地 IPC(Inter-Process Communication,進 程間通信)機制,也可以采用 RPC(Remote Procedure Call,遠程過程調(diào)用)機制;另 一種情況是雙方不在同一臺主機,此時只能采用 RPC 機制進行通信。
提到 RPC, 讀者應(yīng)該對 Geth 啟動參數(shù)有點印象,Geth 啟動時可以選擇開啟 RPC 服務(wù),對應(yīng)的 默認服務(wù)端口是 8545。。
接著,我們來了解一下智能合約運行的過程。
智能合約的運行過程是后端服務(wù)連接某節(jié)點,將 智能合約的調(diào)用(交易)發(fā)送給節(jié)點,節(jié)點在驗證了交易的合法性后進行全網(wǎng)廣播,被礦工打包到 區(qū)塊中代表此交易得到確認,至此交易才算完成。
就像數(shù)據(jù)庫一樣,每個區(qū)塊鏈平臺都會提供主流 開發(fā)語言的 SDK(Software Development Kit,軟件開發(fā)工具包),由于 Geth 本身就是用 Go 語言 編寫的,因此若想使用 Go 語言連接節(jié)點、發(fā)交易,直接在工程內(nèi)導(dǎo)入 go-ethereum(Geth 源碼) 包就可以了,剩下的問題就是流程和 API 的事情了。
總結(jié)一下,智能合約被調(diào)用的兩個關(guān)鍵點是節(jié)點和 SDK。
由于 IPC 要求后端與節(jié)點必須在同一主機,所以很多時候開發(fā)者都會采用 RPC 模式。除了 RPC,以太坊也為開發(fā)者提供了 json- rpc 接口,本文就不展開討論了。
接下來介紹如何使用 Go 語言,借助 go-ethereum 源碼庫來實現(xiàn)智能合約的調(diào)用。這是有固定 步驟的,我們先來說一下總體步驟,以下面的合約為例。
步驟 01:編譯合約,獲取合約 ABI(Application Binary Interface,應(yīng)用二進制接口)。 單擊【ABI】按鈕拷貝合約 ABI 信息,將其粘貼到文件 calldemo.abi 中(可使用 Go 語言IDE 創(chuàng)建該文件,文件名可自定義,后綴最好使用 abi)。
最好能將 calldemo.abi 單獨保存在一個目錄下,輸入“l(fā)s”命令只能看到 calldemo.abi 文件,參 考效果如下:
步驟 02:獲得合約地址。注意要將合約部署到 Geth 節(jié)點。因此 Environment 選擇為 Web3 Provider。
在【Environment】選項框中選擇“Web3 Provider”,然后單擊【Deploy】按鈕。
部署后,獲得合約地址為:0xa09209c28AEf59a4653b905792a9a910E78E7407。
步驟 03:利用 abigen 工具(Geth 工具包內(nèi)的可執(zhí)行程序)編譯智能合約為 Go 代碼。abigen 工具的作用是將 abi 文件轉(zhuǎn)換為 Go 代碼,命令如下:
其中各參數(shù)的含義如下。 (1)abi:是指定傳入的 abi 文件。 (2)type:是指定輸出文件中的基本結(jié)構(gòu)類型。 (3)pkg:指定輸出文件 package 名稱。 (4)out:指定輸出文件名。 執(zhí)行后,將在代碼目錄下看到 funcdemo.go 文件,讀者可以打開該文件欣賞一下,注意不要修改它。
步驟 04:創(chuàng)建 main.go,填入如下代碼。 注意代碼中 HexToAddress 函數(shù)內(nèi)要傳入該合約部署后的地址,此地址在步驟 01 中獲得。
步驟 04:設(shè)置 go mod,以便工程自動識別。
前面有所提及,若要使用 Go 語言調(diào)用智能合約,需要下載 go-ethereum 工程,可以使用下面 的指令:
該指令會自動將 go-ethereum 下載到“$GOPATH/src/github.com/ethereum/go-ethereum”,這樣還算 不錯。不過,Go 語言自 1.11 版本后,增加了 module 管理工程的模式。只要設(shè)置好了 go mod,下載 依賴工程的事情就不必關(guān)心了。
接下來設(shè)置 module 生效和 GOPROXY,命令如下:
在項目工程內(nèi),執(zhí)行初始化,calldemo 可以自定義名稱。
步驟 05:運行代碼。執(zhí)行代碼,將看到下面的效果,以及最終輸出的 2020。
上述輸出信息中,可以看到 Go 語言會自動下載依賴文件,這就是 go mod 的神奇之處??吹?2020,相信讀者也知道運行結(jié)果是正確的了。
開發(fā)框架。level1編程項目這是一個基于go語言編寫的,自動化測試以太坊智能合約的開發(fā)框架,使用此框架,可以自動化的部署合約,自動測試合約內(nèi)的功能函數(shù)。
女媧版平臺使用Go語言開發(fā),包括主鏈節(jié)點軟件包(VNODE)和子鏈節(jié)點軟件包(SCS)兩部分。本次發(fā)布的主要更新有:1. VNODE節(jié)點可以提供SCS節(jié)點接入服務(wù);2. VNODE節(jié)點可以在提供SCS節(jié)點接入服務(wù)時,獲得子鏈挖礦獎勵;3.?提供對IPFS的子鏈支持;?4.?提供對無幣區(qū)塊鏈的子鏈支持;5.?發(fā)布接入墨客moac主網(wǎng)的SCS節(jié)點軟件;6.?發(fā)布子鏈(MicroChain)協(xié)議智能合約;7.?發(fā)布基于POS共識的子鏈合約ProcWind;
怎么設(shè)置區(qū)塊鏈的名字,分2步驟,第一是打開瀏覽器輸入?yún)^(qū)塊鏈網(wǎng),第二是再輸入你需要設(shè)置的名字。
鏈代碼(Chaincode)又稱為智能合約,是用Go語言編寫的程序,主要用于操作賬本上的數(shù)據(jù)。鏈代碼是運行在區(qū)塊鏈上的、特定條件下自動執(zhí)行的代碼邏輯,是用戶利用區(qū)塊鏈實現(xiàn)業(yè)務(wù)邏輯的重要途徑?;趨^(qū)塊鏈特點,智能合約的運行結(jié)果是可信的,其結(jié)果是無法被偽造和篡改的。在使用區(qū)塊鏈服務(wù)時,用戶需要開發(fā)自己的鏈代碼和應(yīng)用程序。用戶的應(yīng)用程序通過區(qū)塊
區(qū)塊鏈服務(wù)狀態(tài)為“彈性IP異?!?。排查項:彈性公網(wǎng)IP已“解綁”或被釋放。在BCS控制臺,服務(wù)管理頁面中的目標服務(wù)卡片中,單擊“更多 更新訪問地址”查看彈性公網(wǎng)IP。登錄網(wǎng)絡(luò)控制臺,查找目標彈性公網(wǎng)IP并查看其狀態(tài)。若彈性公網(wǎng)IP已“解綁”:登錄網(wǎng)絡(luò)控制臺,在目標彈性公網(wǎng)IP的操作列單擊“綁定”,完成后刷新BCS控制臺的服務(wù)管理頁面。
智能合約具有在執(zhí)行期間“發(fā)出”事件的能力。 事件在以太坊中也稱為“日志”。 事件的輸出存儲在日志部分下的事務(wù)處理中。 事件已經(jīng)在以太坊智能合約中被廣泛使用,以便在發(fā)生相對重要的動作時記錄,特別是在代幣合約(即ERC-20)中,以指示代幣轉(zhuǎn)賬已經(jīng)發(fā)生。 這些部分將引導(dǎo)您完成從區(qū)塊鏈中讀取事件以及訂閱事件的過程,以便交易事務(wù)被礦工打包入塊的時候及時收到通知。
為了訂閱事件日志,我們需要做的第一件事就是撥打啟用websocket的以太坊客戶端。 幸運的是,Infura支持websockets。
下一步是創(chuàng)建篩選查詢。 在這個例子中,我們將閱讀來自我們在之前課程中創(chuàng)建的示例合約中的所有事件。
我們接收事件的方式是通過Go channel。 讓我們從go-ethereum core/types 包創(chuàng)建一個類型為 Log 的channel。
現(xiàn)在我們所要做的就是通過從客戶端調(diào)用 SubscribeFilterLogs 來訂閱,它接收查詢選項和輸出通道。 這將返回包含unsubscribe和error方法的訂閱結(jié)構(gòu)。
最后,我們要做的就是使用select語句設(shè)置一個連續(xù)循環(huán)來讀入新的日志事件或訂閱錯誤。
我們會在下個章節(jié)介紹如何解析日志。
Commands
Store.sol
event_subscribe.go
智能合約可以可選地釋放“事件”,其作為交易收據(jù)的一部分存儲日志。讀取這些事件相當簡單。首先我們需要構(gòu)造一個過濾查詢。我們從go-ethereum包中導(dǎo)入 FilterQuery 結(jié)構(gòu)體并用過濾選項初始化它。我們告訴它我們想過濾的區(qū)塊范圍并指定從中讀取此日志的合約地址。在示例中,我們將從在 智能合約章節(jié) 創(chuàng)建的智能合約中讀取特定區(qū)塊所有日志。
下一步是調(diào)用ethclient的 FilterLogs ,它接收我們的查詢并將返回所有的匹配事件日志。
返回的所有日志將是ABI編碼,因此它們本身不會非常易讀。為了解碼日志,我們需要導(dǎo)入我們智能合約的ABI。為此,我們導(dǎo)入編譯好的智能合約Go包,它將包含名稱格式為 ContractABI 的外部屬性。之后,我們使用go-ethereum中的 accounts/abi 包的 abi.JSON 函數(shù)返回一個我們可以在Go應(yīng)用程序中使用的解析過的ABI接口。
現(xiàn)在我們可以通過日志進行迭代并將它們解碼為我么可以使用的類型。若您回憶起我們的樣例合約釋放的日志在Solidity中是類型為 bytes32 ,那么Go中的等價物將是 [32]byte 。我們可以使用這些類型創(chuàng)建一個匿名結(jié)構(gòu)體,并將指針作為第一個參數(shù)傳遞給解析后的ABI接口的 Unpack 函數(shù),以解碼原始的日志數(shù)據(jù)。第二個參數(shù)是我們嘗試解碼的事件名稱,最后一個參數(shù)是編碼的日志數(shù)據(jù)。
此外,日志結(jié)構(gòu)體包含附加信息,例如,區(qū)塊摘要,區(qū)塊號和交易摘要。
若您的solidity事件包含 indexed 事件類型,那么它們將成為 主題 而不是日志的數(shù)據(jù)屬性的一部分。在solidity中您最多只能有4個主題,但只有3個可索引的事件類型。第一個主題總是事件的簽名。我們的示例合約不包含可索引的事件,但如果它確實包含,這是如何讀取事件主題。
正如您所見,首個主題只是被哈希過的事件簽名。
這就是閱讀和解析日志的全部內(nèi)容。要學(xué)習如何訂閱日志,閱讀上個章節(jié)。
命令
Store.sol
event_read.go
首先,創(chuàng)建ERC-20智能合約的事件日志的interface文件 erc20.sol :
然后在給定abi使用 abigen 創(chuàng)建Go包
現(xiàn)在在我們的Go應(yīng)用程序中,讓我們創(chuàng)建與ERC-20事件日志簽名類型相匹配的結(jié)構(gòu)類型:
初始化以太坊客戶端
按照ERC-20智能合約地址和所需的塊范圍創(chuàng)建一個“FilterQuery”。這個例子我們會用 ZRX 代幣:
用 FilterLogs 來過濾日志:
接下來我們將解析JSON abi,稍后我們將使用解壓縮原始日志數(shù)據(jù):
為了按某種日志類型進行過濾,我們需要弄清楚每個事件日志函數(shù)簽名的keccak256哈希值。 事件日志函數(shù)簽名哈希始終是 topic [0] ,我們很快就會看到。 以下是使用go-ethereum crypto 包計算keccak256哈希的方法:
現(xiàn)在我們將遍歷所有日志并設(shè)置switch語句以按事件日志類型進行過濾:
現(xiàn)在要解析 Transfer 事件日志,我們將使用 abi.Unpack 將原始日志數(shù)據(jù)解析為我們的日志類型結(jié)構(gòu)。 解包不會解析 indexed 事件類型,因為它們存儲在 topics 下,所以對于那些我們必須單獨解析,如下例所示:
Approval 日志也是類似的方法:
最后,把所有的步驟放一起:
我們可以把解析的日志與etherscan的數(shù)據(jù)對比:
Commands
erc20.sol
event_read_erc20.go
solc version used for these examples
要讀取 0x Protocol 事件日志,我們必須首先將solidity智能合約編譯為一個Go包。
安裝solc版本 0.4.11
為例如 Exchange.sol 的事件日志創(chuàng)建0x Protocol交易所智能合約接口:
Create the 0x protocol exchange smart contract interface for event logs as Exchange.sol :
接著給定abi,使用 abigen 來創(chuàng)建Go exchange 包:
Then use abigen to create the Go exchange package given the abi:
現(xiàn)在在我們的Go應(yīng)用程序中,讓我們創(chuàng)建與0xProtocol事件日志簽名類型匹配的結(jié)構(gòu)體類型:
初始化以太坊客戶端:
創(chuàng)建一個 FilterQuery ,并為其傳遞0x Protocol智能合約地址和所需的區(qū)塊范圍:
用 FilterLogs 查詢?nèi)罩荆?/p>
接下來我們將解析JSON abi,我們后續(xù)將使用解壓縮原始日志數(shù)據(jù):
為了按某種日志類型過濾,我們需要知曉每個事件日志函數(shù)簽名的keccak256摘要。正如我們很快所見到的那樣,事件日志函數(shù)簽名摘要總是 topic[0] :
現(xiàn)在我們迭代所有的日志并設(shè)置一個switch語句來按事件日志類型過濾:
現(xiàn)在要解析 LogFill ,我們將使用 abi.Unpack 將原始數(shù)據(jù)類型解析為我們自定義的日志類型結(jié)構(gòu)體。Unpack不會解析 indexed 事件類型,因為這些它們存儲在 topics 下,所以對于那些我們必須單獨解析,如下例所示:
對于 LogCancel 類似:
最后是 LogError :
將它們放在一起并運行我們將看到以下輸出:
將解析后的日志輸出與etherscan上的內(nèi)容進行比較:
命令
Exchange.sol
event_read_0xprotocol.go
這些示例使用的solc版本