用于生成签名的组件是:签名者私钥,以及将要签名的数据的哈希。 只要输出为32字节,就可以使用任何哈希算法。 我们将使用Keccak-256作为哈希算法,这是以太坊常常使用的算法。
首先,我们将加载私钥。

  1. privateKey, err := crypto.HexToECDSA("fad9c8855b740a0b7ed4c221dbad0f33a83a49cad6b3fe8d5817ac83d38b6a19")
  2. if err != nil {
  3. log.Fatal(err)
  4. }

接下来我们将获取我们希望签名的数据的Keccak-256,在这个例子里,它将是hello。 go-ethereumcrypto包提供了一个方便的Keccak256Hash方法来实现这一目的。

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

最后,我们使用私钥签名哈希,得到签名。

  1. signature, err := crypto.Sign(hash.Bytes(), privateKey)
  2. if err != nil {
  3. log.Fatal(err)
  4. }
  5. fmt.Println(hexutil.Encode(signature)) // 0x789a80053e4927d0a898db8e065e948f5cf086e32f9ccaa54c1908e22ac430c62621578113ddbb62d509bf6049b8fb544ab06d36f916685a2eb8e57ffadde02301

现在我们已经成功生成了签名,在下个章节中,我们将学习如何验证签名确实是由该私钥的持有者签名的。


完整代码

signature_generate.go

  1. package main
  2. import (
  3. "fmt"
  4. "log"
  5. "github.com/ethereum/go-ethereum/common/hexutil"
  6. "github.com/ethereum/go-ethereum/crypto"
  7. )
  8. func main() {
  9. privateKey, err := crypto.HexToECDSA("fad9c8855b740a0b7ed4c221dbad0f33a83a49cad6b3fe8d5817ac83d38b6a19")
  10. if err != nil {
  11. log.Fatal(err)
  12. }
  13. data := []byte("hello")
  14. hash := crypto.Keccak256Hash(data)
  15. fmt.Println(hash.Hex()) // 0x1c8aff950685c2ed4bc3174f3472287b56d9517b9c948127319a09a7a36deac8
  16. signature, err := crypto.Sign(hash.Bytes(), privateKey)
  17. if err != nil {
  18. log.Fatal(err)
  19. }
  20. fmt.Println(hexutil.Encode(signature)) // 0x789a80053e4927d0a898db8e065e948f5cf086e32f9ccaa54c1908e22ac430c62621578113ddbb62d509bf6049b8fb544ab06d36f916685a2eb8e57ffadde02301
  21. }