go語言實(shí)現(xiàn)簡(jiǎn)易比特幣系統(tǒng)之交易簽名及校驗(yàn)功能
介紹
簽名的輸入:
- 待簽名的交易數(shù)據(jù),包括輸入和輸出
- 引用的UTXO信息
- 私鑰
簽名的輸出:
- 數(shù)字?jǐn)?shù)字簽名
- 公鑰
簽名的目的
- 證明交易所引用的UTXO的確屬于付款人
- 證明交易的所有數(shù)據(jù)的確是付款人提供的,且未被修改過
簽名中需要的數(shù)據(jù)
- UTXO中的PubKeyHash,這描述了付款人
- 新生成UTXO中的PubKeyHash,這描述了收款人
- 由于每一筆交易都可能引用多個(gè)UTXO,因?yàn)槎鄠€(gè)UTXO可能存在于多條交易中。所以我們需要遍歷所以的引用交易,并對(duì)他們逐個(gè)簽名
簽名過程
用解鎖腳本解鎖對(duì)應(yīng)的UTXO鎖定腳本
簽名
//簽名的具體實(shí)現(xiàn), 參數(shù):私鑰,inputs里面所有引用的交易的結(jié)構(gòu)map[string]Transaction func (tx *Transaction) Sign(privateKey *ecdsa.PrivateKey, prevTXs map[string]Transaction){ //1. 創(chuàng)建一個(gè)當(dāng)前交易的副本:txCopy,使用函數(shù):TrimmedCopy:要把Signature和PubKey字段設(shè)置為null //2. 循環(huán)遍歷txCopy的inputs,得到這個(gè)input索引的output的公鑰哈希 //3. 生成簽名的數(shù)據(jù),要簽名的數(shù)據(jù)一定是哈希值 //a. 我們對(duì)每一個(gè)input都簽名一次,簽名的數(shù)據(jù)是由當(dāng)前input引用的output的哈希+當(dāng)前的outputs(都存在當(dāng)前這個(gè)txCopy里面) //b. 對(duì)拼好的txCopy進(jìn)行哈希處理,SetHash得到TXID,這個(gè)TXID就是我們要簽名的最終數(shù)據(jù) //4. 執(zhí)行簽名動(dòng)作,得到r,s字節(jié)流 //5. 放到我們簽名的inputs的Signature中 if tx.IsCoinbase(){ return } //1. txCopy := tx.TrimmedCopy() //2. for i, input := range txCopy.TXInputs{ prevTX := prevTXs[string(input.Txid)] if len(prevTX.TXID) == 0{ log.Panic("引用的交易無效\n") } //不要對(duì)input進(jìn)行賦值,這是一個(gè)副本,要對(duì)txCopy.TXInput[xx]進(jìn)行操作,否則無法把pubKeyHash傳進(jìn)來 txCopy.TXInputs[i].PubKey = prevTX.TXOutputs[input.Index].PubKeyHash //3. //ab. //所需要的三個(gè)數(shù)據(jù)都具備了,開始做哈希處理 txCopy.SetHash() //還原,以免影響后面的input簽名 txCopy.TXInputs[i].PubKey = nil signDataHash := txCopy.TXID //4. r, s, err := ecdsa.Sign(rand.Reader, privateKey, signDataHash) if err != nil{ log.Panic(err) } //5. signature := append(r.Bytes(), s.Bytes()...) tx.TXInputs[i].Signature = signature } }
校驗(yàn)
func (tx *Transaction) Verify (prevTXs map[string]Transaction) bool{ if tx.IsCoinbase(){ return true } //1. 得到簽名的數(shù)據(jù) //2. 得到signature,反退回r,s //3. 拆解PubKey, X,Y得到原生公鑰 //4. Verify //1. txCopy := tx.TrimmedCopy() for i, input := range tx.TXInputs{ prevTX := prevTXs[string(input.Txid)] if len(prevTX.TXID) == 0{ log.Panic("引用的交易無效\n") } txCopy.TXInputs[i].PubKey = prevTX.TXOutputs[input.Index].PubKeyHash txCopy.SetHash() dataHash := txCopy.TXID //2 signature := input.Signature //拆r,s //3 pubKey := input.PubKey //拆r,s r := big.Int{} s := big.Int{} r.SetBytes(signature[:len(signature)/2]) s.SetBytes(signature[len(signature)/2:]) X := big.Int{} Y := big.Int{} //b. pubKey平均分,前半部分給X,后半部分給Y X.SetBytes(pubKey[:len(pubKey)/2]) Y.SetBytes(pubKey[len(pubKey)/2:]) pubKeyOrigin := ecdsa.PublicKey{elliptic.P256(), &X, &Y} //4 if !ecdsa.Verify(&pubKeyOrigin, dataHash, &r, &s){ return false } } return true }
拷貝交易
//拷貝方法,用來引用交易 func (tx *Transaction) TrimmedCopy() Transaction{ var inputs []TXInput var outputs []TXOutput for _, input := range tx.TXInputs{ inputs = append(inputs, TXInput{input.Txid, input.Index, nil, nil}) } for _, output := range tx.TXOutputs{ outputs = append(outputs, output) } return Transaction{tx.TXID, inputs, outputs} }
最后
本套源碼來源于黑馬程序員,在此十分感謝黑馬程序員的教程!
源碼:https://gitee.com/xiaoshengdada/go_bitcoin/tree/master/v6
如果有任何問題可以來微信群交流,另外群里有學(xué)習(xí)資料,可以自行下載。一起學(xué)習(xí)進(jìn)步。
到此這篇關(guān)于go語言實(shí)現(xiàn)簡(jiǎn)易比特幣系統(tǒng)之交易簽名及校驗(yàn)功能的文章就介紹到這了,更多相關(guān)go語言比特幣交易簽名校驗(yàn)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Golang實(shí)現(xiàn)解析JSON的三種方法總結(jié)
這篇文章主要為大家詳細(xì)介紹了Golang實(shí)現(xiàn)解析JSON的三種方法,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)了解JSON有一定幫助,需要的可以參考一下2022-09-09jenkins構(gòu)建go及java項(xiàng)目的方法
這篇文章主要介紹了jenkins構(gòu)建go及java項(xiàng)目,本文通過圖文實(shí)例相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值了,需要的朋友可以參考下2021-04-04深入理解Go高級(jí)并發(fā)模式編寫更高效可擴(kuò)展的應(yīng)用程序
Go對(duì)并發(fā)提供了強(qiáng)大的原生支持,本文討論Go的高級(jí)并發(fā)模式,理解這些并發(fā)模式,可以幫助我們編寫高效的Go應(yīng)用程序,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-02-02深入了解Golang網(wǎng)絡(luò)編程N(yùn)et包的使用
net包主要是增加?context?控制,封裝了一些不同的連接類型以及DNS?查找等等,同時(shí)在有需要的地方引入?goroutine?提高處理效率。本文主要和大家分享下在Go中網(wǎng)絡(luò)編程的實(shí)現(xiàn),需要的可以參考一下2022-07-07golang常用加密解密算法總結(jié)(AES、DES、RSA、Sha1、MD5)
在項(xiàng)目開發(fā)過程中,當(dāng)操作一些用戶的隱私信息,本文主要主要介紹了golang常用加密解密算法總結(jié)(AES、DES、RSA、Sha1MD5),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04Go語言實(shí)現(xiàn)棧與隊(duì)列基本操作學(xué)家
go語言中,并沒有棧與隊(duì)列相關(guān)的數(shù)據(jù)結(jié)構(gòu),但是我們可以借助切片來實(shí)現(xiàn)棧與隊(duì)列的操作;接下來我們一起實(shí)現(xiàn)棧與隊(duì)列基本操作,感興趣的可以了解一下2022-11-11Golang 中的可測(cè)試示例函數(shù)(Example Function)詳解
這篇文章詳細(xì)講解了 Golang 中的可測(cè)試示例函數(shù),示例函數(shù)類似于單元測(cè)試函數(shù),但沒有 *testing 類型的參數(shù),編寫示例函數(shù)也是很容易的,本文就通過代碼示例給大家介紹一下Golang的可測(cè)試示例函數(shù),需要的朋友可以參考下2023-07-07詳解如何在Go中實(shí)現(xiàn)優(yōu)雅停止
和其他語言相比,Go 中有相同也有不同,相同的是實(shí)現(xiàn)思路上和其他語言沒啥差異,不同在于 Go 采用的是 goroutine + channel 的并發(fā)模型,與傳統(tǒng)的進(jìn)程線程相比,實(shí)現(xiàn)細(xì)節(jié)上存在差異,本文將從實(shí)際場(chǎng)景和它的一般實(shí)現(xiàn)方式展開,逐步討論這個(gè)話題,需要的朋友可以參考下2024-04-04