|
|
- package evidence
-
- import (
- "fmt"
- "testing"
- "time"
-
- "github.com/stretchr/testify/assert"
- "github.com/stretchr/testify/mock"
-
- "github.com/tendermint/tendermint/crypto"
- "github.com/tendermint/tendermint/crypto/tmhash"
- "github.com/tendermint/tendermint/evidence/mocks"
- tmrand "github.com/tendermint/tendermint/libs/rand"
- "github.com/tendermint/tendermint/types"
- )
-
- func TestVerifyEvidenceWrongAddress(t *testing.T) {
- var height int64 = 4
- val := types.NewMockPV()
- stateStore := initializeValidatorState(val, height)
- state := stateStore.LoadState()
- blockStore := &mocks.BlockStore{}
- blockStore.On("LoadBlockMeta", mock.AnythingOfType("int64")).Return(
- &types.BlockMeta{Header: types.Header{Time: defaultEvidenceTime}},
- )
- evidence := types.NewMockDuplicateVoteEvidence(1, defaultEvidenceTime, evidenceChainID)
- err := VerifyEvidence(evidence, state, stateStore, blockStore)
- errMsg := fmt.Sprintf("address %X was not a validator at height 1", evidence.Address())
- if assert.Error(t, err) {
- assert.Equal(t, err.Error(), errMsg)
- }
- }
-
- func TestVerifyEvidenceExpiredEvidence(t *testing.T) {
- var height int64 = 4
- val := types.NewMockPV()
- stateStore := initializeValidatorState(val, height)
- state := stateStore.LoadState()
- state.ConsensusParams.Evidence.MaxAgeNumBlocks = 1
- expiredEvidenceTime := time.Date(2018, 1, 1, 0, 0, 0, 0, time.UTC)
- blockStore := &mocks.BlockStore{}
- blockStore.On("LoadBlockMeta", mock.AnythingOfType("int64")).Return(
- &types.BlockMeta{Header: types.Header{Time: expiredEvidenceTime}},
- )
-
- expiredEv := types.NewMockDuplicateVoteEvidenceWithValidator(1, expiredEvidenceTime, val, evidenceChainID)
- err := VerifyEvidence(expiredEv, state, stateStore, blockStore)
- errMsg := "evidence from height 1 (created at: 2018-01-01 00:00:00 +0000 UTC) is too old"
- if assert.Error(t, err) {
- assert.Equal(t, err.Error()[:len(errMsg)], errMsg)
- }
- }
-
- func TestVerifyEvidenceInvalidTime(t *testing.T) {
- height := int64(4)
- val := types.NewMockPV()
- stateStore := initializeValidatorState(val, height)
- state := stateStore.LoadState()
- blockStore := &mocks.BlockStore{}
- blockStore.On("LoadBlockMeta", mock.AnythingOfType("int64")).Return(
- &types.BlockMeta{Header: types.Header{Time: defaultEvidenceTime}},
- )
-
- differentTime := time.Date(2019, 2, 1, 0, 0, 0, 0, time.UTC)
- ev := types.NewMockDuplicateVoteEvidenceWithValidator(height, differentTime, val, evidenceChainID)
- err := VerifyEvidence(ev, state, stateStore, blockStore)
- errMsg := "evidence time (2019-02-01 00:00:00 +0000 UTC) is different to the time" +
- " of the header we have for the same height (2019-01-01 00:00:00 +0000 UTC)"
- if assert.Error(t, err) {
- assert.Equal(t, errMsg, err.Error())
- }
- }
-
- func TestVerifyEvidenceWithLunaticValidatorEvidence(t *testing.T) {
- var height int64 = 4
- val := types.NewMockPV()
- stateStore := initializeValidatorState(val, height)
- blockID := types.BlockID{
- Hash: tmrand.Bytes(tmhash.Size),
- PartSetHeader: types.PartSetHeader{
- Total: 1,
- Hash: tmrand.Bytes(tmhash.Size),
- },
- }
- h := &types.Header{
- ChainID: evidenceChainID,
- Height: 3,
- Time: defaultEvidenceTime,
- LastBlockID: blockID,
- LastCommitHash: tmhash.Sum([]byte("last_commit_hash")),
- DataHash: tmhash.Sum([]byte("data_hash")),
- ValidatorsHash: tmhash.Sum([]byte("validators_hash")),
- NextValidatorsHash: tmhash.Sum([]byte("next_validators_hash")),
- ConsensusHash: tmhash.Sum([]byte("consensus_hash")),
- AppHash: tmhash.Sum([]byte("app_hash")),
- LastResultsHash: tmhash.Sum([]byte("last_results_hash")),
- EvidenceHash: tmhash.Sum([]byte("evidence_hash")),
- ProposerAddress: crypto.AddressHash([]byte("proposer_address")),
- }
- blockStore := &mocks.BlockStore{}
- blockStore.On("LoadBlockMeta", mock.AnythingOfType("int64")).Return(
- &types.BlockMeta{Header: *h},
- )
-
- validH1 := *h
- validH1.ValidatorsHash = tmhash.Sum([]byte("different_validators_hash"))
-
- validH2 := validH1
- validH2.Time = time.Date(2020, 1, 1, 0, 0, 0, 0, time.UTC)
-
- badH1 := validH1
- badH1.ChainID = "different_chain_id"
-
- badH2 := *h
- badH2.DataHash = tmhash.Sum([]byte("different_data_hash"))
-
- testCases := []struct {
- Header *types.Header
- ExpErr bool
- ErrMsg string
- }{
- {
- h,
- true,
- "ValidatorsHash matches committed hash",
- },
- {
- &validH1,
- false,
- "",
- },
- {
- &validH2,
- false,
- "",
- },
- {
- &badH1,
- true,
- "chainID do not match: test_chain vs different_chain_id",
- },
- {
- &badH2,
- true,
- "ValidatorsHash matches committed hash", // it doesn't recognise that the data hashes are different
- },
- }
-
- for idx, tc := range testCases {
- ev := types.NewLunaticValidatorEvidence(tc.Header,
- makeValidVoteForHeader(tc.Header, val), "ValidatorsHash", defaultEvidenceTime)
- err := VerifyEvidence(ev, stateStore.LoadState(), stateStore, blockStore)
- if tc.ExpErr {
- if assert.Error(t, err, fmt.Sprintf("expected an error for case: %d", idx)) {
- assert.Equal(t, tc.ErrMsg, err.Error(), fmt.Sprintf("case: %d", idx))
- }
- } else {
- assert.NoError(t, err, fmt.Sprintf("did not expect an error for case: %d", idx))
- }
-
- }
- }
-
- func makeValidVoteForHeader(header *types.Header, val types.MockPV) *types.Vote {
- vote := makeVote(header.Height, 1, 0, val.PrivKey.PubKey().Address(), types.BlockID{
- Hash: header.Hash(),
- PartSetHeader: types.PartSetHeader{
- Total: 100,
- Hash: crypto.CRandBytes(tmhash.Size),
- },
- }, defaultEvidenceTime)
- v := vote.ToProto()
- err := val.SignVote(evidenceChainID, v)
- if err != nil {
- panic("verify_test: failed to sign vote for header")
- }
- vote.Signature = v.Signature
- return vote
- }
|