如何从R ||中的散列消息和签名正确恢复ECDSA公钥。 S || V格式?

时间:2018-06-30 04:00:14

标签: go cryptography ethereum ecdsa go-ethereum

我已使用以下代码生成ecdsa密钥对(privKeypubKey),对其进行编码,然后再解码回去:https://stackoverflow.com/a/41315404/1901320

接下来,我使用txnData.Payload()为消息([]byte类型为crypto.Keccak256())创建哈希,并使用以太坊的加密软件包({{1} }。这会在R ||中创建一个65位ECDSA签名。 S || V格式。

crypto.Sign()

当我尝试使用github.com/ethereum/go-ethereum/crypto hashData := crypto.Keccak256(txnData.Payload) sig, _ := crypto.Sign(hashData, privKey) pkey, _ := crypto.Ecrecover(hashData, sig) // This and pubKey do not match 和ECDSA签名中获取公钥并将其与所使用的hashData对应的公钥crypto.Ecrecover()进行比较时要创建签名,我发现公钥不匹配。这似乎不应该发生。关于我在哪里出错的任何想法吗?

1 个答案:

答案 0 :(得分:3)

这是一个完整的工作示例,说明如何使用以太坊生成和验证签名。

package main

import (
    "bytes"
    "crypto/ecdsa"
    "fmt"
    "log"

    "github.com/ethereum/go-ethereum/common/hexutil"
    "github.com/ethereum/go-ethereum/crypto"
)

func main() {
    privateKey, err := crypto.HexToECDSA("fad9c8855b740a0b7ed4c221dbad0f33a83a49cad6b3fe8d5817ac83d38b6a19")
    if err != nil {
        log.Fatal(err)
    }

    publicKey := privateKey.PublicKey

    publicKeyBytes := crypto.FromECDSAPub(&publicKey)

    data := []byte("hello")
    hash := crypto.Keccak256Hash(data)
    fmt.Println(hash.Hex()) // 0x1c8aff950685c2ed4bc3174f3472287b56d9517b9c948127319a09a7a36deac8

    signature, err := crypto.Sign(hash.Bytes(), privateKey)
    if err != nil {
        log.Fatal(err)
    }

    fmt.Println(hexutil.Encode(signature)) // 0x789a80053e4927d0a898db8e065e948f5cf086e32f9ccaa54c1908e22ac430c62621578113ddbb62d509bf6049b8fb544ab06d36f916685a2eb8e57ffadde02301

    sigPublicKey, err := crypto.Ecrecover(hash.Bytes(), signature)
    if err != nil {
        log.Fatal(err)
    }

    matches := bytes.Equal(sigPublicKey, publicKeyBytes)
    fmt.Println(matches) // true

    sigPublicKeyECDSA, err := crypto.SigToPub(hash.Bytes(), signature)
    if err != nil {
        log.Fatal(err)
    }

    sigPublicKeyBytes := crypto.FromECDSAPub(sigPublicKeyECDSA)
    matches = bytes.Equal(sigPublicKeyBytes, publicKeyBytes)
    fmt.Println(matches) // true

    signatureNoRecoverID := signature[:len(signature)-1] // remove recovery id
    verified := crypto.VerifySignature(publicKeyBytes, hash.Bytes(), signatureNoRecoverID)
    fmt.Println(verified) // true
}

请查看Ethereum Development with Go指南,以获取有关使用以太坊的更多示例。