mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-05 01:04:29 +00:00
209 lines
6.9 KiB
Go
209 lines
6.9 KiB
Go
package sync
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
libp2pcore "github.com/libp2p/go-libp2p/core"
|
|
"github.com/libp2p/go-libp2p/core/peer"
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/db"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/execution"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/p2p/types"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/sync/verify"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/blocks"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/interfaces"
|
|
eth "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v4/runtime/version"
|
|
)
|
|
|
|
// sendRecentBeaconBlocksRequest sends a recent beacon blocks request to a peer to get
|
|
// those corresponding blocks from that peer.
|
|
func (s *Service) sendRecentBeaconBlocksRequest(ctx context.Context, blockRoots *types.BeaconBlockByRootsReq, id peer.ID) error {
|
|
ctx, cancel := context.WithTimeout(ctx, respTimeout)
|
|
defer cancel()
|
|
|
|
blks, err := SendBeaconBlocksByRootRequest(ctx, s.cfg.clock, s.cfg.p2p, id, blockRoots, func(blk interfaces.ReadOnlySignedBeaconBlock) error {
|
|
blkRoot, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
s.pendingQueueLock.Lock()
|
|
defer s.pendingQueueLock.Unlock()
|
|
if err := s.insertBlockToPendingQueue(blk.Block().Slot(), blk, blkRoot); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
})
|
|
|
|
for _, blk := range blks {
|
|
// Skip blocks before deneb because they have no blob.
|
|
if blk.Version() < version.Deneb {
|
|
continue
|
|
}
|
|
blkRoot, err := blk.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := s.requestPendingBlobs(ctx, blk, blkRoot, id); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
// beaconBlocksRootRPCHandler looks up the request blocks from the database from the given block roots.
|
|
func (s *Service) beaconBlocksRootRPCHandler(ctx context.Context, msg interface{}, stream libp2pcore.Stream) error {
|
|
ctx, cancel := context.WithTimeout(ctx, ttfbTimeout)
|
|
defer cancel()
|
|
SetRPCStreamDeadlines(stream)
|
|
log := log.WithField("handler", "beacon_blocks_by_root")
|
|
|
|
rawMsg, ok := msg.(*types.BeaconBlockByRootsReq)
|
|
if !ok {
|
|
return errors.New("message is not type BeaconBlockByRootsReq")
|
|
}
|
|
blockRoots := *rawMsg
|
|
if err := s.rateLimiter.validateRequest(stream, uint64(len(blockRoots))); err != nil {
|
|
return err
|
|
}
|
|
if len(blockRoots) == 0 {
|
|
// Add to rate limiter in the event no
|
|
// roots are requested.
|
|
s.rateLimiter.add(stream, 1)
|
|
s.writeErrorResponseToStream(responseCodeInvalidRequest, "no block roots provided in request", stream)
|
|
return errors.New("no block roots provided")
|
|
}
|
|
|
|
if uint64(len(blockRoots)) > params.BeaconNetworkConfig().MaxRequestBlocks {
|
|
s.cfg.p2p.Peers().Scorers().BadResponsesScorer().Increment(stream.Conn().RemotePeer())
|
|
s.writeErrorResponseToStream(responseCodeInvalidRequest, "requested more than the max block limit", stream)
|
|
return errors.New("requested more than the max block limit")
|
|
}
|
|
s.rateLimiter.add(stream, int64(len(blockRoots)))
|
|
|
|
for _, root := range blockRoots {
|
|
blk, err := s.cfg.beaconDB.Block(ctx, root)
|
|
if err != nil {
|
|
log.WithError(err).Debug("Could not fetch block")
|
|
s.writeErrorResponseToStream(responseCodeServerError, types.ErrGeneric.Error(), stream)
|
|
return err
|
|
}
|
|
if err := blocks.BeaconBlockIsNil(blk); err != nil {
|
|
continue
|
|
}
|
|
|
|
if blk.Block().IsBlinded() {
|
|
blk, err = s.cfg.executionPayloadReconstructor.ReconstructFullBlock(ctx, blk)
|
|
if err != nil {
|
|
if errors.Is(err, execution.EmptyBlockHash) {
|
|
log.WithError(err).Warn("Could not reconstruct block from header with syncing execution client. Waiting to complete syncing")
|
|
} else {
|
|
log.WithError(err).Error("Could not get reconstruct full block from blinded body")
|
|
}
|
|
s.writeErrorResponseToStream(responseCodeServerError, types.ErrGeneric.Error(), stream)
|
|
return err
|
|
}
|
|
}
|
|
|
|
if err := s.chunkBlockWriter(stream, blk); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
closeStream(stream, log)
|
|
return nil
|
|
}
|
|
|
|
// requestPendingBlobs handles the request for pending blobs based on the given beacon block.
|
|
func (s *Service) requestPendingBlobs(ctx context.Context, block interfaces.ReadOnlySignedBeaconBlock, blockRoot [32]byte, peerID peer.ID) error {
|
|
if block.Version() < version.Deneb {
|
|
return nil // Block before deneb has no blob.
|
|
}
|
|
|
|
commitments, err := block.Block().Body().BlobKzgCommitments()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if len(commitments) == 0 {
|
|
return nil // No operation if the block has no blob commitments.
|
|
}
|
|
|
|
contextByte, err := ContextByteVersionsForValRoot(s.cfg.chain.GenesisValidatorsRoot())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
request, err := s.constructPendingBlobsRequest(ctx, blockRoot, len(commitments))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return s.sendAndSaveBlobSidecars(ctx, request, contextByte, peerID, block)
|
|
}
|
|
|
|
// sendAndSaveBlobSidecars sends the blob request and saves received sidecars.
|
|
func (s *Service) sendAndSaveBlobSidecars(ctx context.Context, request types.BlobSidecarsByRootReq, contextByte ContextByteVersions, peerID peer.ID, block interfaces.ReadOnlySignedBeaconBlock) error {
|
|
if len(request) == 0 {
|
|
return nil
|
|
}
|
|
|
|
sidecars, err := SendBlobSidecarByRoot(ctx, s.cfg.clock, s.cfg.p2p, peerID, contextByte, &request)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
RoBlock, err := blocks.NewROBlock(block)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if len(sidecars) != len(request) {
|
|
return fmt.Errorf("received %d blob sidecars, expected %d for RPC", len(sidecars), len(request))
|
|
}
|
|
for _, sidecar := range sidecars {
|
|
if err := verify.BlobAlignsWithBlock(sidecar, RoBlock); err != nil {
|
|
return err
|
|
}
|
|
log.WithFields(blobFields(sidecar)).Debug("Received blob sidecar RPC")
|
|
}
|
|
|
|
return s.cfg.beaconDB.SaveBlobSidecar(ctx, sidecars)
|
|
}
|
|
|
|
// constructPendingBlobsRequest creates a request for BlobSidecars by root, considering blobs already in DB.
|
|
func (s *Service) constructPendingBlobsRequest(ctx context.Context, blockRoot [32]byte, count int) (types.BlobSidecarsByRootReq, error) {
|
|
knownBlobs, err := s.cfg.beaconDB.BlobSidecarsByRoot(ctx, blockRoot)
|
|
if err != nil && !errors.Is(err, db.ErrNotFound) {
|
|
return nil, err
|
|
}
|
|
|
|
knownIndices := indexSetFromBlobs(knownBlobs)
|
|
requestedIndices := filterUnknownIndices(knownIndices, count, blockRoot)
|
|
|
|
return requestedIndices, nil
|
|
}
|
|
|
|
// Helper function to create a set of known indices.
|
|
func indexSetFromBlobs(blobs []*eth.BlobSidecar) map[uint64]struct{} {
|
|
indices := make(map[uint64]struct{})
|
|
for _, blob := range blobs {
|
|
indices[blob.Index] = struct{}{}
|
|
}
|
|
return indices
|
|
}
|
|
|
|
// Helper function to filter out known indices.
|
|
func filterUnknownIndices(knownIndices map[uint64]struct{}, count int, blockRoot [32]byte) []*eth.BlobIdentifier {
|
|
var ids []*eth.BlobIdentifier
|
|
for i := uint64(0); i < uint64(count); i++ {
|
|
if _, exists := knownIndices[i]; exists {
|
|
continue
|
|
}
|
|
ids = append(ids, ð.BlobIdentifier{Index: i, BlockRoot: blockRoot[:]})
|
|
}
|
|
return ids
|
|
}
|