2020-07-06 22:27:42 +00:00
|
|
|
package blocks
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2020-09-01 01:29:27 +00:00
|
|
|
"context"
|
2020-07-06 22:27:42 +00:00
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
|
|
|
stateTrie "github.com/prysmaticlabs/prysm/beacon-chain/state"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ProcessBlockHeader validates a block by its header.
|
|
|
|
//
|
|
|
|
// Spec pseudocode definition:
|
|
|
|
//
|
|
|
|
// def process_block_header(state: BeaconState, block: BeaconBlock) -> None:
|
|
|
|
// # Verify that the slots match
|
|
|
|
// assert block.slot == state.slot
|
|
|
|
// # Verify that proposer index is the correct index
|
|
|
|
// assert block.proposer_index == get_beacon_proposer_index(state)
|
|
|
|
// # Verify that the parent matches
|
|
|
|
// assert block.parent_root == hash_tree_root(state.latest_block_header)
|
|
|
|
// # Save current block as the new latest block
|
|
|
|
// state.latest_block_header = BeaconBlockHeader(
|
|
|
|
// slot=block.slot,
|
|
|
|
// parent_root=block.parent_root,
|
|
|
|
// # state_root: zeroed, overwritten in the next `process_slot` call
|
|
|
|
// body_root=hash_tree_root(block.body),
|
|
|
|
// # signature is always zeroed
|
|
|
|
// )
|
|
|
|
// # Verify proposer is not slashed
|
|
|
|
// proposer = state.validators[get_beacon_proposer_index(state)]
|
|
|
|
// assert not proposer.slashed
|
|
|
|
// # Verify proposer signature
|
|
|
|
// assert bls_verify(proposer.pubkey, signing_root(block), block.signature, get_domain(state, DOMAIN_BEACON_PROPOSER))
|
|
|
|
func ProcessBlockHeader(
|
2020-10-12 08:11:05 +00:00
|
|
|
_ context.Context,
|
2020-07-06 22:27:42 +00:00
|
|
|
beaconState *stateTrie.BeaconState,
|
|
|
|
block *ethpb.SignedBeaconBlock,
|
|
|
|
) (*stateTrie.BeaconState, error) {
|
2020-09-08 15:45:20 +00:00
|
|
|
beaconState, err := ProcessBlockHeaderNoVerify(beaconState, block.Block)
|
2020-07-06 22:27:42 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify proposer signature.
|
|
|
|
if err := VerifyBlockSignature(beaconState, block); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return beaconState, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ProcessBlockHeaderNoVerify validates a block by its header but skips proposer
|
|
|
|
// signature verification.
|
|
|
|
//
|
|
|
|
// WARNING: This method does not verify proposer signature. This is used for proposer to compute state root
|
|
|
|
// using a unsigned block.
|
|
|
|
//
|
|
|
|
// Spec pseudocode definition:
|
|
|
|
// def process_block_header(state: BeaconState, block: BeaconBlock) -> None:
|
|
|
|
// # Verify that the slots match
|
|
|
|
// assert block.slot == state.slot
|
|
|
|
// # Verify that proposer index is the correct index
|
|
|
|
// assert block.proposer_index == get_beacon_proposer_index(state)
|
|
|
|
// # Verify that the parent matches
|
|
|
|
// assert block.parent_root == hash_tree_root(state.latest_block_header)
|
|
|
|
// # Save current block as the new latest block
|
|
|
|
// state.latest_block_header = BeaconBlockHeader(
|
|
|
|
// slot=block.slot,
|
|
|
|
// parent_root=block.parent_root,
|
|
|
|
// # state_root: zeroed, overwritten in the next `process_slot` call
|
|
|
|
// body_root=hash_tree_root(block.body),
|
|
|
|
// # signature is always zeroed
|
|
|
|
// )
|
|
|
|
// # Verify proposer is not slashed
|
|
|
|
// proposer = state.validators[get_beacon_proposer_index(state)]
|
|
|
|
// assert not proposer.slashed
|
|
|
|
func ProcessBlockHeaderNoVerify(
|
|
|
|
beaconState *stateTrie.BeaconState,
|
2020-09-08 15:45:20 +00:00
|
|
|
block *ethpb.BeaconBlock,
|
2020-07-06 22:27:42 +00:00
|
|
|
) (*stateTrie.BeaconState, error) {
|
2020-09-08 15:45:20 +00:00
|
|
|
if block == nil {
|
2020-07-06 22:27:42 +00:00
|
|
|
return nil, errors.New("nil block")
|
|
|
|
}
|
|
|
|
if beaconState.Slot() != block.Slot {
|
|
|
|
return nil, fmt.Errorf("state slot: %d is different than block slot: %d", beaconState.Slot(), block.Slot)
|
|
|
|
}
|
|
|
|
idx, err := helpers.BeaconProposerIndex(beaconState)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if block.ProposerIndex != idx {
|
|
|
|
return nil, fmt.Errorf("proposer index: %d is different than calculated: %d", block.ProposerIndex, idx)
|
|
|
|
}
|
|
|
|
parentHeader := beaconState.LatestBlockHeader()
|
|
|
|
if parentHeader.Slot >= block.Slot {
|
|
|
|
return nil, fmt.Errorf("block.Slot %d must be greater than state.LatestBlockHeader.Slot %d", block.Slot, parentHeader.Slot)
|
|
|
|
}
|
2020-08-27 18:13:32 +00:00
|
|
|
parentRoot, err := parentHeader.HashTreeRoot()
|
2020-07-06 22:27:42 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !bytes.Equal(block.ParentRoot, parentRoot[:]) {
|
|
|
|
return nil, fmt.Errorf(
|
|
|
|
"parent root %#x does not match the latest block header signing root in state %#x",
|
|
|
|
block.ParentRoot, parentRoot)
|
|
|
|
}
|
|
|
|
|
|
|
|
proposer, err := beaconState.ValidatorAtIndexReadOnly(idx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if proposer.Slashed() {
|
|
|
|
return nil, fmt.Errorf("proposer at index %d was previously slashed", idx)
|
|
|
|
}
|
|
|
|
|
2020-08-27 18:13:32 +00:00
|
|
|
bodyRoot, err := block.Body.HashTreeRoot()
|
2020-07-06 22:27:42 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if err := beaconState.SetLatestBlockHeader(ðpb.BeaconBlockHeader{
|
|
|
|
Slot: block.Slot,
|
|
|
|
ProposerIndex: block.ProposerIndex,
|
|
|
|
ParentRoot: block.ParentRoot,
|
|
|
|
StateRoot: params.BeaconConfig().ZeroHash[:],
|
|
|
|
BodyRoot: bodyRoot[:],
|
|
|
|
}); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return beaconState, nil
|
|
|
|
}
|