mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-11 04:00:05 +00:00
d17996f8b0
* Update V3 from V4 * Fix build v3 -> v4 * Update ssz * Update beacon_chain.pb.go * Fix formatter import * Update update-mockgen.sh comment to v4 * Fix conflicts. Pass build and tests * Fix test
264 lines
10 KiB
Go
264 lines
10 KiB
Go
package blocks
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/helpers"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/signing"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
|
fieldparams "github.com/prysmaticlabs/prysm/v4/config/fieldparams"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/interfaces"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v4/crypto/bls"
|
|
"github.com/prysmaticlabs/prysm/v4/crypto/hash"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/ssz"
|
|
ethpbv2 "github.com/prysmaticlabs/prysm/v4/proto/eth/v2"
|
|
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v4/runtime/version"
|
|
)
|
|
|
|
const executionToBLSPadding = 12
|
|
|
|
// ProcessBLSToExecutionChanges processes a list of BLS Changes and validates them. However,
|
|
// the method doesn't immediately verify the signatures in the changes and prefers to extract
|
|
// a signature set from them at the end of the transition and then verify them via the
|
|
// signature set.
|
|
func ProcessBLSToExecutionChanges(
|
|
st state.BeaconState,
|
|
signed interfaces.ReadOnlySignedBeaconBlock) (state.BeaconState, error) {
|
|
if signed.Version() < version.Capella {
|
|
return st, nil
|
|
}
|
|
changes, err := signed.Block().Body().BLSToExecutionChanges()
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get BLSToExecutionChanges")
|
|
}
|
|
// Return early if no changes
|
|
if len(changes) == 0 {
|
|
return st, nil
|
|
}
|
|
for _, change := range changes {
|
|
st, err = processBLSToExecutionChange(st, change)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not process BLSToExecutionChange")
|
|
}
|
|
}
|
|
return st, nil
|
|
}
|
|
|
|
// processBLSToExecutionChange validates a SignedBLSToExecution message and
|
|
// changes the validator's withdrawal address accordingly.
|
|
//
|
|
// Spec pseudocode definition:
|
|
//
|
|
// def process_bls_to_execution_change(state: BeaconState, signed_address_change: SignedBLSToExecutionChange) -> None:
|
|
//
|
|
// validator = state.validators[address_change.validator_index]
|
|
//
|
|
// assert validator.withdrawal_credentials[:1] == BLS_WITHDRAWAL_PREFIX
|
|
// assert validator.withdrawal_credentials[1:] == hash(address_change.from_bls_pubkey)[1:]
|
|
//
|
|
// domain = get_domain(state, DOMAIN_BLS_TO_EXECUTION_CHANGE)
|
|
// signing_root = compute_signing_root(address_change, domain)
|
|
// assert bls.Verify(address_change.from_bls_pubkey, signing_root, signed_address_change.signature)
|
|
//
|
|
// validator.withdrawal_credentials = (
|
|
// ETH1_ADDRESS_WITHDRAWAL_PREFIX
|
|
// + b'\x00' * 11
|
|
// + address_change.to_execution_address
|
|
// )
|
|
func processBLSToExecutionChange(st state.BeaconState, signed *ethpb.SignedBLSToExecutionChange) (state.BeaconState, error) {
|
|
// Checks that the message passes the validation conditions.
|
|
val, err := ValidateBLSToExecutionChange(st, signed)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
message := signed.Message
|
|
newCredentials := make([]byte, executionToBLSPadding)
|
|
newCredentials[0] = params.BeaconConfig().ETH1AddressWithdrawalPrefixByte
|
|
val.WithdrawalCredentials = append(newCredentials, message.ToExecutionAddress...)
|
|
err = st.UpdateValidatorAtIndex(message.ValidatorIndex, val)
|
|
return st, err
|
|
}
|
|
|
|
// ValidateBLSToExecutionChange validates the execution change message against the state and returns the
|
|
// validator referenced by the message.
|
|
func ValidateBLSToExecutionChange(st state.ReadOnlyBeaconState, signed *ethpb.SignedBLSToExecutionChange) (*ethpb.Validator, error) {
|
|
if signed == nil {
|
|
return nil, errNilSignedWithdrawalMessage
|
|
}
|
|
message := signed.Message
|
|
if message == nil {
|
|
return nil, errNilWithdrawalMessage
|
|
}
|
|
|
|
val, err := st.ValidatorAtIndex(message.ValidatorIndex)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
cred := val.WithdrawalCredentials
|
|
if cred[0] != params.BeaconConfig().BLSWithdrawalPrefixByte {
|
|
return nil, errInvalidBLSPrefix
|
|
}
|
|
|
|
// hash the public key and verify it matches the withdrawal credentials
|
|
fromPubkey := message.FromBlsPubkey
|
|
hashFn := ssz.NewHasherFunc(hash.CustomSHA256Hasher())
|
|
digest := hashFn.Hash(fromPubkey)
|
|
if !bytes.Equal(digest[1:], cred[1:]) {
|
|
return nil, errInvalidWithdrawalCredentials
|
|
}
|
|
return val, nil
|
|
}
|
|
|
|
// ProcessWithdrawals processes the validator withdrawals from the provided execution payload
|
|
// into the beacon state.
|
|
//
|
|
// Spec pseudocode definition:
|
|
//
|
|
// def process_withdrawals(state: BeaconState, payload: ExecutionPayload) -> None:
|
|
//
|
|
// expected_withdrawals = get_expected_withdrawals(state)
|
|
// assert len(payload.withdrawals) == len(expected_withdrawals)
|
|
//
|
|
// for expected_withdrawal, withdrawal in zip(expected_withdrawals, payload.withdrawals):
|
|
// assert withdrawal == expected_withdrawal
|
|
// decrease_balance(state, withdrawal.validator_index, withdrawal.amount)
|
|
//
|
|
// # Update the next withdrawal index if this block contained withdrawals
|
|
// if len(expected_withdrawals) != 0:
|
|
// latest_withdrawal = expected_withdrawals[-1]
|
|
// state.next_withdrawal_index = WithdrawalIndex(latest_withdrawal.index + 1)
|
|
//
|
|
// # Update the next validator index to start the next withdrawal sweep
|
|
// if len(expected_withdrawals) == MAX_WITHDRAWALS_PER_PAYLOAD:
|
|
// # Next sweep starts after the latest withdrawal's validator index
|
|
// next_validator_index = ValidatorIndex((expected_withdrawals[-1].validator_index + 1) % len(state.validators))
|
|
// state.next_withdrawal_validator_index = next_validator_index
|
|
// else:
|
|
// # Advance sweep by the max length of the sweep if there was not a full set of withdrawals
|
|
// next_index = state.next_withdrawal_validator_index + MAX_VALIDATORS_PER_WITHDRAWALS_SWEEP
|
|
// next_validator_index = ValidatorIndex(next_index % len(state.validators))
|
|
// state.next_withdrawal_validator_index = next_validator_index
|
|
func ProcessWithdrawals(st state.BeaconState, executionData interfaces.ExecutionData) (state.BeaconState, error) {
|
|
expectedWithdrawals, err := st.ExpectedWithdrawals()
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get expected withdrawals")
|
|
}
|
|
|
|
var wdRoot [32]byte
|
|
if executionData.IsBlinded() {
|
|
r, err := executionData.WithdrawalsRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get withdrawals root")
|
|
}
|
|
wdRoot = bytesutil.ToBytes32(r)
|
|
} else {
|
|
wds, err := executionData.Withdrawals()
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get withdrawals")
|
|
}
|
|
wdRoot, err = ssz.WithdrawalSliceRoot(wds, fieldparams.MaxWithdrawalsPerPayload)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get withdrawals root")
|
|
}
|
|
}
|
|
|
|
expectedRoot, err := ssz.WithdrawalSliceRoot(expectedWithdrawals, fieldparams.MaxWithdrawalsPerPayload)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get expected withdrawals root")
|
|
}
|
|
if expectedRoot != wdRoot {
|
|
return nil, fmt.Errorf("expected withdrawals root %#x, got %#x", expectedRoot, wdRoot)
|
|
}
|
|
|
|
for _, withdrawal := range expectedWithdrawals {
|
|
err := helpers.DecreaseBalance(st, withdrawal.ValidatorIndex, withdrawal.Amount)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not decrease balance")
|
|
}
|
|
}
|
|
if len(expectedWithdrawals) > 0 {
|
|
if err := st.SetNextWithdrawalIndex(expectedWithdrawals[len(expectedWithdrawals)-1].Index + 1); err != nil {
|
|
return nil, errors.Wrap(err, "could not set next withdrawal index")
|
|
}
|
|
}
|
|
var nextValidatorIndex primitives.ValidatorIndex
|
|
if uint64(len(expectedWithdrawals)) < params.BeaconConfig().MaxWithdrawalsPerPayload {
|
|
nextValidatorIndex, err = st.NextWithdrawalValidatorIndex()
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get next withdrawal validator index")
|
|
}
|
|
nextValidatorIndex += primitives.ValidatorIndex(params.BeaconConfig().MaxValidatorsPerWithdrawalsSweep)
|
|
nextValidatorIndex = nextValidatorIndex % primitives.ValidatorIndex(st.NumValidators())
|
|
} else {
|
|
nextValidatorIndex = expectedWithdrawals[len(expectedWithdrawals)-1].ValidatorIndex + 1
|
|
if nextValidatorIndex == primitives.ValidatorIndex(st.NumValidators()) {
|
|
nextValidatorIndex = 0
|
|
}
|
|
}
|
|
if err := st.SetNextWithdrawalValidatorIndex(nextValidatorIndex); err != nil {
|
|
return nil, errors.Wrap(err, "could not set next withdrawal validator index")
|
|
}
|
|
return st, nil
|
|
}
|
|
|
|
// BLSChangesSignatureBatch extracts the relevant signatures from the provided execution change
|
|
// messages and transforms them into a signature batch object.
|
|
func BLSChangesSignatureBatch(
|
|
st state.ReadOnlyBeaconState,
|
|
changes []*ethpb.SignedBLSToExecutionChange,
|
|
) (*bls.SignatureBatch, error) {
|
|
// Return early if no changes
|
|
if len(changes) == 0 {
|
|
return bls.NewSet(), nil
|
|
}
|
|
batch := &bls.SignatureBatch{
|
|
Signatures: make([][]byte, len(changes)),
|
|
PublicKeys: make([]bls.PublicKey, len(changes)),
|
|
Messages: make([][32]byte, len(changes)),
|
|
Descriptions: make([]string, len(changes)),
|
|
}
|
|
c := params.BeaconConfig()
|
|
domain, err := signing.ComputeDomain(c.DomainBLSToExecutionChange, c.GenesisForkVersion, st.GenesisValidatorsRoot())
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not compute signing domain")
|
|
}
|
|
for i, change := range changes {
|
|
batch.Signatures[i] = change.Signature
|
|
publicKey, err := bls.PublicKeyFromBytes(change.Message.FromBlsPubkey)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not convert bytes to public key")
|
|
}
|
|
batch.PublicKeys[i] = publicKey
|
|
htr, err := signing.SigningData(change.Message.HashTreeRoot, domain)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not compute BLSToExecutionChange signing data")
|
|
}
|
|
batch.Messages[i] = htr
|
|
batch.Descriptions[i] = signing.BlsChangeSignature
|
|
}
|
|
return batch, nil
|
|
}
|
|
|
|
// VerifyBLSChangeSignature checks the signature in the SignedBLSToExecutionChange message.
|
|
// It validates the signature with the Capella fork version if the passed state
|
|
// is from a previous fork.
|
|
func VerifyBLSChangeSignature(
|
|
st state.ReadOnlyBeaconState,
|
|
change *ethpbv2.SignedBLSToExecutionChange,
|
|
) error {
|
|
c := params.BeaconConfig()
|
|
domain, err := signing.ComputeDomain(c.DomainBLSToExecutionChange, c.GenesisForkVersion, st.GenesisValidatorsRoot())
|
|
if err != nil {
|
|
return errors.Wrap(err, "could not compute signing domain")
|
|
}
|
|
publicKey := change.Message.FromBlsPubkey
|
|
return signing.VerifySigningRoot(change.Message, publicKey, change.Signature, domain)
|
|
}
|