mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-15 14:38:20 +00:00
f49720209e
Co-authored-by: Raul Jordan <raul@prysmaticlabs.com>
1109 lines
40 KiB
Go
1109 lines
40 KiB
Go
package beacon
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strconv"
|
|
|
|
"github.com/golang/protobuf/ptypes/empty"
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/core/feed"
|
|
blockfeed "github.com/prysmaticlabs/prysm/v3/beacon-chain/core/feed/block"
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/core/helpers"
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/db/filters"
|
|
rpchelpers "github.com/prysmaticlabs/prysm/v3/beacon-chain/rpc/eth/helpers"
|
|
fieldparams "github.com/prysmaticlabs/prysm/v3/config/fieldparams"
|
|
"github.com/prysmaticlabs/prysm/v3/config/params"
|
|
"github.com/prysmaticlabs/prysm/v3/consensus-types/blocks"
|
|
"github.com/prysmaticlabs/prysm/v3/consensus-types/interfaces"
|
|
"github.com/prysmaticlabs/prysm/v3/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/ssz/detect"
|
|
"github.com/prysmaticlabs/prysm/v3/network/forks"
|
|
ethpbv1 "github.com/prysmaticlabs/prysm/v3/proto/eth/v1"
|
|
ethpbv2 "github.com/prysmaticlabs/prysm/v3/proto/eth/v2"
|
|
"github.com/prysmaticlabs/prysm/v3/proto/migration"
|
|
"github.com/prysmaticlabs/prysm/v3/time/slots"
|
|
"go.opencensus.io/trace"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/metadata"
|
|
"google.golang.org/grpc/status"
|
|
"google.golang.org/protobuf/types/known/emptypb"
|
|
)
|
|
|
|
const versionHeader = "eth-consensus-version"
|
|
|
|
var (
|
|
errNilBlock = errors.New("nil block")
|
|
)
|
|
|
|
// blockIdParseError represents an error scenario where a block ID could not be parsed.
|
|
type blockIdParseError struct {
|
|
message string
|
|
}
|
|
|
|
// newBlockIdParseError creates a new error instance.
|
|
func newBlockIdParseError(reason error) blockIdParseError {
|
|
return blockIdParseError{
|
|
message: errors.Wrapf(reason, "could not parse block ID").Error(),
|
|
}
|
|
}
|
|
|
|
// Error returns the underlying error message.
|
|
func (e *blockIdParseError) Error() string {
|
|
return e.message
|
|
}
|
|
|
|
// 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.
|
|
func (bs *Server) GetWeakSubjectivity(ctx context.Context, _ *empty.Empty) (*ethpbv1.WeakSubjectivityResponse, error) {
|
|
if err := rpchelpers.ValidateSync(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
|
|
}
|
|
|
|
// GetBlockHeader retrieves block header for given block id.
|
|
func (bs *Server) GetBlockHeader(ctx context.Context, req *ethpbv1.BlockRequest) (*ethpbv1.BlockHeaderResponse, error) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon.GetBlockHeader")
|
|
defer span.End()
|
|
|
|
blk, err := bs.blockFromBlockID(ctx, req.BlockId)
|
|
err = handleGetBlockError(blk, err)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
v1alpha1Header, err := blk.Header()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get block header from block: %v", err)
|
|
}
|
|
header := migration.V1Alpha1SignedHeaderToV1(v1alpha1Header)
|
|
headerRoot, err := header.Message.HashTreeRoot()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not hash block header: %v", err)
|
|
}
|
|
blkRoot, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not hash block: %v", err)
|
|
}
|
|
canonical, err := bs.ChainInfoFetcher.IsCanonical(ctx, blkRoot)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not determine if block root is canonical: %v", err)
|
|
}
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, blkRoot)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not check if block is optimistic: %v", err)
|
|
}
|
|
|
|
return ðpbv1.BlockHeaderResponse{
|
|
Data: ðpbv1.BlockHeaderContainer{
|
|
Root: headerRoot[:],
|
|
Canonical: canonical,
|
|
Header: ðpbv1.BeaconBlockHeaderContainer{
|
|
Message: header.Message,
|
|
Signature: header.Signature,
|
|
},
|
|
},
|
|
ExecutionOptimistic: isOptimistic,
|
|
Finalized: bs.FinalizationFetcher.IsFinalized(ctx, blkRoot),
|
|
}, nil
|
|
}
|
|
|
|
// ListBlockHeaders retrieves block headers matching given query. By default it will fetch current head slot blocks.
|
|
func (bs *Server) ListBlockHeaders(ctx context.Context, req *ethpbv1.BlockHeadersRequest) (*ethpbv1.BlockHeadersResponse, error) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon.ListBlockHeaders")
|
|
defer span.End()
|
|
|
|
var err error
|
|
var blks []interfaces.ReadOnlySignedBeaconBlock
|
|
var blkRoots [][32]byte
|
|
if len(req.ParentRoot) == 32 {
|
|
blks, blkRoots, err = bs.BeaconDB.Blocks(ctx, filters.NewFilter().SetParentRoot(req.ParentRoot))
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve blocks: %v", err)
|
|
}
|
|
} else {
|
|
slot := bs.ChainInfoFetcher.HeadSlot()
|
|
if req.Slot != nil {
|
|
slot = *req.Slot
|
|
}
|
|
blks, err = bs.BeaconDB.BlocksBySlot(ctx, slot)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve blocks for slot %d: %v", req.Slot, err)
|
|
}
|
|
_, blkRoots, err = bs.BeaconDB.BlockRootsBySlot(ctx, slot)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve block roots for slot %d: %v", req.Slot, err)
|
|
}
|
|
}
|
|
if len(blks) == 0 {
|
|
return nil, status.Error(codes.NotFound, "Could not find requested blocks")
|
|
}
|
|
|
|
isOptimistic := false
|
|
isFinalized := true
|
|
blkHdrs := make([]*ethpbv1.BlockHeaderContainer, len(blks))
|
|
for i, bl := range blks {
|
|
v1alpha1Header, err := bl.Header()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get block header from block: %v", err)
|
|
}
|
|
header := migration.V1Alpha1SignedHeaderToV1(v1alpha1Header)
|
|
headerRoot, err := header.Message.HashTreeRoot()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not hash block header: %v", err)
|
|
}
|
|
canonical, err := bs.ChainInfoFetcher.IsCanonical(ctx, blkRoots[i])
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not determine if block root is canonical: %v", err)
|
|
}
|
|
if !isOptimistic {
|
|
isOptimistic, err = bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, blkRoots[i])
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not check if block is optimistic: %v", err)
|
|
}
|
|
}
|
|
if isFinalized {
|
|
isFinalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoots[i])
|
|
}
|
|
blkHdrs[i] = ðpbv1.BlockHeaderContainer{
|
|
Root: headerRoot[:],
|
|
Canonical: canonical,
|
|
Header: ðpbv1.BeaconBlockHeaderContainer{
|
|
Message: header.Message,
|
|
Signature: header.Signature,
|
|
},
|
|
}
|
|
}
|
|
|
|
return ðpbv1.BlockHeadersResponse{Data: blkHdrs, ExecutionOptimistic: isOptimistic, Finalized: isFinalized}, nil
|
|
}
|
|
|
|
// SubmitBlock instructs the beacon node to broadcast a newly signed beacon block to the beacon network, to be
|
|
// included in the beacon chain. The beacon node is not required to validate the signed ReadOnlyBeaconBlock, and a successful
|
|
// response (20X) only indicates that the broadcast has been successful. The beacon node is expected to integrate the
|
|
// new block into its state, and therefore validate the block internally, however blocks which fail the validation are
|
|
// still broadcast but a different status code is returned (202).
|
|
func (bs *Server) SubmitBlock(ctx context.Context, req *ethpbv2.SignedBeaconBlockContainer) (*emptypb.Empty, error) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon.SubmitBlock")
|
|
defer span.End()
|
|
|
|
switch blkContainer := req.Message.(type) {
|
|
case *ethpbv2.SignedBeaconBlockContainer_Phase0Block:
|
|
if err := bs.submitPhase0Block(ctx, blkContainer.Phase0Block, req.Signature); err != nil {
|
|
return nil, err
|
|
}
|
|
case *ethpbv2.SignedBeaconBlockContainer_AltairBlock:
|
|
if err := bs.submitAltairBlock(ctx, blkContainer.AltairBlock, req.Signature); err != nil {
|
|
return nil, err
|
|
}
|
|
case *ethpbv2.SignedBeaconBlockContainer_BellatrixBlock:
|
|
if err := bs.submitBellatrixBlock(ctx, blkContainer.BellatrixBlock, req.Signature); err != nil {
|
|
return nil, err
|
|
}
|
|
case *ethpbv2.SignedBeaconBlockContainer_CapellaBlock:
|
|
if err := bs.submitCapellaBlock(ctx, blkContainer.CapellaBlock, req.Signature); err != nil {
|
|
return nil, err
|
|
}
|
|
default:
|
|
return nil, status.Errorf(codes.InvalidArgument, "Unsupported block container type %T", blkContainer)
|
|
}
|
|
|
|
return &emptypb.Empty{}, nil
|
|
}
|
|
|
|
// SubmitBlockSSZ instructs the beacon node to broadcast a newly signed beacon block to the beacon network, to be
|
|
// included in the beacon chain. The beacon node is not required to validate the signed ReadOnlyBeaconBlock, and a successful
|
|
// response (20X) only indicates that the broadcast has been successful. The beacon node is expected to integrate the
|
|
// new block into its state, and therefore validate the block internally, however blocks which fail the validation are
|
|
// still broadcast but a different status code is returned (202).
|
|
//
|
|
// The provided block must be SSZ-serialized.
|
|
func (bs *Server) SubmitBlockSSZ(ctx context.Context, req *ethpbv2.SSZContainer) (*emptypb.Empty, error) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon.SubmitBlockSSZ")
|
|
defer span.End()
|
|
|
|
md, ok := metadata.FromIncomingContext(ctx)
|
|
if !ok {
|
|
return &emptypb.Empty{}, status.Errorf(codes.Internal, "Could not read "+versionHeader+" header")
|
|
}
|
|
ver := md.Get(versionHeader)
|
|
if len(ver) == 0 {
|
|
return &emptypb.Empty{}, status.Errorf(codes.Internal, "Could not read "+versionHeader+" header")
|
|
}
|
|
schedule := forks.NewOrderedSchedule(params.BeaconConfig())
|
|
forkVer, err := schedule.VersionForName(ver[0])
|
|
if err != nil {
|
|
return &emptypb.Empty{}, status.Errorf(codes.Internal, "Could not determine fork version: %v", err)
|
|
}
|
|
unmarshaler, err := detect.FromForkVersion(forkVer)
|
|
if err != nil {
|
|
return &emptypb.Empty{}, status.Errorf(codes.Internal, "Could not create unmarshaler: %v", err)
|
|
}
|
|
block, err := unmarshaler.UnmarshalBeaconBlock(req.Data)
|
|
if err != nil {
|
|
return &emptypb.Empty{}, status.Errorf(codes.Internal, "Could not unmarshal request data into block: %v", err)
|
|
}
|
|
root, err := block.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return &emptypb.Empty{}, status.Errorf(codes.Internal, "Could not compute block's hash tree root: %v", err)
|
|
}
|
|
return &emptypb.Empty{}, bs.submitBlock(ctx, root, block)
|
|
}
|
|
|
|
// GetBlock retrieves block details for given block ID.
|
|
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.blockFromBlockID(ctx, req.BlockId)
|
|
err = 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.
|
|
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.blockFromBlockID(ctx, req.BlockId)
|
|
err = 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.blockFromBlockID(ctx, req.BlockId)
|
|
err = 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
|
|
}
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if !errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
return nil, status.Errorf(codes.Internal, "Could not get signed beacon block: %v", err)
|
|
}
|
|
result, err = getBlockAltair(blk)
|
|
if result != nil {
|
|
result.Finalized = bs.FinalizationFetcher.IsFinalized(ctx, blkRoot)
|
|
return result, nil
|
|
}
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if !errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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
|
|
}
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if !errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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
|
|
}
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if !errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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.blockFromBlockID(ctx, req.BlockId)
|
|
err = 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
|
|
}
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if !errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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
|
|
}
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if !errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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
|
|
}
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if !errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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
|
|
}
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if !errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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)
|
|
}
|
|
|
|
// GetBlockRoot retrieves hashTreeRoot of ReadOnlyBeaconBlock/BeaconBlockHeader.
|
|
func (bs *Server) GetBlockRoot(ctx context.Context, req *ethpbv1.BlockRequest) (*ethpbv1.BlockRootResponse, error) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon.GetBlockRoot")
|
|
defer span.End()
|
|
|
|
var root []byte
|
|
var err error
|
|
switch string(req.BlockId) {
|
|
case "head":
|
|
root, err = bs.ChainInfoFetcher.HeadRoot(ctx)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve head block: %v", err)
|
|
}
|
|
if root == nil {
|
|
return nil, status.Errorf(codes.NotFound, "No head root was found")
|
|
}
|
|
case "finalized":
|
|
finalized := bs.ChainInfoFetcher.FinalizedCheckpt()
|
|
root = finalized.Root
|
|
case "genesis":
|
|
blk, err := bs.BeaconDB.GenesisBlock(ctx)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve blocks for genesis slot: %v", err)
|
|
}
|
|
if err := blocks.BeaconBlockIsNil(blk); err != nil {
|
|
return nil, status.Errorf(codes.NotFound, "Could not find genesis block: %v", err)
|
|
}
|
|
blkRoot, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, status.Error(codes.Internal, "Could not hash genesis block")
|
|
}
|
|
root = blkRoot[:]
|
|
default:
|
|
if len(req.BlockId) == 32 {
|
|
blk, err := bs.BeaconDB.Block(ctx, bytesutil.ToBytes32(req.BlockId))
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve block for block root %#x: %v", req.BlockId, err)
|
|
}
|
|
if err := blocks.BeaconBlockIsNil(blk); err != nil {
|
|
return nil, status.Errorf(codes.NotFound, "Could not find block: %v", err)
|
|
}
|
|
root = req.BlockId
|
|
} else {
|
|
slot, err := strconv.ParseUint(string(req.BlockId), 10, 64)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.InvalidArgument, "Could not parse block ID: %v", err)
|
|
}
|
|
hasRoots, roots, err := bs.BeaconDB.BlockRootsBySlot(ctx, primitives.Slot(slot))
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve blocks for slot %d: %v", slot, err)
|
|
}
|
|
|
|
if !hasRoots {
|
|
return nil, status.Error(codes.NotFound, "Could not find any blocks with given slot")
|
|
}
|
|
root = roots[0][:]
|
|
if len(roots) == 1 {
|
|
break
|
|
}
|
|
for _, blockRoot := range roots {
|
|
canonical, err := bs.ChainInfoFetcher.IsCanonical(ctx, blockRoot)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not determine if block root is canonical: %v", err)
|
|
}
|
|
if canonical {
|
|
root = blockRoot[:]
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
b32Root := bytesutil.ToBytes32(root)
|
|
isOptimistic, err := bs.OptimisticModeFetcher.IsOptimisticForRoot(ctx, b32Root)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not check if block is optimistic: %v", err)
|
|
}
|
|
|
|
return ðpbv1.BlockRootResponse{
|
|
Data: ðpbv1.BlockRootContainer{
|
|
Root: root,
|
|
},
|
|
ExecutionOptimistic: isOptimistic,
|
|
Finalized: bs.FinalizationFetcher.IsFinalized(ctx, b32Root),
|
|
}, nil
|
|
}
|
|
|
|
// 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.blockFromBlockID(ctx, req.BlockId)
|
|
err = 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 (bs *Server) blockFromBlockID(ctx context.Context, blockId []byte) (interfaces.ReadOnlySignedBeaconBlock, error) {
|
|
var err error
|
|
var blk interfaces.ReadOnlySignedBeaconBlock
|
|
switch string(blockId) {
|
|
case "head":
|
|
blk, err = bs.ChainInfoFetcher.HeadBlock(ctx)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not retrieve head block")
|
|
}
|
|
case "finalized":
|
|
finalized := bs.ChainInfoFetcher.FinalizedCheckpt()
|
|
finalizedRoot := bytesutil.ToBytes32(finalized.Root)
|
|
blk, err = bs.BeaconDB.Block(ctx, finalizedRoot)
|
|
if err != nil {
|
|
return nil, errors.New("could not get finalized block from db")
|
|
}
|
|
case "genesis":
|
|
blk, err = bs.BeaconDB.GenesisBlock(ctx)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not retrieve blocks for genesis slot")
|
|
}
|
|
default:
|
|
if len(blockId) == 32 {
|
|
blk, err = bs.BeaconDB.Block(ctx, bytesutil.ToBytes32(blockId))
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not retrieve block")
|
|
}
|
|
} else {
|
|
slot, err := strconv.ParseUint(string(blockId), 10, 64)
|
|
if err != nil {
|
|
e := newBlockIdParseError(err)
|
|
return nil, &e
|
|
}
|
|
blks, err := bs.BeaconDB.BlocksBySlot(ctx, primitives.Slot(slot))
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not retrieve blocks for slot %d", slot)
|
|
}
|
|
_, roots, err := bs.BeaconDB.BlockRootsBySlot(ctx, primitives.Slot(slot))
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not retrieve block roots for slot %d", slot)
|
|
}
|
|
numBlks := len(blks)
|
|
if numBlks == 0 {
|
|
return nil, nil
|
|
}
|
|
for i, b := range blks {
|
|
canonical, err := bs.ChainInfoFetcher.IsCanonical(ctx, roots[i])
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not determine if block root is canonical: %v", err)
|
|
}
|
|
if canonical {
|
|
blk = b
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return blk, nil
|
|
}
|
|
|
|
func handleGetBlockError(blk interfaces.ReadOnlySignedBeaconBlock, err error) error {
|
|
if invalidBlockIdErr, ok := err.(*blockIdParseError); ok {
|
|
return status.Errorf(codes.InvalidArgument, "Invalid block ID: %v", invalidBlockIdErr)
|
|
}
|
|
if err != nil {
|
|
return status.Errorf(codes.Internal, "Could not get block from block ID: %v", err)
|
|
}
|
|
if err := blocks.BeaconBlockIsNil(blk); err != nil {
|
|
return status.Errorf(codes.NotFound, "Could not find requested block: %v", err)
|
|
}
|
|
return 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 {
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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 {
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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 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 {
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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 {
|
|
// ErrUnsupportedGetter means that we have another block type
|
|
if errors.Is(err, blocks.ErrUnsupportedGetter) {
|
|
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) submitPhase0Block(ctx context.Context, phase0Blk *ethpbv1.BeaconBlock, sig []byte) error {
|
|
v1alpha1Blk, err := migration.V1ToV1Alpha1SignedBlock(ðpbv1.SignedBeaconBlock{Block: phase0Blk, Signature: sig})
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not convert block to v1 block")
|
|
}
|
|
wrappedPhase0Blk, err := blocks.NewSignedBeaconBlock(v1alpha1Blk)
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not prepare block")
|
|
}
|
|
root, err := phase0Blk.HashTreeRoot()
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not tree hash block: %v", err)
|
|
}
|
|
|
|
return bs.submitBlock(ctx, root, wrappedPhase0Blk)
|
|
}
|
|
|
|
func (bs *Server) submitAltairBlock(ctx context.Context, altairBlk *ethpbv2.BeaconBlockAltair, sig []byte) error {
|
|
v1alpha1Blk, err := migration.AltairToV1Alpha1SignedBlock(ðpbv2.SignedBeaconBlockAltair{Message: altairBlk, Signature: sig})
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not convert block to v1 block")
|
|
}
|
|
wrappedAltairBlk, err := blocks.NewSignedBeaconBlock(v1alpha1Blk)
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not prepare block")
|
|
}
|
|
|
|
root, err := altairBlk.HashTreeRoot()
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not tree hash block: %v", err)
|
|
}
|
|
|
|
return bs.submitBlock(ctx, root, wrappedAltairBlk)
|
|
}
|
|
|
|
func (bs *Server) submitBellatrixBlock(ctx context.Context, bellatrixBlk *ethpbv2.BeaconBlockBellatrix, sig []byte) error {
|
|
v1alpha1Blk, err := migration.BellatrixToV1Alpha1SignedBlock(ðpbv2.SignedBeaconBlockBellatrix{Message: bellatrixBlk, Signature: sig})
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not convert block to v1 block")
|
|
}
|
|
wrappedBellatrixBlk, err := blocks.NewSignedBeaconBlock(v1alpha1Blk)
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not prepare block")
|
|
}
|
|
|
|
root, err := bellatrixBlk.HashTreeRoot()
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not tree hash block: %v", err)
|
|
}
|
|
|
|
return bs.submitBlock(ctx, root, wrappedBellatrixBlk)
|
|
}
|
|
|
|
func (bs *Server) submitCapellaBlock(ctx context.Context, capellaBlk *ethpbv2.BeaconBlockCapella, sig []byte) error {
|
|
v1alpha1Blk, err := migration.CapellaToV1Alpha1SignedBlock(ðpbv2.SignedBeaconBlockCapella{Message: capellaBlk, Signature: sig})
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not convert block to v1 block")
|
|
}
|
|
wrappedCapellaBlk, err := blocks.NewSignedBeaconBlock(v1alpha1Blk)
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not prepare block")
|
|
}
|
|
|
|
root, err := capellaBlk.HashTreeRoot()
|
|
if err != nil {
|
|
return status.Errorf(codes.InvalidArgument, "Could not tree hash block: %v", err)
|
|
}
|
|
|
|
return bs.submitBlock(ctx, root, wrappedCapellaBlk)
|
|
}
|
|
|
|
func (bs *Server) submitBlock(ctx context.Context, blockRoot [fieldparams.RootLength]byte, block interfaces.ReadOnlySignedBeaconBlock) error {
|
|
// Do not block proposal critical path with debug logging or block feed updates.
|
|
defer func() {
|
|
log.WithField("blockRoot", fmt.Sprintf("%#x", bytesutil.Trunc(blockRoot[:]))).Debugf(
|
|
"Block proposal received via RPC")
|
|
bs.BlockNotifier.BlockFeed().Send(&feed.Event{
|
|
Type: blockfeed.ReceivedBlock,
|
|
Data: &blockfeed.ReceivedBlockData{SignedBlock: block},
|
|
})
|
|
}()
|
|
|
|
// Broadcast the new block to the network.
|
|
blockPb, err := block.Proto()
|
|
if err != nil {
|
|
return errors.Wrap(err, "could not get protobuf block")
|
|
}
|
|
if err := bs.Broadcaster.Broadcast(ctx, blockPb); err != nil {
|
|
return status.Errorf(codes.Internal, "Could not broadcast block: %v", err)
|
|
}
|
|
|
|
if err := bs.BlockReceiver.ReceiveBlock(ctx, block, blockRoot); err != nil {
|
|
return status.Errorf(codes.Internal, "Could not process beacon block: %v", err)
|
|
}
|
|
|
|
return nil
|
|
}
|