1 Star 4 Fork 8

duke.du/go-bitcoin-demo

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
wallet.go 2.89 KB
一键复制 编辑 原始数据 按行查看 历史
duke.du 提交于 2022-08-12 23:53 . add README docs
package main
import (
"bytes"
"crypto/ecdsa"
"crypto/elliptic"
"crypto/rand"
"crypto/sha256"
"fmt"
"github.com/btcsuite/btcutil/base58"
"golang.org/x/crypto/ripemd160"
)
// - 结构定义
type wallet struct {
//私钥
PriKey *ecdsa.PrivateKey
//公钥原型定义
// type PublicKey struct {
// elliptic.Curve
// X, Y *big.Int
// }
// 公钥, X,Y类型一致,长度一致,我们将X和Y拼接成字节流,赋值给pubKey字段,用于传输
// 验证时,将X,Y截取出来(类似r,s),再创建一条曲线,就可以还原公钥,进一步进行校验
PubKey []byte
}
// - ==创建秘钥对==
func newWalletKeyPair() *wallet {
curve := elliptic.P256()
//创建私钥
priKey, err := ecdsa.GenerateKey(curve, rand.Reader)
if err != nil {
fmt.Println("ecdsa.GenerateKey err:", err)
return nil
}
//获取公钥
pubKeyRaw := priKey.PublicKey
//将公钥X,Y拼接到一起
pubKey := append(pubKeyRaw.X.Bytes(), pubKeyRaw.Y.Bytes()...)
//创建wallet结构返回
wallet := wallet{priKey, pubKey}
return &wallet
}
// - ==根据私钥生成地址==
func (w *wallet) getAddress() string {
//公钥
// pubKey := w.PubKey
pubKeyHash := getPubKeyHashFromPubKey(w.PubKey)
//拼接version和公钥哈希,得到21字节的数据
payload := append([]byte{byte(0x00)}, pubKeyHash...)
//生成4字节的校验码
checksum := checkSum(payload)
//25字节数据
payload = append(payload, checksum...)
address := base58.Encode(payload)
return address
}
//给定公钥,得到公钥哈希值
func getPubKeyHashFromPubKey(pubKey []byte) []byte {
hash1 := sha256.Sum256(pubKey)
//hash160处理
hasher := ripemd160.New()
hasher.Write(hash1[:])
// 公钥哈希,锁定output时就是使用这值
pubKeyHash := hasher.Sum(nil)
return pubKeyHash
}
//通过地址,反推出公钥哈希,注意不是公钥
func getPubKeyHashFromAddress(address string) []byte {
//base58解码
decodeInfo := base58.Decode(address)
if len(decodeInfo) != 25 {
fmt.Println("getPubKeyHashFromAddress, 传入地址无效")
return nil
}
//需要校验一下地址
//截取
pubKeyHash := decodeInfo[1 : len(decodeInfo)-4]
return pubKeyHash
}
//得到4字节的校验码
func checkSum(payload []byte) []byte {
first := sha256.Sum256(payload)
second := sha256.Sum256(first[:])
//4字节checksum
checksum := second[0:4]
return checksum
}
func isValidAddress(address string) bool {
// 解码,得到25字节数据
decodeInfo := base58.Decode(address)
if len(decodeInfo) != 25 {
fmt.Println("isValidAddress, 传入地址长度无效")
return false
}
// 截取前21字节payload,截取后四字节checksum1
payload := decodeInfo[:len(decodeInfo)-4] //21字节
checksum1 := decodeInfo[len(decodeInfo)-4:] //4字节
// 对palyload计算,得到checksum2,与checksum1对比,true校验成功,反之失败
checksum2 := checkSum(payload)
return bytes.Equal(checksum1, checksum2)
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
Go
1
https://gitee.com/bwcs/go-bitcoin-demo.git
[email protected]:bwcs/go-bitcoin-demo.git
bwcs
go-bitcoin-demo
go-bitcoin-demo
main

搜索帮助