mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-14 22:18:20 +00:00
049e608c75
* HTTP Beacon API: `/eth/v1/validator/contribution_and_proofs` * add comment to invalid test case * fix validation and test * review * in progress * implementation * remove test file * remove duplicate * tests * HTTP Beacon API: `/eth/v1/validator/sync_committee_subscriptions` * pointers, pointers everywhere * fix * fix after merge * review * James' review --------- Co-authored-by: james-prysm <90280386+james-prysm@users.noreply.github.com>
191 lines
6.2 KiB
Go
191 lines
6.2 KiB
Go
package helpers
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/api/grpc"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/blockchain"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/db"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/rpc/eth/shared"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/rpc/lookup"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/sync"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/v4/time/slots"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/status"
|
|
)
|
|
|
|
// ValidateSyncGRPC checks whether the node is currently syncing and returns an error if it is.
|
|
// It also appends syncing info to gRPC headers.
|
|
func ValidateSyncGRPC(
|
|
ctx context.Context,
|
|
syncChecker sync.Checker,
|
|
headFetcher blockchain.HeadFetcher,
|
|
timeFetcher blockchain.TimeFetcher,
|
|
optimisticModeFetcher blockchain.OptimisticModeFetcher,
|
|
) error {
|
|
if !syncChecker.Syncing() {
|
|
return nil
|
|
}
|
|
headSlot := headFetcher.HeadSlot()
|
|
isOptimistic, err := optimisticModeFetcher.IsOptimistic(ctx)
|
|
if err != nil {
|
|
return status.Errorf(codes.Internal, "Could not check optimistic status: %v", err)
|
|
}
|
|
|
|
syncDetailsContainer := &shared.SyncDetailsContainer{
|
|
Data: &shared.SyncDetails{
|
|
HeadSlot: strconv.FormatUint(uint64(headSlot), 10),
|
|
SyncDistance: strconv.FormatUint(uint64(timeFetcher.CurrentSlot()-headSlot), 10),
|
|
IsSyncing: true,
|
|
IsOptimistic: isOptimistic,
|
|
},
|
|
}
|
|
|
|
err = grpc.AppendCustomErrorHeader(ctx, syncDetailsContainer)
|
|
if err != nil {
|
|
return status.Errorf(
|
|
codes.Internal,
|
|
"Syncing to latest head, not ready to respond. Could not prepare sync details: %v",
|
|
err,
|
|
)
|
|
}
|
|
return status.Error(codes.Unavailable, "Syncing to latest head, not ready to respond")
|
|
}
|
|
|
|
// IsOptimistic checks whether the beacon state's block is optimistic.
|
|
func IsOptimistic(
|
|
ctx context.Context,
|
|
stateId []byte,
|
|
optimisticModeFetcher blockchain.OptimisticModeFetcher,
|
|
stateFetcher lookup.Stater,
|
|
chainInfo blockchain.ChainInfoFetcher,
|
|
database db.ReadOnlyDatabase,
|
|
) (bool, error) {
|
|
stateIdString := strings.ToLower(string(stateId))
|
|
switch stateIdString {
|
|
case "head":
|
|
return optimisticModeFetcher.IsOptimistic(ctx)
|
|
case "genesis":
|
|
return false, nil
|
|
case "finalized":
|
|
fcp := chainInfo.FinalizedCheckpt()
|
|
if fcp == nil {
|
|
return true, errors.New("received nil finalized checkpoint")
|
|
}
|
|
// Special genesis case in the event our checkpoint root is a zerohash.
|
|
if bytes.Equal(fcp.Root, params.BeaconConfig().ZeroHash[:]) {
|
|
return false, nil
|
|
}
|
|
return optimisticModeFetcher.IsOptimisticForRoot(ctx, bytesutil.ToBytes32(fcp.Root))
|
|
case "justified":
|
|
jcp := chainInfo.CurrentJustifiedCheckpt()
|
|
if jcp == nil {
|
|
return true, errors.New("received nil justified checkpoint")
|
|
}
|
|
// Special genesis case in the event our checkpoint root is a zerohash.
|
|
if bytes.Equal(jcp.Root, params.BeaconConfig().ZeroHash[:]) {
|
|
return false, nil
|
|
}
|
|
return optimisticModeFetcher.IsOptimisticForRoot(ctx, bytesutil.ToBytes32(jcp.Root))
|
|
default:
|
|
if len(stateId) == 32 {
|
|
return isStateRootOptimistic(ctx, stateId, optimisticModeFetcher, stateFetcher, chainInfo, database)
|
|
} else {
|
|
optimistic, err := optimisticModeFetcher.IsOptimistic(ctx)
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "could not check optimistic status")
|
|
}
|
|
if !optimistic {
|
|
return false, nil
|
|
}
|
|
slotNumber, parseErr := strconv.ParseUint(stateIdString, 10, 64)
|
|
if parseErr != nil {
|
|
// ID format does not match any valid options.
|
|
e := lookup.NewStateIdParseError(parseErr)
|
|
return true, &e
|
|
}
|
|
fcp := chainInfo.FinalizedCheckpt()
|
|
if fcp == nil {
|
|
return true, errors.New("received nil finalized checkpoint")
|
|
}
|
|
finalizedSlot, err := slots.EpochStart(fcp.Epoch)
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "could not get head state's finalized slot")
|
|
}
|
|
lastValidatedCheckpoint, err := database.LastValidatedCheckpoint(ctx)
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "could not get last validated checkpoint")
|
|
}
|
|
validatedSlot, err := slots.EpochStart(lastValidatedCheckpoint.Epoch)
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "could not get last validated slot")
|
|
}
|
|
if primitives.Slot(slotNumber) <= validatedSlot {
|
|
return false, nil
|
|
}
|
|
// if the finalized checkpoint is higher than the last
|
|
// validated checkpoint, we are syncing and have synced
|
|
// a finalization being optimistic
|
|
if validatedSlot < finalizedSlot {
|
|
return true, nil
|
|
}
|
|
if primitives.Slot(slotNumber) == chainInfo.HeadSlot() {
|
|
// We know the head is optimistic because we checked it above.
|
|
return true, nil
|
|
}
|
|
headRoot, err := chainInfo.HeadRoot(ctx)
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "could not get head root")
|
|
}
|
|
r, err := chainInfo.Ancestor(ctx, headRoot, primitives.Slot(slotNumber))
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "could not get ancestor root")
|
|
}
|
|
return optimisticModeFetcher.IsOptimisticForRoot(ctx, bytesutil.ToBytes32(r))
|
|
}
|
|
}
|
|
}
|
|
|
|
func isStateRootOptimistic(
|
|
ctx context.Context,
|
|
stateId []byte,
|
|
optimisticModeFetcher blockchain.OptimisticModeFetcher,
|
|
stateFetcher lookup.Stater,
|
|
chainInfo blockchain.ChainInfoFetcher,
|
|
database db.ReadOnlyDatabase,
|
|
) (bool, error) {
|
|
st, err := stateFetcher.State(ctx, stateId)
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "could not fetch state")
|
|
}
|
|
if st.Slot() == chainInfo.HeadSlot() {
|
|
return optimisticModeFetcher.IsOptimistic(ctx)
|
|
}
|
|
has, roots, err := database.BlockRootsBySlot(ctx, st.Slot())
|
|
if err != nil {
|
|
return true, errors.Wrapf(err, "could not get block roots for slot %d", st.Slot())
|
|
}
|
|
if !has {
|
|
return true, errors.New("no block roots returned from the database")
|
|
}
|
|
for _, r := range roots {
|
|
b, err := database.Block(ctx, r)
|
|
if err != nil {
|
|
return true, errors.Wrapf(err, "could not obtain block")
|
|
}
|
|
if bytesutil.ToBytes32(stateId) != b.Block().StateRoot() {
|
|
continue
|
|
}
|
|
return optimisticModeFetcher.IsOptimisticForRoot(ctx, r)
|
|
}
|
|
// No block matching requested state root, return true.
|
|
return true, nil
|
|
}
|