美文网首页
打造公链(3)

打造公链(3)

作者: 中小本的葱 | 来源:发表于2018-06-08 14:14 被阅读0次

    持久化和命令行接口

    已经构建出一个PoW机制的区块链,但区块链的数据需要持久化到一个数据库,还需要提供一个简单的命令行接口,用户完成一些
    与区块链的交互操作,既然认为区块链本质上是一个分布式数据库,那么就要完成存储和读取。

    选择数据库

    原则上,选用什么数据都是可以的,但go语言我们选择BoltDB

    BoltDB

    简洁
    go实现
    不需要运行一个服务器
    能够允许构造想要的数据结构
    Bolt使用键值存储,没有像SQL RDBMS的表,没有行和列。数据被存储为键值对。键值对被存储在bucket中,这是为了将相似的键值对
    进行分组。因此,为了获取一个值,需要知道一个bucket和一个键(key)。

    Bolt数据库没有数据类型:键和值都是字节数组(byte array)。需要存储go的Block,就需要进行序列化。实现一个从go struct转换
    到一个byte array的机制,同时还要能转回struct。在这里我们选用go标准库 encoding/gob来完成这一目标。

    数据库结构

    在进行序列化存储之前,我们想要搞明白,到底什么数据存储到数据库中。

    首先,我们看看Bitcoin是如何做的:

    Bitcoin Core使用两个“bucket”来存储数据:

    blocks,存储了描述一条链中所有块的元数据
    chainstate,存储了一条链的状态,当前所有的未花费和交易输出,和一些元数据
    在此版轮子中,还未进行交易,只需要blocks bucket。也简单将整个数据库存储为单个文件,而没有将区块存储在不同的文件中。
    也不需要文件编号(file number)相关的东西,我们会用到的键值对有:

    32字节的block-hash -> block结构
    l -> 链中最后一个块的hash

    持久化

    从前一个轮子的NewBlockchain函数开始,此函数创建一个新的区块链实例,并且会添加一个创世块。加入数据库功能后,我们希望
    做更多的事情:

    1.打开一个数据库文件
    2.检查文件里面是否已经存储了一个区块链
    3.如果已经存储了一个区块链:

    • 创建一个新的Blockchain实例
    • 设置Blockchain实例的tip为数据库中存储的最后一个块的哈希
      4.如果没有区块链
    • 创建创世块
    • 存储到数据库
    • 将创世块哈希保存为最后一个块的哈希
    • 创建一个新的Blockchain实例,初始时tip指向创世块

    检查区块链

    首先构造一个能遍历区块的区块链迭代器(BlockchainIterator),迭代器的初始状态为链中的tip,然后从尾到头(创世块)进行
    迭代获取区块。实际上,选择一个tip就是意味着给一条链“投票”,怎么解释呢,一条链可能有很多分支,最长的那条链会被认为
    是主分支,获得一个tip(可以是链中任意一个块)后,就可以重新构造整条链,所以说,一个tip就是区块链的一种标识符。

    命令行接口

    说有相关命令在这个版本的轮子中,都会通过CLI struct进行处理

    type CLI struct{
        blockchain *Blockchain
    }
    

    go 命令

    go build -o blockchain_go
    
    ./blockchain_go printchain
    ./blockchain_go addblock -data "Send 1 BTC to Tom"
    ./blockchain_go printchain
    

    项目代码分为下面几个部分

    block.go

    package main
    
    import (
        "bytes"
        "encoding/gob"
        "log"
        "time"
    )
    
    type Block struct {
        Timestamp   int64
        Data    []byte
        PrevBlockHash   []byte
        Hash    []byte
        Nonce   int
    }
    
    // 将Block序列化为一个字节数组
    func (block *Block) Serialize() []byte{
        var result bytes.Buffer
        encoder := gob.NewEncoder(&result)
        err := encoder.Encode(block)
        if err!=nil{
            log.Panic(err)
        }
        return result.Bytes()
    }
    // 将字节数组反序列化为一个Block
    func DeserializeBlock(d []byte) *Block{
        var block Block
        decoder := gob.NewDecoder(bytes.NewReader(d))
        err := decoder.Decode(&block)
        if err!=nil{
            log.Panic(err)
        }
        return &block
    }
    
    func NewBlock(data string,prevBlockHash []byte) *Block {
        block := &Block{
            Timestamp:time.Now().Unix(),
            Data:[]byte(data),
            PrevBlockHash:prevBlockHash,
            Hash:[]byte{},
            Nonce:0}
        pow := NewProofOfWork(block)
        nonce,hash := pow.Run()
        block.Hash = hash[:]
        block.Nonce = nonce
        return block
    }
    
    func NewGenesisBlock() *Block {
        return NewBlock("Genesis Block", []byte{})
    }
    

    blockchain.go

    package main
    
    import (
        "github.com/boltdb/bolt"
        "log"
        "fmt"
    )
    
    const dbFile = "blockchain.db"
    const blocksBucket = "blocks"
    
    // tip 尾部的意思,这里是存储最后一个块的hash值 ,存储最后的tip就能推导出整条chain
    // 在链的尾端可能会短暂分叉的情况,所以选择tip其实是选择那条链
    // db 存储数据库连接
    type Blockchain struct {
        tip []byte
        db *bolt.DB
    }
    
    func NewBlockchain() *Blockchain{
        var tip []byte
        // 打开一个BoltDB文件
        db,err := bolt.Open(dbFile,0600,nil)
        if err!=nil{
            log.Panic(err)
        }
        err = db.Update(func(tx *bolt.Tx) error {
            bucket := tx.Bucket([]byte(blocksBucket))
            // 如果数据库中不存在区块链就创建一个,否则直接读取最后一个块的hash值
            if bucket==nil{
                fmt.Println("No existing blockchain found. Creating a new one...")
                genesis := NewGenesisBlock()
                bucket,err := tx.CreateBucket([]byte(blocksBucket))
                if err!=nil{
                    log.Panic(err)
                }
                err = bucket.Put(genesis.Hash, genesis.Serialize())
                if err!=nil{
                    log.Panic(err)
                }
                err = bucket.Put([]byte("1"),genesis.Hash)
                if err!=nil{
                    log.Panic(err)
                }
                tip = genesis.Hash
            }else{
                tip = bucket.Get([]byte("1"))
            }
            return nil
        })
        if err != nil {
            log.Panic(err)
        }
        blockchain := Blockchain{tip,db}
        return &blockchain
    }
    
    // 加入区块时,需要将区块持久化到数据库中
    func (blockchain *Blockchain) AddBlock(data string){
        var lastHash []byte
        // 首先获取最后一个块的哈希用于生成新的哈希
        err := blockchain.db.View(func(tx *bolt.Tx) error {
            bucket := tx.Bucket([]byte(blocksBucket))
            lastHash = bucket.Get([]byte("1"))
            return nil
        })
        if err != nil {
            log.Panic(err)
        }
        newBlock := NewBlock(data,lastHash)
        err = blockchain.db.Update(func(tx *bolt.Tx) error {
            bucket := tx.Bucket([]byte(blocksBucket))
            err := bucket.Put(newBlock.Hash,newBlock.Serialize())
            if err != nil {
                log.Panic(err)
            }
            err = bucket.Put([]byte("1"),newBlock.Hash)
            if err != nil {
                log.Panic(err)
            }
            blockchain.tip = newBlock.Hash
            return nil
        })
    }
    
    type BlockchainIterator struct {
        currentHash []byte
        db  *bolt.DB
    }
    
    func (blockchain *Blockchain) Iterator() *BlockchainIterator{
        blockchainiterator := &BlockchainIterator{blockchain.tip,blockchain.db}
        return blockchainiterator
    }
    
    // 返回链中的下一个块
    func (i *BlockchainIterator) Next() *Block{
        var block *Block
        err := i.db.View(func(tx *bolt.Tx) error {
            bucket := tx.Bucket([]byte(blocksBucket))
            encodedBlock := bucket.Get(i.currentHash)
            block = DeserializeBlock(encodedBlock)
            return nil
        })
        if err!=nil{
            log.Panic(err)
        }
        i.currentHash = block.PrevBlockHash
        return block
    }
    

    cli.go

    package main
    
    import (
        "fmt"
        "os"
        "flag"
        "log"
    )
    
    type CLI struct {
        blockchain *Blockchain
    }
    const usage = `
    Usage:
        addblock -data BLOCK_DATA   add a block to the blockchain
        printchain    print all the blocks of the blockchain
    `
    func (cli *CLI) printUsage(){
        fmt.Println(usage)
    }
    func (cli *CLI) validateArgs(){
        if len(os.Args)<2{
            cli.printUsage()
            os.Exit(1)
        }
    }
    func (cli *CLI) Run(){
        cli.validateArgs()
        addBlockCmd := flag.NewFlagSet("addblock",flag.ExitOnError)
        printChainCmd := flag.NewFlagSet("printchain",flag.ExitOnError)
        addBlockData := addBlockCmd.String("data","","Block data")
        switch os.Args[1]{
        case "addblock":
            err := addBlockCmd.Parse(os.Args[2:])
            if err != nil{
                log.Panic(err)
            }
        case    "printchain":
            err := printChainCmd.Parse(os.Args[2:])
            if err != nil {
                log.Panic(err)
            }
        default:
            cli.printUsage()
            os.Exit(1)
        }
        if addBlockCmd.Parsed(){
            if *addBlockData == ""{
                addBlockCmd.Usage()
                os.Exit(1)
            }
            cli.blockchain.AddBlock(*addBlockData)
        }
        if printChainCmd.Parsed(){
            cli.printChain()
        }
    }
    

    commands.go

    package main
    
    import (
        "fmt"
        "strconv"
    )
    
    func (cli *CLI) addBlock(data string){
        cli.blockchain.AddBlock(data)
        fmt.Println("add block success!")
    }
    func (cli *CLI) printChain(){
        blockchainiterator := cli.blockchain.Iterator()
        for {
            block := blockchainiterator.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
            }
        }
    }
    

    main.go

    package main
    
    func main()  {
        blockchain := NewBlockchain()
        defer blockchain.db.Close()
        cli := CLI{blockchain}
        cli.Run()
    }
    

    proofofwork.go

    package main
    
    import (
        "math"
        "math/big"
        "bytes"
        "fmt"
        "crypto/sha256"
    )
    
    const targetBits = 24
    var (
        maxNonce = math.MaxInt64
    )
    type ProofOfWork struct {
        block *Block
        target *big.Int
    }
    func NewProofOfWork(block *Block) *ProofOfWork{
        target := big.NewInt(1)
        target.Lsh(target,uint(256-targetBits))
        pow := &ProofOfWork{block,target}
        return pow
    }
    
    func (pow *ProofOfWork) prepareData(nonce int) []byte{
        data := bytes.Join(
            [][]byte{
                pow.block.PrevBlockHash,
                pow.block.Data,
                IntToHex(pow.block.Timestamp),
                IntToHex(int64(targetBits)),
                IntToHex(int64(nonce)),
            },
            []byte{},
        )
        return data
    }
    
    func (pow *ProofOfWork) Run() (int,[]byte){
        var hashInt big.Int
        var hash [32]byte
        nonce := 0
        fmt.Printf("Mining the block containing \"%s\"\n", pow.block.Data)
        for nonce<maxNonce{
            data := pow.prepareData(nonce)
            hash = sha256.Sum256(data)
            hashInt.SetBytes(hash[:])
            if hashInt.Cmp(pow.target)== -1{
                fmt.Printf("\r%x", hash)
                break
            }else{
                nonce++
            }
        }
        fmt.Print("\n\n")
        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
    }
    

    utils.go

    package main
    
    import (
        "bytes"
        "encoding/binary"
        "log"
    )
    
    // Convert an int64 to a byte array
    func IntToHex(num int64) []byte{
        buff := new(bytes.Buffer)
        err := binary.Write(buff,binary.BigEndian,num)
        if err!=nil{
            log.Panic(err)
        }
        return buff.Bytes()
    }
    

    相关文章

      网友评论

          本文标题:打造公链(3)

          本文链接:https://www.haomeiwen.com/subject/viqasftx.html