btcd源码解析——从“新区块的生成”开始(2)

4.2 作为客户端的btcwallet

我们接着上一篇博客,继续来讲解作为客户端的btcwallet是如何工作的。
也即,作为客户端的btcwallet是如何与btcd进程交互的。

// walletMain[btcwallet.go] -> rpcClientConnnectLoop[btcwallet.go]
func rpcClientConnectLoop(legacyRPCServer *legacyrpc.Server, loader 
*wallet.Loader) {
...
chainClient, err = startChainRPC(certs)                     // L188
...
}

rpcClientConnectLoop函数主要通过startChainRPC函数创建并启动了一个连接btcdRPC客户端。startChainRPC函数的定义如下所示:

// walletMain[btcwallet.go] -> rpcClientConnnectLoop[btcwallet.go] -> startChainRPC[btcwallet.go]
func startChainRPC(certs []byte) (*chain.RPCClient, error) {
...
rpcc, err := chain.NewRPCClient(activeNet.Params, cfg.RPCConnect,       cfg.BtcdUsername, cfg.BtcdPassword, certs, cfg.DisableClientTLS, 0)                 // L268
...
err = rpcc.Start()                          // L272
...
}

其中L268创建了该客户端,而L272行启动了该客户端。以下分别分两小节介绍客户端的创建和启动过程。

4.2.1 创建连接btcd的RPC客户端

rpc.go中的NewRPCClient函数又调用了btcsuite套件中的infrastructure.go文件中的New函数(补充说明:btcsuite是独立于btcdbtcwallet源代码的一套共用的套件)。

// walletMain[btcwallet.go] -> rpcClientConnnectLoop[btcwallet.go] -> startChainRPC[btcwallet.go] -> NewRPCClient[rpc.go] -> New[infrastructure.go]
func New(config *ConnConfig, ntfnHandlers *NotificationHandlers) (*Client, error) {
    ...
    client := &Client{
        ...
        wsConn:          wsConn,        // L1252
        ...
    }
    
    client.start()                      // L1269
}

需要注意的是,尽管该函数的名称是新建了一个client,但client的启动其实也是在该函数中的L1269行完成的。
在L1252行初始化了一个wsConn连接,该连接用于真实向btcd发送数据,但此处的初始化其实是将一个nil值赋值给wsConn。在后面,我们将会看到对该连接的重新赋值,那时候是赋予了一个真实可用的值。
start函数的源码如下所示:

// walletMain[btcwallet.go] -> rpcClientConnnectLoop[btcwallet.go] -> startChainRPC[btcwallet.go] -> NewRPCClient[rpc.go] -> New[infrastructure.go] -> start[infrastructure.go]
func (c *Client) start() {
    ...
    go c.wsInHandler()              // L1040
    go c.wsOutHandler()             // L1041
    ...
}

L1040行和L1041行的的代码非常关键。1)L1040行的代码启动了一个协程,该协程开启了一个for循环,用于接收来自websocket连接上的数据,这里主要是接收来自btcdresponse数据(如新生成的区块的hash)。2)L1041行的代码也启动了一个for循环的协程,用于接受btcwallet内部产生的发送数据的请求,并在接收到请求后向btcd发送数据。

通常地,btcwallet会利用4.1节中的代码接收到来自btcctl发送的请求,然后通过wsOutHandler中的函数将请求发送给btcd,接着利用wsInHandler接收btcd的返回数据。
下面我们将分为两个小节分别介绍wsInHandlerwsOutHandler方法。

4.2.1.1 wsInHandler函数相关

我们首先来看一下wsInHandler函数中的

// walletMain[btcwallet.go] -> rpcClientConnnectLoop[btcwallet.go] -> startChainRPC[btcwallet.go] -> NewRPCClient[rpc.go] -> New[infrastructure.go] -> start[infrastructure.go] -> wsInHandler[infrastructure.go]
func (c *Client) wsInHandler() {
out:       
    for {                  
        ...
        _, msg, err := c.wsConn.ReadMessage()           // L412
        ...
        c.handleMessage(msg)                // L421
        ...
    }
}

L412行的代码主要用于从websocket连接中接收btcd返回的信息(比如生成的新区块的hash值),并在L421利用handleMessage方法激活之前的responseChan管道。

// walletMain[btcwallet.go] -> rpcClientConnnectLoop[btcwallet.go] -> startChainRPC[btcwallet.go] -> NewRPCClient[rpc.go] -> New[infrastructure.go] -> start[infrastructure.go] -> wsInHandler[infrastructure.go] -> handlerMessage[infrastructure.go]
func (c *Client) handleMessage(msg []byte) {
    var in inMessage
    ...
    err := json.Unmarshal(msg, &in)
    ...
    request := c.removeRequest(id)              // L357
    ...
    result, err := in.rawResponse.result()
    request.responseChan <- &response{result: result, err: err}     // L373
}

handleMessage首先通过解析msg变量生成result变量,然后在L373行中向responseChan管道中填充数据。下面我们重点来关注一下这个responseChan管道。

我们注意到request是在L357行生成的,removeRequest方法如下所示:

// 关注responseChan管道, removeRequest[infrastructure.go]
func (c *Client) removeRequest(id uint64) *jsonRequest {
    ...
    request := c.requestList.Remove(element).(*jsonRequest)         //L212
    ...
}

L212是从一个list中取出一个jsonRequest,相应地,这些jsonRequest是在addRequest方法中被加入的。addRequest方法又是在sendRequest方法中被调用的,如下所示:

// 关注responseChan管道, sendRequest[infrastructure.go]
func (c *Client) sendRequest(jReq *jsonRequest) {
    ...
    if err := c.addRequest(jReq); err != nil {        //L855
    ...
}

jReq是在RawRequestAsync函数中定义的,其在L64行对jsonRequest中的responseChan字段进行了赋值,并在L68行将responseChan作为返回值返回,如下所示:

// 关注responseChan管道, RawRequestAsync[rawrequest.go]
func (c *Client) RawRequestAsync(method string, params []json.RawMessage) FutureRawResult {
    ...
   jReq := &jsonRequest{       
        ...
        responseChan:   responseChan,               // L64
   }
   ...
   return responseChan                              // L68
}

该管道变量responseChan最终返回到了RawRequest函数中,并通过Receive方法对该管道进行接收,具体内容已经在上一篇博客中的4.1.2节进行了讲解。这样responseChan就把这两部分内容串起来了,本节对responseChan管道填充btcd的返回值,4.1.2节从responseChan中接收该值。

4.2.1.2 wsOutHandler函数相关

下面我们来看一下wsOutHandler函数中的实现细节。总地来说,wsOutHandler是利用sendChan管道接收"由4.1节接收到的btcctl发送来的请求",并将该请求进一步发送给btcd

// walletMain[btcwallet.go] -> rpcClientConnnectLoop[btcwallet.go] -> startChainRPC[btcwallet.go] -> NewRPCClient[rpc.go] -> New[infrastructure.go] -> start[infrastructure.go] -> wsOutHandler[infrastruture.go]
func (c *Client) wsOutHandler() {
out:       
    for {                  
        select {              
        case msg := <-c.sendChan:                            // L449    
            err := c.wsConn.WriteMessage(websocket.TextMessage, msg)             // L450       
            if err != nil {                           
                c.Disconnect()                           
                break out                 
            }
         ...
        }    
    }
...
}

L449行的代码被sendChan通道阻塞了,其由4.1.1节中sendMessage函数的L481行激活,其向sendChan管道中存放的是从btcctl发送来的请求,前面也已经介绍过。
L450行的代码完成了向btcd发送数据的任务。发送数据借助于wsConn这个websocket连接。

4.2.2 启动连接btcd的RPC客户端

回到startChainRPC函数中的L272行,我们来看一下在btcwallet中用于连接btcd的客户端是如何启动的。RPC客户端的start函数定义如下所示:

// walletMain[btcwallet.go] -> rpcClientConnnectLoop[btcwallet.go] -> startChainRPC[btcwallet.go] -> Start[rpc.go]
func (c *RPCClient) Start() error {
    err := c.Connect(c.reconnectAttempts)           // L101
    ...
}

Start函数调用了RPC客户端的Connect函数, 该函数真实建立了和btcdwebsocket连接,也即对RPC客户端中的wsConn变量进行了重新赋值。

// walletMain[btcwallet.go] -> rpcClientConnnectLoop[btcwallet.go] -> startChainRPC[btcwallet.go] -> Start[rpc.go] -> Connect[infrastructure.go]
func (c *Client) Connect(tries int) error {
...
wsConn, err = dial(c.config)            // L1309
...
c.wsConn = wsConn                   // L1324
...
}

在L1324行对wsConn变量重新进行了赋值。

5. btcd中的相关代码

同样地,为避免这一篇博客过长,第5节的内容将放在下一篇博客中。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
2008年爆发全球金融危机,同年11月1日,一个自称中本聪(Satoshi Nakamoto)的人在P2P foundation网站上发布了比特币白皮书《比特币:一种点对点的电子现金系统》 [6]  ,陈述了他对电子货币的设想——比特币就此面世。2009年1月3日,比特币创世区块诞生。 和法定货币相比,比特币没有一个集中的发行方,而是由网络节点的计算生成,谁都有可能参与制造比特币,而且可以全世界流通,可以在任意一台接入互联网的电脑上买卖,不管身处何方,任何人都可以挖掘、购买、出售或收取比特币,并且在交易过程中外人无法辨认用户身份信息。2009年1月5日,不受央行和任何金融机构控制的比特币诞生。比特币是一种数字货币,由计算机生成的一串串复杂代码组成,比特币通过预设的程序制造。 每当比特币进入主流媒体的视野时,主流媒体总会请一些主流经济学家分析一下比特币。早先,这些分析总是集中在比特币是不是骗局。而现如今的分析总是集中在比特币能否成为未来的主流货币。而这其中争论的焦点又往往集中在比特币的通缩特性上。 [7]  不少比特币玩家是被比特币的不能随意增发所吸引的。和比特币玩家的态度截然相反,经济学家们对比特币2100万固定总量的态度两极分化。 凯恩斯学派的经济学家们认为政府应该积极调控货币总量,用货币政策的松紧来为经济适时的加油或者刹车。因此,他们认为比特币固定总量货币牺牲了可调控性,而且更糟糕的是将不可避免地导致通货紧缩,进而伤害整体经济。奥地利学派经济学家们的观点却截然相反,他们认为政府对货币的干预越少越好,货币总量的固定导致的通缩并没什么大不了的,甚至是社会进步的标志。 比特币网络通过“挖矿”来生成比特币。所谓“挖矿”实质上是用计算机解决一项复杂的数学问题,来保证比特币网络分布式记账系统的一致性。比特币网络会自动调整数学问题的难度,让整个网络约每10分钟得到一个合格答案。随后比特币网络会生成一定量的比特币作为区块奖励,奖励获得答案的人。 [6]  2009年,比特币诞生的时候,区块奖励是50个比特币。诞生10分钟后,第一批50个比特币生成了,而此时的货币总量就是50。随后比特币就以约每10分钟50个的速度增长。当总量达到1050万时(2100万的50%),区块奖励减半为25个。当总量达到1575万(产出525万,即1050的50%)时,区块奖励再减半为12.5个。该货币系统曾在4年内只有不超过1050万个,之后的总数量将被永久限制在约2100万个。 [3]  [8]  比特币是一种虚拟货币,数量有限,但是可以用来套现:可以兑换成大多数国家的货币。你可以使用比特币购买一些虚拟的物品,比如网络游戏当中的衣服、帽子、装备等,只要有人接受,你也可以使用比特币购买现实生活当中的物品。 2014年2月25日,“比特币中国”的比特币开盘价格为3562.41元,截至下午4点40分,价格已下跌至3185元,跌幅逾10%。根据该平台的历史行情数据显示,在2014年1月27日,1比特币还能兑换5032元人民币。这意味着,该平台上不到一个月,比特币价格已下跌了36.7%。 同年9月9日,美国电商巨头eBay宣布,该公司旗下支付处理子公司Braintree将开始接受比特币支付。该公司已与比特币交易平台Coinbase达成合作,开始接受这种相对较的支付手段。 虽然eBay市场交易平台和PayPal业务还不接受比特币支付,但旅行房屋租赁社区Airbnb和租车服务Uber等Braintree客户将可开始接受这种虚拟货币。Braintree的主要业务是面向企业提供支付处理软件,该公司在2013年被eBay以大约8亿美元的价格收购。 2017年1月22日晚间,火币网、比特币中国与OKCoin币行相继在各自官网发布公告称,为进一步抑制投机,防止价格剧烈波动,各平台将于1月24日中午12:00起开始收取交易服务费,服务费按成交金额的0.2%固定费率收取,且主动成交和被动成交费率一致。 [9]  5月5日,OKCoin币行网的数据显示,比特币的价格刚刚再度刷历史,截止发稿前高触及9222元人民币高位。1月24日中午12:00起,中国三大比特币平台正式开始收取交易费。9月4日,央行等七部委发公告称中国禁止虚拟货币交易。同年12月17日,比特币达到历史高价19850美元。 2018年11月25日,比特币跌破4000美元大关,后稳定在3000多美元。 [10]  11月19日,加密货币恢复跌势,比特币自2017年10月以来首次下探5000美元大关,原因是之前BCH出现硬分叉,且监管部门对首次代币发行(ICO)加强了审查。 [10]  11月21日凌晨4点半,coinbase平台比特币报价跌破4100美元,创下了13个月以来的低。 2019年4月,比特币再次突破5000美元大关,创年内高。 [11]  5月12日,比特币近八个月来首次突破7000美元。 [12]  5月14日,据coinmarketcap报价显示,比特币站上8000美元,24小时内上涨14.68%。 [13]  6月22日 ,比特币价格突破10000美元大关。比特币价格在10200左右震荡,24小时涨幅近7%。 [14]  6月26日,比特币价格一举突破12000美元,创下自去年1月来近17个月高点。 [15]  6月27日早间,比特币价格一度接近14000美元,再创年内高。 [16]  2020年2月10日,比特币突破了一万美元。据交易数据,比特币的价格涨幅突破3% [17]  。3月12日,据加密货币交易平台Bitstamp数据显示,19点44分,比特币低价格已跌至5731美元 [18]  。5月8日,比特币突破10000美元关口,创下2月份以来的高 [19]  。5月10日早上8点开始,比特币单价在半小时内从9500美元价位瞬间下跌了上千美元,低价格跌破8200美元,高价差超1400美元 [20]  。7月26日下午6点,比特币短时极速拉升,高触及10150.15USDT,日内大涨幅超过4%,这是2020年6月2日以来首次突破1万美元关口 [21]  。11月4日,比特币价格正式突破14000美元 [22]  。11月12日晚,比特币价格突破16000美元,刷2018年1月以来高,一周涨超8.6%。比特币总市值突破2915亿美元 [23]  。11月18日,比特币价格突破17000美元 [24]  。12月1日,比特币价格报19455.31美元,24小时涨幅为5.05%。 [25]  12月17日,比特币价格突破23000美元整数关口,刷历史高,日内涨幅超7.5%。 [26]  截至12月27日19时20分,比特币报价28273.06美元。 [27]  2021年1月8日,比特币涨至4万美元关口上方,高至40402美元

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值