初始化生成配置文件

人们常说,“阅读源代码”是学习编程的一种重要方法。作为程序员,我们在平时的学习工作中,都应该阅读过不少源代码。但是对于大多数人来说,阅读的可能更多是一些代码片断、示例,或者在老师、同事的指导下,先对要阅读的项目代码有了整体的了解之后,再进行针对性的阅读。

但是如果我们面对的是一个像比原这样比较庞大的项目,身边又没有人指导,只能靠自己去看,这时应该怎么来阅读呢?也许每个人也都能找到自己的办法,或高效,或低效,或放弃。

我在这次阅读比原源代码的过程中,尝试的是这样一种方法:从外部入手,通过与比原节点进行数据交互,来一步步了解比原的内部原理。就像剥石榴一样,一点点小心翼翼的下手,最后才能吃到鲜美的果肉。

所以这个文章系列叫作“剥开比原看代码”。

说明

在系列中的每一章,我通常都会由一个或者几个相关的问题入手,然后通过对源代码进行分析,来说明比原的代码是如何实现的。对于与当前问题关系不大的代码,则会简单带过,等真正需要它们出场的时候再详细解说。

为了保证文章中引用代码的稳定性,我将基于比原的v1.0.1代码进行分析。随着时间推移,比原的代码也将快速更新,但是我觉得,只要把这个版本的代码理解了,再去看新的代码,应该是一件很容易的事情。

在文章中,将会有一些直接指向github上bytom源代码的链接。为了方便,我专门将bytom v1.0.1的代码放到了一个新的仓库中,这样就不容易与比原官方的最新代码混淆。该仓库地址为:https://github.com/freewind/bytom-v1.0.1

当然,你不必clone这个仓库(clone官方仓库http://github.com/Bytom/bytom就够了),然后在必要的时候,使用以下命令将代码切换到v1.0.1的tag,以便与本系列引用的代码一致:

git fetch

git checkout -b v1.0.1

不论采用哪种阅读方法,我想第一步都应该先在本地把比原节点跑起来,试试各种功能。

对于如何下载、配置和安装的问题,请直接参看官方文档https://github.com/Bytom/bytom/tree/v1.0.1(注意我这里给出的是v1.0.1的文档),这里不多说。

本篇问题

当我们本地使用make bytomd编译完比原后,我们可以使用下面的命令来进行初始化:

./bytomd init --chain_id testnet

这里指定了使用的chain是testnet(还有别的选项,如mainnet等等)。运行成功后,它将会在本地文件系统生成一些配置文件,供比原启动时使用。

所以我的问题是:

比原初始化时,产生了什么样的配置文件,放在了哪个目录下?

下面我将结合源代码,来回答这个问题。

目录位置

首先比原在本地会有一个目录专门用于放置各种数据,比如密钥、配置文件、数据库文件等。这个目录对应的代码位于config/config.go#L190-L205

funcDefaultDataDir()string{// Try to place the data folder in the user's home dirhome:=homeDir()dataDir:="./.bytom"ifhome!=""{switchruntime.GOOS{case"darwin":dataDir=filepath.Join(home,"Library","Bytom")case"windows":dataDir=filepath.Join(home,"AppData","Roaming","Bytom")default:dataDir=filepath.Join(home,".bytom")}}returndataDir}

可以看到,在不同的操作系统上,数据目录的位置也不同:

苹果系统(darwin):~/Library/Bytom

Windows(windows): ~/AppData/Roaming/Bytom

其它(如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,预定义了不同的模板内容:

vardefaultConfigTmpl=`# 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"

`varmainNetConfigTmpl=`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"

`vartestNetConfigTmpl=`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"

`varsoloNetConfigTmpl=`chain_id = "solonet"

[p2p]

laddr = "tcp://0.0.0.0:46658"

seeds = ""

`

可以看到,原来这些端口号和seed的地址,都是事先写好在模板里的。

而且,通过观察这些配置,我们可以发现,如果chain_id不同,则监听的端口和连接的种子都不同:

mainnet(连接到主网): 46657,会主动连接6个种子

testnet(连接到测试网): 46656,会主动连接3个种子

solonet(本地单独节点): 46658,不会主动连接别人(也因此不会被别人连接上),适合单机研究

写入文件

这里我们需要快速的把bytomd init的执行流程过一遍,才能清楚配置文件的写入时机,也同时把前面的内容串在了一起。

首先,当我们运行bytomd init时,它对应的代码入口为cmd/bytomd/main.go#L54:

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

其中的config.DefaultDataDir()就对应于前面提到数据目录位置。

然后执行cmd.Execute(),将根据传入的参数init,选择下面的函数来执行:cmd/bytomd/commands/init.go#L25-L24

funcinitFiles(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}ifconfig.ChainID=="mainnet"{cfg.EnsureRoot(config.RootDir,"mainnet")}elseifconfig.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

funcEnsureRoot(rootDirstring,networkstring){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:

funcselectNetwork(networkstring)string{ifnetwork=="testnet"{returndefaultConfigTmpl+testNetConfigTmpl}elseifnetwork=="mainnet"{returndefaultConfigTmpl+mainNetConfigTmpl}else{returndefaultConfigTmpl+soloNetConfigTmpl}}

果然就是一个简单的字符串拼接,其中的defaultConfigTmpl和*NetConfgTmpl在前面已经出现,这里不重复。

最后调用第三方函数cmn.MustWriteFile(configFilePath, []byte(selectNetwork(network)), 0644),把拼接出来的配置文件内容以权限0644写入到指定的文件地址。

到这里,我们这个问题就算回答完毕了。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,080评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,422评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,630评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,554评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,662评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,856评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,014评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,752评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,212评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,541评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,687评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,347评论 4 331
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,973评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,777评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,006评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,406评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,576评论 2 349

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,633评论 18 139
  • 但她的勇敢却不是时时可以给她力量的,反而像敏感一样,用力过猛就会自伤,他们之间的最后一面是那样不堪入目,连接着彼此...
    121跑起来阅读 419评论 2 0
  • 第一步. 下载 首先打开python官网:https://www.python.org/ 这里以win7X6...
    小飞牛_666阅读 2,639评论 0 2
  • 终于断断续续地用一个礼拜的时间看完了BENJAMIN写的《地下室》。 BENJAMIN是个不羁的漫画...
    墨香黎阅读 223评论 0 1
  • 近日,崔龙洙面对媒体采访,直言不讳的说,希望在足协杯决赛中战胜恒大队夺冠,更放言可以在明年亚冠上要战胜自己的前东家...
    王子和白龙马阅读 230评论 1 0