blockchain_go/blockchain.go

365 lines
7.0 KiB
Go
Raw Normal View History

package main
2017-08-28 01:57:27 -05:00
import (
"bytes"
"crypto/ecdsa"
2017-09-02 23:35:36 -05:00
"encoding/hex"
"errors"
"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_%s.db"
2017-08-28 01:57:27 -05:00
const blocksBucket = "blocks"
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
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
}
// CreateBlockchain creates a new blockchain DB
func CreateBlockchain(address, nodeID string) *Blockchain {
dbFile := fmt.Sprintf(dbFile, nodeID)
if dbExists(dbFile) {
fmt.Println("Blockchain already exists.")
os.Exit(1)
}
var tip []byte
cbtx := NewCoinbaseTX(address, genesisCoinbaseData)
genesis := NewGenesisBlock(cbtx)
db, err := bolt.Open(dbFile, 0600, nil)
2017-08-28 04:28:23 -05:00
if err != nil {
log.Panic(err)
}
err = db.Update(func(tx *bolt.Tx) error {
b, err := tx.CreateBucket([]byte(blocksBucket))
2017-08-28 01:57:27 -05:00
if err != nil {
log.Panic(err)
}
err = b.Put(genesis.Hash, genesis.Serialize())
2017-08-28 01:57:27 -05:00
if err != nil {
log.Panic(err)
}
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
})
if err != nil {
log.Panic(err)
}
bc := Blockchain{tip, db}
return &bc
}
// NewBlockchain creates a new Blockchain with genesis Block
func NewBlockchain(nodeID string) *Blockchain {
dbFile := fmt.Sprintf(dbFile, nodeID)
if dbExists(dbFile) == 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
}
// Reset removes all blockchain data
func (bc *Blockchain) Reset() {
}
2017-10-01 08:30:21 -05:00
// AddBlock saves the block into the blockchain
func (bc *Blockchain) AddBlock(block *Block) {
err := bc.db.Update(func(tx *bolt.Tx) error {
b := tx.Bucket([]byte(blocksBucket))
blockInDb := b.Get(block.Hash)
if blockInDb != nil {
return nil
}
blockData := block.Serialize()
err := b.Put(block.Hash, blockData)
if err != nil {
log.Panic(err)
}
lastHash := b.Get([]byte("l"))
lastBlockData := b.Get(lastHash)
lastBlock := DeserializeBlock(lastBlockData)
if block.Height > lastBlock.Height {
err = b.Put([]byte("l"), block.Hash)
if err != nil {
log.Panic(err)
}
2017-10-01 08:30:21 -05:00
}
return nil
})
if err != nil {
log.Panic(err)
}
}
// 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-17 00:44:43 -05:00
// FindUTXO finds all unspent transaction outputs and returns transactions with spent outputs removed
func (bc *Blockchain) FindUTXO() map[string]TXOutputs {
2017-09-16 22:04:28 -05:00
UTXO := make(map[string]TXOutputs)
spentTXOs := make(map[string][]int)
bci := bc.Iterator()
for {
block := bci.Next()
for _, tx := range block.Transactions {
txID := hex.EncodeToString(tx.ID)
Outputs:
for outIdx, out := range tx.Vout {
// Was the output spent?
if spentTXOs[txID] != nil {
for _, spentOutIdx := range spentTXOs[txID] {
if spentOutIdx == outIdx {
continue Outputs
}
}
}
outs := UTXO[txID]
outs.Outputs = append(outs.Outputs, out)
UTXO[txID] = outs
}
if tx.IsCoinbase() == false {
for _, in := range tx.Vin {
inTxID := hex.EncodeToString(in.Txid)
spentTXOs[inTxID] = append(spentTXOs[inTxID], in.Vout)
}
}
}
if len(block.PrevBlockHash) == 0 {
break
}
}
return UTXO
}
// Iterator returns a BlockchainIterat
func (bc *Blockchain) Iterator() *BlockchainIterator {
bci := &BlockchainIterator{bc.tip, bc.db}
return bci
}
// GetBestHeight returns the height of the latest block
func (bc *Blockchain) GetBestHeight() int {
var lastBlock Block
err := bc.db.View(func(tx *bolt.Tx) error {
b := tx.Bucket([]byte(blocksBucket))
lastHash := b.Get([]byte("l"))
blockData := b.Get(lastHash)
lastBlock = *DeserializeBlock(blockData)
return nil
})
if err != nil {
log.Panic(err)
}
return lastBlock.Height
}
// GetBlock finds a block by its hash and returns it
func (bc *Blockchain) GetBlock(blockHash []byte) (Block, error) {
var block Block
err := bc.db.View(func(tx *bolt.Tx) error {
b := tx.Bucket([]byte(blocksBucket))
blockData := b.Get(blockHash)
if blockData == nil {
return errors.New("Block is not found.")
}
block = *DeserializeBlock(blockData)
return nil
})
if err != nil {
return block, err
}
return block, nil
}
2017-09-30 23:19:43 -05:00
// GetBlockHashes returns a list of hashes of all the blocks in the chain
func (bc *Blockchain) GetBlockHashes() [][]byte {
var blocks [][]byte
bci := bc.Iterator()
for {
block := bci.Next()
blocks = append(blocks, block.Hash)
if len(block.PrevBlockHash) == 0 {
break
}
}
return blocks
}
// MineBlock mines a new block with the provided transactions
func (bc *Blockchain) MineBlock(transactions []*Transaction) *Block {
var lastHash []byte
2017-09-30 23:02:38 -05:00
var lastHeight int
2017-09-02 23:01:02 -05:00
for _, tx := range transactions {
2017-10-03 04:21:15 -05:00
// TODO: ignore transaction if it's not valid
if bc.VerifyTransaction(tx) != true {
log.Panic("ERROR: Invalid transaction")
2017-09-02 23:01:02 -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"))
2017-08-28 01:57:27 -05:00
2017-09-30 23:02:38 -05:00
blockData := b.Get(lastHash)
block := DeserializeBlock(blockData)
lastHeight = block.Height
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-30 23:02:38 -05:00
newBlock := NewBlock(transactions, lastHash, lastHeight+1)
2017-09-02 23:17:10 -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
err = b.Put([]byte("l"), newBlock.Hash)
if err != nil {
log.Panic(err)
}
2017-08-28 01:57:27 -05:00
bc.tip = newBlock.Hash
2017-08-28 01:57:27 -05:00
2017-09-02 23:17:10 -05:00
return nil
})
if err != nil {
log.Panic(err)
}
return newBlock
2017-09-02 23:17:10 -05:00
}
// SignTransaction signs inputs of a Transaction
func (bc *Blockchain) SignTransaction(tx *Transaction, privKey ecdsa.PrivateKey) {
prevTXs := make(map[string]Transaction)
2017-09-02 23:17:10 -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)
}
prevTXs[hex.EncodeToString(prevTX.ID)] = prevTX
}
2017-09-02 23:17:10 -05:00
tx.Sign(privKey, prevTXs)
}
2017-09-02 23:17:10 -05:00
// VerifyTransaction verifies transaction input signatures
func (bc *Blockchain) VerifyTransaction(tx *Transaction) bool {
2017-09-16 21:16:50 -05:00
if tx.IsCoinbase() {
return true
}
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
}
prevTXs[hex.EncodeToString(prevTX.ID)] = prevTX
}
2017-08-28 04:28:23 -05:00
return tx.Verify(prevTXs)
}
2017-08-28 04:28:23 -05:00
func dbExists(dbFile string) bool {
if _, err := os.Stat(dbFile); os.IsNotExist(err) {
return false
}
return true
}