Skip to content

Commit

Permalink
Enforce proper check of ParentAgregatedSeal during preprepare phase (c…
Browse files Browse the repository at this point in the history
…elo-org#1829)

## Problem

During consensus, when validator has to verify that a proposed block is valid, the validators not checking the `ParentAggregatedSeal`. Thus, they can potentially accept an invalid block as valid.

Full Nodes during regular block syncing don't have the same problem, and thus reject the block. In a scenario with validators depending on proxies (full nodes) this also implies that after such a block, network connections between validators and proxies will break, thus the network will stall.

## How?

To verify a block, validators have to check the header, but ignore the `AggreatedSeal` since this field is the result of consensus and will only be added as the final step of it.

To achieve this, `backend.VerifyProposal()` calls `engine.VerifyHeader` and it will mark the block as invalid if this call fails, unless the error is related to the AggregatedSeal. This is incorrect, since if `engine.VerifyHeader()` fails due to the missing AggregatedSeal it does not imply that the rest of the header is valid, just that the first error on it found is that.

In particular, `engine.VerifyHeader()` will not verify the `ParentAggregatedSeal` if `AggreatedSeal` is missing.

## Fix

In this PR, instead of calling `VerifyHeader` we call a new function `verifyHeaderFromProposal` that checks that the `AggregatedSeal`'s signature is empty, and checks all others fields. As expected from a consensus proposal.


This PR is applied on top of the 1.5.x version branch
  • Loading branch information
hbandura authored Feb 18, 2022
1 parent b77c485 commit 300e836
Show file tree
Hide file tree
Showing 2 changed files with 41 additions and 19 deletions.
5 changes: 2 additions & 3 deletions consensus/istanbul/backend/backend.go
Original file line number Diff line number Diff line change
Expand Up @@ -590,10 +590,9 @@ func (sb *Backend) Verify(proposal istanbul.Proposal) (*istanbulCore.StateProces
}
}

err := sb.VerifyHeader(sb.chain, block.Header(), false)
err := sb.verifyHeaderFromProposal(sb.chain, block.Header())

// ignore errEmptyAggregatedSeal error because we don't have the committed seals yet
if err != nil && err != errEmptyAggregatedSeal {
if err != nil {
if err == consensus.ErrFutureBlock {
return nil, time.Unix(int64(block.Header().Time), 0).Sub(now()), consensus.ErrFutureBlock
} else {
Expand Down
55 changes: 39 additions & 16 deletions consensus/istanbul/backend/engine.go
Original file line number Diff line number Diff line change
Expand Up @@ -76,8 +76,10 @@ var (
errInvalidVotingChain = errors.New("invalid voting chain")
// errInvalidAggregatedSeal is returned if the aggregated seal is invalid.
errInvalidAggregatedSeal = errors.New("invalid aggregated seal")
// errInvalidAggregatedSeal is returned if the aggregated seal is missing.
// errEmptyAggregatedSeal is returned if the aggregated seal is missing.
errEmptyAggregatedSeal = errors.New("empty aggregated seal")
// errNonEmptyAggregatedSeal is returned if the aggregated seal is not empty during preprepase proposal phase.
errNonEmptyAggregatedSeal = errors.New("Non empty aggregated seal during preprepare")
// errMismatchTxhashes is returned if the TxHash in header is mismatch.
errMismatchTxhashes = errors.New("mismatch transactions hashes")
// errInvalidValidatorSetDiff is returned if the header contains invalid validator set diff
Expand All @@ -103,14 +105,22 @@ func (sb *Backend) Author(header *types.Header) (common.Address, error) {
// VerifyHeader checks whether a header conforms to the consensus rules of a
// given engine. Verifies the seal regardless of given "seal" argument.
func (sb *Backend) VerifyHeader(chain consensus.ChainHeaderReader, header *types.Header, seal bool) error {
return sb.verifyHeader(chain, header, nil)
return sb.verifyHeader(chain, header, false, nil)
}

// verifyHeaderFromProposal checks whether a header conforms to the consensus rules from the
// preprepare istanbul phase.
func (sb *Backend) verifyHeaderFromProposal(chain consensus.ChainHeaderReader, header *types.Header) error {
return sb.verifyHeader(chain, header, true, nil)
}

// verifyHeader checks whether a header conforms to the consensus rules.The
// caller may optionally pass in a batch of parents (ascending order) to avoid
// looking those up from the database. This is useful for concurrently verifying
// a batch of new headers.
func (sb *Backend) verifyHeader(chain consensus.ChainHeaderReader, header *types.Header, parents []*types.Header) error {
// If emptyAggregatedSeal is set, the aggregatedSeal will be checked to be completely empty. Otherwise
// it will be checked as a normal aggregated seal.
func (sb *Backend) verifyHeader(chain consensus.ChainHeaderReader, header *types.Header, emptyAggregatedSeal bool, parents []*types.Header) error {
if header.Number == nil {
return errUnknownBlock
}
Expand All @@ -132,7 +142,7 @@ func (sb *Backend) verifyHeader(chain consensus.ChainHeaderReader, header *types
return errInvalidExtraDataFormat
}

return sb.verifyCascadingFields(chain, header, parents)
return sb.verifyCascadingFields(chain, header, emptyAggregatedSeal, parents)
}

// A sanity check for lightest mode. Checks that the correct epoch block exists for this header
Expand Down Expand Up @@ -160,7 +170,9 @@ func (sb *Backend) checkEpochBlockExists(chain consensus.ChainHeaderReader, head
// rather depend on a batch of previous headers. The caller may optionally pass
// in a batch of parents (ascending order) to avoid looking those up from the
// database. This is useful for concurrently verifying a batch of new headers.
func (sb *Backend) verifyCascadingFields(chain consensus.ChainHeaderReader, header *types.Header, parents []*types.Header) error {
// If emptyAggregatedSeal is set, the aggregatedSeal will be checked to be completely empty. Otherwise
// it will be checked as a normal aggregated seal.
func (sb *Backend) verifyCascadingFields(chain consensus.ChainHeaderReader, header *types.Header, emptyAggregatedSeal bool, parents []*types.Header) error {
// The genesis block is the always valid dead-end
number := header.Number.Uint64()
if number == 0 {
Expand Down Expand Up @@ -189,7 +201,7 @@ func (sb *Backend) verifyCascadingFields(chain consensus.ChainHeaderReader, head
return err
}

return sb.verifyAggregatedSeals(chain, header, parents)
return sb.verifyAggregatedSeals(chain, header, emptyAggregatedSeal, parents)
}

// VerifyHeaders is similar to VerifyHeader, but verifies a batch of headers
Expand All @@ -206,7 +218,7 @@ func (sb *Backend) VerifyHeaders(chain consensus.ChainHeaderReader, headers []*t
if errored {
err = consensus.ErrUnknownAncestor
} else {
err = sb.verifyHeader(chain, header, headers[:i])
err = sb.verifyHeader(chain, header, false, headers[:i])
}

if err != nil {
Expand Down Expand Up @@ -252,7 +264,9 @@ func (sb *Backend) verifySigner(chain consensus.ChainHeaderReader, header *types

// verifyAggregatedSeals checks whether the aggregated seal and parent seal in the header is
// signed on by the block's validators and the parent block's validators respectively
func (sb *Backend) verifyAggregatedSeals(chain consensus.ChainHeaderReader, header *types.Header, parents []*types.Header) error {
// If emptyAggregatedSeal is set, the aggregatedSeal will be checked to be completely empty. Otherwise
// it will be checked as a normal aggregated seal.
func (sb *Backend) verifyAggregatedSeals(chain consensus.ChainHeaderReader, header *types.Header, emptyAggregatedseal bool, parents []*types.Header) error {
number := header.Number.Uint64()
// We don't need to verify committed seals in the genesis block
if number == 0 {
Expand All @@ -264,20 +278,29 @@ func (sb *Backend) verifyAggregatedSeals(chain consensus.ChainHeaderReader, head
return err
}

// The length of Committed seals should be larger than 0
if len(extra.AggregatedSeal.Signature) == 0 {
return errEmptyAggregatedSeal
}

// Check the signatures on the current header
snap, err := sb.snapshot(chain, number-1, header.ParentHash, parents)
if err != nil {
return err
}
validators := snap.ValSet.Copy()
err = sb.verifyAggregatedSeal(header.Hash(), validators, extra.AggregatedSeal)
if err != nil {
return err

if emptyAggregatedseal {
// The length of Committed seals should be exactly 0 (preprepare proposal check)
if len(extra.AggregatedSeal.Signature) != 0 {
return errNonEmptyAggregatedSeal
}
// Should we also verify that the bitmap and round are nil?
} else {
// The length of Committed seals should be larger than 0
if len(extra.AggregatedSeal.Signature) == 0 {
return errEmptyAggregatedSeal
}

err = sb.verifyAggregatedSeal(header.Hash(), validators, extra.AggregatedSeal)
if err != nil {
return err
}
}

// The genesis block is skipped since it has no parents.
Expand Down

0 comments on commit 300e836

Please sign in to comment.