Go学习第八天

签名

java 复制代码
func (a *Account) Sign(message []byte) ([]byte, error) {
	hash := crypto.Keccak256Hash(message)
	signature, err := crypto.Sign(hash.Bytes(), a.privateKeyECDSA)
	if err != nil {
		log.Fatal(err)
	}
	signMsg := []byte(hexutil.Encode(signature))
	return signMsg, err
}

验签

java 复制代码
func (a *Account) VerifySign(signatureHex, message string) (bool, error) {
	// 签名 Hex 转 Bytes
	signature, err := hexutil.Decode(signatureHex)
	if err != nil {
		return false, errors.New("签名后的数据解析异常")
	}
	// msg 转 hash
	newHash := crypto.Keccak256Hash([]byte(message))

	// 由于前端工具差异, 可能需要重新 Hash
	//var newHash []byte
	if signature[64] > 30 {
		signature[64] -= 31
		//newHash = signHash(hash.Bytes())
	} else {
		signature[64] -= 27
		//newHash = hash.Bytes()
	}

	sigPublicKeyECDSA, err := crypto.SigToPub(newHash.Bytes(), signature)
	if err != nil {
		return false, errors.New("签名解出公钥椭圆曲线异常")
	}

	// 一个重要的问题是我们必须首先删除signture的最后一个字节,因为它是ECDSA恢复ID,不能包含它。
	sigPublicKeyBytes := crypto.FromECDSAPub(sigPublicKeyECDSA)
	signatureNoRecoverID := signature[:len(signature)-1]

	recoveredAddr := crypto.PubkeyToAddress(*sigPublicKeyECDSA)
	if string(a.PublicKey()) != recoveredAddr.Hex() {
		return false, errors.New("签名解出公钥异常")
	}

	// 该函数接收原始数据的签名,哈希值和字节格式的公钥。 它返回一个布尔值,如果公钥与签名的签名者匹配,则为true。
	verified := crypto.VerifySignature(sigPublicKeyBytes, newHash.Bytes(), signatureNoRecoverID)
	if !verified {
		return false, errors.New("验签失败")
	}
	return verified, nil
}
相关推荐
Hello eveybody20 小时前
介绍一下背包DP(C++)
开发语言·c++·动态规划·dp·背包dp
05候补工程师20 小时前
[线性代数] 判定线性相关性的“降维打击”:从基本定理到速通特殊法
经验分享·笔记·学习·线性代数·考研
Run_Teenage20 小时前
Linux:线程互斥,线程锁
运维·开发语言·jvm
太阳上的雨天20 小时前
AI学习ing~
学习·ai·ai编程
小小de风呀20 小时前
de风——【从零开始学C++】(四):类和对象(下)
开发语言·c++·算法
覆东流20 小时前
第10天:python元组
开发语言·后端·python
CSCN新手听安20 小时前
【Qt】系统相关(一)内容简介,事件概念,事件的处理
开发语言·c++·qt
不想写代码的星星20 小时前
重识 std::tuple:一个被低估的编译期异构容器
开发语言·c++
麦田里的粮仓21 小时前
文档站点生成器 - Fumadocs
学习
techdashen21 小时前
用 Rust 写生产级服务要踩多少坑——Cloudflare 把答案做成了一个开源库
开发语言·rust·开源