剝開比原看代碼01:初始化時產生的設定檔在哪兒

來源:互聯網
上載者:User

作者:freewind

比原項目倉庫:

Github地址:https://github.com/Bytom/bytom

Gitee地址:https://gitee.com/BytomBlockc...

人們常說,“閱讀原始碼”是學習編程的一種重要方法。作為程式員,我們在平時的學習工作中,都應該閱讀過不少原始碼。但是對於大多數人來說,閱讀的可能更多是一些代碼片斷、樣本,或者在老師、同事的指導下,先對要閱讀的項目代碼有了整體的瞭解之後,再進行針對性的閱讀。

但是如果我們面對的是一個像比原這樣比較龐大的項目,身邊又沒有人指導,只能靠自己去看,這時應該怎麼來閱讀呢?也許每個人也都能找到自己的辦法,或高效,或低效,或放棄。

我在這次閱讀比原原始碼的過程中,嘗試的是這樣一種方法:從外部入手,通過與比原節點進行資料互動,來一步步瞭解比原的內部原理。就像剝石榴一樣,一點點小心翼翼的下手,最後才能吃到鮮美的果肉。

所以這個文章系列叫作“剝開比原看代碼”。

說明

在系列中的每一章,我通常都會由一個或者幾個相關的問題入手,然後通過對原始碼進行分析,來說明比原的代碼是如何?的。對於與當前問題關係不大的代碼,則會簡單帶過,等真正需要它們出場的時候再詳細解說。

為了保證文章中引用代碼的穩定性,我將基於比原的v1.0.1代碼進行分析。隨著時間推移,比原的代碼也將快速更新,但是我覺得,只要把這個版本的代碼理解了,再去看新的代碼,應該是一件很容易的事情。

在文章中,將會有一些直接指向github上bytom原始碼的連結。為了方便,我專門將bytom v1.0.1的代碼放到了一個新的倉庫中,這樣就不容易與比原官方的最新代碼混淆。該倉庫地址為:https://github.com/freewind/b...

當然,你不必clone這個倉庫(clone官方倉庫http://github.com/Bytom/bytom就夠了),然後在必要的時候,使用以下命令將代碼切換到v1.0.1的tag,以便與本系列引用的代碼一致:

git fetchgit checkout -b v1.0.1

不論採用哪種閱讀方法,我想第一步都應該先在本地把比原節點跑起來,試試各種功能。

對於如何下載、配置和安裝的問題,請直接參看官方文檔https://github.com/Bytom/byto...(注意我這裡給出的是v1.0.1的文檔),這裡不多說。

本篇問題

當我們本地使用make bytomd編譯完比原後,我們可以使用下面的命令來進行初始化:

./bytomd init --chain_id testnet

這裡指定了使用的chain是testnet(還有別的選項,如mainnet等等)。運行成功後,它將會在本地檔案系統產生一些設定檔,供比原啟動時使用。

所以我的問題是:

比原初始化時,產生了什麼樣的設定檔,放在了哪個目錄下?

下面我將結合原始碼,來回答這個問題。

目錄位置

首先比原在本地會有一個目錄專門用於放置各種資料,比如密鑰、設定檔、資料庫檔案等。這個目錄對應的代碼位於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}

可以看到,在不同的作業系統上,資料目錄的位置也不同:

  1. 蘋果系統(darwin):~/Library/Bytom
  2. Windows(windows): ~/AppData/Roaming/Bytom
  3. 其它(如Linux):~/.bytom

設定檔內容

我們根據自己的作業系統開啟相應的目錄(我的是~/Library/Bytom),可以看到有一個config.toml,內容大約如下:

$ cat config.toml# This is a TOML config file.# For more information, see https://github.com/toml-lang/tomlfast_sync = truedb_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"

它已經把一些基本資料告訴我們了,比如:

  • db_backend = "leveldb":說明比原內部使用了leveldb作為資料庫(用來儲存塊資料、帳號、交易資訊等)
  • api_addr = "0.0.0.0:9888":我們可以在瀏覽器中開啟http://localhost:9888來訪問dashboard頁面,進行查看與管理
  • chain_id = "testnet":當前串連的是testnet,即測試網,裡面挖出來的比原幣是不值錢的
  • laddr = "tcp://0.0.0.0:46656":本地監聽46656連接埠,別的節點如果想連我,就需要訪問我的46656連接埠
  • seeds = "47.96.42.1:46656,172.104.224.219:46656,45.118.132.164:46656":比原啟動後,會主動串連這幾個地址擷取資料

內容範本

使用不同的chain_id去初始化時,會產生不同內容的設定檔,那麼這些內容來自於哪裡呢?

原來在config/toml.go#L22-L45,預定義了不同的模板內容:

var defaultConfigTmpl = `# This is a TOML config file.# For more information, see https://github.com/toml-lang/tomlfast_sync = truedb_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 = ""`

可以看到,原來這些連接埠號碼和seed的地址,都是事先寫好在模板裡的。

而且,通過觀察這些配置,我們可以發現,如果chain_id不同,則監聽的連接埠和串連的種子都不同:

  1. mainnet(串連到主網): 46657,會主動串連6個種子
  2. testnet(串連到測試網): 46656,會主動串連3個種子
  3. solonet(本地單獨節點): 46658,不會主動串連別人(也因此不會被別人串連上),適合單機研究

寫入檔案

這裡我們需要快速的把bytomd init的執行流程過一遍,才能清楚設定檔的寫入時機,也同時把前面的內容串在了一起。

首先,當我們運行bytomd init時,它對應的代碼入口為cmd/bytomd/main.go#L54:

func main() {    cmd := cli.PrepareBaseCmd(commands.RootCmd, "TM", os.ExpandEnv(config.DefaultDataDir()))    cmd.Execute()}

其中的config.DefaultDataDir()就對應於前面提到資料目錄位置。

然後執行cmd.Execute(),將根據傳入的參數init,選擇下面的函數來執行: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的寫入地址,即我們前面所說的資料目錄下的config.toml檔案。

cfg.EnsureRoot將用來確認資料目錄是有效,並且將根據傳入的chain_id不同,來產生不同的內容寫入到設定檔中。

它對應的代碼是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)    }}

可以看到,它對資料目錄進行了許可權上的確認,並且發現當設定檔存在的時候,不會做任何更改。所以如果我們需要產生新的設定檔,就需要把舊的刪除(或改名)。

其中的selectNetwork(network)函數,實現了根據chain_id的不同來組裝不同的設定檔內容,它對應於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    }}

果然就是一個簡單的字串拼接,其中的defaultConfigTmpl*NetConfgTmpl在前面已經出現,這裡不重複。

最後調用第三方函數cmn.MustWriteFile(configFilePath, []byte(selectNetwork(network)), 0644),把拼接出來的設定檔內容以許可權0644寫入到指定的檔案地址。

到這裡,我們這個問題就算回答完畢了。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.