以太坊地址和公钥_以太坊交易签名解析源码解读

上篇文章《以太坊交易签名过程源码解析[1]》从源码角度分析了一个合约调用的的签名过程,签名后的交易发送到以太坊节点后,节点需要从签名交易中还原出公钥(从公钥中单向计算出账号地址),进而将交易放入交易池中。本文从go-ethereum源码的出发,看看如何从签名交易中还原出公钥。

一、准备工作

我们使用上文中最后得到的签名交易串来进行解析,这里我写的解析代码如下所示。

package mainimport (    "fmt"    "github.com/ethereum/go-ethereum/common/hexutil"    "github.com/ethereum/go-ethereum/core/types"    "github.com/ethereum/go-ethereum/rlp"    "math/big")func main() {    // 还原交易对象    encodedTxStr := "0xf889188504a817c800832dc6c09405e56888360ae54acf2a389bab39bd41e3934d2b80a4ee919d50000000000000000000000000000000000000000000000000000000000000007b25a041c4a2eb073e6df89c3f467b3516e9c313590d8d57f7c217fe7e72a7b4a6b8eda05f20a758396a5e681ce1ab4cec749f8560e28c9eb91072ec7a8acc002a11bb1d"    encodedTx, err := hexutil.Decode(encodedTxStr)    if err != nil {        fmt.Println("hexutil.Decode failed: ", err.Error())        return    }    // rlp解码    tx := new(types.Transaction)    if err := rlp.DecodeBytes(encodedTx, tx); err != nil {        fmt.Println("rlp.DecodeBytes failed: ", err.Error())        return    }    // chainId为1的EIP155签名器    signer := types.NewEIP155Signer(big.NewInt(1))    // 使用签名器从已签名的交易中还原账户公钥    from, err := types.Sender(signer, tx)    if err != nil {        fmt.Println("types.Sender: ", err.Error())        return    }    fmt.Println("from: ", from.Hex())    jsonTx, _ := tx.MarshalJSON()    fmt.Println("tx: ", string(jsonTx))}

其中:

•encodedTxStr是上篇文章得到的具有签名的交易对象的rlp编码•最终还原得到的from值为0xA2088F51Ea1f9BA308F5014150961e5a6E0A4E13,正是签名私钥对应的账号地址(私钥单向生成公钥,公钥单向生成地址)•签名解析核心使用的是Sender方法

二、签名解析

types.Sender方法中核心调用了EIP155签名器的Sender方法,其源码如下。

// go-ethereum/core/types/transaction_signing.gofunc (s EIP155Signer) Sender(tx *Transaction) (common.Address, error) {    if !tx.Protected() {//①        return HomesteadSigner{}.Sender(tx)    }    if tx.ChainId().Cmp(s.chainId) != 0 {//②        return common.Address{}, ErrInvalidChainId    }    //③    V := new(big.Int).Sub(tx.data.V, s.chainIdMul)    V.Sub(V, big8)    return recoverPlain(s.Hash(tx), tx.data.R, tx.data.S, V, true)}

Sender方法中:

•①首先判断了交易是否是受保护的(是否是EIP155签名器进行的签名),如果不是,则使用HomesteadSigner签名器校验•②接着判断了交易中的链ID与签名器的链ID是否一致,如果不一致则返回空地址•③根据V的计算方法还原recid为27(37-1*2-8),在recoverPlain方法会按照homestead签名方式继续解析签名。

recoverPlain源码如下所示。

// go-ethereum/core/types/transaction_signing.gofunc recoverPlain(sighash common.Hash, R, S, Vb *big.Int, homestead bool) (common.Address, error) {    if Vb.BitLen() > 8 {        return common.Address{}, ErrInvalidSig    }    V := byte(Vb.Uint64() - 27)    if !crypto.ValidateSignatureValues(V, R, S, homestead) {        return common.Address{}, ErrInvalidSig    }    // encode the signature in uncompressed format    r, s := R.Bytes(), S.Bytes()    sig := make([]byte, crypto.SignatureLength)    copy(sig[32-len(r):32], r)    copy(sig[64-len(s):64], s)    sig[64] = V //①    fmt.Println("sig: ", common.Bytes2Hex(sig))    // recover the public key from the signature    pub, err := crypto.Ecrecover(sighash[:], sig) //②    if err != nil {        return common.Address{}, err    }    if len(pub) == 0 || pub[0] != 4 {        return common.Address{}, errors.New("invalid public key")    }    fmt.Println("pub: ", common.Bytes2Hex(pub))    var addr common.Address    copy(addr[:], crypto.Keccak256(pub[1:])[12:])//③    return addr, nil}

其中recoverPlain方法的参数分别为:

•sighash是交易对象tx的rlp编码,hex值为0x9ef7f101dae55081553998d52d0ce57c4cf37271f800b70c0863c4a749977ef1,与我们上文中需要签名的交易hash是一致的。•R,hex值为41c4a2eb073e6df89c3f467b3516e9c313590d8d57f7c217fe7e72a7b4a6b8ed•S hex值为5f20a758396a5e681ce1ab4cec749f8560e28c9eb91072ec7a8acc002a11bb1d•Vb,十进制值为27•bool类型的homestead,值为true

在recoverPlain方法中:

•①,根据R、S、V拼接得到的sign,hex值为:41c4a2eb073e6df89c3f467b3516e9c313590d8d57f7c217fe7e72a7b4a6b8ed5f20a758396a5e681ce1ab4cec749f8560e28c9eb91072ec7a8acc002a11bb1d00•②,调用加密包中的Ecrecover方法根据签名还原公钥,该方法会调用secp256k1包中的RecoverPubkey方法。还原得到的公钥hex值为045762d11bad6617b5eef31fefd6aff1391dab0a2380817eaf882874b1d50823b13e4934f923f4b7e6a3d19219e92a04678a8fb7029c2ecf7256672b57a6cb77b0 。•③,根据公钥计算账号地址,取公钥pub第一位之后的值计算Keccak256,然后在取后12位以后,得到的账号地址为:0xA2088F51Ea1f9BA308F5014150961e5a6E0A4E13

至此,我们已经从签名中还原出了账号地址(公钥)。如果需要校验签名是否正确,可以通过调用secp256k1包中的VerifySignature方法,传入公钥、交易hash和签名,通过比对R值是否一致进行验证。

References

[1] 以太坊交易签名过程源码解析: https://learnblockchain.cn/article/1225


本文作者:六天

作者主页:

https://learnblockchain.cn/people/436


515387c891330876862f4548dcd5ffef.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/329325.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

IIS负载均衡-Application Request Route详解第三篇:使用ARR进行Http请求的负载均衡

在前两篇文章中,我们已经讲述如何配置与安装ARR,从本篇文章开始,我们将重点的来讲述如何在使用ARR进行负载均衡。 本篇文章的目的主要是一步步的带领大家如何配置和使用ARR来进行Http请求的负载均衡,从而实现高可用与高扩展性。同…

Oracle入门(十四.17)之procedure传递参数

一、PROCEDURE 参数模式参数模式在形式参数声明中指定,位于参数名称之后,位于其数据类型之前。 参数传递模式: •IN参数(默认值)为子程序提供处理值。 •OUT参数将值返回给调用者。•IN OUT参数提供一个输入值&#xf…

1.elasticsearch文档存储(保存|修改|删除)

【README】 0.本文部分内容(数据)总结自 es 开发文档, Document APIs | Elasticsearch Guide [7.2] | Elastic ; 1.本文的es版本是7.2.1; 2.elasticsearch 是一个数据存储,检索和分析引擎;本…

db9针232接口波特率标准_理解串口通信以及232,485,422常见问题

下面先讲一讲串口通信的一些基本概念,术语。如果对串口通信比较熟悉的,就当复习复习,如果哪里讲错或不到位,也可以及时指出,当作一块交流交流。这里并不对串口的编程作讲解,主要是从应用的角度去讲一讲。因…

理解 .NET Platform Standard

.NET Platform Standard:https://github.com/dotnet/corefx/blob/master/Documentation/architecture/net-platform-standard.md .NET Platform Standard 是什么?直译过来就是 .NET 平台规范或标准,它的目的就是使 .NET 各个平台之间更加统一…

Oracle入门(十四.18)之使用动态SQL

一、SQL的执行流程数据库中的所有SQL语句都经历了不同的阶段: •解析:预执行“这可能吗?”检查包括语法,对象存在,权限等 •绑定:获取语句中引用的任何变量的实际值 •执行:语句被执行。•提取&…

天平游码读数例题_初二上册物理实验——托盘天平使用的注意事项

今天给大家讲讲托盘天平的使用和注意事项。(1)首先,我们回忆一下什么是托盘天平如下图托盘天平由底座、横梁、分度盘、托盘、平衡螺母、标尺、游码、托盘等组成,当然还有砝码。托盘天平是称量物体的质量的工具(质量符号m,单位:国际…

3.elasticsearch文档查询dsl

【README】 1.本文elasticsearch版本是 7.2.1; 2.文档查询语句叫做 DSL, domain structure language, 领域特定语言;dsl,参见 Query DSL | Elasticsearch Guide [7.2] | Elastic 3.elasticsearch 基于json 提供了完…

Oracle入门(十四.19)之触发器简介

一、触发器的需求让我们从一个例子开始吧:一条业务规则规定,只要员工的工资发生变化,变更就必须记录在日志记录表中。 可以创建两个过程来执行此操作:UPD_EMP_SAL更新工资,LOG_SAL_CHANGE将行插入日志表。可以从UPD_EM…

搜狐视频Redis私有云cachecloud开源了

项目地址:https://github.com/sohutv/cachecloud/ 一、CacheCloud是做什么的 CacheCloud提供一个Redis云管理平台:实现多种类型(Redis Standalone、Redis Sentinel、Redis Cluster)自动部署、解决Redis实例碎片化现象、提供完善统计、监控、运维功能、减…

springboot 订单重复提交_瞬间几千次的重复提交,我用Spring Boot+Redis扛住了

在实际的开发项目中,一个对外暴露的接口往往会面临,瞬间大量的重复的请求提交,如果想过滤掉重复请求造成对业务的伤害,那就需要实现幂等!我们来解释一下幂等的概念:任意多次执行所产生的影响均与一次执行的…

Oracle入门(十四.21)之创建DML触发器:第二部分

一、使用条件谓词 在上文中,看到了一个触发器,可以防止在周末插入EMPLOYEES: CREATE OR REPLACE TRIGGER secure_emp BEFORE INSERT ON employees BEGINIF TO_CHAR(SYSDATE,DY) IN (SAT,SUN) THENRAISE_APPLICATION_ERROR(-20500,You may ins…

【直播预告】创享未来 2016微软开发者峰会

感谢所有中国开发者对2016微软开发者峰会的热情关注,目前活动已经截止报名了,不过M姐为大家带来新的福利: 2016微软开发者峰会将全程线上直播! 2016微软开发者峰会将全程线上直播! 2016微软开发者峰会将全程线上直播…

定时任务重启后执行策略_C语言操作时间函数time.ctime,实现定时执行某个任务小例子...

时间操作函数在实际项目开发中会经常用到,最近做项目也正好用到就正好顺便整理一下。时间概述由上图可知:通过系统调用函数time()可以从内核获得一个类型为time_t的1个值,该值叫calendar时间,即从1970年1月1日的UTC时间从0时0分0妙…

Oracle入门(十四.20)之创建DML触发器:第一部分

一、什么是DML触发器?DML触发器是执行SQL DML语句(INSERT,UPDATE或DELETE)时自动触发(执行)的触发器。 您可以通过两种方法对DML触发器进行分类: •执行时间:BEFORE,AFTE…

IIS负载均衡-Application Request Route详解第四篇:使用ARR实现三层部署架构

本篇的主要目的是带领大家一起来使用ARR来实现一个三层部署架构。这里的三层部署架构主要是由:服务层,应用程序服务器层已经数据层实现。如下图所示: 每次一提到“层”这个字的时候,似乎感觉这个字特别的惹火。很多朋友开始讨论起…

c遗传算法的终止条件一般_KDD比赛之遗传算法(举例理解)

求最大值问题是这样的:求解函数 f(x) x 10*sin(5*x) 7*cos(4*x) 在区间[0,9]的最大值。这个函数大概长这样:那么如何应用遗传算法如何来找到这个奇怪的函数的最大值呢?事实上,不管一个函数的形状多么奇怪,遗传算法都…

6.elasticsearch查询与过滤上下文(query context与filter contenxt)以及term术语查询

【README】 1.本文总结自: Query and filter context | Elasticsearch Guide [7.2] | Elastichttps://www.elastic.co/guide/en/elasticsearch/reference/7.2/query-filter-context.html2.文档相关性分数是否被计算,取决于查询子句是在查询上下文&…

Oracle入门(十四.22)之创建DDL和数据库事件触发器

一、什么是DDL和数据库事件触发器?DDL语句触发DDL触发器:CREATE,ALTER或DROP。 数据库事件触发器由数据库中的非SQL事件触发,例如: •用户连接到数据库或与数据库断开连接。 •DBA启动或关闭数据库。•用户会话中引发了…

Visual Studio上开发Python?你不可不知道的六大功能!

Visual Studio 2013/2015 搭配 Python Tools for Visual Studio 扩充套件让 Visual Studio 能提供对 Python 程序语言高度整合的开发环境,并完整发挥 Visual Studio 强大的功能,协助您在 Visual Studio 内开发 Python 程序上如虎添翼,提升开发…