Hi 游客

更多精彩,请登录!

比特池塘 区块链技术 正文
人们常说,“阅读源代码”是学习编程的一种重要方法。作为程序员,我们在平时的学习工作中,都应该阅读过不少源代码。但是对于大多数人来说,阅读的可能更多是一些代码片断、示例,或者在老师、同事的指导下,先对要阅读的项目代码有了整体的了解之后,再进行针对性的阅读。: J3 T3 x* V4 s" s
但是如果我们面对的是一个像比原这样比较庞大的项目,身边又没有人指导,只能靠自己去看,这时应该怎么来阅读呢?也许每个人也都能找到自己的办法,或高效,或低效,或放弃。) H( J& n* O9 b6 z
我在这次阅读比原源代码的过程中,尝试的是这样一种方法:从外部入手,通过与比原节点进行数据交互,来一步步了解比原的内部原理。就像剥石榴一样,一点点小心翼翼的下手,最后才能吃到鲜美的果肉。3 o6 w2 n& l& D, I: w2 v
所以这个文章系列叫作“剥开比原看代码”。5 i2 T( `; j( X" F2 B) \6 g
说明  X2 W, |5 y& i! b" R
在系列中的每一章,我通常都会由一个或者几个相关的问题入手,然后通过对源代码进行分析,来说明比原的代码是如何实现的。对于与当前问题关系不大的代码,则会简单带过,等真正需要它们出场的时候再详细解说。
0 S* C+ H7 b( x  i, W6 z2 F5 ~为了保证文章中引用代码的稳定性,我将基于比原的v1.0.1代码进行分析。随着时间推移,比原的代码也将快速更新,但是我觉得,只要把这个版本的代码理解了,再去看新的代码,应该是一件很容易的事情。
  }+ u% Q" |$ }5 p% Q在文章中,将会有一些直接指向github上bytom源代码的链接。为了方便,我专门将bytom v1.0.1的代码放到了一个新的仓库中,这样就不容易与比原官方的最新代码混淆。该仓库地址为:https://github.com/freewind/bytom-v1.0.1
0 m4 d$ y2 z2 a2 y2 T' t当然,你不必clone这个仓库(clone官方仓库http://github.com/Bytom/bytom就够了),然后在必要的时候,使用以下命令将代码切换到v1.0.1的tag,以便与本系列引用的代码一致:
- y" t3 J5 N2 i  c* Ogit fetch  L  S% l3 X2 V# ]
git checkout -b v1.0.1( s4 _+ @3 ^0 o6 V& D6 l5 C
不论采用哪种阅读方法,我想第一步都应该先在本地把比原节点跑起来,试试各种功能。
( F1 l: `9 c7 a1 r3 W) P对于如何下载、配置和安装的问题,请直接参看官方文档https://github.com/Bytom/bytom/tree/v1.0.1(注意我这里给出的是v1.0.1的文档),这里不多说。# O1 v* [2 ?- P7 S* w
本篇问题; N* z6 v7 G0 Z, ]+ H8 @
当我们本地使用make bytomd编译完比原后,我们可以使用下面的命令来进行初始化:% G0 H1 |+ F0 g+ o" a( b, W
./bytomd init --chain_id testnet' D, n! q4 T( G# W: C& i
这里指定了使用的chain是testnet(还有别的选项,如mainnet等等)。运行成功后,它将会在本地文件系统生成一些配置文件,供比原启动时使用。
- s1 O0 n! ^$ M* W, R所以我的问题是:
( E3 u: k9 \5 r比原初始化时,产生了什么样的配置文件,放在了哪个目录下?" }6 d, |/ c+ U1 r* J- d
下面我将结合源代码,来回答这个问题。
/ ~! r% f9 y) ^目录位置1 f! R) y( |! O2 O8 }0 `7 C
首先比原在本地会有一个目录专门用于放置各种数据,比如密钥、配置文件、数据库文件等。这个目录对应的代码位于config/config.go#L190-L205:0 W! |7 A' p3 o  m* k
func DefaultDataDir() string {1 H, }( c+ U8 ?+ }' s# n7 E" ?
    // Try to place the data folder in the user's home dir; v% {/ {; _( R! j# a
    home := homeDir()
! l$ G& ~' I0 N. |    dataDir := "./.bytom"
9 g7 Z. q- E: e$ m$ U  w    if home != "" {5 {7 x. H! e4 Y, W2 l+ r0 ]5 _
        switch runtime.GOOS {3 h: r$ {5 x) o* {7 m# H( y
        case "darwin":' i. R- X+ i8 |9 W
            dataDir = filepath.Join(home, "Library", "Bytom"), i2 H  a0 v0 W* a2 V" R
        case "windows":
$ G- E, L/ G6 H) H5 g            dataDir = filepath.Join(home, "AppData", "Roaming", "Bytom")
. R( O8 B/ f  R+ C2 q  U/ j2 P3 a        default:
- H5 j" s# y/ [- \, I            dataDir = filepath.Join(home, ".bytom")
1 `" p: z; t9 V% y  g& B6 R8 u- D        }2 A+ U0 y8 j* L! Q% V7 z% P! s' C: o0 o
    }7 h, x" H2 `% O
    return dataDir
) C! M) I5 p0 h# J/ g1 Z}
" u' ?/ u( z- M. b! I可以看到,在不同的操作系统上,数据目录的位置也不同:; R) s1 x7 p6 M3 ?- P* o7 w6 R
  • 苹果系统(darwin):~/Library/Bytom
  • Windows(windows): ~/AppData/Roaming/Bytom
  • 其它(如Linux):~/.bytom5 ]0 |, {4 r% I8 |3 }" i: H3 |' D

    . u& T* U! z  L5 s) ^, m/ O3 s配置文件内容4 r3 _; y3 o+ }# s1 p5 A
    我们根据自己的操作系统打开相应的目录(我的是~/AppData/Roaming/Bytom),可以看到有一个config.toml,内容大约如下:- D" [5 K$ Y; R$ S6 }" b1 ?
    $ cat config.toml: F" n4 X# g+ |. @: r" q
    # This is a TOML config file.
    ( h) a7 N7 f( E! I3 G# For more information, see https://github.com/toml-lang/toml6 n, R, T; i! O, ^. z' r: k
    fast_sync = true
    : W. m0 o8 u! C% J- H3 t& Ddb_backend = "leveldb"
    0 ]/ t4 }) t' `' t2 Q& Lapi_addr = "0.0.0.0:9888"
    0 b% J( G& [5 C) B, t( L/ G2 tchain_id = "testnet"
    - ^3 e7 v- K+ U1 s3 ?1 [[p2p]0 j1 j" A9 D0 u& I  W
    laddr = "tcp://0.0.0.0:46656"
    " c$ {# r" w# c( d: f. ~seeds = "47.96.42.1:46656,172.104.224.219:46656,45.118.132.164:46656"! Z; V: |4 r( `4 {/ {
    它已经把一些基本信息告诉我们了,比如:
    2 }+ C/ C; Y. p3 x2 h% _' t% I
  • 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":比原启动后,会主动连接这几个地址获取数据) U2 ]$ F9 U. h+ e! T
    ! \" Z8 S, v1 z& v8 D
    内容模板( Q, y5 V% L7 B7 P, E3 `
    使用不同的chain_id去初始化时,会生成不同内容的配置文件,那么这些内容来自于哪里呢?! t+ Y! p+ E! i, B
    原来在config/toml.go#L22-L45,预定义了不同的模板内容:& h9 R( ?$ j" `. E
    var defaultConfigTmpl = `# This is a TOML config file.
    % c+ q# T" Z: n1 y# For more information, see https://github.com/toml-lang/toml
    & N8 ^7 j; O3 `* B. U0 a, S; ?fast_sync = true5 y  W- W7 `# @; c1 ^9 C
    db_backend = "leveldb", ~4 J8 O4 H4 M) N. k4 Z3 b
    api_addr = "0.0.0.0:9888"9 ~6 x/ L6 m' B4 P: \' E: ]8 L
    `" k  \% ]; I+ d2 T
    var mainNetConfigTmpl = `chain_id = "mainnet"
    9 f" j7 c& A  T. m3 W- Q1 z2 n[p2p]
    + l) Z) b+ M, {laddr = "tcp://0.0.0.0:46657"
    # ~' B5 U2 y5 `5 u* m' bseeds = "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": J+ V% L3 S; Y- D* r
    `
    ) b( G. t" }' f- Y- Wvar testNetConfigTmpl = `chain_id = "testnet") ^) P" ?: V& E2 N7 S
    [p2p]
    % [3 p% g+ E3 i6 t: Qladdr = "tcp://0.0.0.0:46656"
    , p3 d6 [' W. M. Lseeds = "47.96.42.1:46656,172.104.224.219:46656,45.118.132.164:46656"
    3 b. Z( C9 c% u! f5 R8 h2 F`
      K. z$ c1 J- k2 rvar soloNetConfigTmpl = `chain_id = "solonet"4 A5 s/ L) i  j
    [p2p]( r; u' Y) l# r) x( J, q  ?: O
    laddr = "tcp://0.0.0.0:46658"
    0 _+ Q  V  G1 Q  \4 J6 useeds = ""
    1 _8 _, }6 c( E0 \`5 Y/ o  ?! v, o' s6 h) @9 }2 F6 J
    可以看到,原来这些端口号和seed的地址,都是事先写好在模板里的。
    2 ~0 j3 u, Y' g9 [( b而且,通过观察这些配置,我们可以发现,如果chain_id不同,则监听的端口和连接的种子都不同:
    9 C. e. u6 l3 C' w5 A5 B' Y5 c! I* g  [
  • mainnet(连接到主网): 46657,会主动连接6个种子
  • testnet(连接到测试网): 46656,会主动连接3个种子
  • solonet(本地单独节点): 46658,不会主动连接别人(也因此不会被别人连接上),适合单机研究- O( ]+ ~7 E' m( W
    ; s; n1 h- x! o# x" }0 \6 m: L
    写入文件' ~/ y# X* a' S! b
    这里我们需要快速的把bytomd init的执行流程过一遍,才能清楚配置文件的写入时机,也同时把前面的内容串在了一起。8 p9 r) D. B7 `9 h
    首先,当我们运行bytomd init时,它对应的代码入口为cmd/bytomd/main.go#L54:& ^+ Q" _' R/ L( T0 \' G
    func main() {# C9 L4 A8 G* M* J; |
        cmd := cli.PrepareBaseCmd(commands.RootCmd, "TM", os.ExpandEnv(config.DefaultDataDir()))
    9 ]8 M: Z4 ^8 T  k    cmd.Execute()
    , G$ L" D9 w; W9 x- S5 w}
    " z7 ?' E  V8 J, L$ I9 o0 k  C: G7 W其中的config.DefaultDataDir()就对应于前面提到数据目录位置。4 d; F/ e! f- m+ w1 N6 P6 @& y# M
    然后执行cmd.Execute(),将根据传入的参数init,选择下面的函数来执行:cmd/bytomd/commands/init.go#L25-L24
    # D" r# A$ @: r+ T8 O4 Yfunc initFiles(cmd *cobra.Command, args []string) {& X& s6 s2 O1 B1 R7 A
        configFilePath := path.Join(config.RootDir, "config.toml")
    1 H. Y, g9 Z- L' Y$ r* l    if _, err := os.Stat(configFilePath); !os.IsNotExist(err) {
      K$ T" j/ C. ~; l: i+ ]        log.WithField("config", configFilePath).Info("Already exists config file.")4 ^* \) y% H; f- k7 @' O1 t
            return  v% [4 t. T4 Y7 D3 d
        }
    , Z: S; D$ b. f# X    if config.ChainID == "mainnet" {
      h! O) Q: k# d4 ?% U        cfg.EnsureRoot(config.RootDir, "mainnet")
    . {1 ]7 V/ w4 P9 ^( W# H    } else if config.ChainID == "testnet" {
    : K$ k1 V( g* C        cfg.EnsureRoot(config.RootDir, "testnet")7 }1 ?# L) ~* w2 j5 x% W1 O- X
        } else {& V$ }: X. G, g6 ~9 q
            cfg.EnsureRoot(config.RootDir, "solonet")
    - O) |+ s1 k: \: t% J0 \( R    }
    1 x- s! u2 o: A# c; S    log.WithField("config", configFilePath).Info("Initialized bytom")- @1 S) x, k! W* f+ j
    }
    0 t5 c* `, f' l! r: w其中的configFilePath,就是config.toml的写入地址,即我们前面所说的数据目录下的config.toml文件。- I) {" V6 j8 z  e& @$ ?; u' {
    cfg.EnsureRoot将用来确认数据目录是有效的,并且将根据传入的chain_id不同,来生成不同的内容写入到配置文件中。* i5 `5 G7 ^, _
    它对应的代码是config/toml.go#L10% W" p. F( H. F2 c, ]! z
    func EnsureRoot(rootDir string, network string) {
    . P0 F' {. S3 p# a7 X9 B5 S    cmn.EnsureDir(rootDir, 0700)6 p( X4 b0 `8 G9 H
        cmn.EnsureDir(rootDir+"/data", 0700)
    . V0 \; j$ A' j# v    configFilePath := path.Join(rootDir, "config.toml")
    ( C6 [- x( F/ G" N, M7 m    // Write default config file if missing.
    : b. W8 w1 W; M- o    if !cmn.FileExists(configFilePath) {
    : O. I+ c2 Y: L% I' D$ r2 O1 a* }        cmn.MustWriteFile(configFilePath, []byte(selectNetwork(network)), 0644)- V. X. h6 z1 A+ N; Y" z6 n
        }
    * `( Z1 k4 Q1 ]6 q- w! V}' R9 r9 b/ L# P# U: C
    可以看到,它对数据目录进行了权限上的确认,并且发现当配置文件存在的时候,不会做任何更改。所以如果我们需要生成新的配置文件,就需要把旧的删除(或改名)。
    ' O, w6 c5 p9 W# ]* r其中的selectNetwork(network)函数,实现了根据chain_id的不同来组装不同的配置文件内容,它对应于master/config/toml.go#L48:
    3 w4 h  ^4 }! u+ C; b5 ofunc selectNetwork(network string) string {
    5 ~2 f4 T# K( q& D4 i( |: k    if network == "testnet" {
    % M. {$ z: m: t  I1 y4 z( U        return defaultConfigTmpl + testNetConfigTmpl: h) j# x+ A* i  L  o6 w; ?
        } else if network == "mainnet" {
    / Q) T& @$ B) J- }8 T( X: L3 K( O        return defaultConfigTmpl + mainNetConfigTmpl% {- G$ V+ L  |* q
        } else {: c7 b$ [9 H) J% e7 [5 x. d
            return defaultConfigTmpl + soloNetConfigTmpl
    $ |& x% Q$ c1 `7 N  V    }
    5 A( i9 d) u0 N/ g$ b$ @7 ^}
    ) Q- k* x9 H  \; @果然就是一个简单的字符串拼接,其中的defaultConfigTmpl和*NetConfgTmpl在前面已经出现,这里不重复。
    + M0 b7 E, o, H% @2 j& @1 {8 ~/ g最后调用第三方函数cmn.MustWriteFile(configFilePath, []byte(selectNetwork(network)), 0644),把拼接出来的配置文件内容以权限0644写入到指定的文件地址。
    & S( X! q1 I% J* Q/ X/ [到这里,我们这个问题就算回答完毕了。
    ! b7 W0 e0 n9 \# M4 D感谢社区用户Freewind的辛勤写作
  • BitMere.com 比特池塘系信息发布平台,比特池塘仅提供信息存储空间服务。
    声明:该文观点仅代表作者本人,本文不代表比特池塘立场,且不构成建议,请谨慎对待。
    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    成为第一个吐槽的人

    一夜雨十年灯潞 初中生
    • 粉丝

      0

    • 关注

      7

    • 主题

      11