Zhuang's Diary

言之有物,持之以恒

2020年1月21日,国际清算银行(BIS)、加拿大央行、英国央行、日本央行、欧央行、瑞典央行和瑞士央行官网同时发布一则信息:成立央行小组,开展CBDC应用案例研发。

加拿大

加拿大央行与瑞典央行已进入CBDC的试验测试阶段:2016年加拿大银行和新加坡金融管理局分别启动了Jasper和Ubin项目,探索分布式账户技术(DLT)在银行间大额支付系统的应用。双方都采用了R3联盟的DLT技术,开发了国内银行间支付结算的批发型数字货币原型系统,并经过几个阶段的后续试验验证,但仅限于在国内尝试。2019年加拿大银行和新加坡金管局合作开发Jasper-Ubin项目,在没有中介代理的环境下,实现了跨境、跨币种和跨平台支付中应用CBDC的试验成功。可以说在批发型CBDC应用于银行间大额支付结算和跨境支付方面,加拿大央行已经进入了概念验证试验的阶段,走在国际前列。

瑞典

瑞典央行的情况与加拿大不同,其无现金化趋势和速度为全球关注。发行CBDC“电子克朗“的目的是确保国家货币发行权、维护国内支付结算体系稳定发展、让金融服务普惠社会各类群体。从2017年启动电子克朗研发计划到目前进入测试阶段。它着眼的方向是零售型CBDC。

英国央行与欧央行对CBDC进行了原型设计与概念验证:笔者前文《英国:发行数字货币,还是发行塑料钞,这是一个问题》对英国开展数字货币的研发有介绍。简单说,英国作为全球首个央行数字货币模型RSCoin的创始者,虽然后来没有像加拿大、瑞典那样有更实质性的动作,但研究的步子一直没有停顿。加新两国的联合项目,也有英国的参与:2018年下,英格兰银行和加拿大央行、新加坡金管局以及4家商业银行、2家第三方机构参与了Jasper项目第4阶段的研究活动,对解决跨境支付存在的挑战和摩擦,创建交易新模式提出了建议,为2019年加新两国的DLT系统对接奠定了研究基础。此外,英国央行前行长马克·卡尼提出了一个“合成霸权货币”的概念,意欲以此取代目前以美元为主导的国际货币组合。

欧央行

对CBDC的研发早已做了大量准备工作,拉加德就任行长以来,欧央行在CBDC方面的研发大幅度提速,最近在它主导下,十几个欧盟国家央行与埃森哲和R3联盟合作,开发设计了一种基于DLT的“欧洲链”系统,解决CBDC小额支付中隐私与反洗钱的平衡问题。并经过概念验证。

日本

日本央行与瑞士央行暂处于CBDC理论研究阶段:日本是一个现金偏好明显的国家,80%日常购物仍使用现金(《日本时报》,2019)。甚至在2019政府为抵销消费税上涨而推出的非现金支付积分奖励情况下,民调50-70岁以上的人仍有31%至 54%无意使用非现支付手段(《每日新闻》,2019)。显然缺乏开展零售型CBDC研发的动机。天秤币发行计划推出后,日本官方态度保持低调,直到12月央行行长黑田东彦还称,当前没有推出CBDC的需要。但最近,其副行长雨宫正佳(Masayoshi Amamiya)表示:“在日本,公众对CBDC的需求可能会激增,这取决于结算系统的发展情况。如果发生这种情况,我们必须做好应对的准备。”

去年底央行发表了一个有关CBDC法律问题的报告,对CBDC发行后给现行私法和刑法、日本银行法、数据收集法、行政法和竞争法等法律领域带来的影响进行了探讨,是目前所见CBDC研究中讨论法律问题最为全面、深入的报告,说明准备工作并没有停顿。

瑞士

瑞士情况和日本相似,公众有强烈的现金偏好,即使是年轻人如此。其他国家纷纷出于反洗钱和反恐融资等目的而取消大面额现钞,降低现金消费限额,而瑞士法郎仍然保留最大面额1000瑞郎,现金交易上限是10万瑞郎。对于私人加密货币的态度它也和日本一样比较友好,有众多加密货币机构在瑞士设立。央行高管曾表态数字货币还是私有化比较合适,去年底瑞士联邦委员会仍称CBDC目前不会给瑞士带来好处,相反将引发金融稳定等风险。

美国

DIGITAL DOLLARS: ELECTRONIC CASH VS DIGITAL CURRENCY
New technological capabilities have created a dynamic, via tokenization, that allows for a digital version of central bank currency, without sacrificing stability, security or privacy. This is about more than a federal government payments infrastructure.

In addition to coins and paper money, a dollar-based Central Bank Digital Currency (CBDC) could operate as a third form of money backed by the full faith and credit of the United States of America, but brings it into the truly portable digital world, opening up new possible use cases and applications.

Considering the importance of the U.S. dollar to the global economy, our work on a U.S. dollar-based CBDC via the Digital Dollar Project is a logical and important step as we drive towards a digital transformation of the global financial system.

The Digital Dollar Project will be driven by an open, innovative, collaborative mindset, which will explore the full range of possibilities on how to best bring this to fruition

数字美元:电子现金与数字货币
新的技术通过Token化创建了一种动态的功能,该功能允许在不牺牲稳定性,安全性或私密性的情况下使用数字版本的中央银行货币。这远远超出了美国联邦政府的支付基础设施。

除硬币和纸币外,以美元为基础的中央银行数字货币(CBDC)可以作为第三种形式的货币,以美利坚合众国的充分信仰和信誉为后盾,但将其带入真正的便携式数字世界,打开新的可能的用例和应用程序。

考虑到美元对全球经济的重要性,在我们朝着全球金融体系的数字化转型迈进的过程中,我们通过“数字美元项目”在以美元为基础的CBDC方面的工作是合乎逻辑且重要的一步。

数字美元项目将以开放,创新,协作的心态驱动,它将探索各种方式来最大程度地实现这一目标。

和中国CBDC的设计几乎完全一致。本白皮书内容中并未提及”blockchain”。白皮书作者为Digital Dollar Project办公室与埃森哲

中国

中国人民银行拟推出的CBDC本质上是法定货币人民币的数字化版本,由法币储备(M0)按照1比1的比例进行兑换。

“从央行角度来讲,无论你是区块链还是集中账户体系,是电子支付还是所谓的移动货币,你采取任何一种技术路线,央行都可以适应。当然,你的技术路线要符合我们的门槛,比如因为是针对零售,至少要满足高并发需求,至少达到30万笔/秒。” 穆长春在会议上说到。

易纲在新闻发布会上表示,人民银行把数字货币和电子支付工具结合起来,将推出一揽子计划,目标是替代一部分现金。

作为理论上无成本的交换媒介,CBDC将能够提高支付系统的效率。 同时,在跨境金融交易的情况下,CBDC可以促成更快,更安全的结算,并允许数字金融服务领域进行更多的创新。此外,CBDC的推出还有助于打击假币,因为每个令牌和交易都需要由集中式系统进行实时的密码验证。

农业银行的PoC如下:

姚前

2020-04刊文

目前大多数国家的央行数字货币(Central Bank Digital Currency,CBDC)实验都是基于区块链技术展开的。但时至今日,CBDC是否采用区块链技术依然存有争议,一种典型的观点是区块链的去中心化与中央银行的集中管理存在冲突,因此不建议CBDC采用该技术。笔者认为,区块链技术正以前所未有的速度在发展,并与各项主流技术在深度融合,因此无论从技术角度还是业务角度,现实应用中的区块链都与“原教旨主义”的理解有所不同。如何运用区块链技术来更好的服务于中心化管理下的分布式运营,可能是CBDC当前需要重点探索的方向。本文以三个典型场景为例,讨论了区块链在CBDC中的可能应用和解决方案,指出虽然区块链的技术特点是不依赖中心机构,但不代表其不能纳入到现有中心机构的体系内,只要通过合理的设计,中央银行恰恰可以利用区块链将分布式运营有效整合起来,更好地实现对CBDC的中心化管控,两者并不存在必然冲突。

场景一:CBDC验钞

笔者曾提出“一币、两库、三中心”的央行数字货币体系。“一币”即央行数字货币,是由央行担保并签名发行的代表具体金额的加密数字串。“两库”指数字货币发行库和数字货币商业银行库,前者是中央银行在CBDC私有云上存放CBDC发行基金的数据库,按照中央银行的现金运营管理体系进行管理,后者是商业银行存放CBDC的数据库,可以在商业银行的数据中心也可以在CBDC私有云上,遵循商业银行现金运营管理规范。“三中心”则包括认证中心、登记中心和大数据分析中心。

其中,登记中心记录CBDC及对应用户身份,完成权属登记,并记录流水,完成CBDC产生、流通、清点核对及消亡全过程登记。其主要功能组件分为发行登记、确权发布、确权查询网站应用、分布式账本服务几个部分。发行登记进行CBDC的发行、流通、回笼过程及权属记录;确权发布将发行登记的权属信息进行脱敏后异步发布到CBDC确权分布式账本中;确权查询网站依托分布式账本面向公众提供在线权属查询服务;分布式账本服务保证中央银行与商业银行的CBDC权属信息的一致。

通俗来说,可以理解为我们在登记中心利用分布式账本不可篡改、不可伪造特性,构建了一个“网上验钞机”,即CBDC确权账本,对外通过互联网提供查询服务。这种设计对当前分布式账本技术而言,在中央银行和商业银行既集中又分散的二元模式下,提供了一种巧妙的应用思路,一方面将核心的发行登记账本对外界进行隔离和保护,同时利用分布式账本优势,提高确权查询数据和系统的安全性和可信度;另一方面,由于分布式账本仅用于对外提供查询访问,交易处理仍由发行登记系统来完成,以细化原子交易颗粒度的方式来进行交易的分布式计算处理,这样可以通过业务设计的方式有效规避现有分布式账本在交易处理上的技术性能瓶颈。显然,这样的设计充分发挥了区块链的技术优势,保障CBDC验钞的可信,但并未影响中央银行对CBDC的全局管控。

尤其是,这种双账本包容性设计,既延续了传统技术的成熟稳定性,又为新的分布式账本技术留有空间,使得两种分布式技术相互兼容、并行不悖、优势互补,并在演进过程中,竞争择优。

场景二:批发端支付结算

目前各国正在开展的CBDC实验,主要针对批发端场景,且大多基于区块链技术。比如,加拿大的 Jasper 项目,试验基于区块链技术的大额支付系统;新加坡的Ubin项目,评估在分布式账本上以数字新元的代币形式进行支付结算的效果;欧洲央行和日本央行的Stella项目,旨在研究分布式账本技术(DLT)在金融市场基础设施中的应用,评估现有支付体系的特定功能是否能够在DLT环境下安全高效地运转。还有中国香港的LionRock项目、泰国的Inthanon项目等均是试验基于区块链技术的CBDC。这些区块链技术的应用都在中央银行的集中管理和严格控制下展开。

以新加坡的Ubin项目为例,其采用了与加拿大Jasper项目一样的数字存托凭证(Digital Deposit Receipt,DDR)模式。为了支持分布式账本中DDR的发行,现有新加坡电子支付系统(MEPS +),也就是新加坡的RTGS系统,专门建立一个DDR资金抵押账户,每日开始时,参与银行请求中央银行将其RTGS账户中的资金转移到DDR资金抵押账户,以此作为抵押,分布式账本创建相应等值的DDR,发送到各银行的DDR钱包,由此参与银行之间可开展基于分布式账本的转账和支付。日终,分布式账本系统将向MEPS+发送一个网络结算文件,MEPS+依此调整DDR资金抵押账户余额,匹配参与者在DLT网络中的DDR余额。

可见,去中心化的分布式账本与现有成熟的中央主导的金融基础设施并不排斥,完全可以相互融合,相互补充。一方面,基于区块链的DDR支付系统为现有RTGS系统提供了一种不依赖传统账户的新型支付方式,有效补充了现有支付清算体系。另一方面,DDR作为RTGS中电子化法定货币的数字化形态延伸,其最终可以转换回RTGS账户价值,并通过RTGS系统对外结算,也就是说,RTGS系统解决了区块链DDR到传统账户资金的结算最终性问题,这侧面也说明了区块链的结算最终性可以有机融合到现有清结算体系中。此外,由于DDR 是通过100%资金抵押生成,不影响货币供应量,因此分布式账本也不会影响到中央银行对货币的总量管控。

显然,在技术逻辑上,中央银行主导的基于区块链的新型支付系统是完全可行的。某种意义上,参照Ubin项目的数字存托凭证模式,可以无需借助类似网联支付平台这样的中间渠道,各家支付机构和商业银行可以通过在金融专网中构建对等网络的方式,以统一的区块链网络连接起来,开展支付清算。考虑到目前区块链技术的交易性能还在演进的过程中,上述清算业务宜在批发层面展开。

应该说,区块链的去中心化是指去中介,但不去监管。在联盟链的环境下,中央银行等监管部门不但可以对区块链所承载的业务及其风险进行中心化管控,而且还可以实现穿透式非现场监管。

场景三:现金数字化

似乎现金的数字化与准备金的数字化(即前述的数字存托凭证)没有本质上的差别,只是前者面向社会公众,而后者仅局限于银行间流通,但面向社会公众就引发了一个难题,倘若允许公众在中央银行开户,中央银行将面临着极大的服务压力,并可能引发存款搬家,导致狭义银行。

一种解决思路是100%备付准备金模式。代理运营机构向中央银行存缴100%备付准备金,随后在其账本上发行相应数额的数字货币可视为央行数字货币。IMF经济学家把它称为合成央行数字货币(sCBDC)。据此,我国第三方支付机构100%备付准备金存缴中央银行之后,它们虚拟账户中的资金就是央行数字货币了。若此,则中国早就是全球首个实现法币数字化的泱泱大国。

但仔细琢磨,这一思路存在着缺陷:一是技术上,100%准备金存缴意味着数字货币的发行、流通、回笼等全生命周期均要依附于传统账户体系,尤其是跨机构CBDC的流通,除了CBDC账本更新外,还要处理相应准备金账户间的清结算,只能牺牲系统灵活性,加以额度管控的方式去应对,而且还需要成立专门的清算机构提供互联互通服务。这不仅增大中央银行中心系统的压力和复杂性,也就是说,还是没有解决央行的服务压力,而且难以实现“账户松耦合”的要求;二是管理上,这种方式央行和运营机构在发行流通过程中是紧绑定的,央行依然承担中心化压力。如何保证代理运营机构100%备付准备金后没有超发货币,尤其是当代理运营机构运营的支付网络不受中心化管控时,中央银行更难以掌控运营代理层的货币发行量,这在一定程度上也构成了某些反对区块链技术应用于CBDC的理由。

视角决定思路,如果换一个角度看,会得到另一种完全不同的更优的解决方案。现在提到CBDC,许多人是自顶向下,从中央银行发行到商业银行,再从商业银行发行到个人的视角来理解CBDC的技术逻辑,所以总有一个乱发票子的担忧。实物货币受制于印钞造币环节,非如此不可,但数字货币的“印钞造币”可以瞬间完成,无需这种制约,而这才是其优势所在。如果以自底向上的视角看,可以惊讶地发现,数字货币最终用户并没有“发行”的概念,而是“兑换”的理念,是手里有多少现金,有多少存款,去兑换CBDC。所以从这角度看,乱发票子的问题并没有那么突出,代理运营机构兑换出的CBDC,不是中央银行给与的货币发行额度,而是用户用实实在在的真金白银等额兑换的结果,中央银行只是站在全局的角度统计相关信息并予以监管。实际上,目前无论是私人的稳定代币,还是各国研发的CBDC,都是按需兑换的思路,而不是扩表发行,这是一个非常关键的点。这一点对货币政策而言,意义重大,表明其没有根本性的变化;对于技术路线而言,意味着可以不拘泥于实物货币的发行流程,系统的设计可以更为简洁,局面因此大为改观。

基于自底向上的兑换视角,可以提出一个CBDC简化版实现方案。具体思路是:业务由底层客户发起,客户申请兑换CBDC并将其托管至代理运营机构。代理运营机构记录客户托管CBDC的明细账本,为每个托管客户单独建立明细账。代理运营机构收到客户兑换并托管CBDC请求后,在收取现金或扣减客户存款的同时,将等额CBDC记录在该客户明细账下,然后向中央银行缴回现金或扣减存款准备金,并以批量方式混同托管至中央银行。中央银行记录代理运营机构的总账本,是一个总量的概念,与代理运营机构的明细账本构成上下两级双账本结构。当同一家代理运营机构的客户之间发生CBDC支付时,只需在该机构的明细账本上变更权属,无需变更中央银行总账本。当发生跨代理运营机构的CBDC支付时,首先由相关的代理运营机构交互处理,在各自明细账本上完成CBDC的权属变更,然后由中央银行在总账本上定期批量变更各机构总账。为提高效率,减少风险,可考虑引入持续净额头寸调整、流动性节约(LSM)等机制。

这一方案有以下优点:一是明确了持有者对CBDC具有完全掌控权。未经持有者的签名或同意,其他任何主体均不能动用CBDC。这就使CBDC真正具备现金属性,与存款类货币本质不同。二是央行不对底层客户单独建档,也就是说,普通公众不在中央银行“开户”,降低了中央银行的服务压力,同时真正实现了“账户松耦合”的要求,因准备金账户批量调整,CBDC系统相对独立于RTGS系统。三是各家代理运营机构可以根据自己的理解,在满足统一标准的基础上,发挥各自特长构建自身的数字货币代理运营系统,有助于竞争,便于客户选择。由于是按需兑换,而不是扩表发行,因此就没有了运营代理层超发货币的担忧。另外,虽然底层客户交易信息只存储在中间层,不存储在中央银行账本上,出于政策需要或监管需要,中央银行有权向下一层的代理运营机构提取信息明细,从而在分布式运营条件下实现了中心化管控。

结语

区块链作为一种可能成为未来金融基础设施的新兴技术,对于中央银行和商业银行二元模式而言,有助于实现分布式运营,同时并不会影响集中管理。本文通过三个典型场景进一步论证了区块链技术的去中心化特点可以纳入到CBDC的分布式运营与央行的集中管理体系中。可将区块链技术应用于CBDC的登记账本,对CBDC验钞,保障可信。在批发端场景,各国开展的实验也表明,基于区块链技术的CBDC和支付系统具有可行性。而在现金数字化的零售场景,本文认为之所以目前CBDC研发方案一直无法发挥出央行中心管控下的分布式运营应有的优势,问题在于自顶向下的“发行”视角,对此,本文基于自底向上的“兑换”视角,提出了全新的CBDC实现方案,这一方案同时实现了“管控中心化,运营分布式”的目标。

“物物而不物于物”,“形而上者之为道,形而下者之为器,以道御器”,这是我国古代哲人的思想。集中管理与分布处理历来需要辩证统一地看待,不宜“先入为主”地将制度层面的中心化管控与技术层面的分布式处理简单对立起来。当前,各国基于区块链技术的央行数字货币实验进展迅速,内容已涉及隐私保护、数据安全、交易性能、身份认证、券款对付、款款对付等广泛议题。作为一项崭新的技术,区块链当然还有这样那样的缺点与不足,但这不是我们轻言放弃的理由。Facebook的Libra项目已在研发基于安全、可扩展和可靠区块链的新一代金融基础设施,这是一个全新的赛道,机遇与挑战并存,“逆水行舟,不进则退”。

CheckTx

在burrow项目中,/consensus/abci/app.go:

1
2
3
4
5
func (app *App) CheckTx(req types.RequestCheckTx) types.ResponseCheckTx {
......
checkTx := ExecuteTx(logHeader, app.checker, app.txDecoder, req.GetTx())
......
}

此处的 ExecuteTx 位于 /execution/execution.go :

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
// If the tx is invalid, an error will be returned.
// Unlike ExecBlock(), state will not be altered.
func (exe *executor) Execute(txEnv *txs.Envelope) (txe *exec.TxExecution, err error) {
......
// Verify transaction signature against inputs
err = txEnv.Verify(exe.params.ChainID)
if err != nil {
logger.InfoMsg("Transaction Verify failed", structure.ErrorKey, err)
return nil, err
}

if txExecutor, ok := exe.contexts[txEnv.Tx.Type()]; ok {
// Establish new TxExecution
txe := exe.block.Tx(txEnv)
defer func() {
if r := recover(); r != nil {
err = fmt.Errorf("recovered from panic in executor.Execute(%s): %v\n%s", txEnv.String(), r,
debug.Stack())
}
}()

err = exe.validateInputsAndStorePublicKeys(txEnv)
if err != nil {
logger.InfoMsg("Transaction validate failed", structure.ErrorKey, err)
txe.PushError(err)
return nil, err
}

err = txExecutor.Execute(txe, txe.Envelope.Tx.Payload)
if err != nil {
logger.InfoMsg("Transaction execution failed", structure.ErrorKey, err)
txe.PushError(err)
return nil, err
}

// Increment sequence numbers for Tx inputs
err = exe.updateSequenceNumbers(txEnv)
if err != nil {
logger.InfoMsg("Updating sequences failed", structure.ErrorKey, err)
txe.PushError(err)
return nil, err
}
// Return execution for this tx
return txe, nil
}
return nil, fmt.Errorf("unknown transaction type: %v", txEnv.Tx.Type())
}

此处 txExecutor.Execute(txe, txe.Envelope.Tx.Payload)txExecutor 类型的不同,分别对应至 /txs/payload/payload.go 中的 Type ,如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// Account transactions
TypeSend = Type(0x01)
TypeCall = Type(0x02)
TypeName = Type(0x03)
TypeBatch = Type(0x04)

// Validation transactions
TypeBond = Type(0x11)
TypeUnbond = Type(0x12)

// Admin transactions
TypePermissions = Type(0x21)
TypeGovernance = Type(0x22)
TypeProposal = Type(0x23)
TypeIdentify = Type(0x24)

根据 Type 不同,在如下 contexts 对应的 Context 的 Execute 方法则会不同,分别对应为如下代码文件中的 Execute 方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
├── burrow
│ ├── execution
│ │ ├── contexts
│ │ │ ├── bond_context.go
│ │ │ ├── bond_context_test.go
│ │ │ ├── call_context.go
│ │ │ ├── governance_context.go
│ │ │ ├── identify_context.go
│ │ │ ├── name_context.go
│ │ │ ├── permissions_context.go
│ │ │ ├── proposal_context.go
│ │ │ ├── send_context.go
│ │ │ ├── shared.go
│ │ │ └── unbond_context.go

例如,在执行发送 TypeCall 类型的交易时, txExecutor.Execute(txe, txe.Envelope.Tx.Payload) 对应的具体逻辑为 /execution/contexts/call_context.go 中的 Execute 方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
func (ctx *CallContext) Execute(txe *exec.TxExecution, p payload.Payload) error {
var ok bool
ctx.tx, ok = p.(*payload.CallTx)
if !ok {
return fmt.Errorf("payload must be CallTx, but is: %v", p)
}
ctx.txe = txe
inAcc, outAcc, err := ctx.Precheck()
if err != nil {
return err
}
// That the fee less than the input amount is checked by Precheck to be greater than or equal to fee
value := ctx.tx.Input.Amount - ctx.tx.Fee

if ctx.RunCall {
return ctx.Deliver(inAcc, outAcc, value)
}
return ctx.Check(inAcc, value)
}

DeliverTx

在执行 ExecuteTx(logHeader, app.committer, app.txDecoder, req.GetTx()) 中,executor execution.Executor 被指定为 app.committer , 与 CheckTX 执行同样的 Execute 逻辑

Commit

具体执行逻辑为 /execution/execution.go 中的 func (exe *executor) Commit(header *abciTypes.Header) (stateHash []byte, err error)

./burrow natives , Dump Solidity interface contracts for Burrow native contracts.

抽出Burrow内置的solidity接口合约。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
➜  script ./burrow natives       
pragma solidity >=0.4.24;

/**
* acmstate.ReaderWriter for managing Secure Native authorizations.
* @dev This interface describes the functions exposed by the native permissions layer in burrow.
* @dev These functions can be accessed as if this contract were deployed at a special address (0x0A758FEB535243577C1A79AE55BED8CA03E226EC).
* @dev This special address is defined as the last 20 bytes of the sha3 hash of the the contract name.
* @dev To instantiate the contract use:
* @dev Permissions permissions = Permissions(address(uint256(keccak256("Permissions"))));
*/
interface Permissions {
/**
* @notice Adds a role to an account
* @param Account account address
* @param Role role name
* @return result whether role was added
*/
function addRole(address _account, string memory _role) public view returns (bool _result);

/**
* @notice Removes a role from an account
* @param Account account address
* @param Role role name
* @return result whether role was removed
*/
function removeRole(address _account, string memory _role) public view returns (bool _result);

/**
* @notice Indicates whether an account has a role
* @param Account account address
* @param Role role name
* @return result whether account has role
*/
function hasRole(address _account, string memory _role) public view returns (bool _result);

/**
* @notice Sets the permission flags for an account. Makes them explicitly set (on or off).
* @param Account account address
* @param Permission the base permissions flags to set for the account
* @param Set whether to set or unset the permissions flags at the account level
* @return The permission flag that was set as uint64
*/
function setBase(address _account, uint64 _permission, bool _set) public view returns (uint64 _result);

/**
* @notice Unsets the permissions flags for an account. Causes permissions being unset to fall through to global permissions.
* @param Account account address
* @param Permission the permissions flags to unset for the account
* @return The permission flag that was unset as uint64
*/
function unsetBase(address _account, uint64 _permission) public view returns (uint64 _result);

/**
* @notice Indicates whether an account has a subset of permissions set
* @param Account account address
* @param Permission the permissions flags (mask) to check whether enabled against base permissions for the account
* @return result whether account has the passed permissions flags set
*/
function hasBase(address _account, uint64 _permission) public view returns (bool _result);

/**
* @notice Sets the global (default) permissions flags for the entire chain
* @param Permission the permissions flags to set
* @param Set whether to set (or unset) the permissions flags
* @return The permission flag that was set as uint64
*/
function setGlobal(uint64 _permission, bool _set) public view returns (uint64 _result);
}

./burrow dump , Dump chain state to backup

1
2
3
4
5
6
7
8
➜  script ./burrow dump local backup_dump
Sourcing config from first of: defaults
Sourcing config from defaults
Sourcing config from first of: genesis file at genesis.json
Sourcing config from genesis file at genesis.json
{"log_channel":"Info","message":"Dumping accounts"}
{"log_channel":"Info","message":"Dumping names"}
{"log_channel":"Info","message":"Dumping events"}

./burrow restore, Restore new chain from backup

从backup文件恢复区块链

文档链接=>https://hyperledger.github.io/burrow/

github链接=>https://github.com/hyperledger/burrow/

接前文《体验Hyperledger-Burrow-1》,启动4个节点如下:

1
2
3
4
5
6
➜  script ps aux | grep burrow
USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
will 21409 2.1 0.8 133268 66036 pts/1 SNl 08:52 1:02 ./burrow start --config=burrow000.toml
will 21410 1.9 0.7 133268 59212 pts/1 SNl 08:52 0:56 ./burrow start --config=burrow001.toml
will 21411 2.0 0.8 133268 65924 pts/1 SNl 08:52 0:59 ./burrow start --config=burrow002.toml
will 21416 2.0 0.7 133268 63416 pts/1 SNl 08:52 1:01 ./burrow start --config=burrow003.toml

制作增加节点的配置文件

1
./burrow spec -v1 | ./burrow configure -s- > burrow_add.toml

运行完成后,增加公私钥文档如下:

1
2
3
4
5
6
7
8
➜  data pwd
/home/will/documents/burrow/script/.keys/data
➜ data cat 035B1B192A16FF8086C36304024010502CEB2DE0.json
{"CurveType":"ed25519","Address":"5ECA9967F5D363F21C4606CC971710200A64C5BC","PublicKey":"4F3530024CF75A7AFC4D168CF2975C285D03CEBB1CBE82E21AD9C76531A63128","AddressHash":"go-crypto-0.5.0","PrivateKey":{"Crypto":"none","Plain":"486059FB274248423F92B7522D5A237AB99B38C80F84D46F720EAE1506D8E2DF4F3530024CF75A7AFC4D168CF2975C285D03CEBB1CBE82E21AD9C76531A63128"}}
➜ names pwd
/home/will/documents/burrow/script/.keys/names
➜ names cat Validator_0
035B1B192A16FF8086C36304024010502CEB2DE0

修改burrow_add.toml配置:
1.删除GenesisDoc的内容,因为genesis.json中已经包含了节点们的GenesisDoc;
2.增加PersistentPeers,PersistentPeers在burrow000.toml中可以找到;
3关闭grpc端口,关闭rpc.info等,否则会跟已有的4个节点端口冲突

发出增加validator的Tx

制作增加validator Tx 交易的文件:

1
2
3
4
5
6
jobs:
- name: AddValidator
update-account:
target: 4F3530024CF75A7AFC4D168CF2975C285D03CEBB1CBE82E21AD9C76531A63128
power: 222222
permissions: [root, send, call, createContract, createAccount, bond, name, proposal, input, batch, identify, hasBase, setBase, unsetBase, setGlobal, hasRole, addRole, removeRole]

其中4F3530024CF75A7AFC4D168CF2975C285D03CEBB1CBE82E21AD9C76531A63128 是新节点的publickey。

执行交易命令(DCF0F3BD45EA59B2D471729EC22838DF7B119012 是增加节点前已经启动的4个节点中的Full Account节点。),即在已经启动的节点上执行本交易:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
./burrow deploy -c 127.0.0.1:10997 --mempool-signing=true --address=DCF0F3BD45EA59B2D471729EC22838DF7B119012 deploy.yaml
log_channel=Info message="Using chain" Chain=127.0.0.1:10997 Signer=
log_channel=Info message="Loading Playbook File."
log_channel=Info message="Loading playbook file" path=/home/will/documents/burrow/script filename=/home/will/documents/burrow/script/deploy.yaml
log_channel=Info message="*****Executing Job*****" JobName=defaultAddr Type=Account
log_channel=Info message="Setting Account" account=DCF0F3BD45EA59B2D471729EC22838DF7B119012
log_channel=Info message="*****Executing Job*****" JobName=AddValidator Type=UpdateAccount
log_channel=Info message=GovTx account="unsupported value type"
log_channel=Info message="Using mempool signing since no keyClient set, pass --keys to sign locally or elsewhere"
log_channel=Info message="Using mempool signing"
log_channel=Info message="Tx Return" TransactionHash=5a1a57e975e5ead5079c12c63c473e1c04c59cf82fe9974f191faaf1d8aeb8b8 BlockHeight=18
log_channel=Info message="Job Vars" name=input value=DCF0F3BD45EA59B2D471729EC22838DF7B119012
log_channel=Info message="Job Vars" name=native value=
log_channel=Info message="Job Vars" name=power value=222222
log_channel=Info message="Job Vars" name=sequence value=
log_channel=Info message="Job Vars" name=address value=
log_channel=Info message="Job Vars" name=publicKey value=D90E05FAD677699B4FEE56B567F85D3A90438B4B627C27CED0D4C850BDC97876
log_channel=Info message="Writing to current directory" output=/home/will/documents/burrow/script/deploy.output.json
log_channel=Info message="JOBS THAT SUCCEEEDED" count=1
log_channel=Info message="Playbook result" jobNo=0 file=deploy.yaml time=230.744184ms

注意:新创建的validator没有余额。

启动新增加的节点

1
./burrow start --config=burrow_add.toml --genesis=genesis.json --address=5ECA9967F5D363F21C4606CC971710200A64C5BC

5ECA9967F5D363F21C4606CC971710200A64C5BC 是被增加节点的地址(address)。

1
2
3
4
5
6
7
➜  script ps aux | grep burrow
USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
will 21409 2.1 0.8 133268 66036 pts/1 SNl 08:52 1:02 ./burrow start --config=burrow000.toml
will 21410 1.9 0.7 133268 59212 pts/1 SNl 08:52 0:56 ./burrow start --config=burrow001.toml
will 21411 2.0 0.8 133268 65924 pts/1 SNl 08:52 0:59 ./burrow start --config=burrow002.toml
will 21416 2.0 0.7 133268 63416 pts/1 SNl 08:52 1:01 ./burrow start --config=burrow003.toml
will 21505 2.2 0.7 133268 58864 pts/1 Sl+ 08:55 1:01 ./burrow start --config=burrow_add.toml --genesis=genesis.json --address=DCF0F3BD45EA59B2D471729EC22838DF7B119012

ed25519是tendermint选择的加密曲线,secp256k1是ethereum选择的加密曲线。

两者都可以在./burrow keys list 中看到账户地址和名称。在web3.getAccounts中只能够看到secp256k1曲线下生成的账户地址,metamask和remix也是如此。

burrow keys gen -n -t secp256k1 --name zhuang 如果不指定 -n 的话,需要输入密码。

发送转账交易

创建转账交易 deploy_sendToken.yaml

1
2
3
4
5
jobs:
- name: sendTxTest1
send:
destination: 0DE8C14FE07FA693D3A6F65D826647D39E9A3BB0
amount: 2000

发起

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
➜  script ./burrow keys gen -n -t secp256k1 --name zhuang1
0DE8C14FE07FA693D3A6F65D826647D39E9A3BB0
➜ script ./burrow deploy --address CCC93802D38A33B636D8B2B74218021A571CF252 deploy_sendToken.yaml
log_channel=Info message="Using chain" Chain=127.0.0.1:10997 Signer=
log_channel=Info message="Loading Playbook File."
log_channel=Info message="Loading playbook file" path=/home/will/documents/burrow/script filename=/home/will/documents/burrow/script/deploy_sendToken.yaml
log_channel=Info message="*****Executing Job*****" JobName=defaultAddr Type=Account
log_channel=Info message="Setting Account" account=CCC93802D38A33B636D8B2B74218021A571CF252
log_channel=Info message="*****Executing Job*****" JobName=sendTxTest1 Type=Send
log_channel=Info message="Sending Transaction" source=CCC93802D38A33B636D8B2B74218021A571CF252 destination=0DE8C14FE07FA693D3A6F65D826647D39E9A3BB0 amount=2000
log_channel=Info message=SendTx send="unsupported value type"
log_channel=Info message="Using mempool signing since no keyClient set, pass --keys to sign locally or elsewhere"
log_channel=Info message="Using mempool signing"
log_channel=Info message="Tx Return" TransactionHash=a0abda87ea4751beaeae04d5ac67c35114b9f068720db6e1795ec7083c4a872e BlockHeight=27
log_channel=Info message="Writing to current directory" output=/home/will/documents/burrow/script/deploy_sendToken.output.json
log_channel=Info message="JOBS THAT SUCCEEEDED" count=1
log_channel=Info message="Playbook result" jobNo=0 file=deploy_sendToken.yaml time=347.234362ms