From 7d7ba354ec262345e5391c632fb569d574b444d5 Mon Sep 17 00:00:00 2001 From: Jerry Date: Wed, 8 Mar 2023 14:53:05 -0800 Subject: [PATCH 1/4] Verify validator set against local contract on receiving an end-of-sprint block --- consensus/bor/bor.go | 24 ++++++++++++++++++++++++ 1 file changed, 24 insertions(+) diff --git a/consensus/bor/bor.go b/consensus/bor/bor.go index 5b32263762..050f39c09c 100644 --- a/consensus/bor/bor.go +++ b/consensus/bor/bor.go @@ -454,6 +454,30 @@ func (c *Bor) verifyCascadingFields(chain consensus.ChainHeaderReader, header *t return err } + // Verify the validator list match the local contract + if IsSprintStart(number+1, c.config.CalculateSprint(number)) { + newValidators, err := c.spanner.GetCurrentValidators(context.Background(), header.ParentHash, number+1) + if err != nil { + return errUnknownValidators + } + + sort.Sort(valset.ValidatorsByAddress(newValidators)) + + headerVals, _ := valset.ParseValidators(header.Extra[extraVanity : len(header.Extra)-extraSeal]) + + sort.Sort(valset.ValidatorsByAddress(headerVals)) + + if len(newValidators) != len(headerVals) { + return errInvalidSpanValidators + } + + for i, val := range newValidators { + if !bytes.Equal(val.HeaderBytes(), headerVals[i].HeaderBytes()) { + return errInvalidSpanValidators + } + } + } + // verify the validator list in the last sprint block if IsSprintStart(number, c.config.CalculateSprint(number)) { parentValidatorBytes := parent.Extra[extraVanity : len(parent.Extra)-extraSeal] From bb8a1d55c3cf7ad03e131c9ae78435284c9db6b6 Mon Sep 17 00:00:00 2001 From: Jerry Date: Thu, 9 Mar 2023 13:28:58 -0800 Subject: [PATCH 2/4] Fix tests --- consensus/bor/bor.go | 12 +++++++++++- tests/bor/bor_test.go | 20 ++++++++++++++++++++ tests/bor/helper.go | 10 ++++++++++ 3 files changed, 41 insertions(+), 1 deletion(-) diff --git a/consensus/bor/bor.go b/consensus/bor/bor.go index 050f39c09c..5f6b94bccf 100644 --- a/consensus/bor/bor.go +++ b/consensus/bor/bor.go @@ -298,6 +298,14 @@ func (c *Bor) VerifyHeader(chain consensus.ChainHeaderReader, header *types.Head return c.verifyHeader(chain, header, nil) } +func (c *Bor) GetSpanner() Spanner { + return c.spanner +} + +func (c *Bor) SetSpanner(spanner Spanner) { + c.spanner = spanner +} + // VerifyHeaders is similar to VerifyHeader, but verifies a batch of headers. The // method returns a quit channel to abort the operations and a results channel to // retrieve the async verifications (the order is that of the input slice). @@ -456,7 +464,9 @@ func (c *Bor) verifyCascadingFields(chain consensus.ChainHeaderReader, header *t // Verify the validator list match the local contract if IsSprintStart(number+1, c.config.CalculateSprint(number)) { - newValidators, err := c.spanner.GetCurrentValidators(context.Background(), header.ParentHash, number+1) + parentHeader := chain.GetHeaderByNumber(number - 1) + newValidators, err := c.spanner.GetCurrentValidators(context.Background(), parentHeader.Hash(), number+1) + if err != nil { return errUnknownValidators } diff --git a/tests/bor/bor_test.go b/tests/bor/bor_test.go index 2dc20a915e..e6e8188ce0 100644 --- a/tests/bor/bor_test.go +++ b/tests/bor/bor_test.go @@ -392,12 +392,18 @@ func TestInsertingSpanSizeBlocks(t *testing.T) { currentValidators := []*valset.Validator{valset.NewValidator(addr, 10)} + spanner := getMockedSpanner(t, currentValidators) + _bor.SetSpanner(spanner) + // Insert sprintSize # of blocks so that span is fetched at the start of a new sprint for i := uint64(1); i <= spanSize; i++ { block = buildNextBlock(t, _bor, chain, block, nil, init.genesis.Config.Bor, nil, currentValidators) insertNewBlock(t, chain, block) } + spanner = getMockedSpanner(t, currentSpan.ValidatorSet.Validators) + _bor.SetSpanner(spanner) + validators, err := _bor.GetCurrentValidators(context.Background(), block.Hash(), spanSize) // check validator set at the first block of new span if err != nil { t.Fatalf("%s", err) @@ -427,6 +433,9 @@ func TestFetchStateSyncEvents(t *testing.T) { currentValidators := []*valset.Validator{valset.NewValidator(addr, 10)} + spanner := getMockedSpanner(t, currentValidators) + _bor.SetSpanner(spanner) + // Insert sprintSize # of blocks so that span is fetched at the start of a new sprint for i := uint64(1); i < sprintSize; i++ { if IsSpanEnd(i) { @@ -528,6 +537,9 @@ func TestFetchStateSyncEvents_2(t *testing.T) { currentValidators = []*valset.Validator{valset.NewValidator(addr, 10)} } + spanner := getMockedSpanner(t, currentValidators) + _bor.SetSpanner(spanner) + block = buildNextBlock(t, _bor, chain, block, nil, init.genesis.Config.Bor, nil, currentValidators) insertNewBlock(t, chain, block) } @@ -554,6 +566,9 @@ func TestFetchStateSyncEvents_2(t *testing.T) { currentValidators = []*valset.Validator{valset.NewValidator(addr, 10)} } + spanner := getMockedSpanner(t, currentValidators) + _bor.SetSpanner(spanner) + block = buildNextBlock(t, _bor, chain, block, nil, init.genesis.Config.Bor, nil, res.Result.ValidatorSet.Validators) insertNewBlock(t, chain, block) } @@ -580,6 +595,8 @@ func TestOutOfTurnSigning(t *testing.T) { h.EXPECT().Close().AnyTimes() + spanner := getMockedSpanner(t, heimdallSpan.ValidatorSet.Validators) + _bor.SetSpanner(spanner) _bor.SetHeimdallClient(h) db := init.ethereum.ChainDb() @@ -1082,6 +1099,9 @@ func TestJaipurFork(t *testing.T) { res, _ := loadSpanFromFile(t) + spanner := getMockedSpanner(t, res.Result.ValidatorSet.Validators) + _bor.SetSpanner(spanner) + for i := uint64(1); i < sprintSize; i++ { block = buildNextBlock(t, _bor, chain, block, nil, init.genesis.Config.Bor, nil, res.Result.ValidatorSet.Validators) insertNewBlock(t, chain, block) diff --git a/tests/bor/helper.go b/tests/bor/helper.go index e28076a3b1..e1b83858b3 100644 --- a/tests/bor/helper.go +++ b/tests/bor/helper.go @@ -352,6 +352,16 @@ func getMockedHeimdallClient(t *testing.T, heimdallSpan *span.HeimdallSpan) (*mo return h, ctrl } +func getMockedSpanner(t *testing.T, validators []*valset.Validator) *bor.MockSpanner { + t.Helper() + + spanner := bor.NewMockSpanner(gomock.NewController(t)) + spanner.EXPECT().GetCurrentValidators(gomock.Any(), gomock.Any(), gomock.Any()).Return(validators, nil).AnyTimes() + spanner.EXPECT().GetCurrentSpan(gomock.Any(), gomock.Any()).Return(&span.Span{0, 0, 0}, nil).AnyTimes() + spanner.EXPECT().CommitSpan(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(nil).AnyTimes() + return spanner +} + func generateFakeStateSyncEvents(sample *clerk.EventRecordWithTime, count int) []*clerk.EventRecordWithTime { events := make([]*clerk.EventRecordWithTime, count) event := *sample From 3aed7109ce162b40440ab256de6d39c7a4c1c3b3 Mon Sep 17 00:00:00 2001 From: Jerry Date: Thu, 9 Mar 2023 13:31:16 -0800 Subject: [PATCH 3/4] Respect error returned by ParseValidators --- consensus/bor/bor.go | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/consensus/bor/bor.go b/consensus/bor/bor.go index 5f6b94bccf..d15e8d715a 100644 --- a/consensus/bor/bor.go +++ b/consensus/bor/bor.go @@ -473,7 +473,11 @@ func (c *Bor) verifyCascadingFields(chain consensus.ChainHeaderReader, header *t sort.Sort(valset.ValidatorsByAddress(newValidators)) - headerVals, _ := valset.ParseValidators(header.Extra[extraVanity : len(header.Extra)-extraSeal]) + headerVals, err := valset.ParseValidators(header.Extra[extraVanity : len(header.Extra)-extraSeal]) + + if err != nil { + return err + } sort.Sort(valset.ValidatorsByAddress(headerVals)) From af8223d7735ab1c420e46bba27f224332e40d5db Mon Sep 17 00:00:00 2001 From: Jerry Date: Thu, 9 Mar 2023 17:11:38 -0800 Subject: [PATCH 4/4] Keep going back until a parent block presents --- consensus/bor/bor.go | 11 +++++++++-- consensus/bor/heimdall/span/spanner.go | 2 +- 2 files changed, 10 insertions(+), 3 deletions(-) diff --git a/consensus/bor/bor.go b/consensus/bor/bor.go index d15e8d715a..cffe692843 100644 --- a/consensus/bor/bor.go +++ b/consensus/bor/bor.go @@ -464,8 +464,15 @@ func (c *Bor) verifyCascadingFields(chain consensus.ChainHeaderReader, header *t // Verify the validator list match the local contract if IsSprintStart(number+1, c.config.CalculateSprint(number)) { - parentHeader := chain.GetHeaderByNumber(number - 1) - newValidators, err := c.spanner.GetCurrentValidators(context.Background(), parentHeader.Hash(), number+1) + parentBlockNumber := number - 1 + + var newValidators []*valset.Validator + + var err error + + for newValidators, err = c.spanner.GetCurrentValidators(context.Background(), chain.GetHeaderByNumber(parentBlockNumber).Hash(), number+1); err != nil && parentBlockNumber > 0; { + parentBlockNumber-- + } if err != nil { return errUnknownValidators diff --git a/consensus/bor/heimdall/span/spanner.go b/consensus/bor/heimdall/span/spanner.go index e0f2d66c6b..5001b0f738 100644 --- a/consensus/bor/heimdall/span/spanner.go +++ b/consensus/bor/heimdall/span/spanner.go @@ -116,7 +116,7 @@ func (c *ChainSpanner) GetCurrentValidators(ctx context.Context, headerHash comm Data: &msgData, }, blockNr, nil) if err != nil { - panic(err) + return nil, err } var (