区块链进阶之路

以太坊交易签名解析源码解读

2020-07-12  本文已影响0人  六天天天向上

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

一、准备工作

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

package main
import (
    "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))
}

其中:

二、签名解析

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

// go-ethereum/core/types/transaction_signing.go
func (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方法中:

recoverPlain源码如下所示。

// go-ethereum/core/types/transaction_signing.go
func 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方法的参数分别为:

在recoverPlain方法中:

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

上一篇 下一篇

猜你喜欢

热点阅读