mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-04 08:44:28 +00:00
155 lines
4.8 KiB
Go
155 lines
4.8 KiB
Go
package initialsync
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"sort"
|
|
|
|
peer "github.com/libp2p/go-libp2p-peer"
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/go-ssz"
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
ethpb "github.com/prysmaticlabs/prysm/proto/eth/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/shared/p2p"
|
|
"github.com/sirupsen/logrus"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
// processBlock is the main method that validates each block which is received
|
|
// for initial sync. It checks if the blocks are valid and then will continue to
|
|
// process and save it into the db.
|
|
func (s *InitialSync) processBlock(ctx context.Context, block *ethpb.BeaconBlock, chainHead *pb.ChainHeadResponse) error {
|
|
ctx, span := trace.StartSpan(ctx, "beacon-chain.sync.initial-sync.processBlock")
|
|
defer span.End()
|
|
recBlock.Inc()
|
|
|
|
if block.Slot == chainHead.CanonicalSlot {
|
|
if err := s.exitInitialSync(s.ctx, block, chainHead); err != nil {
|
|
log.Errorf("Could not exit initial sync: %v", err)
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
if err := s.validateAndSaveNextBlock(ctx, block); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// processBatchedBlocks processes all the received blocks from
|
|
// the p2p message.
|
|
func (s *InitialSync) processBatchedBlocks(msg p2p.Message, chainHead *pb.ChainHeadResponse) error {
|
|
ctx, span := trace.StartSpan(msg.Ctx, "beacon-chain.sync.initial-sync.processBatchedBlocks")
|
|
defer span.End()
|
|
batchedBlockReq.Inc()
|
|
|
|
response := msg.Data.(*pb.BatchedBeaconBlockResponse)
|
|
batchedBlocks := response.BatchedBlocks
|
|
if len(batchedBlocks) == 0 {
|
|
// Do not process empty responses.
|
|
s.p2p.Reputation(msg.Peer, p2p.RepPenalityInitialSyncFailure)
|
|
return nil
|
|
}
|
|
|
|
log.WithField("blocks", len(batchedBlocks)).Info("Processing batched block response")
|
|
// Sort batchBlocks in ascending order.
|
|
sort.Slice(batchedBlocks, func(i, j int) bool {
|
|
return batchedBlocks[i].Slot < batchedBlocks[j].Slot
|
|
})
|
|
|
|
for _, block := range batchedBlocks {
|
|
if err := s.processBlock(ctx, block, chainHead); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
log.Debug("Finished processing batched blocks")
|
|
return nil
|
|
}
|
|
|
|
// requestBatchedBlocks sends out a request for multiple blocks that's between finalized roots
|
|
// and head roots.
|
|
func (s *InitialSync) requestBatchedBlocks(ctx context.Context, FinalizedRoot []byte, canonicalRoot []byte, peer peer.ID) {
|
|
ctx, span := trace.StartSpan(ctx, "beacon-chain.sync.initial-sync.requestBatchedBlocks")
|
|
defer span.End()
|
|
sentBatchedBlockReq.Inc()
|
|
|
|
log.WithFields(logrus.Fields{
|
|
"finalizedBlkRoot": fmt.Sprintf("%#x", bytesutil.Trunc(FinalizedRoot[:])),
|
|
"headBlkRoot": fmt.Sprintf("%#x", bytesutil.Trunc(canonicalRoot[:]))},
|
|
).Debug("Requesting batched blocks")
|
|
if err := s.p2p.Send(ctx, &pb.BatchedBeaconBlockRequest{
|
|
FinalizedRoot: FinalizedRoot,
|
|
CanonicalRoot: canonicalRoot,
|
|
}, peer); err != nil {
|
|
log.Errorf("Could not send batch block request to peer %s: %v", peer.Pretty(), err)
|
|
}
|
|
}
|
|
|
|
// validateAndSaveNextBlock will validate whether blocks received from the blockfetcher
|
|
// routine can be added to the chain.
|
|
func (s *InitialSync) validateAndSaveNextBlock(ctx context.Context, block *ethpb.BeaconBlock) error {
|
|
ctx, span := trace.StartSpan(ctx, "beacon-chain.sync.initial-sync.validateAndSaveNextBlock")
|
|
defer span.End()
|
|
if block == nil {
|
|
return errors.New("received nil block")
|
|
}
|
|
root, err := ssz.SigningRoot(block)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if s.db.HasBlock(root) {
|
|
log.WithField("block", fmt.Sprintf("%#x", root)).
|
|
Warn("Skipping block in db already")
|
|
return nil
|
|
}
|
|
|
|
if err := s.checkBlockValidity(ctx, block); err != nil {
|
|
return err
|
|
}
|
|
log.WithFields(logrus.Fields{
|
|
"root": fmt.Sprintf("%#x", bytesutil.Trunc(root[:])),
|
|
"slot": block.Slot,
|
|
}).Info("Saving block")
|
|
|
|
s.mutex.Lock()
|
|
defer s.mutex.Unlock()
|
|
parentRoot := bytesutil.ToBytes32(block.ParentRoot)
|
|
parentBlock, err := s.db.Block(parentRoot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if parentBlock == nil {
|
|
return fmt.Errorf("parent block with root %#x doesnt exist in the db", parentRoot)
|
|
}
|
|
|
|
state, err := s.db.HistoricalStateFromSlot(ctx, parentBlock.Slot, parentRoot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := s.chainService.VerifyBlockValidity(ctx, block, state); err != nil {
|
|
return err
|
|
}
|
|
if err := s.db.SaveBlock(block); err != nil {
|
|
return err
|
|
}
|
|
if err := s.db.SaveAttestationTarget(ctx, &pb.AttestationTarget{
|
|
Slot: block.Slot,
|
|
BeaconBlockRoot: root[:],
|
|
ParentRoot: block.ParentRoot,
|
|
}); err != nil {
|
|
return errors.Wrap(err, "could not to save attestation target")
|
|
}
|
|
state, err = s.chainService.AdvanceState(ctx, state, block)
|
|
if err != nil {
|
|
return errors.Wrap(err, "could not apply block state transition")
|
|
}
|
|
if err := s.chainService.CleanupBlockOperations(ctx, block); err != nil {
|
|
return err
|
|
}
|
|
return s.db.UpdateChainHead(ctx, block, state)
|
|
}
|