Skip to content

Understanding Ethereum(Go version)|理解以太坊: Go-Ethereum 源码剖析

License

Notifications You must be signed in to change notification settings

kong9527/Understanding-Ethereum-Go-version

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Understanding-Ethereum-Go-version

  • Title: Understanding Ethereum: Starting with go-ethereum Source Code|理解以太坊:Go-Ethereum 源码剖析
  • Subject: Ethereum Source Code Analysis | 以太坊源码剖析
  • Author: Siyuan Han
  • Go-Ethereum Version: v1.10.15 (London Upgrade)
  • Updated date: 2022-04

Preface

Background

自中本聪发表比特币白皮书至今,已经过了十几年的时光,Blockchain 这一理念,从最开始仅作为支持 Bitcoin 的分布式账本,在这十几年中也在不断演化发展。随着参与人数的增加,不断有新的思想涌入社区。Blockchain 逐渐成为了集成了包括数据库分布式系统密码学点对点网络编译原理静态软件分析众包经济学货币金融学社会学在内的等多个学科知识的一个全新技术领域。Blockchain 也逐渐从小众的去中心化社区逐渐走向了主流社会的舞台,目前仍是当下最热度最高技术迭代最快最能引起社会讨论的技术话题之一。

在 Blockchain 固有的 decentralized 的思想下,市面上绝大多数的 Blockchain 系统都已经开源,并会继续以开源的形式持续开发。这就为我们提供了一种的极好的学习 Blockchain 技术的方式:通过结合文档,基于 State-of-the-arts 的 Blockchain Systems 的源代码出发开始学习。

一个热门的技术总会有大量不同视角的作者,在不同的时间记录下来的文档资料。作为学习者,不管是探究以加密货币导向(Crypto-based)的 Bitcoin, 还是了解致力于实现通用 Web3.0 框架(General-Purpose)的 Ethereum,社区中有丰厚的文档从 high-level 的角度来讲述它们的基础概念,以及设计的思想。比如,技术社区有非常多的资料来讲述什么是梅克尔树 (Merkle Hash Tree),什么是梅克尔帕特里夏树 (Merkle Patricia Trie),什么是有向无环图 (Directed acyclic Graph); BFT (Byzantine Fault Tolerance) 和 PoW (Proof-Of-Work) 共识算法算法的区别;以及介绍 Blockchain 系统为什么可以抵抗双花攻击 (Double-Spending),或者为什么 Ethereum 会遇到 DAO Attack (Decentralized autonomous organization) 等具体问题。

但是,只了解关键组件的实现细节,或者高度抽象的系统工作流,并不代表着可以让读者从真正搞清楚 Blockchain 的工作原理,反而很容易让读者在一些关键细节上一头雾水,似懂非懂。比如,当我们谈到 Blockchain 系统中 Transaction 的生命周期时,在文档中经常会读到,“Miner 节点批量地从自己维护的 Transaction pool 中选择一些 Transaction 并打包成一个新的 Block 中”。那么究竟 miner 是怎么从网络中获取到 Transaction?又是继续什么样的策略从 Transaction pool 中选取多少 Transaction?最终按照又按照什么样的 Order 把 transaction 打包进区块链中的呢?如何打包成功的 Block 是怎么交互给其他节点呢?在我学习的过程中,尝试去搜索了一下,发现鲜有文章从整体的系统工作流的角度出发,以细粒度的视角对区块链系统中的具体的实现细节进行解析。与数据库系统 (Database Management System) 相似,Blockchain 系统同样是一个包含网络层,业务逻辑层,任务解析层,存储层的复杂数据管理系统。对它研究同样需要从系统的实现细节出发,从宏观到微观的了解每个执行逻辑的工作流,才能彻底理解和掌握这门技术的秘密。

笔者坚信,随着网络基础架构的不断完善,将带来显著的带宽增加和通信延迟下降。同时随着存储技术和分布式算法的不断发展,未来系统的运行效率将会不断逼近硬件极限。在未来的是五到十年内,云端服务/去中心化系统的效率以及覆盖场景一定还会有很大的提升。未来技术世界一定是两极分化的。一极是以大云计算公司(i.e, Google,MS,Oracle,Snowflake,and Alibaba)为代表的中心化服务商。另一极就是以 Blockchain 技术作为核心的去中心化的世界。在这个世界中,Ethereum 及其生态系统是当之无愧的领头羊。Ethereum 作为通用型 Public Chain 中的翘楚取得了巨大的成功,构建了强大的生态系统。并且吸引到了一大批世界上最优秀的工程师,研究人员的持续的发力,不断的将新思想,新理念,新技术引入到 Ethereum 中,持续的引领 Blockchain 技术发展。同时 Go-Ethereum 作为其优秀的开源实现,已经被广泛的订制,来适应不同的私有/联盟/Layer-2 的场景 (e.g., Quorum, Binance Smart Chain, Optimism)。因此,要想真正掌握好区块链系统的原理,达到可以设计开发区块链系统的水平,研究好 Ethereum 的原理以及其设计思想是非常有必要。

本系列文章作为我在博士期间学习/研究的记录,将会从 Ethereum 中具体业务的工作的视角出发,在源码的层面细粒度的解析以太坊系统中各个模块的实现的细节,以及背后的蕴含的技术和设计思想。同时,在阅读源代码中发现的问题也可以会提交 Pr 来贡献社区。本系列基于的代码库是 Go-ethereum version 1.10.*版本。Go-ethereum 是以太坊协议的 Go 语言实现版本,目前由以太坊基金会维护。目前除了 Go-ethereum 之外,Ethereum 还有 C++, Python,Java, Rust 等基于其他语言实现的版本。相比于其他的社区版实现,Go-ethereum 的用户数量最多,开发人员最多,版本更新最频繁,issues 的发现和处理都较快。其他语言的 Ethereum 实现版本因为用户与开发人员的数量相对较少,更新频率相对较低,隐藏问题出现的可能性更高。因此我们选择从 Go-ethereum 代码库作为我们的主要学习资料。

为什么要阅读区块链系统的源代码?

  1. 文档资料相对较少,且内容浅尝辄止相互独立。比如,很多的科普文章都提到,在打包新的 Block 的时候,miner 负责把 a batch of transactions 从 transaction pool 中打包到新的 block 中。那么我们希望读者思考如下的几个问题:

    • Miner 是基于什么样策略从 Transaction Pool 中选择 Transaction 呢?
    • 被选择的 Transactions 又是以怎样的顺序 (Order) 被打包到区块中的呢?
    • 在执行 Transaction 的 EVM 是怎么计算 gas used,从而限定 Block 中 Transaction 的数量的呢?
    • 剩余的 gas 又是怎么返还给 Transaction Proposer 的呢?
    • EVM 是怎么解释 Contract 的 Message Call 并执行的呢?
    • 在执行 Transaction 中是哪个模块,是怎样去修改 Contract 中的持久化变量呢?
    • Smart Contract 中的持久化变量又是以什么样的形式存储在什么地方的呢?
    • 当新的 Block 加入到 Blockchain 中时,World State 又是何时怎样更新的呢?
    • 哪些数据常驻内存,哪些数据又需要保存在 Disk 中呢?
  2. 目前的 Blockchain 系统并没有像数据库系统 (DBMS) 那样统一的形成系统性的方法论。在 Ethereum 中每个不同的模块中都集成了大量的细节。从源码的角度出发可以了解到很多容易被忽视的细节。简单的说,一个完整的区块链系统至少包含以下的模块:

    • 密码学模块:加解密,签名,安全 hash,Mining
    • 网络模块:P2P 节点通信
    • 分布式共识模块:PoW, BFT,PoA
    • 智能合约解释器模块:Solidity 编译语言,EVM 解释器
    • 数据存储模块:数据库,Caching,数据存储,Index,LevelDB
    • Log 日志模块
    • etc.

而在具体实现中,由于设计理念,以及 go 语言的特性(没有继承派生关系),Go-ethereum 中的模块之间相互调用关系相对复杂。因此,只有通过阅读源码的方式才能更好理解不同模块之间的调用关系,以及业务的 workflow 中执行流程/细节。

Blockchain System (BCS) VS Database Management System (DBMS)

Blockchain 系统在设计层面借鉴了很多数据库系统中的设计逻辑。

  • Blockchain 系统同样也从 Transaction 作为基本的操作载核,包含一个 Parser 模块,Transaction Executor 模块,和一个 Storage 管理模块。

Contents(暂定)

PART ONE - General Source Code Analysis: Basic Workflow and Data Components

PART TWO - General Source Code Analysis: Lower-level Services

  • [10_状态数据优化:Batch and Pruning]
  • [11_Blockchain 的数据是如何持久化的:Leveldb in Practice]
  • [12_当 I/O 变成瓶颈:Caching in Practice]
  • [13_深入 EVM: 设计与实现]
  • [14_Signer: 如何证明 Transaction 的合法性]
  • 15_节点的调用 RPC and IPC

PART THREE - Advanced Topics~P

PART FOUR - Ethereum in Practice

PART FIVE - APPENDIX

  • 40_FQA
  • [41_Ethereum System Tunning]
  • [42_go-ethereum 的开发思想]
  • [43_Metrics in Ethereum]
  • [44_Golang & Ethereum]
  • [45_Looking Forward to the Ethereum2.0]

How to measure the level of understanding of a system?

如何衡量对一个系统的理解程度?

  • Level 4: 掌握(Mastering)
    • 在完全理解的基础上,可以设计并编写一个新的系统
  • Level 3: 完全理解(Complete Understanding)
    • 在理解的基础上,完全掌握系统的各项实现的细节,并能做出优化
    • 可以对现有的系统定制化到不同的应用场景
  • Level 2: 理解(Understanding)
    • 熟练使用系统提供的 API
    • 了解系统模块的调用关系
    • 能对系统的部分模块进行简单修改/重构
  • Level 1: 了解(Brief understanding)
    • 了解系统设计的目标,了解系统的应用场景
    • 可以使用系统的部分的 API

我们希望读者在阅读完本系列之后,对 Ethereum 的理解能够达到 Level 2 - Level 3 的水平。

Some Details

  • 以太坊是基于 State 状态机模型的区块链系统,Miner 在 update new Block 的时候,会直接修改自身的状态(添加区块奖励给自己。所以与 Bitcoin 不同的是,Ethereum 的区块中,并没有类似的 Coinbase 的 transaction。
  • 在 core/transaction.go 中,transaction 的的数据结构是有 time.Time 的参数的。但是在下面的 newTransaction 的 function 中只是使用 Local 的 time.now() 对 Transaction.time 进行初始化。
  • 在 core/transaction.go 的 transaction 数据结构定义的时候,在 transaction.time 后面的注释写到(Time first seen locally (spam avoidance), Time 只是用于在本地首次看到的时间。
  • uncle block 中的 transaction 不会被包括到主链上。
  • go-ethereum 有专用函数来控制每次 transaction 执行完,返还给用户的 Gas 的量。有根据 EIP-3529,每次最多返还 50%的 gas.
  • 不同的 Contracts 的数据会混合的保存在底层的一个 LevelDB instance 中。
  • LevelDB 中保存的是 MPT 的 Node 信息,包括 State Trie 和 Storage Trie。
  • 在以太坊更新数据相关的工作流中,通常先调用Finalise函数,然后执行Commit函数。

关键函数

 // 向 leveldb 中更新 Storage 数据
 func WritePreimages(db ethdb.KeyValueWriter, preimages map[common.Hash][]byte)

 // 向 Blockchain 中添加新的 Block,会涉及到 StateDB(Memory)/Trie(Memory)/EthDB(Disk) 的更新
 func (bc *BlockChain) insertChain(chain types.Blocks, verifySeals, setHead bool) (int, error)

 // insertChain 中调用来执行 Block 中的所有的交易
 func (p *StateProcessor) Process(block *types.Block, statedb *state.StateDB, cfg vm.Config) (types.Receipts, []*types.Log, uint64, error)

 //执行单条 Transaction 的调用
 func applyTransaction(msg types.Message, config *params.ChainConfig, bc ChainContext, author *common.Address, gp *GasPool, statedb *state.StateDB, blockNumber *big.Int, blockHash common.Hash, tx *types.Transaction, usedGas *uint64, evm *vm.EVM) (*types.Receipt, error)

 // 状态转移函数
 func (st *StateTransition) TransitionDb() (*ExecutionResult, error)

 // 执行合约内 function
 func (in *EVMInterpreter) Run(contract *Contract, input []byte, readOnly bool) (ret []byte, err error)

 // opSstore 的调用
 func (s *StateDB) SetState(addr common.Address, key, value common.Hash)
 // 被修改的 state 的值会首先被放在 StateObject 的 dirtyStorage 中,而不是直接添加到 Trie 或者 Disk Database 中。
 func (s *stateObject) setState(key, value common.Hash)

 // 根据当前的 State Trie 的值来重新计算 State Trie 的 Root,并返回改 Root 的值
 func (s *StateDB) IntermediateRoot(deleteEmptyObjects bool) common.Hash

 // Finalise 当前内存中的 Cache.
 func (s *StateDB) Finalise(deleteEmptyObjects bool) 

 // Commit StateDB 中的 Cache 到内存数据库中
 func (s *StateDB) Commit(deleteEmptyObjects bool) (common.Hash, error)

 // 将 StateObject 中所有的 dirtyStorage 转存到 PendingStorage 中,并清空 dirtyStorage,并给 prefetcher 赋值
 func (s *stateObject) finalise(prefetch bool)

 // 更新 StorageObject 对应的 Trie, from Pending Storage
 func (s *stateObject) updateTrie(db Database) Trie

 // 最终获取到新的 StateObject 的 Storage Root
 func (t *Trie) hashRoot() (node, node, error)

 // 用于在内存数据库中保存 MPT 节点
 func (c *committer) store(n node, db *Database) node

 // 向 rawdb 对应的数据库写数据 (leveldb)
 func (db *Database) Commit(node common.Hash, report bool, callback func(common.Hash)) error

References

本书主要参考了 Go-Ethereum 代码库,Ethereum Yellow Paper,以及 EIP 的具体 Spec。读者可以从下面的 Link 中找到相关的引用资料。

  • [1] Ethereum Yellow Paper (Paper Link)
  • [2] Ethereum/Go-Ethereum (link)
  • [3] Go-ethereum code analysis (link)
  • [4] Ethereum Improvement Proposals (link)
  • [5] Mastering Bitcoin(Second Edition)
  • [6] Mastering Ethereum (link)
  • [7] EIP-2718: Typed Transaction Envelope(link)
  • [8] EIP-2930: Optional access lists (link)
  • [9] EIP-1559: Fee market change for ETH 1.0 chain (link)
  • [10] Ask about Geth: Snapshot acceleration (link)

Talks

  • Succinct Proofs in Ethereum - Barry Whitehat, Ethereum Foundation (Youtube)

About

Understanding Ethereum(Go version)|理解以太坊: Go-Ethereum 源码剖析

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Go 98.4%
  • Shell 1.6%