mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-11 04:00:05 +00:00
3a1b4a19c2
* Add a new blob channel * fix mock * reset the channel * keep a map of channels * gazelle * do not overwrite map * remove pre-declaration
645 lines
22 KiB
Go
645 lines
22 KiB
Go
package blockchain
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/blockchain/kzg"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/blocks"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/feed"
|
|
statefeed "github.com/prysmaticlabs/prysm/v4/beacon-chain/core/feed/state"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/helpers"
|
|
coreTime "github.com/prysmaticlabs/prysm/v4/beacon-chain/core/time"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/transition"
|
|
forkchoicetypes "github.com/prysmaticlabs/prysm/v4/beacon-chain/forkchoice/types"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
|
"github.com/prysmaticlabs/prysm/v4/config/features"
|
|
fieldparams "github.com/prysmaticlabs/prysm/v4/config/fieldparams"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
consensusblocks "github.com/prysmaticlabs/prysm/v4/consensus-types/blocks"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/interfaces"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v4/crypto/bls"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/v4/monitoring/tracing"
|
|
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1/attestation"
|
|
"github.com/prysmaticlabs/prysm/v4/runtime/version"
|
|
"github.com/prysmaticlabs/prysm/v4/time/slots"
|
|
"github.com/sirupsen/logrus"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
// A custom slot deadline for processing state slots in our cache.
|
|
const slotDeadline = 5 * time.Second
|
|
|
|
// A custom deadline for deposit trie insertion.
|
|
const depositDeadline = 20 * time.Second
|
|
|
|
// This defines size of the upper bound for initial sync block cache.
|
|
var initialSyncBlockCacheSize = uint64(2 * params.BeaconConfig().SlotsPerEpoch)
|
|
|
|
// postBlockProcess is called when a gossip block is received. This function performs
|
|
// several duties most importantly informing the engine if head was updated,
|
|
// saving the new head information to the blockchain package and
|
|
// handling attestations, slashings and similar included in the block.
|
|
func (s *Service) postBlockProcess(ctx context.Context, signed interfaces.ReadOnlySignedBeaconBlock, blockRoot [32]byte, postState state.BeaconState, isValidPayload bool) error {
|
|
ctx, span := trace.StartSpan(ctx, "blockChain.onBlock")
|
|
defer span.End()
|
|
if err := consensusblocks.BeaconBlockIsNil(signed); err != nil {
|
|
return invalidBlock{error: err}
|
|
}
|
|
startTime := time.Now()
|
|
b := signed.Block()
|
|
|
|
if err := s.cfg.ForkChoiceStore.InsertNode(ctx, postState, blockRoot); err != nil {
|
|
return errors.Wrapf(err, "could not insert block %d to fork choice store", signed.Block().Slot())
|
|
}
|
|
if err := s.handleBlockAttestations(ctx, signed.Block(), postState); err != nil {
|
|
return errors.Wrap(err, "could not handle block's attestations")
|
|
}
|
|
|
|
s.InsertSlashingsToForkChoiceStore(ctx, signed.Block().Body().AttesterSlashings())
|
|
if isValidPayload {
|
|
if err := s.cfg.ForkChoiceStore.SetOptimisticToValid(ctx, blockRoot); err != nil {
|
|
return errors.Wrap(err, "could not set optimistic block to valid")
|
|
}
|
|
}
|
|
|
|
start := time.Now()
|
|
headRoot, err := s.cfg.ForkChoiceStore.Head(ctx)
|
|
if err != nil {
|
|
log.WithError(err).Warn("Could not update head")
|
|
}
|
|
if blockRoot != headRoot {
|
|
receivedWeight, err := s.cfg.ForkChoiceStore.Weight(blockRoot)
|
|
if err != nil {
|
|
log.WithField("root", fmt.Sprintf("%#x", blockRoot)).Warn("could not determine node weight")
|
|
}
|
|
headWeight, err := s.cfg.ForkChoiceStore.Weight(headRoot)
|
|
if err != nil {
|
|
log.WithField("root", fmt.Sprintf("%#x", headRoot)).Warn("could not determine node weight")
|
|
}
|
|
log.WithFields(logrus.Fields{
|
|
"receivedRoot": fmt.Sprintf("%#x", blockRoot),
|
|
"receivedWeight": receivedWeight,
|
|
"headRoot": fmt.Sprintf("%#x", headRoot),
|
|
"headWeight": headWeight,
|
|
}).Debug("Head block is not the received block")
|
|
}
|
|
newBlockHeadElapsedTime.Observe(float64(time.Since(start).Milliseconds()))
|
|
|
|
// verify conditions for FCU, notifies FCU, and saves the new head.
|
|
// This function also prunes attestations, other similar operations happen in prunePostBlockOperationPools.
|
|
if _, err := s.forkchoiceUpdateWithExecution(ctx, headRoot, s.CurrentSlot()+1); err != nil {
|
|
return err
|
|
}
|
|
|
|
optimistic, err := s.cfg.ForkChoiceStore.IsOptimistic(blockRoot)
|
|
if err != nil {
|
|
log.WithError(err).Debug("Could not check if block is optimistic")
|
|
optimistic = true
|
|
}
|
|
|
|
// Send notification of the processed block to the state feed.
|
|
s.cfg.StateNotifier.StateFeed().Send(&feed.Event{
|
|
Type: statefeed.BlockProcessed,
|
|
Data: &statefeed.BlockProcessedData{
|
|
Slot: signed.Block().Slot(),
|
|
BlockRoot: blockRoot,
|
|
SignedBlock: signed,
|
|
Verified: true,
|
|
Optimistic: optimistic,
|
|
},
|
|
})
|
|
|
|
defer reportAttestationInclusion(b)
|
|
if headRoot == blockRoot {
|
|
// Updating next slot state cache can happen in the background
|
|
// except in the epoch boundary in which case we lock to handle
|
|
// the shuffling and proposer caches updates.
|
|
// We handle these caches only on canonical
|
|
// blocks, otherwise this will be handled by lateBlockTasks
|
|
slot := postState.Slot()
|
|
if slots.IsEpochEnd(slot) {
|
|
if err := transition.UpdateNextSlotCache(ctx, blockRoot[:], postState); err != nil {
|
|
return errors.Wrap(err, "could not update next slot state cache")
|
|
}
|
|
if err := s.handleEpochBoundary(ctx, slot, postState, blockRoot[:]); err != nil {
|
|
return errors.Wrap(err, "could not handle epoch boundary")
|
|
}
|
|
} else {
|
|
go func() {
|
|
slotCtx, cancel := context.WithTimeout(context.Background(), slotDeadline)
|
|
defer cancel()
|
|
if err := transition.UpdateNextSlotCache(slotCtx, blockRoot[:], postState); err != nil {
|
|
log.WithError(err).Error("could not update next slot state cache")
|
|
}
|
|
}()
|
|
}
|
|
}
|
|
onBlockProcessingTime.Observe(float64(time.Since(startTime).Milliseconds()))
|
|
return nil
|
|
}
|
|
|
|
func getStateVersionAndPayload(st state.BeaconState) (int, interfaces.ExecutionData, error) {
|
|
if st == nil {
|
|
return 0, nil, errors.New("nil state")
|
|
}
|
|
var preStateHeader interfaces.ExecutionData
|
|
var err error
|
|
preStateVersion := st.Version()
|
|
switch preStateVersion {
|
|
case version.Phase0, version.Altair:
|
|
default:
|
|
preStateHeader, err = st.LatestExecutionPayloadHeader()
|
|
if err != nil {
|
|
return 0, nil, err
|
|
}
|
|
}
|
|
return preStateVersion, preStateHeader, nil
|
|
}
|
|
|
|
func (s *Service) onBlockBatch(ctx context.Context, blks []consensusblocks.ROBlock) error {
|
|
ctx, span := trace.StartSpan(ctx, "blockChain.onBlockBatch")
|
|
defer span.End()
|
|
|
|
if len(blks) == 0 {
|
|
return errors.New("no blocks provided")
|
|
}
|
|
|
|
if err := consensusblocks.BeaconBlockIsNil(blks[0]); err != nil {
|
|
return invalidBlock{error: err}
|
|
}
|
|
b := blks[0].Block()
|
|
|
|
// Retrieve incoming block's pre state.
|
|
if err := s.verifyBlkPreState(ctx, b); err != nil {
|
|
return err
|
|
}
|
|
preState, err := s.cfg.StateGen.StateByRootInitialSync(ctx, b.ParentRoot())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if preState == nil || preState.IsNil() {
|
|
return fmt.Errorf("nil pre state for slot %d", b.Slot())
|
|
}
|
|
|
|
// Fill in missing blocks
|
|
if err := s.fillInForkChoiceMissingBlocks(ctx, blks[0].Block(), preState.CurrentJustifiedCheckpoint(), preState.FinalizedCheckpoint()); err != nil {
|
|
return errors.Wrap(err, "could not fill in missing blocks to forkchoice")
|
|
}
|
|
|
|
jCheckpoints := make([]*ethpb.Checkpoint, len(blks))
|
|
fCheckpoints := make([]*ethpb.Checkpoint, len(blks))
|
|
sigSet := bls.NewSet()
|
|
type versionAndHeader struct {
|
|
version int
|
|
header interfaces.ExecutionData
|
|
}
|
|
preVersionAndHeaders := make([]*versionAndHeader, len(blks))
|
|
postVersionAndHeaders := make([]*versionAndHeader, len(blks))
|
|
var set *bls.SignatureBatch
|
|
boundaries := make(map[[32]byte]state.BeaconState)
|
|
for i, b := range blks {
|
|
v, h, err := getStateVersionAndPayload(preState)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
preVersionAndHeaders[i] = &versionAndHeader{
|
|
version: v,
|
|
header: h,
|
|
}
|
|
|
|
set, preState, err = transition.ExecuteStateTransitionNoVerifyAnySig(ctx, preState, b)
|
|
if err != nil {
|
|
return invalidBlock{error: err}
|
|
}
|
|
// Save potential boundary states.
|
|
if slots.IsEpochStart(preState.Slot()) {
|
|
boundaries[b.Root()] = preState.Copy()
|
|
}
|
|
jCheckpoints[i] = preState.CurrentJustifiedCheckpoint()
|
|
fCheckpoints[i] = preState.FinalizedCheckpoint()
|
|
|
|
v, h, err = getStateVersionAndPayload(preState)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
postVersionAndHeaders[i] = &versionAndHeader{
|
|
version: v,
|
|
header: h,
|
|
}
|
|
sigSet.Join(set)
|
|
}
|
|
|
|
var verify bool
|
|
if features.Get().EnableVerboseSigVerification {
|
|
verify, err = sigSet.VerifyVerbosely()
|
|
} else {
|
|
verify, err = sigSet.Verify()
|
|
}
|
|
if err != nil {
|
|
return invalidBlock{error: err}
|
|
}
|
|
if !verify {
|
|
return errors.New("batch block signature verification failed")
|
|
}
|
|
|
|
// blocks have been verified, save them and call the engine
|
|
pendingNodes := make([]*forkchoicetypes.BlockAndCheckpoints, len(blks))
|
|
var isValidPayload bool
|
|
for i, b := range blks {
|
|
root := b.Root()
|
|
isValidPayload, err = s.notifyNewPayload(ctx,
|
|
postVersionAndHeaders[i].version,
|
|
postVersionAndHeaders[i].header, b)
|
|
if err != nil {
|
|
return s.handleInvalidExecutionError(ctx, err, root, b.Block().ParentRoot())
|
|
}
|
|
if isValidPayload {
|
|
if err := s.validateMergeTransitionBlock(ctx, preVersionAndHeaders[i].version,
|
|
preVersionAndHeaders[i].header, b); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
args := &forkchoicetypes.BlockAndCheckpoints{Block: b.Block(),
|
|
JustifiedCheckpoint: jCheckpoints[i],
|
|
FinalizedCheckpoint: fCheckpoints[i]}
|
|
pendingNodes[len(blks)-i-1] = args
|
|
if err := s.saveInitSyncBlock(ctx, root, b); err != nil {
|
|
tracing.AnnotateError(span, err)
|
|
return err
|
|
}
|
|
if err := s.cfg.BeaconDB.SaveStateSummary(ctx, ðpb.StateSummary{
|
|
Slot: b.Block().Slot(),
|
|
Root: root[:],
|
|
}); err != nil {
|
|
tracing.AnnotateError(span, err)
|
|
return err
|
|
}
|
|
if i > 0 && jCheckpoints[i].Epoch > jCheckpoints[i-1].Epoch {
|
|
if err := s.cfg.BeaconDB.SaveJustifiedCheckpoint(ctx, jCheckpoints[i]); err != nil {
|
|
tracing.AnnotateError(span, err)
|
|
return err
|
|
}
|
|
}
|
|
if i > 0 && fCheckpoints[i].Epoch > fCheckpoints[i-1].Epoch {
|
|
if err := s.updateFinalized(ctx, fCheckpoints[i]); err != nil {
|
|
tracing.AnnotateError(span, err)
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
// Save boundary states that will be useful for forkchoice
|
|
for r, st := range boundaries {
|
|
if err := s.cfg.StateGen.SaveState(ctx, r, st); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
lastB := blks[len(blks)-1]
|
|
lastBR := lastB.Root()
|
|
// Also saves the last post state which to be used as pre state for the next batch.
|
|
if err := s.cfg.StateGen.SaveState(ctx, lastBR, preState); err != nil {
|
|
return err
|
|
}
|
|
// Insert all nodes but the last one to forkchoice
|
|
if err := s.cfg.ForkChoiceStore.InsertChain(ctx, pendingNodes); err != nil {
|
|
return errors.Wrap(err, "could not insert batch to forkchoice")
|
|
}
|
|
// Insert the last block to forkchoice
|
|
if err := s.cfg.ForkChoiceStore.InsertNode(ctx, preState, lastBR); err != nil {
|
|
return errors.Wrap(err, "could not insert last block in batch to forkchoice")
|
|
}
|
|
// Set their optimistic status
|
|
if isValidPayload {
|
|
if err := s.cfg.ForkChoiceStore.SetOptimisticToValid(ctx, lastBR); err != nil {
|
|
return errors.Wrap(err, "could not set optimistic block to valid")
|
|
}
|
|
}
|
|
arg := ¬ifyForkchoiceUpdateArg{
|
|
headState: preState,
|
|
headRoot: lastBR,
|
|
headBlock: lastB.Block(),
|
|
}
|
|
if _, err := s.notifyForkchoiceUpdate(ctx, arg); err != nil {
|
|
return err
|
|
}
|
|
return s.saveHeadNoDB(ctx, lastB, lastBR, preState, !isValidPayload)
|
|
}
|
|
|
|
func (s *Service) updateEpochBoundaryCaches(ctx context.Context, st state.BeaconState) error {
|
|
e := coreTime.CurrentEpoch(st)
|
|
if err := helpers.UpdateCommitteeCache(ctx, st, e); err != nil {
|
|
return errors.Wrap(err, "could not update committee cache")
|
|
}
|
|
if err := helpers.UpdateProposerIndicesInCache(ctx, st, e); err != nil {
|
|
return errors.Wrap(err, "could not update proposer index cache")
|
|
}
|
|
go func() {
|
|
// Use a custom deadline here, since this method runs asynchronously.
|
|
// We ignore the parent method's context and instead create a new one
|
|
// with a custom deadline, therefore using the background context instead.
|
|
slotCtx, cancel := context.WithTimeout(context.Background(), slotDeadline)
|
|
defer cancel()
|
|
if err := helpers.UpdateCommitteeCache(slotCtx, st, e+1); err != nil {
|
|
log.WithError(err).Warn("Could not update committee cache")
|
|
}
|
|
if err := helpers.UpdateProposerIndicesInCache(slotCtx, st, e+1); err != nil {
|
|
log.WithError(err).Warn("Failed to cache next epoch proposers")
|
|
}
|
|
}()
|
|
return nil
|
|
}
|
|
|
|
// Epoch boundary tasks: it copies the headState and updates the epoch boundary
|
|
// caches.
|
|
func (s *Service) handleEpochBoundary(ctx context.Context, slot primitives.Slot, headState state.BeaconState, blockRoot []byte) error {
|
|
ctx, span := trace.StartSpan(ctx, "blockChain.handleEpochBoundary")
|
|
defer span.End()
|
|
// return early if we are advancing to a past epoch
|
|
if slot < headState.Slot() {
|
|
return nil
|
|
}
|
|
if !slots.IsEpochEnd(slot) {
|
|
return nil
|
|
}
|
|
copied := headState.Copy()
|
|
copied, err := transition.ProcessSlotsUsingNextSlotCache(ctx, copied, blockRoot, slot+1)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return s.updateEpochBoundaryCaches(ctx, copied)
|
|
}
|
|
|
|
// This feeds in the attestations included in the block to fork choice store. It's allows fork choice store
|
|
// to gain information on the most current chain.
|
|
func (s *Service) handleBlockAttestations(ctx context.Context, blk interfaces.ReadOnlyBeaconBlock, st state.BeaconState) error {
|
|
// Feed in block's attestations to fork choice store.
|
|
for _, a := range blk.Body().Attestations() {
|
|
committee, err := helpers.BeaconCommitteeFromState(ctx, st, a.Data.Slot, a.Data.CommitteeIndex)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
indices, err := attestation.AttestingIndices(a.AggregationBits, committee)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
r := bytesutil.ToBytes32(a.Data.BeaconBlockRoot)
|
|
if s.cfg.ForkChoiceStore.HasNode(r) {
|
|
s.cfg.ForkChoiceStore.ProcessAttestation(ctx, indices, r, a.Data.Target.Epoch)
|
|
} else if err := s.cfg.AttPool.SaveBlockAttestation(a); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// InsertSlashingsToForkChoiceStore inserts attester slashing indices to fork choice store.
|
|
// To call this function, it's caller's responsibility to ensure the slashing object is valid.
|
|
// This function requires a write lock on forkchoice.
|
|
func (s *Service) InsertSlashingsToForkChoiceStore(ctx context.Context, slashings []*ethpb.AttesterSlashing) {
|
|
for _, slashing := range slashings {
|
|
indices := blocks.SlashableAttesterIndices(slashing)
|
|
for _, index := range indices {
|
|
s.cfg.ForkChoiceStore.InsertSlashedIndex(ctx, primitives.ValidatorIndex(index))
|
|
}
|
|
}
|
|
}
|
|
|
|
// This saves post state info to DB or cache. This also saves post state info to fork choice store.
|
|
// Post state info consists of processed block and state. Do not call this method unless the block and state are verified.
|
|
func (s *Service) savePostStateInfo(ctx context.Context, r [32]byte, b interfaces.ReadOnlySignedBeaconBlock, st state.BeaconState) error {
|
|
ctx, span := trace.StartSpan(ctx, "blockChain.savePostStateInfo")
|
|
defer span.End()
|
|
if err := s.cfg.BeaconDB.SaveBlock(ctx, b); err != nil {
|
|
return errors.Wrapf(err, "could not save block from slot %d", b.Block().Slot())
|
|
}
|
|
if err := s.cfg.StateGen.SaveState(ctx, r, st); err != nil {
|
|
return errors.Wrap(err, "could not save state")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// This removes the attestations in block `b` from the attestation mem pool.
|
|
func (s *Service) pruneAttsFromPool(headBlock interfaces.ReadOnlySignedBeaconBlock) error {
|
|
atts := headBlock.Block().Body().Attestations()
|
|
for _, att := range atts {
|
|
if helpers.IsAggregated(att) {
|
|
if err := s.cfg.AttPool.DeleteAggregatedAttestation(att); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
if err := s.cfg.AttPool.DeleteUnaggregatedAttestation(att); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// validateMergeTransitionBlock validates the merge transition block.
|
|
func (s *Service) validateMergeTransitionBlock(ctx context.Context, stateVersion int, stateHeader interfaces.ExecutionData, blk interfaces.ReadOnlySignedBeaconBlock) error {
|
|
// Skip validation if block is older than Bellatrix.
|
|
if blocks.IsPreBellatrixVersion(blk.Block().Version()) {
|
|
return nil
|
|
}
|
|
|
|
// Skip validation if block has an empty payload.
|
|
payload, err := blk.Block().Body().Execution()
|
|
if err != nil {
|
|
return invalidBlock{error: err}
|
|
}
|
|
isEmpty, err := consensusblocks.IsEmptyExecutionData(payload)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if isEmpty {
|
|
return nil
|
|
}
|
|
|
|
// Handle case where pre-state is Altair but block contains payload.
|
|
// To reach here, the block must have contained a valid payload.
|
|
if blocks.IsPreBellatrixVersion(stateVersion) {
|
|
return s.validateMergeBlock(ctx, blk)
|
|
}
|
|
|
|
// Skip validation if the block is not a merge transition block.
|
|
// To reach here. The payload must be non-empty. If the state header is empty then it's at transition.
|
|
empty, err := consensusblocks.IsEmptyExecutionData(stateHeader)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !empty {
|
|
return nil
|
|
}
|
|
return s.validateMergeBlock(ctx, blk)
|
|
}
|
|
|
|
// This routine checks if there is a cached proposer payload ID available for the next slot proposer.
|
|
// If there is not, it will call forkchoice updated with the correct payload attribute then cache the payload ID.
|
|
func (s *Service) runLateBlockTasks() {
|
|
if err := s.waitForSync(); err != nil {
|
|
log.WithError(err).Error("failed to wait for initial sync")
|
|
return
|
|
}
|
|
|
|
attThreshold := params.BeaconConfig().SecondsPerSlot / 3
|
|
ticker := slots.NewSlotTickerWithOffset(s.genesisTime, time.Duration(attThreshold)*time.Second, params.BeaconConfig().SecondsPerSlot)
|
|
for {
|
|
select {
|
|
case <-ticker.C():
|
|
s.lateBlockTasks(s.ctx)
|
|
case <-s.ctx.Done():
|
|
log.Debug("Context closed, exiting routine")
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *Service) isDataAvailable(ctx context.Context, root [32]byte, signed interfaces.ReadOnlySignedBeaconBlock) error {
|
|
if signed.Version() < version.Deneb {
|
|
return nil
|
|
}
|
|
block := signed.Block()
|
|
if block == nil {
|
|
return errors.New("invalid nil beacon block")
|
|
}
|
|
// We are only required to check within MIN_EPOCHS_FOR_BLOB_SIDECARS_REQUESTS
|
|
if slots.ToEpoch(block.Slot())+params.BeaconNetworkConfig().MinEpochsForBlobsSidecarsRequest > primitives.Epoch(s.CurrentSlot()) {
|
|
return nil
|
|
}
|
|
body := block.Body()
|
|
if body == nil {
|
|
return errors.New("invalid nil beacon block body")
|
|
}
|
|
kzgCommitments, err := body.BlobKzgCommitments()
|
|
if err != nil {
|
|
return errors.Wrap(err, "could not get KZG commitments")
|
|
}
|
|
existingBlobs := len(kzgCommitments)
|
|
if existingBlobs == 0 {
|
|
return nil
|
|
}
|
|
|
|
// Read first from db in case we have the blobs
|
|
s.blobNotifier.Lock()
|
|
var nc *blobNotifierChan
|
|
var ok bool
|
|
nc, ok = s.blobNotifier.chanForRoot[root]
|
|
sidecars, err := s.cfg.BeaconDB.BlobSidecarsByRoot(ctx, root)
|
|
if err == nil {
|
|
if len(sidecars) >= existingBlobs {
|
|
delete(s.blobNotifier.chanForRoot, root)
|
|
s.blobNotifier.Unlock()
|
|
return kzg.IsDataAvailable(kzgCommitments, sidecars)
|
|
}
|
|
}
|
|
// Create the channel if it didn't exist already the index map will be
|
|
// created later anyway
|
|
if !ok {
|
|
nc = &blobNotifierChan{channel: make(chan struct{}, fieldparams.MaxBlobsPerBlock)}
|
|
s.blobNotifier.chanForRoot[root] = nc
|
|
}
|
|
// We have more commitments in the block than blobs in database
|
|
// We sync the channel indices with the sidecars
|
|
nc.indices = make(map[uint64]struct{})
|
|
for _, sidecar := range sidecars {
|
|
nc.indices[sidecar.Index] = struct{}{}
|
|
}
|
|
s.blobNotifier.Unlock()
|
|
channelWrites := len(sidecars)
|
|
for {
|
|
select {
|
|
case <-nc.channel:
|
|
channelWrites++
|
|
if channelWrites == existingBlobs {
|
|
s.blobNotifier.Lock()
|
|
delete(s.blobNotifier.chanForRoot, root)
|
|
s.blobNotifier.Unlock()
|
|
sidecars, err := s.cfg.BeaconDB.BlobSidecarsByRoot(ctx, root)
|
|
if err != nil {
|
|
return errors.Wrap(err, "could not get blob sidecars")
|
|
}
|
|
return kzg.IsDataAvailable(kzgCommitments, sidecars)
|
|
}
|
|
case <-ctx.Done():
|
|
return errors.Wrap(err, "context deadline waiting for blob sidecars")
|
|
}
|
|
}
|
|
}
|
|
|
|
// lateBlockTasks is called 4 seconds into the slot and performs tasks
|
|
// related to late blocks. It emits a MissedSlot state feed event.
|
|
// It calls FCU and sets the right attributes if we are proposing next slot
|
|
// it also updates the next slot cache and the proposer index cache to deal with skipped slots.
|
|
func (s *Service) lateBlockTasks(ctx context.Context) {
|
|
currentSlot := s.CurrentSlot()
|
|
if s.CurrentSlot() == s.HeadSlot() {
|
|
return
|
|
}
|
|
s.cfg.StateNotifier.StateFeed().Send(&feed.Event{
|
|
Type: statefeed.MissedSlot,
|
|
})
|
|
|
|
s.headLock.RLock()
|
|
headRoot := s.headRoot()
|
|
headState := s.headState(ctx)
|
|
s.headLock.RUnlock()
|
|
lastRoot, lastState := transition.LastCachedState()
|
|
if lastState == nil {
|
|
lastRoot, lastState = headRoot[:], headState
|
|
}
|
|
// Copy all the field tries in our cached state in the event of late
|
|
// blocks.
|
|
lastState.CopyAllTries()
|
|
if err := transition.UpdateNextSlotCache(ctx, lastRoot, lastState); err != nil {
|
|
log.WithError(err).Debug("could not update next slot state cache")
|
|
}
|
|
if err := s.handleEpochBoundary(ctx, currentSlot, headState, headRoot[:]); err != nil {
|
|
log.WithError(err).Error("lateBlockTasks: could not update epoch boundary caches")
|
|
}
|
|
// Head root should be empty when retrieving proposer index for the next slot.
|
|
_, id, has := s.cfg.ProposerSlotIndexCache.GetProposerPayloadIDs(s.CurrentSlot()+1, [32]byte{} /* head root */)
|
|
// There exists proposer for next slot, but we haven't called fcu w/ payload attribute yet.
|
|
if (!has && !features.Get().PrepareAllPayloads) || id != [8]byte{} {
|
|
return
|
|
}
|
|
|
|
s.headLock.RLock()
|
|
headBlock, err := s.headBlock()
|
|
if err != nil {
|
|
s.headLock.RUnlock()
|
|
log.WithError(err).Debug("could not perform late block tasks: failed to retrieve head block")
|
|
return
|
|
}
|
|
s.headLock.RUnlock()
|
|
_, err = s.notifyForkchoiceUpdate(ctx, ¬ifyForkchoiceUpdateArg{
|
|
headState: headState,
|
|
headRoot: headRoot,
|
|
headBlock: headBlock.Block(),
|
|
})
|
|
if err != nil {
|
|
log.WithError(err).Debug("could not perform late block tasks: failed to update forkchoice with engine")
|
|
}
|
|
}
|
|
|
|
// waitForSync blocks until the node is synced to the head.
|
|
func (s *Service) waitForSync() error {
|
|
select {
|
|
case <-s.syncComplete:
|
|
return nil
|
|
case <-s.ctx.Done():
|
|
return errors.New("context closed, exiting goroutine")
|
|
}
|
|
}
|
|
|
|
func (s *Service) handleInvalidExecutionError(ctx context.Context, err error, blockRoot [32]byte, parentRoot [32]byte) error {
|
|
if IsInvalidBlock(err) && InvalidBlockLVH(err) != [32]byte{} {
|
|
return s.pruneInvalidBlock(ctx, blockRoot, parentRoot, InvalidBlockLVH(err))
|
|
}
|
|
return err
|
|
}
|