摘要:所以這個(gè)文章系列叫作剝開比原看代碼。所以我的問題是比原初始化時(shí),產(chǎn)生了什么樣的配置文件,放在了哪個(gè)目錄下下面我將結(jié)合源代碼,來回答這個(gè)問題。將用來確認(rèn)數(shù)據(jù)目錄是有效的,并且將根據(jù)傳入的不同,來生成不同的內(nèi)容寫入到配置文件中。
作者:freewind
比原項(xiàng)目倉(cāng)庫(kù):
Github地址:https://github.com/Bytom/bytom
Gitee地址:https://gitee.com/BytomBlockc...
人們常說,“閱讀源代碼”是學(xué)習(xí)編程的一種重要方法。作為程序員,我們?cè)谄綍r(shí)的學(xué)習(xí)工作中,都應(yīng)該閱讀過不少源代碼。但是對(duì)于大多數(shù)人來說,閱讀的可能更多是一些代碼片斷、示例,或者在老師、同事的指導(dǎo)下,先對(duì)要閱讀的項(xiàng)目代碼有了整體的了解之后,再進(jìn)行針對(duì)性的閱讀。
但是如果我們面對(duì)的是一個(gè)像比原這樣比較龐大的項(xiàng)目,身邊又沒有人指導(dǎo),只能靠自己去看,這時(shí)應(yīng)該怎么來閱讀呢?也許每個(gè)人也都能找到自己的辦法,或高效,或低效,或放棄。
我在這次閱讀比原源代碼的過程中,嘗試的是這樣一種方法:從外部入手,通過與比原節(jié)點(diǎn)進(jìn)行數(shù)據(jù)交互,來一步步了解比原的內(nèi)部原理。就像剝石榴一樣,一點(diǎn)點(diǎn)小心翼翼的下手,最后才能吃到鮮美的果肉。
所以這個(gè)文章系列叫作“剝開比原看代碼”。
說明在系列中的每一章,我通常都會(huì)由一個(gè)或者幾個(gè)相關(guān)的問題入手,然后通過對(duì)源代碼進(jìn)行分析,來說明比原的代碼是如何實(shí)現(xiàn)的。對(duì)于與當(dāng)前問題關(guān)系不大的代碼,則會(huì)簡(jiǎn)單帶過,等真正需要它們出場(chǎng)的時(shí)候再詳細(xì)解說。
為了保證文章中引用代碼的穩(wěn)定性,我將基于比原的v1.0.1代碼進(jìn)行分析。隨著時(shí)間推移,比原的代碼也將快速更新,但是我覺得,只要把這個(gè)版本的代碼理解了,再去看新的代碼,應(yīng)該是一件很容易的事情。
在文章中,將會(huì)有一些直接指向github上bytom源代碼的鏈接。為了方便,我專門將bytom v1.0.1的代碼放到了一個(gè)新的倉(cāng)庫(kù)中,這樣就不容易與比原官方的最新代碼混淆。該倉(cāng)庫(kù)地址為:https://github.com/freewind/b...
當(dāng)然,你不必clone這個(gè)倉(cāng)庫(kù)(clone官方倉(cāng)庫(kù)http://github.com/Bytom/bytom就夠了),然后在必要的時(shí)候,使用以下命令將代碼切換到v1.0.1的tag,以便與本系列引用的代碼一致:
git fetch git checkout -b v1.0.1
不論采用哪種閱讀方法,我想第一步都應(yīng)該先在本地把比原節(jié)點(diǎn)跑起來,試試各種功能。
對(duì)于如何下載、配置和安裝的問題,請(qǐng)直接參看官方文檔https://github.com/Bytom/byto...(注意我這里給出的是v1.0.1的文檔),這里不多說。
本篇問題當(dāng)我們本地使用make bytomd編譯完比原后,我們可以使用下面的命令來進(jìn)行初始化:
./bytomd init --chain_id testnet
這里指定了使用的chain是testnet(還有別的選項(xiàng),如mainnet等等)。運(yùn)行成功后,它將會(huì)在本地文件系統(tǒng)生成一些配置文件,供比原啟動(dòng)時(shí)使用。
所以我的問題是:
比原初始化時(shí),產(chǎn)生了什么樣的配置文件,放在了哪個(gè)目錄下?下面我將結(jié)合源代碼,來回答這個(gè)問題。
目錄位置首先比原在本地會(huì)有一個(gè)目錄專門用于放置各種數(shù)據(jù),比如密鑰、配置文件、數(shù)據(jù)庫(kù)文件等。這個(gè)目錄對(duì)應(yīng)的代碼位于config/config.go#L190-L205:
func DefaultDataDir() string { // Try to place the data folder in the user"s home dir home := homeDir() dataDir := "./.bytom" if home != "" { switch runtime.GOOS { case "darwin": dataDir = filepath.Join(home, "Library", "Bytom") case "windows": dataDir = filepath.Join(home, "AppData", "Roaming", "Bytom") default: dataDir = filepath.Join(home, ".bytom") } } return dataDir }
可以看到,在不同的操作系統(tǒng)上,數(shù)據(jù)目錄的位置也不同:
蘋果系統(tǒng)(darwin):~/Library/Bytom
Windows(windows): ~/AppData/Roaming/Bytom
其它(如Linux):~/.bytom
配置文件內(nèi)容我們根據(jù)自己的操作系統(tǒng)打開相應(yīng)的目錄(我的是~/Library/Bytom),可以看到有一個(gè)config.toml,內(nèi)容大約如下:
$ cat config.toml # This is a TOML config file. # For more information, see https://github.com/toml-lang/toml fast_sync = true db_backend = "leveldb" api_addr = "0.0.0.0:9888" chain_id = "testnet" [p2p] laddr = "tcp://0.0.0.0:46656" seeds = "47.96.42.1:46656,172.104.224.219:46656,45.118.132.164:46656"
它已經(jīng)把一些基本信息告訴我們了,比如:
db_backend = "leveldb":說明比原內(nèi)部使用了leveldb作為數(shù)據(jù)庫(kù)(用來保存塊數(shù)據(jù)、帳號(hào)、交易信息等)
api_addr = "0.0.0.0:9888":我們可以在瀏覽器中打開http://localhost:9888來訪問dashboard頁面,進(jìn)行查看與管理
chain_id = "testnet":當(dāng)前連接的是testnet,即測(cè)試網(wǎng),里面挖出來的比原幣是不值錢的
laddr = "tcp://0.0.0.0:46656":本地監(jiān)聽46656端口,別的節(jié)點(diǎn)如果想連我,就需要訪問我的46656端口
seeds = "47.96.42.1:46656,172.104.224.219:46656,45.118.132.164:46656":比原啟動(dòng)后,會(huì)主動(dòng)連接這幾個(gè)地址獲取數(shù)據(jù)
內(nèi)容模板使用不同的chain_id去初始化時(shí),會(huì)生成不同內(nèi)容的配置文件,那么這些內(nèi)容來自于哪里呢?
原來在config/toml.go#L22-L45,預(yù)定義了不同的模板內(nèi)容:
var defaultConfigTmpl = `# This is a TOML config file. # For more information, see https://github.com/toml-lang/toml fast_sync = true db_backend = "leveldb" api_addr = "0.0.0.0:9888" ` var mainNetConfigTmpl = `chain_id = "mainnet" [p2p] laddr = "tcp://0.0.0.0:46657" seeds = "45.79.213.28:46657,198.74.61.131:46657,212.111.41.245:46657,47.100.214.154:46657,47.100.109.199:46657,47.100.105.165:46657" ` var testNetConfigTmpl = `chain_id = "testnet" [p2p] laddr = "tcp://0.0.0.0:46656" seeds = "47.96.42.1:46656,172.104.224.219:46656,45.118.132.164:46656" ` var soloNetConfigTmpl = `chain_id = "solonet" [p2p] laddr = "tcp://0.0.0.0:46658" seeds = "" `
可以看到,原來這些端口號(hào)和seed的地址,都是事先寫好在模板里的。
而且,通過觀察這些配置,我們可以發(fā)現(xiàn),如果chain_id不同,則監(jiān)聽的端口和連接的種子都不同:
mainnet(連接到主網(wǎng)): 46657,會(huì)主動(dòng)連接6個(gè)種子
testnet(連接到測(cè)試網(wǎng)): 46656,會(huì)主動(dòng)連接3個(gè)種子
solonet(本地多帶帶節(jié)點(diǎn)): 46658,不會(huì)主動(dòng)連接別人(也因此不會(huì)被別人連接上),適合單機(jī)研究
寫入文件這里我們需要快速的把bytomd init的執(zhí)行流程過一遍,才能清楚配置文件的寫入時(shí)機(jī),也同時(shí)把前面的內(nèi)容串在了一起。
首先,當(dāng)我們運(yùn)行bytomd init時(shí),它對(duì)應(yīng)的代碼入口為cmd/bytomd/main.go#L54:
func main() { cmd := cli.PrepareBaseCmd(commands.RootCmd, "TM", os.ExpandEnv(config.DefaultDataDir())) cmd.Execute() }
其中的config.DefaultDataDir()就對(duì)應(yīng)于前面提到數(shù)據(jù)目錄位置。
然后執(zhí)行cmd.Execute(),將根據(jù)傳入的參數(shù)init,選擇下面的函數(shù)來執(zhí)行:cmd/bytomd/commands/init.go#L25-L24
func initFiles(cmd *cobra.Command, args []string) { configFilePath := path.Join(config.RootDir, "config.toml") if _, err := os.Stat(configFilePath); !os.IsNotExist(err) { log.WithField("config", configFilePath).Info("Already exists config file.") return } if config.ChainID == "mainnet" { cfg.EnsureRoot(config.RootDir, "mainnet") } else if config.ChainID == "testnet" { cfg.EnsureRoot(config.RootDir, "testnet") } else { cfg.EnsureRoot(config.RootDir, "solonet") } log.WithField("config", configFilePath).Info("Initialized bytom") }
其中的configFilePath,就是config.toml的寫入地址,即我們前面所說的數(shù)據(jù)目錄下的config.toml文件。
cfg.EnsureRoot將用來確認(rèn)數(shù)據(jù)目錄是有效的,并且將根據(jù)傳入的chain_id不同,來生成不同的內(nèi)容寫入到配置文件中。
它對(duì)應(yīng)的代碼是config/toml.go#L10
func EnsureRoot(rootDir string, network string) { cmn.EnsureDir(rootDir, 0700) cmn.EnsureDir(rootDir+"/data", 0700) configFilePath := path.Join(rootDir, "config.toml") // Write default config file if missing. if !cmn.FileExists(configFilePath) { cmn.MustWriteFile(configFilePath, []byte(selectNetwork(network)), 0644) } }
可以看到,它對(duì)數(shù)據(jù)目錄進(jìn)行了權(quán)限上的確認(rèn),并且發(fā)現(xiàn)當(dāng)配置文件存在的時(shí)候,不會(huì)做任何更改。所以如果我們需要生成新的配置文件,就需要把舊的刪除(或改名)。
其中的selectNetwork(network)函數(shù),實(shí)現(xiàn)了根據(jù)chain_id的不同來組裝不同的配置文件內(nèi)容,它對(duì)應(yīng)于master/config/toml.go#L48:
func selectNetwork(network string) string { if network == "testnet" { return defaultConfigTmpl + testNetConfigTmpl } else if network == "mainnet" { return defaultConfigTmpl + mainNetConfigTmpl } else { return defaultConfigTmpl + soloNetConfigTmpl } }
果然就是一個(gè)簡(jiǎn)單的字符串拼接,其中的defaultConfigTmpl和*NetConfgTmpl在前面已經(jīng)出現(xiàn),這里不重復(fù)。
最后調(diào)用第三方函數(shù)cmn.MustWriteFile(configFilePath, []byte(selectNetwork(network)), 0644),把拼接出來的配置文件內(nèi)容以權(quán)限0644寫入到指定的文件地址。
到這里,我們這個(gè)問題就算回答完畢了。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/24161.html
摘要:對(duì)應(yīng)于繼續(xù),加入了超時(shí)對(duì)應(yīng)于終于到了包的調(diào)用,開始真正去連接這個(gè)種子節(jié)點(diǎn)了,到這里,我們可以認(rèn)為這個(gè)問題解決了。 作者:freewind 比原項(xiàng)目倉(cāng)庫(kù): Github地址:https://github.com/Bytom/bytom Gitee地址:https://gitee.com/BytomBlockc... 最開始我對(duì)于這個(gè)問題一直有個(gè)疑惑:區(qū)塊鏈?zhǔn)且粋€(gè)分布式的網(wǎng)絡(luò),那么一個(gè)節(jié)點(diǎn)...
摘要:如果傳的是,就會(huì)在內(nèi)部使用默認(rèn)的隨機(jī)數(shù)生成器生成隨機(jī)數(shù)并生成密鑰。使用的是,生成的是一個(gè)形如這樣的全球唯一的隨機(jī)數(shù)把密鑰以文件形式保存在硬盤上。 作者:freewind 比原項(xiàng)目倉(cāng)庫(kù): Github地址:https://github.com/Bytom/bytom Gitee地址:https://gitee.com/BytomBlockc... 在前一篇,我們探討了從瀏覽器的dashb...
摘要:?jiǎn)?dòng)直到進(jìn)入所以我們首先需要知道,比原在源代碼中是如何啟動(dòng),并且一步步走進(jìn)了的世界。后面省略了一些代碼,主要是用來獲取當(dāng)前監(jiān)聽的實(shí)際以及外網(wǎng),并記錄在日志中。 比原是如何監(jiān)聽p2p端口的 我們知道,在使用bytomd init --chain_id mainnet/testnet/solonet初始化比原的時(shí)候,它會(huì)根據(jù)給定的chain_id的不同,使用不同的端口(參看config/t...
摘要:繼續(xù)看生成地址的方法由于這個(gè)方法里傳過來的是而不是對(duì)象,所以還需要再用查一遍,然后,再調(diào)用這個(gè)私有方法創(chuàng)建地址該方法可以分成部分在第塊中主要關(guān)注的是返回值。 作者:freewind 比原項(xiàng)目倉(cāng)庫(kù): Github地址:https://github.com/Bytom/bytom Gitee地址:https://gitee.com/BytomBlockc... 在比原的dashboard中...
摘要:所以在今天我打算通過源代碼分析一下比原的挖礦流程,但是考慮到它肯定會(huì)涉及到比原的核心,所以太復(fù)雜的地方我就會(huì)先跳過,那些地方時(shí)機(jī)成熟的時(shí)候會(huì)徹底研究一下。 作者:freewind 比原項(xiàng)目倉(cāng)庫(kù): Github地址:https://github.com/Bytom/bytom Gitee地址:https://gitee.com/BytomBlockc... 當(dāng)我們以bytom init ...
閱讀 883·2021-11-18 10:02
閱讀 1708·2019-08-30 15:56
閱讀 2578·2019-08-30 13:47
閱讀 2649·2019-08-29 12:43
閱讀 867·2019-08-29 11:19
閱讀 1795·2019-08-28 18:23
閱讀 2680·2019-08-26 12:23
閱讀 3020·2019-08-23 15:29