package sync import ( "context" "time" libp2pcore "github.com/libp2p/go-libp2p/core" "github.com/pkg/errors" p2ptypes "github.com/prysmaticlabs/prysm/v5/beacon-chain/p2p/types" "github.com/prysmaticlabs/prysm/v5/cmd/beacon-chain/flags" "github.com/prysmaticlabs/prysm/v5/config/params" "github.com/prysmaticlabs/prysm/v5/consensus-types/blocks" "github.com/prysmaticlabs/prysm/v5/consensus-types/interfaces" "github.com/prysmaticlabs/prysm/v5/consensus-types/primitives" "github.com/prysmaticlabs/prysm/v5/monitoring/tracing" pb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/v5/time/slots" "go.opencensus.io/trace" ) // beaconBlocksByRangeRPCHandler looks up the request blocks from the database from a given start block. func (s *Service) beaconBlocksByRangeRPCHandler(ctx context.Context, msg interface{}, stream libp2pcore.Stream) error { ctx, span := trace.StartSpan(ctx, "sync.BeaconBlocksByRangeHandler") defer span.End() ctx, cancel := context.WithTimeout(ctx, respTimeout) defer cancel() SetRPCStreamDeadlines(stream) m, ok := msg.(*pb.BeaconBlocksByRangeRequest) if !ok { return errors.New("message is not type *pb.BeaconBlockByRangeRequest") } log.WithField("startSlot", m.StartSlot).WithField("count", m.Count).Debug("Serving block by range request") rp, err := validateRangeRequest(m, s.cfg.clock.CurrentSlot()) if err != nil { s.writeErrorResponseToStream(responseCodeInvalidRequest, err.Error(), stream) s.cfg.p2p.Peers().Scorers().BadResponsesScorer().Increment(stream.Conn().RemotePeer()) tracing.AnnotateError(span, err) return err } available := s.validateRangeAvailability(rp) if !available { log.Debug("error in validating range availability") s.writeErrorResponseToStream(responseCodeResourceUnavailable, p2ptypes.ErrResourceUnavailable.Error(), stream) tracing.AnnotateError(span, err) return nil } blockLimiter, err := s.rateLimiter.topicCollector(string(stream.Protocol())) if err != nil { return err } remainingBucketCapacity := blockLimiter.Remaining(stream.Conn().RemotePeer().String()) span.AddAttributes( trace.Int64Attribute("start", int64(rp.start)), // lint:ignore uintcast -- This conversion is OK for tracing. trace.Int64Attribute("end", int64(rp.end)), // lint:ignore uintcast -- This conversion is OK for tracing. trace.Int64Attribute("count", int64(m.Count)), trace.StringAttribute("peer", stream.Conn().RemotePeer().String()), trace.Int64Attribute("remaining_capacity", remainingBucketCapacity), ) // Ticker to stagger out large requests. ticker := time.NewTicker(time.Second) defer ticker.Stop() batcher, err := newBlockRangeBatcher(rp, s.cfg.beaconDB, s.rateLimiter, s.cfg.chain.IsCanonical, ticker) if err != nil { log.WithError(err).Info("error in BlocksByRange batch") s.writeErrorResponseToStream(responseCodeServerError, p2ptypes.ErrGeneric.Error(), stream) tracing.AnnotateError(span, err) return err } // prevRoot is used to ensure that returned chains are strictly linear for singular steps // by comparing the previous root of the block in the list with the current block's parent. var batch blockBatch var more bool for batch, more = batcher.next(ctx, stream); more; batch, more = batcher.next(ctx, stream) { batchStart := time.Now() if err := s.writeBlockBatchToStream(ctx, batch, stream); err != nil { s.writeErrorResponseToStream(responseCodeServerError, p2ptypes.ErrGeneric.Error(), stream) return err } rpcBlocksByRangeResponseLatency.Observe(float64(time.Since(batchStart).Milliseconds())) } if err := batch.error(); err != nil { log.WithError(err).Debug("error in BlocksByRange batch") s.writeErrorResponseToStream(responseCodeServerError, p2ptypes.ErrGeneric.Error(), stream) tracing.AnnotateError(span, err) return err } closeStream(stream, log) return nil } type rangeParams struct { start primitives.Slot end primitives.Slot size uint64 } func validateRangeRequest(r *pb.BeaconBlocksByRangeRequest, current primitives.Slot) (rangeParams, error) { rp := rangeParams{ start: r.StartSlot, size: r.Count, } maxRequest := params.MaxRequestBlock(slots.ToEpoch(current)) // Ensure all request params are within appropriate bounds if rp.size == 0 || rp.size > maxRequest { return rangeParams{}, p2ptypes.ErrInvalidRequest } // Allow some wiggle room, up to double the MaxRequestBlocks past the current slot, // to give nodes syncing close to the head of the chain some margin for error. maxStart, err := current.SafeAdd(maxRequest * 2) if err != nil { return rangeParams{}, p2ptypes.ErrInvalidRequest } if rp.start > maxStart { return rangeParams{}, p2ptypes.ErrInvalidRequest } rp.end, err = rp.start.SafeAdd(rp.size - 1) if err != nil { return rangeParams{}, p2ptypes.ErrInvalidRequest } limit := uint64(flags.Get().BlockBatchLimit) if limit > maxRequest { limit = maxRequest } if rp.size > limit { rp.size = limit } return rp, nil } func (s *Service) validateRangeAvailability(rp rangeParams) bool { startBlock := rp.start return s.availableBlocker.AvailableBlock(startBlock) } func (s *Service) writeBlockBatchToStream(ctx context.Context, batch blockBatch, stream libp2pcore.Stream) error { ctx, span := trace.StartSpan(ctx, "sync.WriteBlockRangeToStream") defer span.End() blinded := make([]interfaces.ReadOnlySignedBeaconBlock, 0) for _, b := range batch.canonical() { if err := blocks.BeaconBlockIsNil(b); err != nil { continue } if b.IsBlinded() { blinded = append(blinded, b.ReadOnlySignedBeaconBlock) continue } if chunkErr := s.chunkBlockWriter(stream, b); chunkErr != nil { log.WithError(chunkErr).Debug("Could not send a chunked response") return chunkErr } } if len(blinded) == 0 { return nil } reconstructed, err := s.cfg.executionPayloadReconstructor.ReconstructFullBellatrixBlockBatch(ctx, blinded) if err != nil { log.WithError(err).Error("Could not reconstruct full bellatrix block batch from blinded bodies") return err } for _, b := range reconstructed { if err := blocks.BeaconBlockIsNil(b); err != nil { continue } if b.IsBlinded() { continue } if chunkErr := s.chunkBlockWriter(stream, b); chunkErr != nil { log.WithError(chunkErr).Debug("Could not send a chunked response") return chunkErr } } return nil }