aboutsummaryrefslogblamecommitdiffstats
path: root/dex/cache_test.go
blob: 04bca06efe6ed2337d158c287b964c28b8d77100 (plain) (tree)
1
2
3
4
5
6
7
8
9

                                                    
  
                                                                        



                                                                               
                                                                         




                                                                           
                                                      




                                  
                   
                 

                 

                 
                                                                       
                                                                    
                                                                          

 













                                                                                    
                                  
                                          



                                






                                   



                                                              

                                 



                                                              

                                 



                                                              

                                 



                                                              





                                  




                                                      


                                                                          

                                                                                      
         

                                                                                      

                                 
                                









                                                                             

                                



                                                                          




                                                     


                                                                          

                                                                                      
         

                                                                                      



                                   
                                          



                                

















                                                                                       


                                
         
                                            










                                                                                     
                                           















                                                                                     



                                                 
                                                    

                          
                                                                   






                                                                                        



                                                                                     
 
 
                                            




                                          



                                             

                                                       
         



                                             

                                                       
         



                                             

                                                       
         



                                             

                                                       
         


                                       
 







                                                                                           


                 



                                                                 
         

                                                                                

         


                                             
                  
                                                 
         



                                                                         
                                                                    







                                                                                        
                                                  



                                                                        
         
                              

                                             
                                                                 

                                                   
         
                                                                   

                                                                            
                



                                                                                        

                                                     
                                                                         

                                                           


                 







                                                 
// Copyright 2018 The dexon-consensus Authors
// This file is part of the dexon-consensus library.
//
// The dexon-consensus library is free software: you can redistribute it
// and/or modify it under the terms of the GNU Lesser General Public License as
// published by the Free Software Foundation, either version 3 of the License,
// or (at your option) any later version.
//
// The dexon-consensus library is distributed in the hope that it will be
// useful, but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser
// General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with the dexon-consensus library. If not, see
// <http://www.gnu.org/licenses/>.

package dex

import (
    "math/rand"
    "reflect"
    "sort"
    "strings"
    "testing"

    coreCommon "github.com/dexon-foundation/dexon-consensus/common"
    coreDb "github.com/dexon-foundation/dexon-consensus/core/db"
    coreTypes "github.com/dexon-foundation/dexon-consensus/core/types"
)

type byHash []*coreTypes.Vote

func (v byHash) Len() int {
    return len(v)
}

func (v byHash) Less(i int, j int) bool {
    return strings.Compare(v[i].BlockHash.String(), v[j].BlockHash.String()) < 0
}

func (v byHash) Swap(i int, j int) {
    v[i], v[j] = v[j], v[i]
}

func TestCacheVote(t *testing.T) {
    db, err := coreDb.NewMemBackedDB()
    if err != nil {
        panic(err)
    }
    cache := newCache(3, db)
    pos0 := coreTypes.Position{
        Height: uint64(0),
    }
    pos1 := coreTypes.Position{
        Height: uint64(1),
    }
    vote1 := &coreTypes.Vote{
        VoteHeader: coreTypes.VoteHeader{
            BlockHash: coreCommon.NewRandomHash(),
            Position:  pos0,
        },
    }
    vote2 := &coreTypes.Vote{
        VoteHeader: coreTypes.VoteHeader{
            BlockHash: coreCommon.NewRandomHash(),
            Position:  pos0,
        },
    }
    vote3 := &coreTypes.Vote{
        VoteHeader: coreTypes.VoteHeader{
            BlockHash: coreCommon.NewRandomHash(),
            Position:  pos1,
        },
    }
    vote4 := &coreTypes.Vote{
        VoteHeader: coreTypes.VoteHeader{
            BlockHash: coreCommon.NewRandomHash(),
            Position:  pos1,
        },
    }
    cache.addVote(vote1)
    cache.addVote(vote2)
    cache.addVote(vote3)

    votes := cache.votes(pos0)
    sort.Sort(byHash(votes))

    resultVotes := []*coreTypes.Vote{vote1, vote2}
    sort.Sort(byHash(resultVotes))

    if len(votes) != 2 {
        t.Errorf("fail to get votes: have %d, want 2", len(votes))
    }
    if !votes[0].BlockHash.Equal(resultVotes[0].BlockHash) {
        t.Errorf("get wrong vote: have %s, want %s", votes[0], resultVotes[0])
    }
    if !votes[1].BlockHash.Equal(resultVotes[1].BlockHash) {
        t.Errorf("get wrong vote: have %s, want %s", votes[1], resultVotes[1])
    }
    votes = cache.votes(pos1)
    sort.Sort(byHash(votes))
    if len(votes) != 1 {
        t.Errorf("fail to get votes: have %d, want 1", len(votes))
    }
    if !votes[0].BlockHash.Equal(vote3.BlockHash) {
        t.Errorf("get wrong vote: have %s, want %s", votes[0], vote3)
    }

    cache.addVote(vote4)

    votes = cache.votes(pos0)
    sort.Sort(byHash(votes))

    if len(votes) != 0 {
        t.Errorf("fail to get votes: have %d, want 0", len(votes))
    }
    votes = cache.votes(pos1)
    sort.Sort(byHash(votes))

    resultVotes = []*coreTypes.Vote{vote3, vote4}
    sort.Sort(byHash(resultVotes))

    if len(votes) != 2 {
        t.Errorf("fail to get votes: have %d, want 1", len(votes))
    }
    if !votes[0].BlockHash.Equal(resultVotes[0].BlockHash) {
        t.Errorf("get wrong vote: have %s, want %s", votes[0], resultVotes[0])
    }
    if !votes[1].BlockHash.Equal(resultVotes[1].BlockHash) {
        t.Errorf("get wrong vote: have %s, want %s", votes[1], resultVotes[1])
    }
}

func TestCacheBlock(t *testing.T) {
    db, err := coreDb.NewMemBackedDB()
    if err != nil {
        panic(err)
    }
    cache := newCache(3, db)
    block1 := &coreTypes.Block{
        Hash: coreCommon.NewRandomHash(),
    }
    block2 := &coreTypes.Block{
        Hash: coreCommon.NewRandomHash(),
    }
    block3 := &coreTypes.Block{
        Hash: coreCommon.NewRandomHash(),
    }
    block4 := &coreTypes.Block{
        Hash: coreCommon.NewRandomHash(),
    }
    cache.addBlock(block1)
    cache.addBlock(block2)
    cache.addBlock(block3)

    hashes := coreCommon.Hashes{block1.Hash, block2.Hash, block3.Hash, block4.Hash}
    hashMap := map[coreCommon.Hash]struct{}{
        block1.Hash: {},
        block2.Hash: {},
        block3.Hash: {},
    }
    blocks := cache.blocks(hashes, true)
    if len(blocks) != 3 {
        t.Errorf("fail to get blocks: have %d, want 3", len(blocks))
    }
    for _, block := range blocks {
        if _, exist := hashMap[block.Hash]; !exist {
            t.Errorf("get wrong block: have %s, want %v", block, hashMap)
        }
    }

    cache.addBlock(block4)

    blocks = cache.blocks(hashes, true)
    hashMap[block4.Hash] = struct{}{}
    if len(blocks) != 3 {
        t.Errorf("fail to get blocks: have %d, want 3", len(blocks))
    }
    hasNewBlock := false
    for _, block := range blocks {
        if _, exist := hashMap[block.Hash]; !exist {
            t.Errorf("get wrong block: have %s, want %v", block, hashMap)
        }
        if block.Hash.Equal(block4.Hash) {
            hasNewBlock = true
        }
    }
    if !hasNewBlock {
        t.Errorf("expect block %s in cache, have %v", block4, blocks)
    }

    block5 := &coreTypes.Block{
        Hash: coreCommon.NewRandomHash(),
    }
    if err := db.PutBlock(*block5); err != nil {
        panic(err)
    }
    blocks = cache.blocks(coreCommon.Hashes{block5.Hash}, true)
    if len(blocks) != 1 {
        t.Errorf("fail to get blocks: have %d, want 1", len(blocks))
    } else {
        if !blocks[0].Hash.Equal(block5.Hash) {
            t.Errorf("get wrong block: have %s, want %s", blocks[0], block5)
        }
    }
    blocks = cache.blocks(coreCommon.Hashes{block5.Hash}, false)
    if len(blocks) != 0 {
        t.Errorf("unexpected length of blocks: have %d, want 0", len(blocks))
    }
}

func TestCacheFinalizedBlock(t *testing.T) {
    db, err := coreDb.NewMemBackedDB()
    if err != nil {
        panic(err)
    }
    cache := newCache(3, db)
    block1 := &coreTypes.Block{
        Position: coreTypes.Position{
            Height: 1,
        },
        Hash:       coreCommon.NewRandomHash(),
        Randomness: randomBytes(),
    }
    block2 := &coreTypes.Block{
        Position: coreTypes.Position{
            Height: 2,
        },
        Hash:       coreCommon.NewRandomHash(),
        Randomness: randomBytes(),
    }
    block3 := &coreTypes.Block{
        Position: coreTypes.Position{
            Height: 3,
        },
        Hash:       coreCommon.NewRandomHash(),
        Randomness: randomBytes(),
    }
    block4 := &coreTypes.Block{
        Position: coreTypes.Position{
            Height: 4,
        },
        Hash:       coreCommon.NewRandomHash(),
        Randomness: randomBytes(),
    }
    cache.addFinalizedBlock(block1)
    cache.addFinalizedBlock(block2)
    cache.addFinalizedBlock(block3)

    hashes := coreCommon.Hashes{block1.Hash, block2.Hash, block3.Hash, block4.Hash}
    for i := 0; i < 3; i++ {
        pos := coreTypes.Position{
            Height: uint64(i + 1),
        }
        block := cache.finalizedBlock(pos)
        if block.Hash != hashes[i] {
            t.Errorf("failed to get block: have %s, want %s", block, hashes[i])
        }
    }

    cache.addFinalizedBlock(block4)
    block := cache.finalizedBlock(block4.Position)
    if block == nil {
        t.Errorf("should have block %s in cache", block4)
    }
    if block.Hash != block4.Hash {
        t.Errorf("failed to get block: have %s, want %s", block, block4)
    }

    block5 := &coreTypes.Block{
        Position: coreTypes.Position{
            Height: 5,
        },
        Hash: coreCommon.NewRandomHash(),
    }
    cache.addBlock(block5)
    if block := cache.finalizedBlock(block5.Position); block != nil {
        t.Errorf("unexpected block %s in cache", block)
    }
    blocks := cache.blocks(coreCommon.Hashes{block5.Hash}, true)
    if len(blocks) != 1 {
        t.Errorf("fail to get blocks: have %d, want 1", len(blocks))
    } else {
        if !blocks[0].Hash.Equal(block5.Hash) {
            t.Errorf("get wrong block: have %s, want %s", blocks[0], block5)
        }
    }
    finalizedBlock5 := block5.Clone()
    finalizedBlock5.Randomness = randomBytes()
    cache.addFinalizedBlock(finalizedBlock5)
    block = cache.finalizedBlock(block5.Position)
    if block == nil {
        t.Errorf("expecting block %s in cache", finalizedBlock5)
    }
    if !reflect.DeepEqual(
        block.Randomness,
        finalizedBlock5.Randomness) {
        t.Errorf("mismatch randomness, have %s, want %s",
            block.Randomness,
            finalizedBlock5.Randomness)
    }
    blocks = cache.blocks(coreCommon.Hashes{block5.Hash}, true)
    if len(blocks) != 1 {
        t.Errorf("fail to get blocks: have %d, want 1", len(blocks))
    } else {
        if !blocks[0].Hash.Equal(finalizedBlock5.Hash) {
            t.Errorf("get wrong block: have %s, want %s", blocks[0], block5)
        }
        if !reflect.DeepEqual(
            blocks[0].Randomness,
            finalizedBlock5.Randomness) {
            t.Errorf("mismatch randomness, have %s, want %s",
                blocks[0].Randomness,
                finalizedBlock5.Randomness)
        }
    }
}

func randomBytes() []byte {
    bytes := make([]byte, 32)
    for i := range bytes {
        bytes[i] = byte(rand.Int() % 256)
    }
    return bytes
}