December 7, 2017

以太坊交易池逻辑分析

发送一笔以太坊网络转账交易,除了成功之外,会遇到各种问题,比如发送的一笔交易在etherscan中查不到,或者查到是pending的状态,后来过了几个小时消失了,这些都与节点的交易池的处理逻辑相关,而交易池的逻辑可以有各种实现,目前官方的go-ethereum 的交易池逻辑如下。 当一条交易进入挖矿节点的交易池(tx_pool)时,此节点会做出以下逻辑: 通过判断交易hash,当这条交易已经在节点的交易池里面时,就会丢弃掉当前收到的这一个交易。 如果是一个全新的hash,就会更具共识协议,对这一条交易做基本的验证。验证包括:长度,value,是否溢出当前区块的gaslimit,Nonce值,转账提供的gas大小是否太小等。 如果验证不通过就会返回相应的错误代码。 当验证通过的话,需要根据当前交易池的状态来决定,如果交易池满了的话,那么判断当前的转账交易提供的gas是否高于交易池里当前提供gas最低的那条交易。如果低了,就返回错误ErrUnderpriced,这是返回的错误经常在etherscan中看到。如果当前交易提供的gas值高的话,那么就剔除掉了那一条提供最低gas的转账交易,为当前交易腾出空间。这种情况发生之后,往往在etherscan中会观察到正在pending的交易消失了,找不到了。(也有很大机率依然能够查到pending的交易,原因是etherscan连接了很多节点,每个节点的交易池的状态都是不一样的,那条被踢出的交易在别的节点依然pending) 如果当前交易的Nonce已经在交易池里面了,说明了这个用户想替代之前发出的相同的Nonce的交易。这时节点会判断当前交易的gas是否高出上一条相同Nonce的交易gas某一个阈值(比如默认的是10%)。如果高出了,那么就会剔除之前的那个交易,新的交易就会保留,如果没有高出,当前的交易就会返回失败。 以上逻辑都通过了,那么此条交易便成功存在交易池中等待打包,当然也有可能会被别的高gas的转账交易剔除的概率。 总结,如果你发送了一笔交易碰巧赶上了以太坊网络拥堵,那么你的交易很有可能发送失败。原因是所有的挖矿节点都不接受这么低的gas交易了,或者你本来在交易池中,被剔除了。

October 7, 2017

ethereum rpc 调用分析

以太坊以JSON RPC的方式提供API service。本文将从go-ethereum源码中挖掘服务端如何提供JSON RPC 服务。 服务端启动rpc server ➜ go-ethereum git:(master) ✗ tree -d -L 1 ├── cmd ├── geth ├── main.go ├── internal ├── ethapi ├── api.go ├── node ├── rpc ... go-ethereum的代码很多,单从发起一笔转账这样一个api而言,geth节点涉及的代码相对简单。 首先,cmd/geth/main.go是整个geth节点的entrypoint,main函数会实例化一个全功能的节点: func geth(ctx *cli.Context) error { node := makeFullNode(ctx) startNode(ctx, node) node.Wait() return nil } 实例化之后,将调用node/node.go中的Start方法,来配置node相应的服务, 然后启动,等到所有的服务启动完成之后,节点开启RPC服务,根据config将相应的服务注册到RPC服务的白名单中: func (s *Server) RegisterName(name string, rcvr interface{}) error { ... methods, subscriptions := suitableCallbacks(rcvrVal, svc.typ) ... svc.name = name svc. Read more

July 18, 2017

使用nodejs部署智能合约

使用nodejs部署智能合约 实现智能合约的方式很多种,可以用truffle框架来实现,编译,部署。 这里介绍一种简单的使用nodejs来实现,编译,部署的方法。 创建一个nodejs项目,实现一个简单的智能合约。 mkdir sm && cd sm npm init mkdir contracts && cd contracts vi BaseToken.sol //BaseToken.sol contract Token{ address public owner; mapping (address => uint) public balances; event Sent(address from, address to, uint amount) function Token(){ owner = msg.sender; balances[owner] = 100000000; } function send(address receiver, uint amount){ if (balances[msg.sender] < amount) return; balances[msg.sender] -= amount; balances[receiver] += amount; Sent(msg.sender, receiver, amount); } } 这个合约实现了一个造币和转币的逻辑。 我们的合约是运行在evm上面的字节码,solidity是静态语言,需要通过编译器生成evm的字节码。 Read more

© Wade Van 2019

Powered by Hugo & Kiss.