Hi 游客

更多精彩,请登录!

比特池塘 区块链技术 正文
人们常说,“阅读源代码”是学习编程的一种重要方法。作为程序员,我们在平时的学习工作中,都应该阅读过不少源代码。但是对于大多数人来说,阅读的可能更多是一些代码片断、示例,或者在老师、同事的指导下,先对要阅读的项目代码有了整体的了解之后,再进行针对性的阅读。
, g# O9 L' K7 m6 O5 @* Y6 I# R但是如果我们面对的是一个像比原这样比较庞大的项目,身边又没有人指导,只能靠自己去看,这时应该怎么来阅读呢?也许每个人也都能找到自己的办法,或高效,或低效,或放弃。
4 B. O- e: p, ~我在这次阅读比原源代码的过程中,尝试的是这样一种方法:从外部入手,通过与比原节点进行数据交互,来一步步了解比原的内部原理。就像剥石榴一样,一点点小心翼翼的下手,最后才能吃到鲜美的果肉。
, Y0 q2 L/ m- T" ~) `2 }所以这个文章系列叫作“剥开比原看代码”。& ]! w' w! K+ z
说明8 n1 b  P( _$ w2 V
在系列中的每一章,我通常都会由一个或者几个相关的问题入手,然后通过对源代码进行分析,来说明比原的代码是如何实现的。对于与当前问题关系不大的代码,则会简单带过,等真正需要它们出场的时候再详细解说。# _) U' z; I& y* C# C8 e
为了保证文章中引用代码的稳定性,我将基于比原的v1.0.1代码进行分析。随着时间推移,比原的代码也将快速更新,但是我觉得,只要把这个版本的代码理解了,再去看新的代码,应该是一件很容易的事情。- o2 u3 c( {: z' k, d
在文章中,将会有一些直接指向github上bytom源代码的链接。为了方便,我专门将bytom v1.0.1的代码放到了一个新的仓库中,这样就不容易与比原官方的最新代码混淆。该仓库地址为:https://github.com/freewind/bytom-v1.0.1
* v, r# ]. H" Q* T  c& m2 a当然,你不必clone这个仓库(clone官方仓库http://github.com/Bytom/bytom就够了),然后在必要的时候,使用以下命令将代码切换到v1.0.1的tag,以便与本系列引用的代码一致:/ ]# d9 Q# j* ]8 F: d
git fetch4 x; h' B  q2 S& L
git checkout -b v1.0.1) M* u/ ]4 Z- N1 u' `0 W" Y
不论采用哪种阅读方法,我想第一步都应该先在本地把比原节点跑起来,试试各种功能。6 W" A* t/ y' U2 q- @2 b- p$ O
对于如何下载、配置和安装的问题,请直接参看官方文档https://github.com/Bytom/bytom/tree/v1.0.1(注意我这里给出的是v1.0.1的文档),这里不多说。# m% v4 S  m8 c' ?/ L# Y
本篇问题' n, C5 x, M! \( R
当我们本地使用make bytomd编译完比原后,我们可以使用下面的命令来进行初始化:
( G% ^4 ^' p2 Y' u9 g% J3 P; i./bytomd init --chain_id testnet
% W# c: x4 y! D& e这里指定了使用的chain是testnet(还有别的选项,如mainnet等等)。运行成功后,它将会在本地文件系统生成一些配置文件,供比原启动时使用。0 Z4 @/ S8 x9 r) a/ y
所以我的问题是:
+ Z' @! Z+ r" e' j: I, T' @1 E比原初始化时,产生了什么样的配置文件,放在了哪个目录下?2 z& J% X8 d) x- J/ a0 W. O5 X
下面我将结合源代码,来回答这个问题。, q6 M9 [7 Z; s0 [$ u
目录位置& k. ~8 S6 s6 ~+ Z
首先比原在本地会有一个目录专门用于放置各种数据,比如密钥、配置文件、数据库文件等。这个目录对应的代码位于config/config.go#L190-L205:
) x5 z8 h5 k# ?$ Z- [3 Z2 tfunc DefaultDataDir() string {
! Y5 ]1 f8 Y  ~5 V# `    // Try to place the data folder in the user's home dir' Z: ~- ~4 k, O- z
    home := homeDir()
6 y* S# U9 f+ f  k    dataDir := "./.bytom"
9 V/ N& [/ d' E4 z, x, o/ i    if home != "" {4 d, ~1 S  c7 p# ^% N5 f0 `' S
        switch runtime.GOOS {
) M" t. x- h* o8 I% f+ t        case "darwin":4 ^: V5 e/ w4 c# F5 @  Y* [8 X
            dataDir = filepath.Join(home, "Library", "Bytom")9 q3 l5 H" H" J
        case "windows":) v% A( B4 o4 l: l
            dataDir = filepath.Join(home, "AppData", "Roaming", "Bytom")' f8 G2 E. R: t" [2 h9 U& e  d
        default:& y- ?! b+ i1 t8 d; L+ {/ `2 F6 p
            dataDir = filepath.Join(home, ".bytom")
2 ]% R! B- U% C        }$ @8 i6 T) N, X8 ?: M5 b3 l4 F
    }
$ n/ r/ u3 b4 ^9 I5 k" ]    return dataDir) b3 B. q" s  s$ b% E/ i6 L; O+ j
}5 ^: u, g2 H: P; R9 C7 S
可以看到,在不同的操作系统上,数据目录的位置也不同:9 I# C; q. V* ?- I) }( e5 K' U& f
  • 苹果系统(darwin):~/Library/Bytom
  • Windows(windows): ~/AppData/Roaming/Bytom
  • 其它(如Linux):~/.bytom6 L+ X: C8 k$ s6 V

    5 m  X$ C0 S$ C7 q3 U/ |2 }% t配置文件内容
    - w1 i4 v" n+ t& h/ a6 c( C我们根据自己的操作系统打开相应的目录(我的是~/AppData/Roaming/Bytom),可以看到有一个config.toml,内容大约如下:2 T8 r, w9 m+ W! d$ W5 A% x! o! c! P
    $ cat config.toml( ~4 E/ k( T; X' |: Y0 t3 g1 @: s) }
    # This is a TOML config file.9 ^( A' B9 x) t+ T
    # For more information, see https://github.com/toml-lang/toml( k0 D) l7 w% D- |: ^7 ~: X
    fast_sync = true- U$ ?* s$ `9 C5 I
    db_backend = "leveldb"
    6 W6 a  b! M/ S% X3 f2 J5 eapi_addr = "0.0.0.0:9888"
    1 ^. f* u5 G% y8 R1 uchain_id = "testnet"
    4 z' Z3 q5 \; }[p2p]' ]# G6 x  A: x
    laddr = "tcp://0.0.0.0:46656"6 z6 \2 ^! v) j9 }
    seeds = "47.96.42.1:46656,172.104.224.219:46656,45.118.132.164:46656"
    ; c' T! U* {# ^, G& I4 ]它已经把一些基本信息告诉我们了,比如:
    1 ]% g9 v2 s# ^3 E" v
  • 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":比原启动后,会主动连接这几个地址获取数据
    7 c" [. n3 L% M8 _, a1 _

    - G0 W2 {) \+ f* x/ C内容模板
    + D& Q! [# o+ ^: u2 ?* b使用不同的chain_id去初始化时,会生成不同内容的配置文件,那么这些内容来自于哪里呢?4 [9 F( _3 \3 j& h
    原来在config/toml.go#L22-L45,预定义了不同的模板内容:: r# D9 C2 q; B. l( P8 m' r9 i
    var defaultConfigTmpl = `# This is a TOML config file.
    5 a: {/ {. _. L9 B# For more information, see https://github.com/toml-lang/toml2 |6 f' s, o! d
    fast_sync = true
    : Y; t! E+ E1 u- W2 Z. rdb_backend = "leveldb"# X' G+ B4 `2 G- v4 g' M" ~
    api_addr = "0.0.0.0:9888". _$ Y) C' C, R3 L
    `3 C6 A8 H9 j+ H' N# Y! X  J
    var mainNetConfigTmpl = `chain_id = "mainnet"9 Z% n6 D4 ?- O! ]3 p4 Q
    [p2p]
    6 ]8 r. h+ F: q' v  [& E- {laddr = "tcp://0.0.0.0:46657"
    0 A6 {" Y& u. Rseeds = "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"+ V: @* J" `* }" M, j
    `
    * @) ]1 c1 d& f; }var testNetConfigTmpl = `chain_id = "testnet"
    4 u4 u5 K; E2 r, O- b( o4 i[p2p]
    2 k) h# x' k' mladdr = "tcp://0.0.0.0:46656"
    - Y& a% I, J8 F; nseeds = "47.96.42.1:46656,172.104.224.219:46656,45.118.132.164:46656"9 x- y& E6 f8 m" _4 {: q
    `
    - C2 S' N/ B% R4 s/ `/ B/ i+ O4 yvar soloNetConfigTmpl = `chain_id = "solonet"
    * F& m: R& G# a8 Z[p2p]+ y) N/ x! M+ v: \; z0 c; |
    laddr = "tcp://0.0.0.0:46658"
    : `/ w) y0 S) ~& j/ H4 U' K, Useeds = ""8 V0 f0 ^/ x2 A
    `% @9 z; a. y+ V! x  P' C1 V/ X- Z( i$ U
    可以看到,原来这些端口号和seed的地址,都是事先写好在模板里的。7 a$ T! d6 G5 d" `
    而且,通过观察这些配置,我们可以发现,如果chain_id不同,则监听的端口和连接的种子都不同:) \; L- T( Y3 E; v5 X' P& u: u2 R$ P
  • mainnet(连接到主网): 46657,会主动连接6个种子
  • testnet(连接到测试网): 46656,会主动连接3个种子
  • solonet(本地单独节点): 46658,不会主动连接别人(也因此不会被别人连接上),适合单机研究4 I& Z, w9 x3 M4 M( L1 o" P

    . r3 q3 {: z4 F( N" o$ g: P( M7 J写入文件
    2 E+ p3 L  `7 O# H这里我们需要快速的把bytomd init的执行流程过一遍,才能清楚配置文件的写入时机,也同时把前面的内容串在了一起。
    5 x: v& {% }+ H首先,当我们运行bytomd init时,它对应的代码入口为cmd/bytomd/main.go#L54:
    ( i% R6 R' k' W6 Xfunc main() {
    3 m9 B' u: _7 r    cmd := cli.PrepareBaseCmd(commands.RootCmd, "TM", os.ExpandEnv(config.DefaultDataDir()))
    % u8 b5 e1 a& ]    cmd.Execute()1 L  ~3 ^! c: K0 n$ @1 K: j9 L* E
    }
    " u$ v! w# B. X/ t" h其中的config.DefaultDataDir()就对应于前面提到数据目录位置。
    # \8 \* v% ^' w然后执行cmd.Execute(),将根据传入的参数init,选择下面的函数来执行:cmd/bytomd/commands/init.go#L25-L24
    1 G" d0 W: {/ Gfunc initFiles(cmd *cobra.Command, args []string) {" s1 P- e0 t& o7 [' n3 z
        configFilePath := path.Join(config.RootDir, "config.toml")
    $ W: f# ~6 k( ?$ O/ W. g    if _, err := os.Stat(configFilePath); !os.IsNotExist(err) {
    7 g7 G5 y" @' v        log.WithField("config", configFilePath).Info("Already exists config file.")/ |8 L; o4 ~7 d3 T+ N5 x
            return
    + ]$ q; c; J2 V2 Y& T* [6 a    }
    ( s) G+ _9 d/ k% |' y" y    if config.ChainID == "mainnet" {: R: X2 p2 N' |: k! L4 K
            cfg.EnsureRoot(config.RootDir, "mainnet")  k4 v) P& x/ w, v' H7 o9 }
        } else if config.ChainID == "testnet" {
    $ |5 w  R* y5 L        cfg.EnsureRoot(config.RootDir, "testnet")* C& p& N5 H9 N) W
        } else {5 [7 K& g8 ?& u# K4 ^7 l1 m# V/ g
            cfg.EnsureRoot(config.RootDir, "solonet")
    ( ^5 Q$ n0 u/ ~4 l8 {    }
    % s# t" Y+ ?# b) z; O    log.WithField("config", configFilePath).Info("Initialized bytom")
    0 B$ g; D( J1 F0 M8 i}' |) f' V* o& ]2 \/ f! z( |
    其中的configFilePath,就是config.toml的写入地址,即我们前面所说的数据目录下的config.toml文件。( Q- M9 u0 T7 u6 e* D+ H0 z. {
    cfg.EnsureRoot将用来确认数据目录是有效的,并且将根据传入的chain_id不同,来生成不同的内容写入到配置文件中。
    $ {, y+ f7 |( J它对应的代码是config/toml.go#L105 y7 ~' a$ s, e' {2 ]
    func EnsureRoot(rootDir string, network string) {
    ; e' d* k9 X: c' ^: Y9 N( @    cmn.EnsureDir(rootDir, 0700)7 e) j0 G: d* l
        cmn.EnsureDir(rootDir+"/data", 0700)
    2 Y/ f  R8 ?& U- t$ s0 P; M    configFilePath := path.Join(rootDir, "config.toml")
    ' [, X3 H3 M5 ?3 k  p5 v9 @  {7 X: H    // Write default config file if missing.
    & p& I1 y" j: ^4 L4 i. c    if !cmn.FileExists(configFilePath) {
    # h; C& c/ J- \% P# h4 Y        cmn.MustWriteFile(configFilePath, []byte(selectNetwork(network)), 0644)
    6 m. M6 W; F; z& H3 N& Z+ @    }+ U; g# z: M& ]( R- X* a" I" f
    }
    : f4 Q- {% S1 P: ^- y可以看到,它对数据目录进行了权限上的确认,并且发现当配置文件存在的时候,不会做任何更改。所以如果我们需要生成新的配置文件,就需要把旧的删除(或改名)。8 {& _6 ?, T- ^8 a4 S
    其中的selectNetwork(network)函数,实现了根据chain_id的不同来组装不同的配置文件内容,它对应于master/config/toml.go#L48:
    9 [! o0 g( x8 i  |% Qfunc selectNetwork(network string) string {
    2 b5 C7 P) R; o$ t    if network == "testnet" {
    9 G1 H2 i+ }9 o% v8 \8 Y: c; X        return defaultConfigTmpl + testNetConfigTmpl
    & w. \& Q/ W; e( c    } else if network == "mainnet" {
    ) m( F( q, J8 A% \) N9 C, ?        return defaultConfigTmpl + mainNetConfigTmpl
    % X" ]5 c) U0 {% `( j    } else {4 K( @, T+ K" @. x: `- L. p
            return defaultConfigTmpl + soloNetConfigTmpl
      E: V) J* F8 W0 z7 F    }  a& D; _7 U" _4 V+ v
    }. s* p7 Z' y3 E/ y* ~/ B9 g- F' k
    果然就是一个简单的字符串拼接,其中的defaultConfigTmpl和*NetConfgTmpl在前面已经出现,这里不重复。
    2 Q, ^, A: Q; {( m7 h! m最后调用第三方函数cmn.MustWriteFile(configFilePath, []byte(selectNetwork(network)), 0644),把拼接出来的配置文件内容以权限0644写入到指定的文件地址。/ S- ^8 S  N! F* A8 b4 J* I
    到这里,我们这个问题就算回答完毕了。% t' u& |: T' G8 b( h& b9 ~) w1 r
    感谢社区用户Freewind的辛勤写作
  • BitMere.com 比特池塘系信息发布平台,比特池塘仅提供信息存储空间服务。
    声明:该文观点仅代表作者本人,本文不代表比特池塘立场,且不构成建议,请谨慎对待。
    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    成为第一个吐槽的人

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

      0

    • 关注

      7

    • 主题

      11