在開發以太坊時,很多時候需要搭建一條以太坊私有鏈,通過本文一起看看如何在Mac上進行搭建。
寫在前面
閱讀本文前,你應該對以太坊語言有所了解,如果你還不了解,建議你先看以太坊是什么
go-ethereum客戶端安裝
Go-ethereum客戶端通常被稱為Geth,它是個命令行界面,執行在Go上實現的完整以太坊節點。Geth得益於Go語言的多平台特性,支持在多個平台上使用(比如Windows、Linux、Mac)。Geth是以太坊協議的具體落地實現,通過Geth,你可以實現以太坊的各種功能,如賬戶的新建編輯刪除,開啟挖礦,ether幣的轉移,智能合約的部署和執行等等。所以,我們選擇geth工具來進行開發。由於本人是mac,所以優先使用mac進行開發啦。mac中geth安裝如下:
1 |
brew tap ethereum/ethereum |
檢查是否安裝成功
1 |
geth --help |
如果輸出一些幫助提示命令,則說明安裝成功。
其他平台可參考Geth 安裝
搭建私有鏈
以太坊支持自定義創世區塊,要運行私有鏈,我們就需要定義自己的創世區塊,創世區塊信息寫在一個json格式的配置文件中。首先將下面的內容保存到一個json文件中,例如genesis.json。
json文件內容如下:
1 |
{ |
初始化:寫入創世區塊
准備好創世區塊json配置文件后,需要初始化區塊鏈,將上面的創世區塊信息寫入到區塊鏈中。首先要新建一個目錄data0用來存放區塊鏈數據(其實,這個目錄data0就相當於一個根節點。當我們基於genesis.json生成根節點后,其他人就可以來連接此根節點,從而能進行交易)。data0目錄結構如圖所示:
接下來進入privatechain目錄中,執行初始化命令:
1 |
cd privatechain |
上面的命令的主體是 geth init,表示初始化區塊鏈,命令可以帶有選項和參數,其中–datadir選項后面跟一個目錄名,這里為 data0,表示指定數據存放目錄為 data0, genesis.json是init命令的參數。
運行上面的命令,會讀取genesis.json文件,根據其中的內容,將創世區塊寫入到區塊鏈中。如果看到log信息中含有Successfully wrote genesis state
字樣,說明初始化成功。
初始化成功后的目錄如下:
其中geth/chaindata中存放的是區塊數據,keystore中存放的是賬戶數據。
啟動私有鏈節點
初始化完成后,就有了一條自己的私有鏈,之后就可以啟動自己的私有鏈節點並做一些操作,在終端中輸入以下命令即可啟動節點:
1 |
geth --datadir data0 --networkid 1108 console |
上面命令的主體是geth console,表示啟動節點並進入交互式控制台,–datadir選項指定使用data0作為數據目錄,–networkid選項后面跟一個數字,這里是1108,表示指定這個私有鏈的網絡id為1108。網絡id在連接到其他節點的時候會用到,以太坊公網的網絡id是1,為了不與公有鏈網絡沖突,運行私有鏈節點的時候要指定自己的網絡id(上面命令可能會運行失敗,我直接重啟mac,再進入到privateChain目錄中,簡單粗暴)。
運行上面的命令后,就啟動了區塊鏈節點並進入了Javascript Console:
這是一個交互式的Javascript執行環境,在這里面可以執行Javascript代碼,其中>是命令提示符。在這個環境里也內置了一些用來操作以太坊的Javascript對象,可以直接使用這些對象。這些對象主要包括:
eth:包含一些跟操作區塊鏈相關的方法
net:包含以下查看p2p網絡狀態的方法
admin:包含一些與管理節點相關的方法
miner:包含啟動&停止挖礦的一些方法
personal:主要包含一些管理賬戶的方法
txpool:包含一些查看交易內存池的方法
web3:包含了以上對象,還包含一些單位換算的方法
玩轉Javascript Console
進入以太坊Javascript Console后,就可以使用里面的內置對象做一些操作,這些內置對象提供的功能很豐富,比如查看區塊和交易、創建賬戶、挖礦、發送交易、部署智能合約等。接下來介紹幾個常用功能,下面的操作中,前面帶>的表示在Javascript Console中執行的命令。
創建賬戶
前面只是搭建了私有鏈,並沒有自己的賬戶,可以在js console中輸入eth.accounts來驗證:
1 |
> eth.accounts |
此時沒有賬戶,接下來使用personal對象來創建一個賬戶:
1 |
> personal.newAccount() |
Passphrase其實就是密碼的意思,輸入兩次密碼后,就創建了一個賬戶。再次執行命令:
1 |
> personal.newAccount() |
這時候再去看賬戶,就有兩個了。
1 |
> eth.accounts |
賬戶默認會保存在數據目錄的keystore文件夾中。查看目錄結構,發現data0/keystore中多了兩個文件,這兩個文件就對應剛才創建的兩個賬戶,這是json格式的文本文件,可以打開查看,里面存的是私鑰經過密碼加密后的信息。
1 |
{ |
查看賬戶余額
eth對象提供了查看賬戶余額的方法:
1 |
> eth.getBalance(eth.accounts[0]) |
目前兩個賬戶的以太幣余額都是0,要使賬戶有余額,可以從其他賬戶轉賬過來,或者通過挖礦來獲得以太幣獎勵。
啟動&停止挖礦
通過miner.start()來啟動挖礦:
1 |
> miner.start(10) |
其中start的參數表示挖礦使用的線程數。第一次啟動挖礦會先生成挖礦所需的DAG文件,這個過程有點慢,等進度達到100%后,就會開始挖礦,此時屏幕會被挖礦信息刷屏。
如果想停止挖礦,並且進度已經達到100%之后,可以在js console中輸入
1 |
miner.stop(): |
注意:輸入的字符會被挖礦刷屏信息沖掉,沒有關系,只要輸入完整的miner.stop()之后回車,即可停止挖礦。
挖到一個區塊會獎勵5個以太幣,挖礦所得的獎勵會進入礦工的賬戶,這個賬戶叫做coinbase,默認情況下coinbase是本地賬戶中的第一個賬戶:
1 |
> eth.coinbase |
現在的coinbase是賬戶0,要想使挖礦獎勵進入其他賬戶,通過miner.setEtherbase()將其他賬戶設置成coinbase即可:
1 |
> miner.setEtherbase(eth.accounts[1]) |
挖到區塊以后,賬戶0里面應該就有余額了:
1 |
> eth.getBalance(eth.accounts[0]) |
getBalance()返回值的單位是wei,wei是以太幣的最小單位,1個以太幣=10的18次方個wei。要查看有多少個以太幣,可以用web3.fromWei()將返回值換算成以太幣:
1 |
> web3.fromWei(eth.getBalance(eth.accounts[0]),'ether') |
發送交易
截止目前,賬戶一的余額還是0:
1 |
> eth.getBalance(eth.accounts[1]) |
可以通過發送一筆交易,從賬戶0轉移10個以太幣到賬戶1:
1 |
> amount = web3.toWei(10,'ether') |
這里報錯了,原因是賬戶每隔一段時間就會被鎖住,要發送交易,必須先解鎖賬戶,由於我們要從賬戶0發送交易,所以要解鎖賬戶0:
1 |
> personal.unlockAccount(eth.accounts[0]) |
輸入創建賬戶時設置的密碼,就可以成功解鎖賬戶。然后再發送交易:
1 |
> amount = web3.toWei(10,'ether') |
我們去查看賬戶1中的余額:
1 |
> eth.getBalance(eth.accounts[1]) |
發現還沒轉過去,此時交易已經提交到區塊鏈,但還未被處理,這可以通過查看txpool來驗證:
1 |
> txpool.status |
其中有一條pending的交易,pending表示已提交但還未被處理的交易。
要使交易被處理,必須要挖礦。這里我們啟動挖礦,然后等待挖到一個區塊之后就停止挖礦:
1 |
> miner.start(1);admin.sleepBlocks(1);miner.stop(); |
當miner.stop()返回true后,txpool中pending的交易數量應該為0了,說明交易已經被處理了,而賬戶1應該收到幣了:
1 |
> web3.fromWei(eth.getBalance(eth.accounts[1]),'ether') |
查看交易和區塊
eth對象封裝了查看交易和區塊信息的方法。
查看當前區塊總數:
1 |
> eth.blockNumber |
通過區塊號查看區塊:
1 |
> eth.getBlock(66) |
通過交易hash查看交易:
1 |
> eth.getTransaction("0x1b21bba16dd79b659c83594b0c41de42debb2738b447f6b24e133d51149ae2a6") |
深入淺出區塊鏈 - 系統學習區塊鏈,打造最好的區塊鏈技術博客。