Step 3
- Added Persistence (boltdb storage) - Added CLI - Added PoW Validation
This commit is contained in:
		
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -26,3 +26,5 @@ _testmain.go | |||||||
|  |  | ||||||
| # And the binary | # And the binary | ||||||
| blockchain-poc | blockchain-poc | ||||||
|  | # And the database | ||||||
|  | blockchain.db | ||||||
|   | |||||||
							
								
								
									
										21
									
								
								block.go
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								block.go
									
									
									
									
									
								
							| @@ -1,6 +1,9 @@ | |||||||
| package main | package main | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"encoding/gob" | ||||||
|  | 	"log" | ||||||
| 	"math" | 	"math" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
| @@ -33,3 +36,21 @@ func NewBlock(data string, prevBlockHash []byte) *Block { | |||||||
|  |  | ||||||
| 	return block | 	return block | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (b *Block) Serialize() []byte { | ||||||
|  | 	var result bytes.Buffer | ||||||
|  | 	encoder := gob.NewEncoder(&result) | ||||||
|  | 	if err := encoder.Encode(b); err != nil { | ||||||
|  | 		log.Panic(err) | ||||||
|  | 	} | ||||||
|  | 	return result.Bytes() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func DeserializeBlock(d []byte) *Block { | ||||||
|  | 	var block Block | ||||||
|  | 	decoder := gob.NewDecoder(bytes.NewReader(d)) | ||||||
|  | 	if err := decoder.Decode(&block); err != nil { | ||||||
|  | 		log.Panic(err) | ||||||
|  | 	} | ||||||
|  | 	return &block | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										110
									
								
								blockchain.go
									
									
									
									
									
								
							
							
						
						
									
										110
									
								
								blockchain.go
									
									
									
									
									
								
							| @@ -1,11 +1,64 @@ | |||||||
| package main | package main | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"log" | ||||||
|  |  | ||||||
|  | 	"github.com/boltdb/bolt" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | var dbFile = "blockchain.db" | ||||||
|  |  | ||||||
|  | const blocksBucket = "blocks" | ||||||
|  |  | ||||||
| type Blockchain struct { | type Blockchain struct { | ||||||
| 	blocks []*Block | 	tip []byte | ||||||
|  | 	db  *bolt.DB | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type BlockchainIterator struct { | ||||||
|  | 	currentHash []byte | ||||||
|  | 	db          *bolt.DB | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewBlockchain() *Blockchain { | func NewBlockchain() *Blockchain { | ||||||
| 	return &Blockchain{[]*Block{NewGenesisBlock()}} | 	var tip []byte | ||||||
|  | 	db, err := bolt.Open(dbFile, 0600, nil) | ||||||
|  |  | ||||||
|  | 	err = db.Update(func(tx *bolt.Tx) error { | ||||||
|  | 		b := tx.Bucket([]byte(blocksBucket)) | ||||||
|  |  | ||||||
|  | 		if b == nil { | ||||||
|  | 			fmt.Println("No existing blockchain found. Creating a new one...") | ||||||
|  | 			genesis := NewGenesisBlock() | ||||||
|  |  | ||||||
|  | 			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("1"), genesis.Hash) | ||||||
|  | 			if err != nil { | ||||||
|  | 				log.Panic(err) | ||||||
|  | 			} | ||||||
|  | 			tip = genesis.Hash | ||||||
|  | 		} else { | ||||||
|  | 			tip = b.Get([]byte("1")) | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Panic(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bc := Blockchain{tip, db} | ||||||
|  |  | ||||||
|  | 	return &bc | ||||||
| } | } | ||||||
|  |  | ||||||
| // NewGenesisBlock creates a new Genesis Block to start the blockchain | // NewGenesisBlock creates a new Genesis Block to start the blockchain | ||||||
| @@ -15,7 +68,54 @@ func NewGenesisBlock() *Block { | |||||||
|  |  | ||||||
| // AddBlock adds a block to the blockchain | // AddBlock adds a block to the blockchain | ||||||
| func (bc *Blockchain) AddBlock(data string) { | func (bc *Blockchain) AddBlock(data string) { | ||||||
| 	prevBlock := bc.blocks[len(bc.blocks)-1] | 	var lastHash []byte | ||||||
| 	newBlock := NewBlock(data, prevBlock.Hash) |  | ||||||
| 	bc.blocks = append(bc.blocks, newBlock) | 	err := bc.db.View(func(tx *bolt.Tx) error { | ||||||
|  | 		b := tx.Bucket([]byte(blocksBucket)) | ||||||
|  | 		lastHash = b.Get([]byte("1")) | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Panic(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	newBlock := NewBlock(data, lastHash) | ||||||
|  |  | ||||||
|  | 	err = bc.db.Update(func(tx *bolt.Tx) error { | ||||||
|  | 		var err error | ||||||
|  | 		b := tx.Bucket([]byte(blocksBucket)) | ||||||
|  | 		if err = b.Put(newBlock.Hash, newBlock.Serialize()); err != nil { | ||||||
|  | 			log.Panic(err) | ||||||
|  | 		} | ||||||
|  | 		if err = b.Put([]byte("1"), newBlock.Hash); err != nil { | ||||||
|  | 			log.Panic(err) | ||||||
|  | 		} | ||||||
|  | 		bc.tip = newBlock.Hash | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Panic(err) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (bc *Blockchain) Iterator() *BlockchainIterator { | ||||||
|  | 	bci := &BlockchainIterator{bc.tip, bc.db} | ||||||
|  | 	return bci | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (bci *BlockchainIterator) Next() *Block { | ||||||
|  | 	var block *Block | ||||||
|  |  | ||||||
|  | 	err := bci.db.View(func(tx *bolt.Tx) error { | ||||||
|  | 		b := tx.Bucket([]byte(blocksBucket)) | ||||||
|  | 		encodedBlock := b.Get(bci.currentHash) | ||||||
|  | 		block = DeserializeBlock(encodedBlock) | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Panic(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bci.currentHash = block.PrevBlockHash | ||||||
|  | 	return block | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										79
									
								
								cli.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								cli.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,79 @@ | |||||||
|  | package main | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"flag" | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 	"strconv" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type CLI struct { | ||||||
|  | 	bc *Blockchain | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (cli *CLI) printUsage() { | ||||||
|  | 	fmt.Println("Usage:") | ||||||
|  | 	fmt.Println("\taddblock -data BLOCK_DATA\tadd a block to the blockchain") | ||||||
|  | 	fmt.Println("\tprintchain\t\t\tprint all the blocks of the blockchain") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (cli *CLI) validateArgs() { | ||||||
|  | 	if len(os.Args) < 2 { | ||||||
|  | 		cli.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (cli *CLI) Run() { | ||||||
|  | 	cli.validateArgs() | ||||||
|  |  | ||||||
|  | 	// Valid CLI flags | ||||||
|  | 	addBlockCmd := flag.NewFlagSet("addblock", flag.ExitOnError) | ||||||
|  | 	printChainCmd := flag.NewFlagSet("printchain", flag.ExitOnError) | ||||||
|  | 	// CLI Flag Data | ||||||
|  | 	addBlockData := addBlockCmd.String("data", "", "Block data") | ||||||
|  |  | ||||||
|  | 	switch os.Args[1] { | ||||||
|  | 	case "addblock": | ||||||
|  | 		addBlockCmd.Parse(os.Args[2:]) | ||||||
|  | 	case "printchain": | ||||||
|  | 		printChainCmd.Parse(os.Args[2:]) | ||||||
|  | 	default: | ||||||
|  | 		cli.printUsage() | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if addBlockCmd.Parsed() { | ||||||
|  | 		if *addBlockData == "" { | ||||||
|  | 			addBlockCmd.Usage() | ||||||
|  | 			os.Exit(1) | ||||||
|  | 		} | ||||||
|  | 		cli.addBlock(*addBlockData) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if printChainCmd.Parsed() { | ||||||
|  | 		cli.printChain() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (cli *CLI) addBlock(data string) { | ||||||
|  | 	cli.bc.AddBlock(data) | ||||||
|  | 	fmt.Println("Success!") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (cli *CLI) printChain() { | ||||||
|  | 	bci := cli.bc.Iterator() | ||||||
|  | 	for { | ||||||
|  | 		block := bci.Next() | ||||||
|  | 		fmt.Printf("Prev. hash: %x\n", block.PrevBlockHash) | ||||||
|  | 		fmt.Printf("Data: %s\n", block.Data) | ||||||
|  | 		fmt.Printf("Hash: %x\n", block.Hash) | ||||||
|  | 		pow := NewProofOfWork(block) | ||||||
|  | 		fmt.Printf("PoW: %s\n", strconv.FormatBool(pow.Validate())) | ||||||
|  | 		fmt.Println() | ||||||
|  |  | ||||||
|  | 		if len(block.PrevBlockHash) == 0 { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										13
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								main.go
									
									
									
									
									
								
							| @@ -1,16 +1,9 @@ | |||||||
| package main | package main | ||||||
|  |  | ||||||
| import "fmt" |  | ||||||
|  |  | ||||||
| func main() { | func main() { | ||||||
| 	bc := NewBlockchain() | 	bc := NewBlockchain() | ||||||
| 	bc.AddBlock("Data 1") | 	defer bc.db.Close() | ||||||
| 	bc.AddBlock("Data 2") |  | ||||||
|  |  | ||||||
| 	for _, block := range bc.blocks { | 	cli := CLI{bc} | ||||||
| 		fmt.Printf("Prev. hash: %x\n", block.PrevBlockHash) | 	cli.Run() | ||||||
| 		fmt.Printf("Data: %s\n", block.Data) |  | ||||||
| 		fmt.Printf("Hash: %x\n", block.Hash) |  | ||||||
| 		fmt.Println() |  | ||||||
| 	} |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										11
									
								
								proof.go
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								proof.go
									
									
									
									
									
								
							| @@ -58,3 +58,14 @@ func (pow *ProofOfWork) Run() (int, []byte) { | |||||||
| 	fmt.Print("\n\n") | 	fmt.Print("\n\n") | ||||||
| 	return nonce, hash[:] | 	return nonce, hash[:] | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (pow *ProofOfWork) Validate() bool { | ||||||
|  | 	var hashInt big.Int | ||||||
|  |  | ||||||
|  | 	data := pow.prepareData(pow.block.Nonce) | ||||||
|  | 	hash := sha256.Sum256(data) | ||||||
|  | 	hashInt.SetBytes(hash[:]) | ||||||
|  |  | ||||||
|  | 	isValid := hashInt.Cmp(pow.target) == -1 | ||||||
|  | 	return isValid | ||||||
|  | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user