mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-18 15:54:13 +00:00
0e043d55b4
* Use VerifiedROBlobs in initial-sync * Update beacon-chain/das/cache.go Co-authored-by: Justin Traglia <95511699+jtraglia@users.noreply.github.com> * Apply suggestions from code review comment fixes Co-authored-by: Justin Traglia <95511699+jtraglia@users.noreply.github.com> * fix lint error from gh web ui * deepsource fixes * more deepsource * fix init wiring * mark blobless blocks verified in batch mode * move sig check after parent checks * validate block commitment length at start of da check * remove vestigial locking * rm more copy-locksta * rm old comment * fail the entire batch if any sidecar fails * lint * skip redundant checks, fix len check * assume sig and proposer checks passed for block * inherits most checks from processed block * Assume block processing handles most checks * lint * cleanup unused call and gaz * more detailed logging for e2e * fix bad refactor breaking non-finalized init-sync * self-review cleanup * gaz * Update beacon-chain/verification/blob.go Co-authored-by: Justin Traglia <95511699+jtraglia@users.noreply.github.com> * terence and justin feedback --------- Co-authored-by: Kasey Kirkham <kasey@users.noreply.github.com> Co-authored-by: Justin Traglia <95511699+jtraglia@users.noreply.github.com>
150 lines
5.4 KiB
Go
150 lines
5.4 KiB
Go
package das
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
errors "github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/db/filesystem"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/verification"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/blocks"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v4/runtime/logging"
|
|
"github.com/prysmaticlabs/prysm/v4/runtime/version"
|
|
"github.com/prysmaticlabs/prysm/v4/time/slots"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
var (
|
|
errMixedRoots = errors.New("BlobSidecars must all be for the same block")
|
|
)
|
|
|
|
// LazilyPersistentStore is an implementation of AvailabilityStore to be used when batch syncing.
|
|
// This implementation will hold any blobs passed to Persist until the IsDataAvailable is called for their
|
|
// block, at which time they will undergo full verification and be saved to the disk.
|
|
type LazilyPersistentStore struct {
|
|
store *filesystem.BlobStorage
|
|
cache *cache
|
|
verifier BlobBatchVerifier
|
|
}
|
|
|
|
var _ AvailabilityStore = &LazilyPersistentStore{}
|
|
|
|
// BlobBatchVerifier enables LazyAvailabilityStore to manage the verification process
|
|
// going from ROBlob->VerifiedROBlob, while avoiding the decision of which individual verifications
|
|
// to run and in what order. Since LazilyPersistentStore always tries to verify and save blobs only when
|
|
// they are all available, the interface takes a slice of blobs, enabling the implementation to optimize
|
|
// batch verification.
|
|
type BlobBatchVerifier interface {
|
|
VerifiedROBlobs(ctx context.Context, blk blocks.ROBlock, sc []blocks.ROBlob) ([]blocks.VerifiedROBlob, error)
|
|
}
|
|
|
|
// NewLazilyPersistentStore creates a new LazilyPersistentStore. This constructor should always be used
|
|
// when creating a LazilyPersistentStore because it needs to initialize the cache under the hood.
|
|
func NewLazilyPersistentStore(store *filesystem.BlobStorage, verifier BlobBatchVerifier) *LazilyPersistentStore {
|
|
return &LazilyPersistentStore{
|
|
store: store,
|
|
cache: newCache(),
|
|
verifier: verifier,
|
|
}
|
|
}
|
|
|
|
// Persist adds blobs to the working blob cache. Blobs stored in this cache will be persisted
|
|
// for at least as long as the node is running. Once IsDataAvailable succeeds, all blobs referenced
|
|
// by the given block are guaranteed to be persisted for the remainder of the retention period.
|
|
func (s *LazilyPersistentStore) Persist(current primitives.Slot, sc ...blocks.ROBlob) error {
|
|
if len(sc) == 0 {
|
|
return nil
|
|
}
|
|
if len(sc) > 1 {
|
|
first := sc[0].BlockRoot()
|
|
for i := 1; i < len(sc); i++ {
|
|
if first != sc[i].BlockRoot() {
|
|
return errMixedRoots
|
|
}
|
|
}
|
|
}
|
|
if !params.WithinDAPeriod(slots.ToEpoch(sc[0].Slot()), slots.ToEpoch(current)) {
|
|
return nil
|
|
}
|
|
key := keyFromSidecar(sc[0])
|
|
entry := s.cache.ensure(key)
|
|
for i := range sc {
|
|
if err := entry.stash(&sc[i]); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// IsDataAvailable returns nil if all the commitments in the given block are persisted to the db and have been verified.
|
|
// BlobSidecars already in the db are assumed to have been previously verified against the block.
|
|
func (s *LazilyPersistentStore) IsDataAvailable(ctx context.Context, current primitives.Slot, b blocks.ROBlock) error {
|
|
blockCommitments, err := commitmentsToCheck(b, current)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "could check data availability for block %#x", b.Root())
|
|
}
|
|
// Return early for blocks that are pre-deneb or which do not have any commitments.
|
|
if blockCommitments.count() == 0 {
|
|
return nil
|
|
}
|
|
|
|
key := keyFromBlock(b)
|
|
entry := s.cache.ensure(key)
|
|
defer s.cache.delete(key)
|
|
root := b.Root()
|
|
// Verify we have all the expected sidecars, and fail fast if any are missing or inconsistent.
|
|
// We don't try to salvage problematic batches because this indicates a misbehaving peer and we'd rather
|
|
// ignore their response and decrease their peer score.
|
|
sidecars, err := entry.filter(root, blockCommitments)
|
|
if err != nil {
|
|
return errors.Wrap(err, "incomplete BlobSidecar batch")
|
|
}
|
|
// Do thorough verifications of each BlobSidecar for the block.
|
|
// Same as above, we don't save BlobSidecars if there are any problems with the batch.
|
|
vscs, err := s.verifier.VerifiedROBlobs(ctx, b, sidecars)
|
|
if err != nil {
|
|
var me verification.VerificationMultiError
|
|
ok := errors.As(err, &me)
|
|
if ok {
|
|
fails := me.Failures()
|
|
lf := make(log.Fields, len(fails))
|
|
for i := range fails {
|
|
lf[fmt.Sprintf("fail_%d", i)] = fails[i].Error()
|
|
}
|
|
log.WithFields(lf).WithFields(logging.BlockFieldsFromBlob(sidecars[0])).
|
|
Debug("invalid BlobSidecars received")
|
|
}
|
|
return errors.Wrapf(err, "invalid BlobSidecars received for block %#x", root)
|
|
}
|
|
// Ensure that each BlobSidecar is written to disk.
|
|
for i := range vscs {
|
|
if err := s.store.Save(vscs[i]); err != nil {
|
|
return errors.Wrapf(err, "failed to save BlobSidecar index %d for block %#x", vscs[i].Index, root)
|
|
}
|
|
}
|
|
// All BlobSidecars are persisted - da check succeeds.
|
|
return nil
|
|
}
|
|
|
|
func commitmentsToCheck(b blocks.ROBlock, current primitives.Slot) (safeCommitmentArray, error) {
|
|
var ar safeCommitmentArray
|
|
if b.Version() < version.Deneb {
|
|
return ar, nil
|
|
}
|
|
// We are only required to check within MIN_EPOCHS_FOR_BLOB_SIDECARS_REQUESTS
|
|
if !params.WithinDAPeriod(slots.ToEpoch(b.Block().Slot()), slots.ToEpoch(current)) {
|
|
return ar, nil
|
|
}
|
|
kc, err := b.Block().Body().BlobKzgCommitments()
|
|
if err != nil {
|
|
return ar, err
|
|
}
|
|
if len(kc) > len(ar) {
|
|
return ar, errIndexOutOfBounds
|
|
}
|
|
copy(ar[:], kc)
|
|
return ar, nil
|
|
}
|