2019-04-03 15:13:19 +00:00
|
|
|
package initialsync
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
2019-04-16 18:06:53 +00:00
|
|
|
"fmt"
|
2019-04-03 15:13:19 +00:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2019-04-30 19:55:14 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
2019-04-03 15:13:19 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/p2p"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2019-04-21 21:12:03 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2019-04-03 15:13:19 +00:00
|
|
|
"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 *pb.BeaconBlock) {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "beacon-chain.sync.initial-sync.processBlock")
|
|
|
|
defer span.End()
|
|
|
|
recBlock.Inc()
|
|
|
|
|
|
|
|
if block.Slot == s.highestObservedSlot {
|
|
|
|
s.currentSlot = s.highestObservedSlot
|
2019-04-05 03:39:51 +00:00
|
|
|
if err := s.exitInitialSync(s.ctx, block); err != nil {
|
2019-04-03 15:13:19 +00:00
|
|
|
log.Errorf("Could not exit initial sync: %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if block.Slot < s.currentSlot {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// if it isn't the block in the next slot we check if it is a skipped slot.
|
|
|
|
// if it isn't skipped we save it in memory.
|
|
|
|
if block.Slot != (s.currentSlot + 1) {
|
|
|
|
// if parent exists we validate the block.
|
|
|
|
if s.doesParentExist(block) {
|
|
|
|
if err := s.validateAndSaveNextBlock(ctx, block); err != nil {
|
|
|
|
// Debug error so as not to have noisy error logs
|
|
|
|
if strings.HasPrefix(err.Error(), debugError) {
|
|
|
|
log.Debug(strings.TrimPrefix(err.Error(), debugError))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
log.Errorf("Unable to save block: %v", err)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
s.mutex.Lock()
|
|
|
|
defer s.mutex.Unlock()
|
|
|
|
if _, ok := s.inMemoryBlocks[block.Slot]; !ok {
|
|
|
|
s.inMemoryBlocks[block.Slot] = block
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := s.validateAndSaveNextBlock(ctx, block); err != nil {
|
|
|
|
// Debug error so as not to have noisy error logs
|
|
|
|
if strings.HasPrefix(err.Error(), debugError) {
|
|
|
|
log.Debug(strings.TrimPrefix(err.Error(), debugError))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
log.Errorf("Unable to save block: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// processBatchedBlocks processes all the received blocks from
|
|
|
|
// the p2p message.
|
|
|
|
func (s *InitialSync) processBatchedBlocks(msg p2p.Message) {
|
|
|
|
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.
|
|
|
|
return
|
|
|
|
}
|
2019-04-26 15:18:43 +00:00
|
|
|
if msg.Peer != s.bestPeer {
|
|
|
|
// Only process batch block responses that come from the best peer
|
|
|
|
// we originally synced with.
|
|
|
|
log.WithField("peerID", msg.Peer.Pretty()).Debug("Received batch blocks from a different peer")
|
|
|
|
return
|
|
|
|
}
|
2019-04-03 15:13:19 +00:00
|
|
|
|
|
|
|
log.Debug("Processing batched block response")
|
|
|
|
for _, block := range batchedBlocks {
|
|
|
|
s.processBlock(ctx, block)
|
|
|
|
}
|
|
|
|
log.Debug("Finished processing batched blocks")
|
|
|
|
}
|
|
|
|
|
|
|
|
// requestBatchedBlocks sends out a request for multiple blocks till a
|
|
|
|
// specified bound slot number.
|
|
|
|
func (s *InitialSync) requestBatchedBlocks(startSlot uint64, endSlot uint64) {
|
|
|
|
ctx, span := trace.StartSpan(context.Background(), "beacon-chain.sync.initial-sync.requestBatchedBlocks")
|
|
|
|
defer span.End()
|
|
|
|
sentBatchedBlockReq.Inc()
|
|
|
|
if startSlot > endSlot {
|
2019-04-20 04:09:01 +00:00
|
|
|
log.WithFields(logrus.Fields{
|
|
|
|
"slotSlot": startSlot - params.BeaconConfig().GenesisSlot,
|
|
|
|
"endSlot": endSlot - params.BeaconConfig().GenesisSlot},
|
|
|
|
).Debug("Invalid batched block request")
|
2019-04-03 15:13:19 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
blockLimit := params.BeaconConfig().BatchBlockLimit
|
|
|
|
if startSlot+blockLimit < endSlot {
|
|
|
|
endSlot = startSlot + blockLimit
|
|
|
|
}
|
2019-04-20 04:09:01 +00:00
|
|
|
log.WithFields(logrus.Fields{
|
|
|
|
"slotSlot": startSlot - params.BeaconConfig().GenesisSlot,
|
|
|
|
"endSlot": endSlot - params.BeaconConfig().GenesisSlot},
|
|
|
|
).Debug("Requesting batched blocks")
|
2019-04-03 15:13:19 +00:00
|
|
|
s.p2p.Broadcast(ctx, &pb.BatchedBeaconBlockRequest{
|
|
|
|
StartSlot: startSlot,
|
|
|
|
EndSlot: endSlot,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// validateAndSaveNextBlock will validate whether blocks received from the blockfetcher
|
|
|
|
// routine can be added to the chain.
|
|
|
|
func (s *InitialSync) validateAndSaveNextBlock(ctx context.Context, block *pb.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 := hashutil.HashBeaconBlock(block)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := s.checkBlockValidity(ctx, block); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-04-20 04:09:01 +00:00
|
|
|
log.WithFields(logrus.Fields{
|
2019-04-30 19:55:14 +00:00
|
|
|
"root": fmt.Sprintf("%#x", bytesutil.Trunc(root[:])),
|
2019-04-20 04:09:01 +00:00
|
|
|
"slot": block.Slot - params.BeaconConfig().GenesisSlot,
|
|
|
|
}).Info("Saving block")
|
2019-04-03 15:13:19 +00:00
|
|
|
s.currentSlot = block.Slot
|
|
|
|
|
|
|
|
s.mutex.Lock()
|
|
|
|
defer s.mutex.Unlock()
|
|
|
|
// delete block from memory.
|
|
|
|
if _, ok := s.inMemoryBlocks[block.Slot]; ok {
|
|
|
|
delete(s.inMemoryBlocks, block.Slot)
|
|
|
|
}
|
2019-04-05 14:41:49 +00:00
|
|
|
state, err := s.db.HeadState(ctx)
|
2019-04-03 15:13:19 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-04-04 13:30:23 +00:00
|
|
|
if err := s.chainService.VerifyBlockValidity(ctx, block, state); err != nil {
|
2019-04-03 15:13:19 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := s.db.SaveBlock(block); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-04-24 17:21:00 +00:00
|
|
|
if err := s.db.SaveAttestationTarget(ctx, &pb.AttestationTarget{
|
|
|
|
Slot: block.Slot,
|
|
|
|
BlockRoot: root[:],
|
|
|
|
ParentRoot: block.ParentRootHash32,
|
|
|
|
}); err != nil {
|
|
|
|
return fmt.Errorf("could not to save attestation target: %v", err)
|
|
|
|
}
|
2019-04-03 15:13:19 +00:00
|
|
|
state, err = s.chainService.ApplyBlockStateTransition(ctx, block, state)
|
|
|
|
if err != nil {
|
2019-04-21 21:12:03 +00:00
|
|
|
return fmt.Errorf("could not apply block state transition: %v", err)
|
2019-04-03 15:13:19 +00:00
|
|
|
}
|
|
|
|
if err := s.chainService.CleanupBlockOperations(ctx, block); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return s.db.UpdateChainHead(ctx, block, state)
|
|
|
|
}
|