2017-08-15 02:29:52 -05:00
|
|
|
package main
|
|
|
|
|
2017-08-28 01:57:27 -05:00
|
|
|
import (
|
2017-09-09 22:34:39 -05:00
|
|
|
"bytes"
|
|
|
|
"crypto/ecdsa"
|
2017-09-02 23:35:36 -05:00
|
|
|
"encoding/hex"
|
2017-09-09 22:34:39 -05:00
|
|
|
"errors"
|
2017-08-29 04:27:33 -05:00
|
|
|
"fmt"
|
2017-08-28 01:57:27 -05:00
|
|
|
"log"
|
2017-09-02 23:17:10 -05:00
|
|
|
"os"
|
2017-08-28 01:57:27 -05:00
|
|
|
|
|
|
|
"github.com/boltdb/bolt"
|
|
|
|
)
|
|
|
|
|
|
|
|
const dbFile = "blockchain.db"
|
|
|
|
const blocksBucket = "blocks"
|
2017-09-04 23:34:47 -05:00
|
|
|
const genesisCoinbaseData = "The Times 03/Jan/2009 Chancellor on brink of second bailout for banks"
|
2017-08-28 01:57:27 -05:00
|
|
|
|
2017-09-03 23:02:24 -05:00
|
|
|
// Blockchain implements interactions with a DB
|
2017-08-15 02:29:52 -05:00
|
|
|
type Blockchain struct {
|
2017-08-28 04:16:45 -05:00
|
|
|
tip []byte
|
2017-08-28 04:28:23 -05:00
|
|
|
db *bolt.DB
|
2017-08-28 01:57:27 -05:00
|
|
|
}
|
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
// CreateBlockchain creates a new blockchain DB
|
|
|
|
func CreateBlockchain(address string) *Blockchain {
|
|
|
|
if dbExists() {
|
|
|
|
fmt.Println("Blockchain already exists.")
|
|
|
|
os.Exit(1)
|
2017-09-09 22:34:39 -05:00
|
|
|
}
|
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
var tip []byte
|
|
|
|
db, err := bolt.Open(dbFile, 0600, nil)
|
2017-08-28 04:28:23 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
err = db.Update(func(tx *bolt.Tx) error {
|
|
|
|
cbtx := NewCoinbaseTX(address, genesisCoinbaseData)
|
|
|
|
genesis := NewGenesisBlock(cbtx)
|
2017-08-28 01:57:27 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
b, err := tx.CreateBucket([]byte(blocksBucket))
|
2017-08-28 01:57:27 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
err = b.Put(genesis.Hash, genesis.Serialize())
|
2017-08-28 01:57:27 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
err = b.Put([]byte("l"), genesis.Hash)
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
tip = genesis.Hash
|
2017-08-28 01:57:27 -05:00
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
2017-09-09 22:54:58 -05:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
bc := Blockchain{tip, db}
|
|
|
|
|
|
|
|
return &bc
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewBlockchain creates a new Blockchain with genesis Block
|
|
|
|
func NewBlockchain(address string) *Blockchain {
|
|
|
|
if dbExists() == false {
|
|
|
|
fmt.Println("No existing blockchain found. Create one first.")
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
var tip []byte
|
|
|
|
db, err := bolt.Open(dbFile, 0600, nil)
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = db.Update(func(tx *bolt.Tx) error {
|
|
|
|
b := tx.Bucket([]byte(blocksBucket))
|
|
|
|
tip = b.Get([]byte("l"))
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
bc := Blockchain{tip, db}
|
|
|
|
|
|
|
|
return &bc
|
|
|
|
}
|
|
|
|
|
|
|
|
// FindSpendableOutputs finds and returns unspent outputs to reference in inputs
|
|
|
|
func (bc *Blockchain) FindSpendableOutputs(pubKeyHash []byte, amount int) (int, map[string][]int) {
|
|
|
|
unspentOutputs := make(map[string][]int)
|
|
|
|
unspentTXs := bc.FindUnspentTransactions(pubKeyHash)
|
|
|
|
accumulated := 0
|
|
|
|
|
|
|
|
Work:
|
|
|
|
for _, tx := range unspentTXs {
|
|
|
|
txID := hex.EncodeToString(tx.ID)
|
|
|
|
|
|
|
|
for outIdx, out := range tx.Vout {
|
|
|
|
if out.Unlock(pubKeyHash) && accumulated < amount {
|
|
|
|
accumulated += out.Value
|
|
|
|
unspentOutputs[txID] = append(unspentOutputs[txID], outIdx)
|
|
|
|
|
|
|
|
if accumulated >= amount {
|
|
|
|
break Work
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return accumulated, unspentOutputs
|
2017-08-28 01:57:27 -05:00
|
|
|
}
|
|
|
|
|
2017-09-09 22:34:39 -05:00
|
|
|
// FindTransaction finds a transaction by its ID
|
|
|
|
func (bc *Blockchain) FindTransaction(ID []byte) (Transaction, error) {
|
|
|
|
bci := bc.Iterator()
|
|
|
|
|
|
|
|
for {
|
|
|
|
block := bci.Next()
|
|
|
|
|
|
|
|
for _, tx := range block.Transactions {
|
|
|
|
if bytes.Compare(tx.ID, ID) == 0 {
|
|
|
|
return *tx, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(block.PrevBlockHash) == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Transaction{}, errors.New("Transaction is not found")
|
|
|
|
}
|
|
|
|
|
2017-09-03 23:02:24 -05:00
|
|
|
// FindUnspentTransactions returns a list of transactions containing unspent outputs
|
2017-09-07 23:31:34 -05:00
|
|
|
func (bc *Blockchain) FindUnspentTransactions(pubKeyHash []byte) []Transaction {
|
2017-09-05 09:35:45 -05:00
|
|
|
var unspentTXs []Transaction
|
2017-09-03 23:02:24 -05:00
|
|
|
spentTXOs := make(map[string][]int)
|
2017-09-02 22:41:45 -05:00
|
|
|
bci := bc.Iterator()
|
|
|
|
|
|
|
|
for {
|
|
|
|
block := bci.Next()
|
|
|
|
|
|
|
|
for _, tx := range block.Transactions {
|
2017-09-03 23:32:24 -05:00
|
|
|
txID := hex.EncodeToString(tx.ID)
|
2017-09-02 22:41:45 -05:00
|
|
|
|
2017-09-02 23:35:36 -05:00
|
|
|
Outputs:
|
2017-09-03 23:02:24 -05:00
|
|
|
for outIdx, out := range tx.Vout {
|
2017-09-02 22:41:45 -05:00
|
|
|
// Was the output spent?
|
2017-09-03 23:02:24 -05:00
|
|
|
if spentTXOs[txID] != nil {
|
|
|
|
for _, spentOut := range spentTXOs[txID] {
|
|
|
|
if spentOut == outIdx {
|
2017-09-02 23:35:36 -05:00
|
|
|
continue Outputs
|
2017-09-02 22:41:45 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-07 23:31:34 -05:00
|
|
|
if out.Unlock(pubKeyHash) {
|
2017-09-05 09:35:45 -05:00
|
|
|
unspentTXs = append(unspentTXs, *tx)
|
2017-09-02 22:41:45 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if tx.IsCoinbase() == false {
|
|
|
|
for _, in := range tx.Vin {
|
2017-09-07 23:31:34 -05:00
|
|
|
if in.UnlocksOutputWith(pubKeyHash) {
|
2017-09-03 23:02:24 -05:00
|
|
|
inTxID := hex.EncodeToString(in.Txid)
|
|
|
|
spentTXOs[inTxID] = append(spentTXOs[inTxID], in.Vout)
|
2017-09-02 22:41:45 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(block.PrevBlockHash) == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-02 23:01:02 -05:00
|
|
|
return unspentTXs
|
|
|
|
}
|
|
|
|
|
2017-09-05 02:33:33 -05:00
|
|
|
// FindUTXO finds and returns all unspent transaction outputs
|
2017-09-07 23:31:34 -05:00
|
|
|
func (bc *Blockchain) FindUTXO(pubKeyHash []byte) []TXOutput {
|
2017-09-05 02:33:33 -05:00
|
|
|
var UTXOs []TXOutput
|
2017-09-07 23:31:34 -05:00
|
|
|
unspentTransactions := bc.FindUnspentTransactions(pubKeyHash)
|
2017-09-05 02:33:33 -05:00
|
|
|
|
|
|
|
for _, tx := range unspentTransactions {
|
|
|
|
for _, out := range tx.Vout {
|
2017-09-07 23:31:34 -05:00
|
|
|
if out.Unlock(pubKeyHash) {
|
2017-09-05 02:33:33 -05:00
|
|
|
UTXOs = append(UTXOs, out)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return UTXOs
|
|
|
|
}
|
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
// MineBlock mines a new block with the provided transactions
|
|
|
|
func (bc *Blockchain) MineBlock(transactions []*Transaction) {
|
|
|
|
var lastHash []byte
|
2017-09-02 23:01:02 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
for _, tx := range transactions {
|
|
|
|
if bc.VerifyTransaction(tx) != true {
|
|
|
|
log.Panic("ERROR: Invalid transaction")
|
2017-09-02 23:01:02 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
err := bc.db.View(func(tx *bolt.Tx) error {
|
2017-08-28 01:57:27 -05:00
|
|
|
b := tx.Bucket([]byte(blocksBucket))
|
2017-09-09 22:54:58 -05:00
|
|
|
lastHash = b.Get([]byte("l"))
|
2017-08-28 01:57:27 -05:00
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
2017-08-28 04:28:23 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
newBlock := NewBlock(transactions, lastHash)
|
2017-09-02 23:17:10 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
err = bc.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
b := tx.Bucket([]byte(blocksBucket))
|
|
|
|
err := b.Put(newBlock.Hash, newBlock.Serialize())
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
2017-09-02 23:17:10 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
err = b.Put([]byte("l"), newBlock.Hash)
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
2017-08-28 01:57:27 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
bc.tip = newBlock.Hash
|
2017-08-28 01:57:27 -05:00
|
|
|
|
2017-09-02 23:17:10 -05:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
// SignTransaction signs a Transaction
|
|
|
|
func (bc *Blockchain) SignTransaction(tx *Transaction, privKey ecdsa.PrivateKey) {
|
|
|
|
prevTXs := make(map[string]Transaction)
|
2017-09-02 23:17:10 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
for _, vin := range tx.Vin {
|
|
|
|
prevTX, err := bc.FindTransaction(vin.Txid)
|
2017-09-02 23:17:10 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
2017-09-09 22:54:58 -05:00
|
|
|
prevTXs[hex.EncodeToString(prevTX.ID)] = prevTX
|
|
|
|
}
|
2017-09-02 23:17:10 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
tx.Sign(privKey, prevTXs)
|
|
|
|
}
|
2017-09-02 23:17:10 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
// VerifyTransaction verifies transaction
|
|
|
|
func (bc *Blockchain) VerifyTransaction(tx *Transaction) bool {
|
|
|
|
prevTXs := make(map[string]Transaction)
|
|
|
|
|
|
|
|
for _, vin := range tx.Vin {
|
|
|
|
prevTX, err := bc.FindTransaction(vin.Txid)
|
2017-09-02 23:17:10 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
2017-08-28 04:16:45 -05:00
|
|
|
}
|
2017-09-09 22:54:58 -05:00
|
|
|
prevTXs[hex.EncodeToString(prevTX.ID)] = prevTX
|
|
|
|
}
|
2017-08-28 04:28:23 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
for _, tx := range prevTXs {
|
|
|
|
fmt.Println(tx)
|
2017-08-28 01:57:27 -05:00
|
|
|
}
|
2017-09-09 22:54:58 -05:00
|
|
|
// fmt.Println()
|
|
|
|
// fmt.Println(tx)
|
2017-08-28 01:57:27 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
return tx.Verify(prevTXs)
|
|
|
|
}
|
2017-08-28 04:28:23 -05:00
|
|
|
|
2017-09-09 22:54:58 -05:00
|
|
|
// Iterator returns a BlockchainIterat
|
|
|
|
func (bc *Blockchain) Iterator() *BlockchainIterator {
|
|
|
|
bci := &BlockchainIterator{bc.tip, bc.db}
|
|
|
|
|
|
|
|
return bci
|
|
|
|
}
|
|
|
|
|
|
|
|
func dbExists() bool {
|
|
|
|
if _, err := os.Stat(dbFile); os.IsNotExist(err) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
2017-08-15 02:29:52 -05:00
|
|
|
}
|