详解Node.js 中使用 ECDSA 签名遇到的坑


Posted in Javascript onNovember 26, 2018

最近有个朋友问我关于 Node.js 下使用 ECDSA 的问题,主要是使用 Node.js 的 Crypto 模块无法校验网络传输过来的签名结果。在踩坑无数后,终于搞清楚了原因。

坑 0x00:签名输出格式

在排除了证书、消息不一致的可能之后,我开始对比使用 Node.js 签名的结果与网络传输过来的签名,发现长度不一致,大约差了5~7个字节。于是去网上搜索了一下,才知道原来 Node.js (基于 OpenSSL)签名得到的是 DER 格式的内容,而网络上常用的 ECDSA 签名结果是 IEEE P1363 格式的。(也可以写作 R|S)

参考:https://stackoverflow.com/a/39575576

知道问题了就好解决了。但是,DER 和 IEEE P1363 两个格式互转也不是那么容易的。

简单科普一下,ECDSA 是指基于 ECC 椭圆加密算法的签名方式,签名结果是两个整数 R 和 S。 R 和 S 一般长度相同,或者接近。如果长度不同,在各自前面补字节 0x00 直到等长。把 R 和 S 以大头字节序表示,然后依次前后拼接,就是所谓 IEEE P1363 格式。

坑 0x01:DER 的整数问题

先来了解一下 ECDSA 的 DER 输出格式,大概如下:

SEQUENCE <LENGTH>
 INTEGER <INTEGER_LENGTH> <INTEGER_VALUE...> # 整数 R
 INTEGER <INTEGER_LENGTH> <INTEGER_VALUE...> # 整数 S

其中

SEQUENCE 是 DER 数组(串?)标头,用一个字节 0x30 表示

<LENGTH> 是 SEQUENCE 的长度,用一个字节表示,不包括标头和这个长度本身

INTEGER 是整数标头,用一个字节 0x02 表示

<INTEGER_LENGTH> 是整数的字节长度,用一个字节表示。

<INTEGER_VALUE> 是整数的内容,以大头字节序表示。

另一个坑我也已经写出来了,不知道有人发现没有?没想到的话,继续往下。

IEEE P1363 格式下,R 和 S 都是等长的。所以只要把 IEEE P1363 格式的签名从中间切分就可以得到 R 和 S 的内容了。而且 IEEE P1363 格式下,R 和 S 也是以大头字节序表示的,因此没有字节序转换问题了。现在,只需要按上面的格式构造一个 DER 即可。

坑 0x01.0:缺少整数前置字节 0x00

我第一次尝试将 IEEE P1363 格式的签名转换成 DER 格式,并没有失败,但是当我换一个签名结果,却失败了……我对比了 DER 和 IEEE P1363 的区别,发现了一个特点,在 DER 格式下,R 和 S 偶尔会有前置字节 0x00,但不是一定的。

查资料后才明白,DER 下没有“无符号整数”之说,也就是说整数都是有符号的。如果 INTEGER 所表示的整数最高字节大于 0x7F,也就是最高位(符号位)为 1,则表示负数。如果要表示正数,必须在前面补一个字节 0x00……

参考 https://bitcointalk.org/index.php?topic=215205.msg2258789#msg2258789

坑 0x01.1:多余的整数前置字节 0x00

在我修改代码后,虽然提高了成功率,可仍然有失败的情况,仔细看了下,原来是因为 IEEE P1363 格式里,R 和 S 可能被补了不止 1 个字节 0x00……

而 DER 下虽然要求补字节 0x00,却是有且只能有一个字节 0x00。

到此,问题都解决了——直到我测试了 521-bit (是的,你没看错,不是 512) 长度的密钥时,完全失败,毫无例外。

坑 0x02:DER SEQUENCE 的长度超过 0x7F

前面说了,<LENGTH> 只能用一个字节表示,这是一个整数,前文我提到的整数正负问题,这里也存在!

即是说,ECDSA 签名使用 DER 输出格式时,如果使用 521-bit (是的,你没看错,不是 512) 长度的密钥时,DER的长度将超出 0x7F,使得 <LENGTH> 变成了负数!

而解决方案不是补字节 0x00,而是用字节 0x81 填充 <LENGTH>,再在下一个字节用一个无符号整数的表示长度(0 ~ 255)。

参考:https://stackoverflow.com/a/47099047

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
指定位置如果有图片显示图片,无图片显示广告的JS
Jun 05 Javascript
一个分享按钮的插件使用介绍(可扩展,内附开发制作流程)
Sep 19 Javascript
推荐40款强大的 jQuery 导航插件和教程(上篇)
Sep 14 Javascript
浅析Javascript中“==”与“===”的区别
Dec 23 Javascript
jquery实现动静态条形统计图
Aug 17 Javascript
js Canvas实现的日历时钟案例分享
Dec 25 Javascript
原生JS改变透明度实现轮播效果
Mar 24 Javascript
微信小程序实现给嵌套template模板传递数据的方式总结
Dec 18 Javascript
3种vue路由传参的基本模式
Feb 22 Javascript
vue调用语音播放的方法
Sep 27 Javascript
vue基于v-charts封装双向条形图的实现代码
Dec 09 Javascript
create-react-app开发常用配置教程
Jun 25 Javascript
Vue.js的动态组件模板的实现
Nov 26 #Javascript
解决Vue开发中对话框被遮罩层挡住的问题
Nov 26 #Javascript
Vue项目部署在Spring Boot出现页面空白问题的解决方案
Nov 26 #Javascript
electron制作仿制qq聊天界面的示例代码
Nov 26 #Javascript
Vuex的初探与实战小结
Nov 26 #Javascript
微信小程序页面间值传递的两种方法
Nov 26 #Javascript
Vue中的methods、watch、computed的区别
Nov 26 #Javascript
You might like
PHP4中session登录页面的应用
2008/07/25 PHP
PHP面向对象三大特点学习(充分理解抽象、封装、继承、多态)
2012/05/07 PHP
php生成数组的使用示例 php全组合算法
2014/01/16 PHP
PHP面向对象教程之自定义类
2014/06/10 PHP
PHP传参之传值与传址的区别
2015/04/24 PHP
PHP+MySQL之Insert Into数据插入用法分析
2015/09/27 PHP
Yii2 批量插入、更新数据实例
2017/03/15 PHP
ThinkPHP实现的rsa非对称加密类示例
2018/05/29 PHP
用js自动判断浏览器分辨率的代码
2007/01/28 Javascript
jQuery自带的一些常用方法总结
2014/09/03 Javascript
使用jquery解析XML的方法
2014/09/05 Javascript
jQuery移动端图片上传组件
2016/06/12 Javascript
下一代Bootstrap的5个特点 超酷炫!
2016/06/17 Javascript
HTML Table 空白单元格补全的简单实现
2016/10/13 Javascript
在angular 6中使用 less 的实例代码
2018/05/13 Javascript
vue 解除鼠标的监听事件的方法
2019/11/13 Javascript
JS实现transform实现扇子效果
2020/01/17 Javascript
JS中作用域以及变量范围分析
2020/07/18 Javascript
Element Tooltip 文字提示的使用示例
2020/07/26 Javascript
[38:23]完美世界DOTA2联赛循环赛 FTD vs PXG BO2第二场 11.01
2020/11/02 DOTA
Python实现的远程登录windows系统功能示例
2018/06/21 Python
Django框架搭建的简易图书信息网站案例
2019/05/25 Python
python两个_多个字典合并相加的实例代码
2019/12/26 Python
python 按钮点击关闭窗口的实现
2020/03/04 Python
Python Selenium实现无可视化界面过程解析
2020/08/25 Python
html5通过canvas实现刮刮卡效果示例分享
2014/01/27 HTML / CSS
美国体育用品商店:Rally House(NCAA、NFL、MLB、NBA、NHL和MLS)
2018/01/03 全球购物
英国设计师珠宝网站:Joshua James Jewellery
2020/03/01 全球购物
师范应届生教师求职信
2013/11/05 职场文书
市场营销大学生职业规划书
2014/02/25 职场文书
股份合作协议书
2014/04/12 职场文书
档案保密承诺书
2014/06/03 职场文书
关于运动会广播稿50字
2014/10/18 职场文书
信访维稳工作汇报
2014/10/27 职场文书
社区法制宣传日活动总结
2015/05/05 职场文书
Python借助with语句实现代码段只执行有限次
2022/03/23 Python