mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-05 17:22:18 +00:00
2f10b1c7b1
* Remove gogoproto compiler * Remove more gogoproto * Improvements * Fix gengo * More scripts * Gazelle, fix deps * Fix version and errors * Fix gocast for arrays * Fix ethapis * Fixes * Fix compile errors * fix go.mod * //proto/... builds * Update for protov2 * temp fix compilation to move on * Change everything to emptypb.empty * Add grpc to proto/slashings * Fix almost all build failures * Oher build problems * FIX THIS FUCKING THING * gaz literally every .bazel * Final touches * Final final touches * Fix proto * Begin moving proto.Marshal to native * Fix site_data * Fixes * Fix duplicate gateway * Fix gateway target * Fix ethapis * Fixes from review * Update * Fix * Fix status test * Fix fuzz * Add isprotoslice to fun * Change DeepEqual to DeepSSZEqual for proto arrays * Fix build * Fix gaz * Update go * Fixes * Fixes * Add case for nil validators after copy * Fix cast * Fix test * Fix imports * Go mod * Only use extension where needed * Fixes * Split gateway from gengo * gaz * go mod * Add back hydrated state * fix hydrate * Fix proto.clone * Fies * Revert "Split gateway from gengo" This reverts commit 7298bb2054d446e427d9af97e13b8fabe8695085. * Revert "gaz" This reverts commit ca952565701a88727e22302d6c8d60ac48d97255. * Merge all gateway into one target * go mod * Gaz * Add generate v1_gateway files * run pb again * goimports * gaz * Fix comments * Fix protos * Fix PR * Fix protos * Update grpc-gateway and ethapis * Update ethapis and gen-go-cast * Go tidy * Reorder * Fix ethapis * fix spec tests * Fix script * Remove unused import * Fix fuzz * Fix gomod * Update version * Error if the cloned result is nil * Handle optional slots * ADd more empty checks to clone * Undo fuzz changes * Fix build.bazel * Gaz * Redo fuzz changes * Undo some eth1data changes * Update go.mod Co-authored-by: Preston Van Loon <preston@prysmaticlabs.com> * Undo clone beacon state * Remove gogo proto more and unused v1_gateway * Add manual fix for nil vals * Fix gaz * tidy * Tidy again * Add detailed error * Revert "Add detailed error" This reverts commit 59bc053dcd59569a54c95b07739d5a379665ec5d. * Undo varint changes * Fix nil validators in deposit test * Commit * Undo Co-authored-by: Preston Van Loon <preston@prysmaticlabs.com> Co-authored-by: Radosław Kapka <rkapka@wp.pl> Co-authored-by: Nishant Das <nishdas93@gmail.com> Co-authored-by: Raul Jordan <raul@prysmaticlabs.com>
339 lines
11 KiB
Go
339 lines
11 KiB
Go
package beaconv1
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strconv"
|
|
|
|
"github.com/pkg/errors"
|
|
types "github.com/prysmaticlabs/eth2-types"
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1"
|
|
ethpb_alpha "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/feed"
|
|
blockfeed "github.com/prysmaticlabs/prysm/beacon-chain/core/feed/block"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db/filters"
|
|
"github.com/prysmaticlabs/prysm/proto/migration"
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/status"
|
|
"google.golang.org/protobuf/types/known/emptypb"
|
|
)
|
|
|
|
// GetBlockHeader retrieves block header for given block id.
|
|
func (bs *Server) GetBlockHeader(ctx context.Context, req *ethpb.BlockRequest) (*ethpb.BlockHeaderResponse, error) {
|
|
blk, err := bs.blockFromBlockID(ctx, req.BlockId)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get block from block ID: %v", err)
|
|
}
|
|
if blk == nil {
|
|
return nil, status.Errorf(codes.NotFound, "Could not find requested block header")
|
|
}
|
|
|
|
v1BlockHdr, err := migration.V1Alpha1BlockToV1BlockHeader(blk)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get block header from block: %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)
|
|
}
|
|
root, err := v1BlockHdr.HashTreeRoot()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not hash block header: %v", err)
|
|
}
|
|
|
|
return ðpb.BlockHeaderResponse{
|
|
Data: ðpb.BlockHeaderContainer{
|
|
Root: root[:],
|
|
Canonical: canonical,
|
|
Header: ðpb.BeaconBlockHeaderContainer{
|
|
Message: v1BlockHdr.Header,
|
|
Signature: v1BlockHdr.Signature,
|
|
},
|
|
},
|
|
}, 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 *ethpb.BlockHeadersRequest) (*ethpb.BlockHeadersResponse, error) {
|
|
var err error
|
|
var blks []*ethpb_alpha.SignedBeaconBlock
|
|
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 blks == nil {
|
|
return nil, status.Error(codes.NotFound, "Could not find requested blocks")
|
|
}
|
|
|
|
blkHdrs := make([]*ethpb.BlockHeaderContainer, len(blks))
|
|
for i, blk := range blks {
|
|
blkHdr, err := migration.V1Alpha1BlockToV1BlockHeader(blk)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get block header from block: %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)
|
|
}
|
|
root, err := blkHdr.Header.HashTreeRoot()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not hash block header: %v", err)
|
|
}
|
|
blkHdrs[i] = ðpb.BlockHeaderContainer{
|
|
Root: root[:],
|
|
Canonical: canonical,
|
|
Header: ðpb.BeaconBlockHeaderContainer{
|
|
Message: blkHdr.Header,
|
|
Signature: blkHdr.Signature,
|
|
},
|
|
}
|
|
}
|
|
|
|
return ðpb.BlockHeadersResponse{Data: blkHdrs}, 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 BeaconBlock, 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 *ethpb.BeaconBlockContainer) (*emptypb.Empty, error) {
|
|
blk := req.Message
|
|
|
|
v1alpha1Block, err := migration.V1ToV1Alpha1Block(ðpb.SignedBeaconBlock{Block: blk, Signature: req.Signature})
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not convert block to v1")
|
|
}
|
|
|
|
root, err := blk.HashTreeRoot()
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not tree hash block: %v", err)
|
|
}
|
|
|
|
// Do not block proposal critical path with debug logging or block feed updates.
|
|
defer func() {
|
|
log.WithField("blockRoot", fmt.Sprintf("%#x", bytesutil.Trunc(root[:]))).Debugf(
|
|
"Block proposal received via RPC")
|
|
bs.BlockNotifier.BlockFeed().Send(&feed.Event{
|
|
Type: blockfeed.ReceivedBlock,
|
|
Data: &blockfeed.ReceivedBlockData{SignedBlock: v1alpha1Block},
|
|
})
|
|
}()
|
|
|
|
// Broadcast the new block to the network.
|
|
if err := bs.Broadcaster.Broadcast(ctx, v1alpha1Block); err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not broadcast block: %v", err)
|
|
}
|
|
|
|
if err := bs.BlockReceiver.ReceiveBlock(ctx, v1alpha1Block, root); err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not process beacon block: %v", err)
|
|
}
|
|
|
|
return &emptypb.Empty{}, nil
|
|
}
|
|
|
|
// GetBlock retrieves block details for given block id.
|
|
func (bs *Server) GetBlock(ctx context.Context, req *ethpb.BlockRequest) (*ethpb.BlockResponse, error) {
|
|
blk, err := bs.blockFromBlockID(ctx, req.BlockId)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get block from block ID: %v", err)
|
|
}
|
|
if blk == nil {
|
|
return nil, status.Errorf(codes.NotFound, "Could not find requested block")
|
|
}
|
|
|
|
v1Block, err := migration.V1Alpha1ToV1Block(blk)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not convert block to v1")
|
|
}
|
|
|
|
return ðpb.BlockResponse{
|
|
Data: ðpb.BeaconBlockContainer{
|
|
Message: v1Block.Block,
|
|
Signature: blk.Signature,
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
// GetBlockRoot retrieves hashTreeRoot of BeaconBlock/BeaconBlockHeader.
|
|
func (bs *Server) GetBlockRoot(ctx context.Context, req *ethpb.BlockRequest) (*ethpb.BlockRootResponse, error) {
|
|
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 blk == nil {
|
|
return nil, status.Error(codes.NotFound, "Could not find genesis block")
|
|
}
|
|
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 {
|
|
block, 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 block == nil {
|
|
return nil, status.Error(codes.NotFound, "Could not find any blocks with given root")
|
|
}
|
|
|
|
root = req.BlockId
|
|
} else {
|
|
slot, err := strconv.ParseUint(string(req.BlockId), 10, 64)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "could not decode block id: %v", err)
|
|
}
|
|
hasRoots, roots, err := bs.BeaconDB.BlockRootsBySlot(ctx, types.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
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return ðpb.BlockRootResponse{
|
|
Data: ðpb.BlockRootContainer{
|
|
Root: root,
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
// ListBlockAttestations retrieves attestation included in requested block.
|
|
func (bs *Server) ListBlockAttestations(ctx context.Context, req *ethpb.BlockRequest) (*ethpb.BlockAttestationsResponse, error) {
|
|
blk, err := bs.blockFromBlockID(ctx, req.BlockId)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not get block from block ID: %v", err)
|
|
}
|
|
if blk == nil {
|
|
return nil, status.Errorf(codes.NotFound, "Could not find requested block")
|
|
}
|
|
|
|
v1Block, err := migration.V1Alpha1ToV1Block(blk)
|
|
if err != nil {
|
|
return nil, status.Errorf(codes.Internal, "Could not convert block to v1")
|
|
}
|
|
return ðpb.BlockAttestationsResponse{
|
|
Data: v1Block.Block.Body.Attestations,
|
|
}, nil
|
|
}
|
|
|
|
func (bs *Server) blockFromBlockID(ctx context.Context, blockId []byte) (*ethpb_alpha.SignedBeaconBlock, error) {
|
|
var err error
|
|
var blk *ethpb_alpha.SignedBeaconBlock
|
|
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 {
|
|
return nil, errors.Wrap(err, "could not decode block id")
|
|
}
|
|
_, blks, err := bs.BeaconDB.BlocksBySlot(ctx, types.Slot(slot))
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not retrieve blocks for slot %d", slot)
|
|
}
|
|
_, roots, err := bs.BeaconDB.BlockRootsBySlot(ctx, types.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
|
|
}
|
|
blk = blks[0]
|
|
if numBlks == 1 {
|
|
break
|
|
}
|
|
for i, block := 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 = block
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return blk, nil
|
|
}
|