代码拉取完成,页面将自动刷新
同步操作将从 duke.du/go-bitcoin-demo 强制同步,此操作会覆盖自 Fork 仓库以来所做的任何修改,且无法恢复!!!
确定后同步将在后台操作,完成时将刷新页面,请耐心等待。
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)
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。