aboutsummaryrefslogblamecommitdiffstats
path: root/core/leader-selector_test.go
blob: 4738663085728064ada29a4aa72de65aae84cca1 (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 core

import (
    "testing"

    "github.com/stretchr/testify/suite"

    "github.com/dexon-foundation/dexon-consensus/common"
    "github.com/dexon-foundation/dexon-consensus/core/crypto/ecdsa"
    "github.com/dexon-foundation/dexon-consensus/core/types"
    "github.com/dexon-foundation/dexon-consensus/core/utils"
)

type LeaderSelectorTestSuite struct {
    suite.Suite
    mockValidLeaderDefault bool
    mockValidLeaderDB      map[common.Hash]bool
    mockValidLeader        validLeaderFn
}

func (s *LeaderSelectorTestSuite) SetupTest() {
    s.mockValidLeaderDefault = true
    s.mockValidLeaderDB = make(map[common.Hash]bool)
    s.mockValidLeader = func(b *types.Block, _ common.Hash) (bool, error) {
        if ret, exist := s.mockValidLeaderDB[b.Hash]; exist {
            return ret, nil
        }
        return s.mockValidLeaderDefault, nil
    }
}

func (s *LeaderSelectorTestSuite) newLeader() *leaderSelector {
    l := newLeaderSelector(s.mockValidLeader, &common.NullLogger{})
    l.restart(common.NewRandomHash())
    return l
}

func (s *LeaderSelectorTestSuite) TestDistance() {
    leader := s.newLeader()
    hash := common.NewRandomHash()
    prv, err := ecdsa.NewPrivateKey()
    s.Require().NoError(err)
    sig, err := prv.Sign(hash)
    s.Require().NoError(err)
    dis := leader.distance(sig)
    s.Equal(-1, dis.Cmp(maxHash))
}

func (s *LeaderSelectorTestSuite) TestProbability() {
    leader := s.newLeader()
    prv1, err := ecdsa.NewPrivateKey()
    s.Require().NoError(err)
    prv2, err := ecdsa.NewPrivateKey()
    s.Require().NoError(err)
    for {
        hash := common.NewRandomHash()
        sig1, err := prv1.Sign(hash)
        s.Require().NoError(err)
        sig2, err := prv2.Sign(hash)
        s.Require().NoError(err)
        dis1 := leader.distance(sig1)
        dis2 := leader.distance(sig2)
        prob1 := leader.probability(sig1)
        prob2 := leader.probability(sig2)
        s.True(prob1 <= 1 && prob1 >= 0)
        s.True(prob2 <= 1 && prob2 >= 0)
        cmp := dis1.Cmp(dis2)
        if cmp == 0 {
            s.True(dis1.Cmp(dis2) == 0)
            continue
        }
        if cmp == 1 {
            s.True(prob2 > prob1)
        } else if cmp == -1 {
            s.True(prob2 < prob1)
        }
        break
    }
}

func (s *LeaderSelectorTestSuite) TestLeaderBlockHash() {
    leader := s.newLeader()
    blocks := make(map[common.Hash]*types.Block)
    for i := 0; i < 10; i++ {
        prv, err := ecdsa.NewPrivateKey()
        s.Require().NoError(err)
        block := &types.Block{
            ProposerID: types.NewNodeID(prv.PublicKey()),
            Hash:       common.NewRandomHash(),
        }
        s.Require().NoError(
            utils.NewSigner(prv).SignCRS(block, leader.hashCRS))
        s.Require().NoError(leader.processBlock(block))
        blocks[block.Hash] = block
    }
    blockHash := leader.leaderBlockHash()
    leaderBlock, exist := blocks[blockHash]
    s.Require().True(exist)
    leaderDist := leader.distance(leaderBlock.CRSSignature)
    for _, block := range blocks {
        if block == leaderBlock {
            continue
        }
        dist := leader.distance(block.CRSSignature)
        s.Equal(-1, leaderDist.Cmp(dist))
    }
}

func (s *LeaderSelectorTestSuite) TestValidLeaderFn() {
    leader := s.newLeader()
    blocks := make(map[common.Hash]*types.Block)
    for i := 0; i < 10; i++ {
        prv, err := ecdsa.NewPrivateKey()
        s.Require().NoError(err)
        block := &types.Block{
            ProposerID: types.NewNodeID(prv.PublicKey()),
            Hash:       common.NewRandomHash(),
        }
        s.Require().NoError(
            utils.NewSigner(prv).SignCRS(block, leader.hashCRS))
        s.Require().NoError(leader.processBlock(block))
        blocks[block.Hash] = block
    }
    blockHash := leader.leaderBlockHash()

    s.mockValidLeaderDB[blockHash] = false
    leader.restart(leader.hashCRS)
    for _, b := range blocks {
        s.Require().NoError(leader.processBlock(b))
    }
    s.NotEqual(blockHash, leader.leaderBlockHash())
    s.mockValidLeaderDB[blockHash] = true
    s.Equal(blockHash, leader.leaderBlockHash())
    s.Len(leader.pendingBlocks, 0)
}

func (s *LeaderSelectorTestSuite) TestPotentialLeader() {
    leader := s.newLeader()
    blocks := make(map[common.Hash]*types.Block)
    for i := 0; i < 10; i++ {
        if i > 0 {
            s.mockValidLeaderDefault = false
        }
        prv, err := ecdsa.NewPrivateKey()
        s.Require().NoError(err)
        block := &types.Block{
            ProposerID: types.NewNodeID(prv.PublicKey()),
            Hash:       common.NewRandomHash(),
        }
        s.Require().NoError(
            utils.NewSigner(prv).SignCRS(block, leader.hashCRS))
        ok, _ := leader.potentialLeader(block)
        s.Require().NoError(leader.processBlock(block))
        if i > 0 {
            if ok {
                s.Contains(leader.pendingBlocks, block.Hash)
            } else {
                s.NotContains(leader.pendingBlocks, block.Hash)
            }
            blocks[block.Hash] = block
        }
    }
}

func TestLeaderSelector(t *testing.T) {
    suite.Run(t, new(LeaderSelectorTestSuite))
}