diff --git a/cmd/sentry/sentry/downloader.go b/cmd/sentry/sentry/downloader.go index 77542a5d041..5499f290edd 100644 --- a/cmd/sentry/sentry/downloader.go +++ b/cmd/sentry/sentry/downloader.go @@ -480,9 +480,13 @@ func (cs *ControlServerImpl) blockHeaders(ctx context.Context, pkt eth.BlockHead return err } for _, segment := range segments { - if err := cs.Hd.ProcessSegmentPOS(segment, tx); err != nil { + penalties, err := cs.Hd.ProcessSegmentPOS(segment, tx, ConvertH512ToPeerID(peerID)) + if err != nil { return err } + if len(penalties) > 0 { + cs.Penalize(ctx, penalties) + } } } else { var canRequestMore bool diff --git a/turbo/stages/headerdownload/header_algos.go b/turbo/stages/headerdownload/header_algos.go index 68bbc557d9d..f381acffb48 100644 --- a/turbo/stages/headerdownload/header_algos.go +++ b/turbo/stages/headerdownload/header_algos.go @@ -889,38 +889,45 @@ func (hd *HeaderDownload) SetHeaderToDownloadPoS(hash common.Hash, height uint64 } } -func (hd *HeaderDownload) ProcessSegmentPOS(segment ChainSegment, tx kv.Getter) error { +func (hd *HeaderDownload) ProcessSegmentPOS(segment ChainSegment, tx kv.Getter, peerId [64]byte) ([]PenaltyItem, error) { if len(segment) == 0 { - return nil + return nil, nil } - hd.lock.Lock() - defer hd.lock.Unlock() + // We may have received answer from old request so not enough evidence for penalizing. + if segment[0].Number != hd.posAnchor.blockHeight-1 { + return nil, nil + } + // Handle request after closing collectors if hd.headersCollector == nil { - return nil + return nil, nil } + hd.lock.Lock() + defer hd.lock.Unlock() + log.Trace("Collecting...", "from", segment[0].Number, "to", segment[len(segment)-1].Number, "len", len(segment)) for _, segmentFragment := range segment { header := segmentFragment.Header if header.Hash() != hd.posAnchor.parentHash { - return fmt.Errorf("unexpected hash %x (expected %x)", header.Hash(), hd.posAnchor.parentHash) + log.Warn("Unexpected header", "hash", header.Hash(), "expected", hd.posAnchor.parentHash) + return []PenaltyItem{{PeerID: peerId, Penalty: BadBlockPenalty}}, nil } headerNumber := header.Number.Uint64() if err := hd.headersCollector.Collect(dbutils.HeaderKey(headerNumber, header.Hash()), segmentFragment.HeaderRaw); err != nil { - return err + return nil, err } hh, err := hd.headerReader.Header(context.Background(), tx, header.ParentHash, headerNumber-1) if err != nil { - return err + return nil, err } if hh != nil { log.Trace("Synced", "requestId", hd.requestId) hd.posAnchor = nil hd.posStatus = Synced hd.BeaconRequestList.Interrupt(engineapi.Synced) - return nil + return nil, nil } hd.posAnchor = &Anchor{ @@ -929,10 +936,10 @@ func (hd *HeaderDownload) ProcessSegmentPOS(segment ChainSegment, tx kv.Getter) } if headerNumber <= 1 { - return errors.New("wrong genesis in PoS sync") + return nil, errors.New("wrong genesis in PoS sync") } } - return nil + return nil, nil } // GrabAnnounces - returns all available announces and forget them