2017-08-15 02:29:52 -05:00
|
|
|
package main
|
|
|
|
|
2017-08-28 01:57:27 -05:00
|
|
|
import (
|
2017-08-29 04:27:33 -05:00
|
|
|
"fmt"
|
2017-08-28 01:57:27 -05:00
|
|
|
"log"
|
|
|
|
|
|
|
|
"github.com/boltdb/bolt"
|
|
|
|
)
|
|
|
|
|
|
|
|
const dbFile = "blockchain.db"
|
|
|
|
const blocksBucket = "blocks"
|
2017-09-02 21:45:49 -05:00
|
|
|
const genesisCoinbase = "The Times 03/Jan/2009 Chancellor on brink of second bailout for banks"
|
2017-08-28 01:57:27 -05:00
|
|
|
|
2017-08-15 02:29:52 -05:00
|
|
|
// Blockchain keeps a sequence of Blocks
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
// BlockchainIterator is used to iterate over blockchain blocks
|
|
|
|
type BlockchainIterator struct {
|
|
|
|
currentHash []byte
|
2017-08-28 04:28:23 -05:00
|
|
|
db *bolt.DB
|
2017-08-15 02:29:52 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// AddBlock saves provided data as a block in the blockchain
|
2017-09-02 21:56:43 -05:00
|
|
|
func (bc *Blockchain) AddBlock(transactions []*Transaction) {
|
2017-08-28 01:57:27 -05:00
|
|
|
var lastHash []byte
|
|
|
|
|
2017-08-28 04:28:23 -05:00
|
|
|
err := bc.db.View(func(tx *bolt.Tx) error {
|
2017-08-28 01:57:27 -05:00
|
|
|
b := tx.Bucket([]byte(blocksBucket))
|
|
|
|
lastHash = b.Get([]byte("l"))
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
2017-08-28 04:28:23 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
2017-09-02 21:56:43 -05:00
|
|
|
newBlock := NewBlock(transactions, lastHash)
|
2017-08-28 01:57:27 -05:00
|
|
|
|
2017-08-28 04:28:23 -05:00
|
|
|
err = bc.db.Update(func(tx *bolt.Tx) error {
|
2017-08-28 01:57:27 -05:00
|
|
|
b := tx.Bucket([]byte(blocksBucket))
|
|
|
|
err := b.Put(newBlock.Hash, newBlock.Serialize())
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = b.Put([]byte("l"), newBlock.Hash)
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
bc.tip = newBlock.Hash
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-09-02 22:41:45 -05:00
|
|
|
// FindUTXOs finds and returns unspend transaction outputs for the address
|
|
|
|
func (bc *Blockchain) FindUTXOs(address string, amount int) (int, map[string][]int) {
|
|
|
|
var spentTXs map[string][]int
|
|
|
|
var unspentTXs map[string][]int
|
|
|
|
accumulated := 0
|
|
|
|
bci := bc.Iterator()
|
|
|
|
|
|
|
|
for {
|
|
|
|
block := bci.Next()
|
|
|
|
|
|
|
|
Work:
|
|
|
|
for _, tx := range block.Transactions {
|
|
|
|
txid := string(tx.GetHash())
|
|
|
|
|
|
|
|
for outid, out := range tx.Vout {
|
|
|
|
// Was the output spent?
|
|
|
|
if spentTXs[txid] != nil {
|
|
|
|
for _, spentOut := range spentTXs[txid] {
|
|
|
|
if spentOut == outid {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if out.Unlock(address) && accumulated < amount {
|
|
|
|
accumulated += out.Value
|
|
|
|
unspentTXs[txid] = append(unspentTXs[txid], outid)
|
|
|
|
|
|
|
|
if accumulated >= amount {
|
|
|
|
break Work
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if tx.IsCoinbase() == false {
|
|
|
|
for _, in := range tx.Vin {
|
|
|
|
if in.LockedBy(address) {
|
|
|
|
spentTXs[string(in.Txid)] = append(spentTXs[string(in.Txid)], in.Vout)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(block.PrevBlockHash) == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return accumulated, unspentTXs
|
|
|
|
}
|
|
|
|
|
2017-08-28 01:57:27 -05:00
|
|
|
// Iterator ...
|
|
|
|
func (bc *Blockchain) Iterator() *BlockchainIterator {
|
2017-08-28 04:28:23 -05:00
|
|
|
bci := &BlockchainIterator{bc.tip, bc.db}
|
2017-08-28 01:57:27 -05:00
|
|
|
|
|
|
|
return bci
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next returns next block starting from the tip
|
|
|
|
func (i *BlockchainIterator) Next() *Block {
|
|
|
|
var block *Block
|
|
|
|
|
2017-08-28 04:28:23 -05:00
|
|
|
err := i.db.View(func(tx *bolt.Tx) error {
|
2017-08-28 01:57:27 -05:00
|
|
|
b := tx.Bucket([]byte(blocksBucket))
|
|
|
|
encodedBlock := b.Get(i.currentHash)
|
|
|
|
block = DeserializeBlock(encodedBlock)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
2017-08-28 04:28:23 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
2017-08-28 01:57:27 -05:00
|
|
|
i.currentHash = block.PrevBlockHash
|
|
|
|
|
|
|
|
return block
|
2017-08-15 02:29:52 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewBlockchain creates a new Blockchain with genesis Block
|
2017-09-02 21:45:49 -05:00
|
|
|
func NewBlockchain(address string) *Blockchain {
|
2017-08-28 04:28:23 -05:00
|
|
|
var tip []byte
|
2017-08-28 04:16:45 -05:00
|
|
|
db, err := bolt.Open(dbFile, 0600, nil)
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
2017-08-28 01:57:27 -05:00
|
|
|
|
2017-08-28 04:16:45 -05:00
|
|
|
err = db.Update(func(tx *bolt.Tx) error {
|
|
|
|
b := tx.Bucket([]byte(blocksBucket))
|
2017-08-28 01:57:27 -05:00
|
|
|
|
2017-08-28 04:16:45 -05:00
|
|
|
if b == nil {
|
2017-08-29 04:27:33 -05:00
|
|
|
fmt.Println("No existing blockchain found. Creating a new one...")
|
2017-09-02 21:45:49 -05:00
|
|
|
cbtx := NewCoinbaseTX(address, genesisCoinbase)
|
|
|
|
genesis := NewGenesisBlock(cbtx)
|
2017-08-28 01:57:27 -05:00
|
|
|
|
|
|
|
b, err := tx.CreateBucket([]byte(blocksBucket))
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = b.Put(genesis.Hash, genesis.Serialize())
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = b.Put([]byte("l"), genesis.Hash)
|
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
|
|
|
}
|
2017-08-28 04:28:23 -05:00
|
|
|
tip = genesis.Hash
|
2017-08-28 04:16:45 -05:00
|
|
|
} else {
|
2017-08-28 04:28:23 -05:00
|
|
|
tip = b.Get([]byte("l"))
|
2017-08-28 04:16:45 -05:00
|
|
|
}
|
2017-08-28 01:57:27 -05:00
|
|
|
|
2017-08-28 04:16:45 -05:00
|
|
|
return nil
|
|
|
|
})
|
2017-08-28 04:28:23 -05:00
|
|
|
|
2017-08-28 04:16:45 -05:00
|
|
|
if err != nil {
|
|
|
|
log.Panic(err)
|
2017-08-28 01:57:27 -05:00
|
|
|
}
|
|
|
|
|
2017-08-28 04:28:23 -05:00
|
|
|
bc := Blockchain{tip, db}
|
|
|
|
|
2017-08-28 01:57:27 -05:00
|
|
|
return &bc
|
2017-08-15 02:29:52 -05:00
|
|
|
}
|