mirror of
https://github.com/OffchainLabs/prysm.git
synced 2026-01-09 23:48:06 -05:00
* generate SyncCommittee SSZ * refactor error handling * rewards: use http2.HandleError * add light client protobuf files * add light client helpers
810 lines
28 KiB
Go
810 lines
28 KiB
Go
package beacon
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/golang/protobuf/ptypes/empty"
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/api"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/helpers"
|
|
rpchelpers "github.com/prysmaticlabs/prysm/v4/beacon-chain/rpc/eth/helpers"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
consensus_types "github.com/prysmaticlabs/prysm/v4/consensus-types"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/interfaces"
|
|
ethpbv1 "github.com/prysmaticlabs/prysm/v4/proto/eth/v1"
|
|
ethpbv2 "github.com/prysmaticlabs/prysm/v4/proto/eth/v2"
|
|
"github.com/prysmaticlabs/prysm/v4/proto/migration"
|
|
"github.com/prysmaticlabs/prysm/v4/runtime/version"
|
|
"github.com/prysmaticlabs/prysm/v4/time/slots"
|
|
"go.opencensus.io/trace"
|
|
"google.golang.org/grpc"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/metadata"
|
|
"google.golang.org/grpc/status"
|
|
)
|
|
|
|
var (
|
|
errNilBlock = errors.New("nil block")
|
|
)
|
|
|
|
// GetWeakSubjectivity computes the starting epoch of the current weak subjectivity period, and then also
|
|
// determines the best block root and state root to use for a Checkpoint Sync starting from that point.
|
|
// DEPRECATED: GetWeakSubjectivity endpoint will no longer be supported
|
|
func (bs *Server) GetWeakSubjectivity(ctx context.Context, _ *empty.Empty) (*ethpbv1.WeakSubjectivityResponse, error) {
|
|
if err := rpchelpers.ValidateSyncGRPC(ctx, bs.SyncChecker, bs.HeadFetcher, bs.GenesisTimeFetcher, bs.OptimisticModeFetcher); err != nil {
|
|
// This is already a grpc error, so we can't wrap it any further
|
|
return nil, err
|
|
}
|
|
|
|
hs, err := bs.HeadFetcher.HeadStateReadOnly(ctx)
|
|
if err != nil {
|
|
return nil, status.Error(codes.Internal, "could not get head state")
|
|
}
|
|
wsEpoch, err := helpers.LatestWeakSubjectivityEpoch(ctx, hs, params.BeaconConfig())
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "could not get weak subjectivity epoch: %v", err)
|
|
}
|
|
wsSlot, err := slots.EpochStart(wsEpoch)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "could not get weak subjectivity slot: %v", err)
|
|
}
|
|
cbr, err := bs.CanonicalHistory.BlockRootForSlot(ctx, wsSlot)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, fmt.Sprintf("could not find highest block below slot %d", wsSlot))
|
|
}
|
|
cb, err := bs.BeaconDB.Block(ctx, cbr)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, fmt.Sprintf("block with root %#x from slot index %d not found in db", cbr, wsSlot))
|
|
}
|
|
stateRoot := cb.Block().StateRoot()
|
|
log.Printf("weak subjectivity checkpoint reported as epoch=%d, block root=%#x, state root=%#x", wsEpoch, cbr, stateRoot)
|
|
return ðpbv1.WeakSubjectivityResponse{
|
|
Data: ðpbv1.WeakSubjectivityData{
|
|
WsCheckpoint: ðpbv1.Checkpoint{
|
|
Epoch: wsEpoch,
|
|
Root: cbr[:],
|
|
},
|
|
StateRoot: stateRoot[:],
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
// GetBlock retrieves block details for given block ID.
|
|
// DEPRECATED: please use GetBlockV2 instead
|
|
func (bs *Server) GetBlock(ctx context.Context, req *ethpbv1.BlockRequest) (*ethpbv1.BlockResponse, error) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon.GetBlock")
|
|
defer span.End()
|
|
|
|
blk, err := bs.Blocker.Block(ctx, req.BlockId)
|
|
err = rpchelpers.HandleGetBlockError(blk, err)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
signedBeaconBlock, err := migration.SignedBeaconBlock(blk)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
|
|
return ðpbv1.BlockResponse{
|
|
Data: ðpbv1.BeaconBlockContainer{
|
|
Message: signedBeaconBlock.Block,
|
|
Signature: signedBeaconBlock.Signature,
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
// GetBlockSSZ returns the SSZ-serialized version of the becaon block for given block ID.
|
|
// DEPRECATED: please use GetBlockV2SSZ instead
|
|
func (bs *Server) GetBlockSSZ(ctx context.Context, req *ethpbv1.BlockRequest) (*ethpbv1.BlockSSZResponse, error) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon.GetBlockSSZ")
|
|
defer span.End()
|
|
|
|
blk, err := bs.Blocker.Block(ctx, req.BlockId)
|
|
err = rpchelpers.HandleGetBlockError(blk, err)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
signedBeaconBlock, err := migration.SignedBeaconBlock(blk)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
sszBlock, err := signedBeaconBlock.MarshalSSZ()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not marshal block into SSZ: %v", err)
|
|
}
|
|
|
|
return ðpbv1.BlockSSZResponse{Data: sszBlock}, nil
|
|
}
|
|
|
|
// GetBlockV2 retrieves block details for given block ID.
|
|
func (bs *Server) GetBlockV2(ctx context.Context, req *ethpbv2.BlockRequestV2) (*ethpbv2.BlockResponseV2, error) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon.GetBlockV2")
|
|
defer span.End()
|
|
|
|
blk, err := bs.Blocker.Block(ctx, req.BlockId)
|
|
err = rpchelpers.HandleGetBlockError(blk, err)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
blkRoot, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
|
|
result, err := getBlockPhase0(blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
// ErrUnsupportedField means that we have another block type
|
|
if !errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
if err := grpc.SetHeader(ctx, metadata.Pairs(api.VersionHeader, version.String(blk.Version()))); err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not set "+api.VersionHeader+" header: %v", err)
|
|
}
|
|
result, err = getBlockAltair(blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
// ErrUnsupportedField means that we have another block type
|
|
if !errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
result, err = bs.getBlockBellatrix(ctx, blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
// ErrUnsupportedField means that we have another block type
|
|
if !errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
result, err = bs.getBlockCapella(ctx, blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
// ErrUnsupportedField means that we have another block type
|
|
if !errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
|
|
result, err = bs.getBlockDeneb(ctx, blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
|
|
// ErrUnsupportedField means that we have another block type
|
|
if !errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
return nil, status.Errorf(codes.Internal, "Unknown block type %T", blk)
|
|
}
|
|
|
|
// GetBlockSSZV2 returns the SSZ-serialized version of the beacon block for given block ID.
|
|
func (bs *Server) GetBlockSSZV2(ctx context.Context, req *ethpbv2.BlockRequestV2) (*ethpbv2.SSZContainer, error) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon.GetBlockSSZV2")
|
|
defer span.End()
|
|
|
|
blk, err := bs.Blocker.Block(ctx, req.BlockId)
|
|
err = rpchelpers.HandleGetBlockError(blk, err)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
blkRoot, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
|
|
result, err := getSSZBlockPhase0(blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
// ErrUnsupportedField means that we have another block type
|
|
if !errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
result, err = getSSZBlockAltair(blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
// ErrUnsupportedField means that we have another block type
|
|
if !errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
result, err = bs.getSSZBlockBellatrix(ctx, blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
// ErrUnsupportedField means that we have another block type
|
|
if !errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
result, err = bs.getSSZBlockCapella(ctx, blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
// ErrUnsupportedField means that we have another block type
|
|
if !errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
|
|
result, err = bs.getSSZBlockDeneb(ctx, blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
// ErrUnsupportedField means that we have another block type
|
|
if !errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
|
|
return nil, status.Errorf(codes.Internal, "Unknown block type %T", blk)
|
|
}
|
|
|
|
// ListBlockAttestations retrieves attestation included in requested block.
|
|
func (bs *Server) ListBlockAttestations(ctx context.Context, req *ethpbv1.BlockRequest) (*ethpbv1.BlockAttestationsResponse, error) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon.ListBlockAttestations")
|
|
defer span.End()
|
|
|
|
blk, err := bs.Blocker.Block(ctx, req.BlockId)
|
|
err = rpchelpers.HandleGetBlockError(blk, err)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
v1Alpha1Attestations := blk.Block().Body().Attestations()
|
|
v1Attestations := make([]*ethpbv1.Attestation, 0, len(v1Alpha1Attestations))
|
|
for _, att := range v1Alpha1Attestations {
|
|
migratedAtt := migration.V1Alpha1AttestationToV1(att)
|
|
v1Attestations = append(v1Attestations, migratedAtt)
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get block root: %v", err)
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not check if block is optimistic: %v", err)
|
|
}
|
|
return ðpbv1.BlockAttestationsResponse{
|
|
Data: v1Attestations,
|
|
ExecutionOptimistic: isOptimistic,
|
|
Finalized: bs.FinalizationFetcher.IsFinalized(ctx, root),
|
|
}, nil
|
|
}
|
|
|
|
func getBlockPhase0(blk interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.BlockResponseV2, error) {
|
|
phase0Blk, err := blk.PbPhase0Block()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if phase0Blk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
v1Blk, err := migration.SignedBeaconBlock(blk)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get signed beacon block")
|
|
}
|
|
return ðpbv2.BlockResponseV2{
|
|
Version: ethpbv2.Version_PHASE0,
|
|
Data: ðpbv2.SignedBeaconBlockContainer{
|
|
Message: ðpbv2.SignedBeaconBlockContainer_Phase0Block{Phase0Block: v1Blk.Block},
|
|
Signature: v1Blk.Signature,
|
|
},
|
|
ExecutionOptimistic: false,
|
|
}, nil
|
|
}
|
|
|
|
func getBlockAltair(blk interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.BlockResponseV2, error) {
|
|
altairBlk, err := blk.PbAltairBlock()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if altairBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockAltairToV2(altairBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get signed beacon block")
|
|
}
|
|
sig := blk.Signature()
|
|
return ðpbv2.BlockResponseV2{
|
|
Version: ethpbv2.Version_ALTAIR,
|
|
Data: ðpbv2.SignedBeaconBlockContainer{
|
|
Message: ðpbv2.SignedBeaconBlockContainer_AltairBlock{AltairBlock: v2Blk},
|
|
Signature: sig[:],
|
|
},
|
|
ExecutionOptimistic: false,
|
|
}, nil
|
|
}
|
|
|
|
func (bs *Server) getBlockBellatrix(ctx context.Context, blk interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.BlockResponseV2, error) {
|
|
bellatrixBlk, err := blk.PbBellatrixBlock()
|
|
if err != nil {
|
|
// ErrUnsupportedField means that we have another block type
|
|
if errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
if blindedBellatrixBlk, err := blk.PbBlindedBellatrixBlock(); err == nil {
|
|
if blindedBellatrixBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
signedFullBlock, err := bs.ExecutionPayloadReconstructor.ReconstructFullBlock(ctx, blk)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not reconstruct full execution payload to create signed beacon block")
|
|
}
|
|
bellatrixBlk, err = signedFullBlock.PbBellatrixBlock()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get signed beacon block")
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockBellatrixToV2(bellatrixBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
return ðpbv2.BlockResponseV2{
|
|
Version: ethpbv2.Version_BELLATRIX,
|
|
Data: ðpbv2.SignedBeaconBlockContainer{
|
|
Message: ðpbv2.SignedBeaconBlockContainer_BellatrixBlock{BellatrixBlock: v2Blk},
|
|
Signature: sig[:],
|
|
},
|
|
ExecutionOptimistic: isOptimistic,
|
|
}, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
if bellatrixBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockBellatrixToV2(bellatrixBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
return ðpbv2.BlockResponseV2{
|
|
Version: ethpbv2.Version_BELLATRIX,
|
|
Data: ðpbv2.SignedBeaconBlockContainer{
|
|
Message: ðpbv2.SignedBeaconBlockContainer_BellatrixBlock{BellatrixBlock: v2Blk},
|
|
Signature: sig[:],
|
|
},
|
|
ExecutionOptimistic: isOptimistic,
|
|
}, nil
|
|
}
|
|
|
|
func (bs *Server) getBlockCapella(ctx context.Context, blk interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.BlockResponseV2, error) {
|
|
capellaBlk, err := blk.PbCapellaBlock()
|
|
if err != nil {
|
|
// ErrUnsupportedField means that we have another block type
|
|
if errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
if blindedCapellaBlk, err := blk.PbBlindedCapellaBlock(); err == nil {
|
|
if blindedCapellaBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
signedFullBlock, err := bs.ExecutionPayloadReconstructor.ReconstructFullBlock(ctx, blk)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not reconstruct full execution payload to create signed beacon block")
|
|
}
|
|
capellaBlk, err = signedFullBlock.PbCapellaBlock()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get signed beacon block")
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockCapellaToV2(capellaBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
return ðpbv2.BlockResponseV2{
|
|
Version: ethpbv2.Version_CAPELLA,
|
|
Data: ðpbv2.SignedBeaconBlockContainer{
|
|
Message: ðpbv2.SignedBeaconBlockContainer_CapellaBlock{CapellaBlock: v2Blk},
|
|
Signature: sig[:],
|
|
},
|
|
ExecutionOptimistic: isOptimistic,
|
|
}, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
if capellaBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockCapellaToV2(capellaBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
return ðpbv2.BlockResponseV2{
|
|
Version: ethpbv2.Version_CAPELLA,
|
|
Data: ðpbv2.SignedBeaconBlockContainer{
|
|
Message: ðpbv2.SignedBeaconBlockContainer_CapellaBlock{CapellaBlock: v2Blk},
|
|
Signature: sig[:],
|
|
},
|
|
ExecutionOptimistic: isOptimistic,
|
|
}, nil
|
|
}
|
|
|
|
func (bs *Server) getBlockDeneb(ctx context.Context, blk interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.BlockResponseV2, error) {
|
|
denebBlk, err := blk.PbDenebBlock()
|
|
if err != nil {
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
if blindedDenebBlk, err := blk.PbBlindedDenebBlock(); err == nil {
|
|
if blindedDenebBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
signedFullBlock, err := bs.ExecutionPayloadReconstructor.ReconstructFullBlock(ctx, blk)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not reconstruct full execution payload to create signed beacon block")
|
|
}
|
|
denebBlk, err = signedFullBlock.PbDenebBlock()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get signed beacon block")
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockDenebToV2(denebBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
return ðpbv2.BlockResponseV2{
|
|
Version: ethpbv2.Version_DENEB,
|
|
Data: ðpbv2.SignedBeaconBlockContainer{
|
|
Message: ðpbv2.SignedBeaconBlockContainer_DenebBlock{DenebBlock: v2Blk},
|
|
Signature: sig[:],
|
|
},
|
|
ExecutionOptimistic: isOptimistic,
|
|
}, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
if denebBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockDenebToV2(denebBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
return ðpbv2.BlockResponseV2{
|
|
Version: ethpbv2.Version_DENEB,
|
|
Data: ðpbv2.SignedBeaconBlockContainer{
|
|
Message: ðpbv2.SignedBeaconBlockContainer_DenebBlock{DenebBlock: v2Blk},
|
|
Signature: sig[:],
|
|
},
|
|
ExecutionOptimistic: isOptimistic,
|
|
}, nil
|
|
}
|
|
|
|
func getSSZBlockPhase0(blk interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.SSZContainer, error) {
|
|
phase0Blk, err := blk.PbPhase0Block()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if phase0Blk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
signedBeaconBlock, err := migration.SignedBeaconBlock(blk)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get signed beacon block")
|
|
}
|
|
sszBlock, err := signedBeaconBlock.MarshalSSZ()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not marshal block into SSZ")
|
|
}
|
|
return ðpbv2.SSZContainer{Version: ethpbv2.Version_PHASE0, ExecutionOptimistic: false, Data: sszBlock}, nil
|
|
}
|
|
|
|
func getSSZBlockAltair(blk interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.SSZContainer, error) {
|
|
altairBlk, err := blk.PbAltairBlock()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if altairBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockAltairToV2(altairBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get signed beacon block")
|
|
}
|
|
sig := blk.Signature()
|
|
data := ðpbv2.SignedBeaconBlockAltair{
|
|
Message: v2Blk,
|
|
Signature: sig[:],
|
|
}
|
|
sszData, err := data.MarshalSSZ()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not marshal block into SSZ")
|
|
}
|
|
return ðpbv2.SSZContainer{Version: ethpbv2.Version_ALTAIR, ExecutionOptimistic: false, Data: sszData}, nil
|
|
}
|
|
|
|
func (bs *Server) getSSZBlockBellatrix(ctx context.Context, blk interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.SSZContainer, error) {
|
|
bellatrixBlk, err := blk.PbBellatrixBlock()
|
|
if err != nil {
|
|
// ErrUnsupportedField means that we have another block type
|
|
if errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
if blindedBellatrixBlk, err := blk.PbBlindedBellatrixBlock(); err == nil {
|
|
if blindedBellatrixBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
signedFullBlock, err := bs.ExecutionPayloadReconstructor.ReconstructFullBlock(ctx, blk)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not reconstruct full execution payload to create signed beacon block")
|
|
}
|
|
bellatrixBlk, err = signedFullBlock.PbBellatrixBlock()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get signed beacon block")
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockBellatrixToV2(bellatrixBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert signed beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
data := ðpbv2.SignedBeaconBlockBellatrix{
|
|
Message: v2Blk,
|
|
Signature: sig[:],
|
|
}
|
|
sszData, err := data.MarshalSSZ()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not marshal block into SSZ")
|
|
}
|
|
return ðpbv2.SSZContainer{
|
|
Version: ethpbv2.Version_BELLATRIX,
|
|
ExecutionOptimistic: isOptimistic,
|
|
Data: sszData,
|
|
}, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
if bellatrixBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockBellatrixToV2(bellatrixBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert signed beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
data := ðpbv2.SignedBeaconBlockBellatrix{
|
|
Message: v2Blk,
|
|
Signature: sig[:],
|
|
}
|
|
sszData, err := data.MarshalSSZ()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not marshal block into SSZ")
|
|
}
|
|
return ðpbv2.SSZContainer{Version: ethpbv2.Version_BELLATRIX, ExecutionOptimistic: isOptimistic, Data: sszData}, nil
|
|
}
|
|
|
|
func (bs *Server) getSSZBlockCapella(ctx context.Context, blk interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.SSZContainer, error) {
|
|
capellaBlk, err := blk.PbCapellaBlock()
|
|
if err != nil {
|
|
// ErrUnsupportedField means that we have another block type
|
|
if errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
if blindedCapellaBlk, err := blk.PbBlindedCapellaBlock(); err == nil {
|
|
if blindedCapellaBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
signedFullBlock, err := bs.ExecutionPayloadReconstructor.ReconstructFullBlock(ctx, blk)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not reconstruct full execution payload to create signed beacon block")
|
|
}
|
|
capellaBlk, err = signedFullBlock.PbCapellaBlock()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get signed beacon block")
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockCapellaToV2(capellaBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert signed beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
data := ðpbv2.SignedBeaconBlockCapella{
|
|
Message: v2Blk,
|
|
Signature: sig[:],
|
|
}
|
|
sszData, err := data.MarshalSSZ()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not marshal block into SSZ")
|
|
}
|
|
return ðpbv2.SSZContainer{
|
|
Version: ethpbv2.Version_CAPELLA,
|
|
ExecutionOptimistic: isOptimistic,
|
|
Data: sszData,
|
|
}, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
if capellaBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockCapellaToV2(capellaBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert signed beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
data := ðpbv2.SignedBeaconBlockCapella{
|
|
Message: v2Blk,
|
|
Signature: sig[:],
|
|
}
|
|
sszData, err := data.MarshalSSZ()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not marshal block into SSZ")
|
|
}
|
|
return ðpbv2.SSZContainer{Version: ethpbv2.Version_CAPELLA, ExecutionOptimistic: isOptimistic, Data: sszData}, nil
|
|
}
|
|
|
|
func (bs *Server) getSSZBlockDeneb(ctx context.Context, blk interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.SSZContainer, error) {
|
|
denebBlk, err := blk.PbDenebBlock()
|
|
if err != nil {
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if errors.Is(err, consensus_types.ErrUnsupportedField) {
|
|
if blindedDenebBlk, err := blk.PbBlindedDenebBlock(); err == nil {
|
|
if blindedDenebBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
signedFullBlock, err := bs.ExecutionPayloadReconstructor.ReconstructFullBlock(ctx, blk)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not reconstruct full execution payload to create signed beacon block")
|
|
}
|
|
denebBlk, err = signedFullBlock.PbDenebBlock()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get signed beacon block")
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockDenebToV2(denebBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert signed beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
data := ðpbv2.SignedBeaconBlockDeneb{
|
|
Message: v2Blk,
|
|
Signature: sig[:],
|
|
}
|
|
sszData, err := data.MarshalSSZ()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not marshal block into SSZ")
|
|
}
|
|
return ðpbv2.SSZContainer{
|
|
Version: ethpbv2.Version_DENEB,
|
|
ExecutionOptimistic: isOptimistic,
|
|
Data: sszData,
|
|
}, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
if denebBlk == nil {
|
|
return nil, errNilBlock
|
|
}
|
|
v2Blk, err := migration.V1Alpha1BeaconBlockDenebToV2(denebBlk.Block)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not convert signed beacon block")
|
|
}
|
|
root, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not get block root")
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, root)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not check if block is optimistic")
|
|
}
|
|
sig := blk.Signature()
|
|
data := ðpbv2.SignedBeaconBlockDeneb{
|
|
Message: v2Blk,
|
|
Signature: sig[:],
|
|
}
|
|
sszData, err := data.MarshalSSZ()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not marshal block into SSZ")
|
|
}
|
|
return ðpbv2.SSZContainer{Version: ethpbv2.Version_DENEB, ExecutionOptimistic: isOptimistic, Data: sszData}, nil
|
|
}
|