2021-10-07 18:50:03 +00:00
|
|
|
package helpers
|
|
|
|
|
|
|
|
import (
|
2022-06-09 19:35:35 +00:00
|
|
|
"bytes"
|
2021-10-07 18:50:03 +00:00
|
|
|
"context"
|
|
|
|
"strconv"
|
|
|
|
|
2022-03-25 16:31:50 +00:00
|
|
|
"github.com/pkg/errors"
|
2022-08-16 12:20:13 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v3/api/grpc"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/blockchain"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/state"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/sync"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/config/params"
|
2021-10-07 18:50:03 +00:00
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ValidateSync checks whether the node is currently syncing and returns an error if it is.
|
|
|
|
// It also appends syncing info to gRPC headers.
|
2022-06-14 17:47:09 +00:00
|
|
|
func ValidateSync(
|
|
|
|
ctx context.Context,
|
|
|
|
syncChecker sync.Checker,
|
|
|
|
headFetcher blockchain.HeadFetcher,
|
|
|
|
timeFetcher blockchain.TimeFetcher,
|
|
|
|
optimisticModeFetcher blockchain.OptimisticModeFetcher,
|
|
|
|
) error {
|
2021-10-07 18:50:03 +00:00
|
|
|
if !syncChecker.Syncing() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
headSlot := headFetcher.HeadSlot()
|
2022-07-21 15:59:51 +00:00
|
|
|
isOptimistic, err := optimisticModeFetcher.IsOptimistic(ctx)
|
2022-06-14 17:47:09 +00:00
|
|
|
if err != nil {
|
|
|
|
return status.Errorf(codes.Internal, "Could not check optimistic status: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
syncDetailsContainer := &syncDetailsContainer{
|
|
|
|
SyncDetails: &SyncDetailsJson{
|
2021-10-07 18:50:03 +00:00
|
|
|
HeadSlot: strconv.FormatUint(uint64(headSlot), 10),
|
|
|
|
SyncDistance: strconv.FormatUint(uint64(timeFetcher.CurrentSlot()-headSlot), 10),
|
|
|
|
IsSyncing: true,
|
2022-06-14 17:47:09 +00:00
|
|
|
IsOptimistic: isOptimistic,
|
2021-10-07 18:50:03 +00:00
|
|
|
},
|
|
|
|
}
|
2022-06-14 17:47:09 +00:00
|
|
|
|
|
|
|
err = grpc.AppendCustomErrorHeader(ctx, syncDetailsContainer)
|
2021-10-07 18:50:03 +00:00
|
|
|
if err != nil {
|
|
|
|
return status.Errorf(
|
2022-03-24 18:32:39 +00:00
|
|
|
codes.Internal,
|
2021-10-07 18:50:03 +00:00
|
|
|
"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")
|
|
|
|
}
|
2022-03-25 16:31:50 +00:00
|
|
|
|
|
|
|
// IsOptimistic checks whether the latest block header of the passed in beacon state is the header of an optimistic block.
|
2022-06-14 17:47:09 +00:00
|
|
|
func IsOptimistic(ctx context.Context, st state.BeaconState, optimisticModeFetcher blockchain.OptimisticModeFetcher) (bool, error) {
|
2022-03-25 16:31:50 +00:00
|
|
|
header := st.LatestBlockHeader()
|
2022-06-09 19:35:35 +00:00
|
|
|
// This happens when the block at the state's slot is not missing.
|
|
|
|
if bytes.Equal(header.StateRoot, params.BeaconConfig().ZeroHash[:]) {
|
|
|
|
root, err := st.HashTreeRoot(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return false, errors.Wrap(err, "could not get state root")
|
|
|
|
}
|
|
|
|
header.StateRoot = root[:]
|
|
|
|
}
|
2022-03-25 16:31:50 +00:00
|
|
|
headRoot, err := header.HashTreeRoot()
|
|
|
|
if err != nil {
|
|
|
|
return false, errors.Wrap(err, "could not get header root")
|
|
|
|
}
|
2022-06-14 17:47:09 +00:00
|
|
|
isOptimistic, err := optimisticModeFetcher.IsOptimisticForRoot(ctx, headRoot)
|
2022-03-25 16:31:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, errors.Wrap(err, "could not check if block is optimistic")
|
|
|
|
}
|
|
|
|
return isOptimistic, nil
|
|
|
|
}
|
2022-06-14 17:47:09 +00:00
|
|
|
|
|
|
|
// SyncDetailsJson contains information about node sync status.
|
|
|
|
type SyncDetailsJson struct {
|
|
|
|
HeadSlot string `json:"head_slot"`
|
|
|
|
SyncDistance string `json:"sync_distance"`
|
|
|
|
IsSyncing bool `json:"is_syncing"`
|
|
|
|
IsOptimistic bool `json:"is_optimistic"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// SyncDetailsContainer is a wrapper for SyncDetails.
|
|
|
|
type syncDetailsContainer struct {
|
|
|
|
SyncDetails *SyncDetailsJson `json:"sync_details"`
|
|
|
|
}
|