-
Notifications
You must be signed in to change notification settings - Fork 1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add Request And Response RPC Methods For Data Columns (#13909)
* Add RPC Handler * Add Column Requests * Update beacon-chain/db/filesystem/blob.go Co-authored-by: Manu NALEPA <enalepa@offchainlabs.com> * Update beacon-chain/p2p/rpc_topic_mappings.go Co-authored-by: Manu NALEPA <enalepa@offchainlabs.com> * Manu's Review * Manu's Review * Interface Fixes * mock manager --------- Co-authored-by: Manu NALEPA <enalepa@offchainlabs.com>
- Loading branch information
Showing
14 changed files
with
263 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,151 @@ | ||
package sync | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
"math" | ||
"sort" | ||
"time" | ||
|
||
libp2pcore "github.com/libp2p/go-libp2p/core" | ||
"github.com/pkg/errors" | ||
"github.com/prysmaticlabs/prysm/v5/beacon-chain/db" | ||
"github.com/prysmaticlabs/prysm/v5/beacon-chain/p2p" | ||
"github.com/prysmaticlabs/prysm/v5/beacon-chain/p2p/types" | ||
"github.com/prysmaticlabs/prysm/v5/cmd/beacon-chain/flags" | ||
"github.com/prysmaticlabs/prysm/v5/config/features" | ||
"github.com/prysmaticlabs/prysm/v5/config/params" | ||
"github.com/prysmaticlabs/prysm/v5/consensus-types/primitives" | ||
"github.com/prysmaticlabs/prysm/v5/encoding/bytesutil" | ||
"github.com/prysmaticlabs/prysm/v5/monitoring/tracing" | ||
"github.com/prysmaticlabs/prysm/v5/monitoring/tracing/trace" | ||
"github.com/prysmaticlabs/prysm/v5/time/slots" | ||
"github.com/sirupsen/logrus" | ||
) | ||
|
||
func (s *Service) dataColumnSidecarByRootRPCHandler(ctx context.Context, msg interface{}, stream libp2pcore.Stream) error { | ||
ctx, span := trace.StartSpan(ctx, "sync.dataColumnSidecarByRootRPCHandler") | ||
defer span.End() | ||
ctx, cancel := context.WithTimeout(ctx, ttfbTimeout) | ||
defer cancel() | ||
SetRPCStreamDeadlines(stream) | ||
log := log.WithField("handler", p2p.DataColumnSidecarsByRootName[1:]) // slice the leading slash off the name var | ||
// We use the same type as for blobs as they are the same data structure. | ||
// TODO: Make the type naming more generic to be extensible to data columns | ||
ref, ok := msg.(*types.BlobSidecarsByRootReq) | ||
if !ok { | ||
return errors.New("message is not type BlobSidecarsByRootReq") | ||
} | ||
|
||
columnIdents := *ref | ||
if err := validateDataColummnsByRootRequest(columnIdents); err != nil { | ||
s.cfg.p2p.Peers().Scorers().BadResponsesScorer().Increment(stream.Conn().RemotePeer()) | ||
s.writeErrorResponseToStream(responseCodeInvalidRequest, err.Error(), stream) | ||
return err | ||
} | ||
// Sort the identifiers so that requests for the same blob root will be adjacent, minimizing db lookups. | ||
sort.Sort(columnIdents) | ||
|
||
// TODO: Customize data column batches too | ||
batchSize := flags.Get().BlobBatchLimit | ||
var ticker *time.Ticker | ||
if len(columnIdents) > batchSize { | ||
ticker = time.NewTicker(time.Second) | ||
} | ||
|
||
// Compute the oldest slot we'll allow a peer to request, based on the current slot. | ||
cs := s.cfg.clock.CurrentSlot() | ||
minReqSlot, err := DataColumnsRPCMinValidSlot(cs) | ||
if err != nil { | ||
return errors.Wrapf(err, "unexpected error computing min valid blob request slot, current_slot=%d", cs) | ||
} | ||
|
||
for i := range columnIdents { | ||
if err := ctx.Err(); err != nil { | ||
closeStream(stream, log) | ||
return err | ||
} | ||
|
||
// Throttle request processing to no more than batchSize/sec. | ||
if ticker != nil && i != 0 && i%batchSize == 0 { | ||
<-ticker.C | ||
} | ||
s.rateLimiter.add(stream, 1) | ||
root, idx := bytesutil.ToBytes32(columnIdents[i].BlockRoot), columnIdents[i].Index | ||
custodiedColumns, err := p2p.ComputeCustodyColumns(s.cfg.p2p.NodeID()) | ||
if err != nil { | ||
log.WithError(err).Errorf("unexpected error retrieving the node id") | ||
s.writeErrorResponseToStream(responseCodeServerError, types.ErrGeneric.Error(), stream) | ||
return err | ||
} | ||
isCustodied := false | ||
for _, col := range custodiedColumns { | ||
if col == idx { | ||
isCustodied = true | ||
break | ||
} | ||
} | ||
if !isCustodied { | ||
s.cfg.p2p.Peers().Scorers().BadResponsesScorer().Increment(stream.Conn().RemotePeer()) | ||
s.writeErrorResponseToStream(responseCodeInvalidRequest, types.ErrInvalidColumnIndex.Error(), stream) | ||
return types.ErrInvalidColumnIndex | ||
} | ||
|
||
// TODO: Differentiate between blobs and columns for our storage engine | ||
sc, err := s.cfg.blobStorage.GetColumn(root, idx) | ||
if err != nil { | ||
if db.IsNotFound(err) { | ||
log.WithError(err).WithFields(logrus.Fields{ | ||
"root": fmt.Sprintf("%#x", root), | ||
"index": idx, | ||
}).Debugf("Peer requested data column sidecar by root not found in db") | ||
continue | ||
} | ||
log.WithError(err).Errorf("unexpected db error retrieving data column, root=%x, index=%d", root, idx) | ||
s.writeErrorResponseToStream(responseCodeServerError, types.ErrGeneric.Error(), stream) | ||
return err | ||
} | ||
|
||
// If any root in the request content references a block earlier than minimum_request_epoch, | ||
// peers MAY respond with error code 3: ResourceUnavailable or not include the data column in the response. | ||
// note: we are deviating from the spec to allow requests for data column that are before minimum_request_epoch, | ||
// up to the beginning of the retention period. | ||
if sc.SignedBlockHeader.Header.Slot < minReqSlot { | ||
s.writeErrorResponseToStream(responseCodeResourceUnavailable, types.ErrDataColumnLTMinRequest.Error(), stream) | ||
log.WithError(types.ErrDataColumnLTMinRequest). | ||
Debugf("requested data column for block %#x before minimum_request_epoch", columnIdents[i].BlockRoot) | ||
return types.ErrDataColumnLTMinRequest | ||
} | ||
|
||
SetStreamWriteDeadline(stream, defaultWriteDuration) | ||
if chunkErr := WriteDataColumnSidecarChunk(stream, s.cfg.chain, s.cfg.p2p.Encoding(), sc); chunkErr != nil { | ||
log.WithError(chunkErr).Debug("Could not send a chunked response") | ||
s.writeErrorResponseToStream(responseCodeServerError, types.ErrGeneric.Error(), stream) | ||
tracing.AnnotateError(span, chunkErr) | ||
return chunkErr | ||
} | ||
} | ||
closeStream(stream, log) | ||
return nil | ||
} | ||
|
||
func validateDataColummnsByRootRequest(colIdents types.BlobSidecarsByRootReq) error { | ||
if uint64(len(colIdents)) > params.BeaconConfig().MaxRequestDataColumnSidecars { | ||
return types.ErrMaxDataColumnReqExceeded | ||
} | ||
return nil | ||
} | ||
|
||
func DataColumnsRPCMinValidSlot(current primitives.Slot) (primitives.Slot, error) { | ||
// Avoid overflow if we're running on a config where deneb is set to far future epoch. | ||
if params.BeaconConfig().DenebForkEpoch == math.MaxUint64 || !features.Get().EnablePeerDAS { | ||
return primitives.Slot(math.MaxUint64), nil | ||
} | ||
minReqEpochs := params.BeaconConfig().MinEpochsForDataColumnSidecarsRequest | ||
currEpoch := slots.ToEpoch(current) | ||
minStart := params.BeaconConfig().DenebForkEpoch | ||
if currEpoch > minReqEpochs && currEpoch-minReqEpochs > minStart { | ||
minStart = currEpoch - minReqEpochs | ||
} | ||
return slots.EpochStart(minStart) | ||
} |
Oops, something went wrong.