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 }