blockchain_go/blockchain.go

244 lines
4.7 KiB
Go
Raw Normal View History

package main
2017-08-28 01:57:27 -05:00
import (
2017-09-02 23:35:36 -05:00
"encoding/hex"
"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"
const genesisCoinbase = "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
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-09-03 23:02:24 -05:00
// MineBlock mines a new block with the provided transactions
func (bc *Blockchain) MineBlock(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-03 23:02:24 -05:00
// FindUnspentTransactions returns a list of transactions containing unspent outputs
2017-09-02 23:01:02 -05:00
func (bc *Blockchain) FindUnspentTransactions(address string) []*Transaction {
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:02:24 -05:00
txID := hex.EncodeToString(tx.GetHash())
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
}
}
}
if out.CanBeUnlockedWith(address) {
2017-09-02 23:01:02 -05:00
unspentTXs = append(unspentTXs, tx)
2017-09-03 23:02:24 -05:00
continue Outputs
2017-09-02 22:41:45 -05:00
}
}
if tx.IsCoinbase() == false {
for _, in := range tx.Vin {
if in.CanUnlockOutputWith(address) {
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
}
// FindUTXOs finds and returns unspend transaction outputs for the address
func (bc *Blockchain) FindUTXOs(address string, amount int) (int, map[string][]int) {
2017-09-02 23:35:36 -05:00
unspentOutputs := make(map[string][]int)
2017-09-02 23:01:02 -05:00
unspentTXs := bc.FindUnspentTransactions(address)
accumulated := 0
Work:
for _, tx := range unspentTXs {
2017-09-03 23:02:24 -05:00
txID := hex.EncodeToString(tx.GetHash())
2017-09-02 23:01:02 -05:00
2017-09-03 23:02:24 -05:00
for outIdx, out := range tx.Vout {
if out.CanBeUnlockedWith(address) && accumulated < amount {
2017-09-02 23:01:02 -05:00
accumulated += out.Value
2017-09-03 23:02:24 -05:00
unspentOutputs[txID] = append(unspentOutputs[txID], outIdx)
2017-09-02 23:01:02 -05:00
if accumulated >= amount {
break Work
}
}
}
}
return accumulated, unspentOutputs
2017-09-02 22:41:45 -05:00
}
2017-09-03 23:02:24 -05:00
// Iterator returns a BlockchainIterat
2017-08-28 01:57:27 -05:00
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-09-02 23:17:10 -05:00
func dbExists() bool {
if _, err := os.Stat(dbFile); os.IsNotExist(err) {
return false
}
return true
}
// NewBlockchain creates a new Blockchain with genesis Block
func NewBlockchain(address string) *Blockchain {
2017-09-02 23:17:10 -05:00
if dbExists() == false {
2017-09-03 23:02:24 -05:00
fmt.Println("No existing blockchain found. Create one first.")
2017-09-02 23:17:10 -05:00
os.Exit(1)
}
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-09-02 23:17:10 -05:00
tip = b.Get([]byte("l"))
2017-08-28 01:57:27 -05:00
2017-09-02 23:17:10 -05:00
return nil
})
2017-08-28 01:57:27 -05:00
2017-09-02 23:17:10 -05:00
if err != nil {
log.Panic(err)
}
2017-08-28 01:57:27 -05:00
2017-09-02 23:17:10 -05:00
bc := Blockchain{tip, db}
2017-08-28 01:57:27 -05:00
2017-09-02 23:17:10 -05:00
return &bc
}
// CreateBlockchain creates a new blockchain DB
func CreateBlockchain(address string) *Blockchain {
if dbExists() {
fmt.Println("Blockchain already exists.")
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 {
cbtx := NewCoinbaseTX(address, genesisCoinbase)
genesis := NewGenesisBlock(cbtx)
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:16:45 -05:00
}
2017-09-02 23:17:10 -05:00
tip = genesis.Hash
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
}