eth/downloader: resolve local header by hash for beacon sync (#24691)
* eth/downlaoder: resolve local header by hash for beacon sync * eth/downloader: fix error message * eth/downloader: cap the reverse header resolving * eth/downloader: re-enable tests * eth/downloader: add warning logs
This commit is contained in:
parent
9f7bcb9d76
commit
f0328f241b
|
@ -263,10 +263,23 @@ func (d *Downloader) findBeaconAncestor() (uint64, error) {
|
||||||
// fetchBeaconHeaders feeds skeleton headers to the downloader queue for scheduling
|
// fetchBeaconHeaders feeds skeleton headers to the downloader queue for scheduling
|
||||||
// until sync errors or is finished.
|
// until sync errors or is finished.
|
||||||
func (d *Downloader) fetchBeaconHeaders(from uint64) error {
|
func (d *Downloader) fetchBeaconHeaders(from uint64) error {
|
||||||
head, _, err := d.skeleton.Bounds()
|
head, tail, err := d.skeleton.Bounds()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
// A part of headers are not in the skeleton space, try to resolve
|
||||||
|
// them from the local chain. Note the range should be very short
|
||||||
|
// and it should only happen when there are less than 64 post-merge
|
||||||
|
// blocks in the network.
|
||||||
|
var localHeaders []*types.Header
|
||||||
|
if from < tail.Number.Uint64() {
|
||||||
|
count := tail.Number.Uint64() - from
|
||||||
|
if count > uint64(fsMinFullBlocks) {
|
||||||
|
return fmt.Errorf("invalid origin (%d) of beacon sync (%d)", from, tail.Number)
|
||||||
|
}
|
||||||
|
localHeaders = d.readHeaderRange(tail, int(count))
|
||||||
|
log.Warn("Retrieved beacon headers from local", "from", from, "count", count)
|
||||||
|
}
|
||||||
for {
|
for {
|
||||||
// Retrieve a batch of headers and feed it to the header processor
|
// Retrieve a batch of headers and feed it to the header processor
|
||||||
var (
|
var (
|
||||||
|
@ -275,8 +288,18 @@ func (d *Downloader) fetchBeaconHeaders(from uint64) error {
|
||||||
)
|
)
|
||||||
for i := 0; i < maxHeadersProcess && from <= head.Number.Uint64(); i++ {
|
for i := 0; i < maxHeadersProcess && from <= head.Number.Uint64(); i++ {
|
||||||
header := d.skeleton.Header(from)
|
header := d.skeleton.Header(from)
|
||||||
|
|
||||||
|
// The header is not found in skeleton space, try to find it in local chain.
|
||||||
|
if header == nil && from < tail.Number.Uint64() {
|
||||||
|
dist := tail.Number.Uint64() - from
|
||||||
|
if len(localHeaders) >= int(dist) {
|
||||||
|
header = localHeaders[dist-1]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// The header is still missing, the beacon sync is corrupted and bail out
|
||||||
|
// the error here.
|
||||||
if header == nil {
|
if header == nil {
|
||||||
header = d.lightchain.GetHeaderByNumber(from)
|
return fmt.Errorf("missing beacon header %d", from)
|
||||||
}
|
}
|
||||||
headers = append(headers, header)
|
headers = append(headers, header)
|
||||||
hashes = append(hashes, headers[i].Hash())
|
hashes = append(hashes, headers[i].Hash())
|
||||||
|
|
|
@ -159,9 +159,6 @@ type LightChain interface {
|
||||||
// GetHeaderByHash retrieves a header from the local chain.
|
// GetHeaderByHash retrieves a header from the local chain.
|
||||||
GetHeaderByHash(common.Hash) *types.Header
|
GetHeaderByHash(common.Hash) *types.Header
|
||||||
|
|
||||||
// GetHeaderByNumber retrieves a block header from the local chain by number.
|
|
||||||
GetHeaderByNumber(number uint64) *types.Header
|
|
||||||
|
|
||||||
// CurrentHeader retrieves the head header from the local chain.
|
// CurrentHeader retrieves the head header from the local chain.
|
||||||
CurrentHeader() *types.Header
|
CurrentHeader() *types.Header
|
||||||
|
|
||||||
|
@ -486,7 +483,15 @@ func (d *Downloader) syncWithPeer(p *peerConnection, hash common.Hash, td, ttd *
|
||||||
// Retrieve the pivot header from the skeleton chain segment but
|
// Retrieve the pivot header from the skeleton chain segment but
|
||||||
// fallback to local chain if it's not found in skeleton space.
|
// fallback to local chain if it's not found in skeleton space.
|
||||||
if pivot = d.skeleton.Header(number); pivot == nil {
|
if pivot = d.skeleton.Header(number); pivot == nil {
|
||||||
pivot = d.lightchain.GetHeaderByNumber(number)
|
_, oldest, _ := d.skeleton.Bounds() // error is already checked
|
||||||
|
if number < oldest.Number.Uint64() {
|
||||||
|
count := int(oldest.Number.Uint64() - number) // it's capped by fsMinFullBlocks
|
||||||
|
headers := d.readHeaderRange(oldest, count)
|
||||||
|
if len(headers) == count {
|
||||||
|
pivot = headers[len(headers)-1]
|
||||||
|
log.Warn("Retrieved pivot header from local", "number", pivot.Number, "hash", pivot.Hash(), "latest", latest.Number, "oldest", oldest.Number)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// Print an error log and return directly in case the pivot header
|
// Print an error log and return directly in case the pivot header
|
||||||
// is still not found. It means the skeleton chain is not linked
|
// is still not found. It means the skeleton chain is not linked
|
||||||
|
@ -1772,3 +1777,25 @@ func (d *Downloader) DeliverSnapPacket(peer *snap.Peer, packet snap.Packet) erro
|
||||||
return fmt.Errorf("unexpected snap packet type: %T", packet)
|
return fmt.Errorf("unexpected snap packet type: %T", packet)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// readHeaderRange returns a list of headers, using the given last header as the base,
|
||||||
|
// and going backwards towards genesis. This method assumes that the caller already has
|
||||||
|
// placed a reasonable cap on count.
|
||||||
|
func (d *Downloader) readHeaderRange(last *types.Header, count int) []*types.Header {
|
||||||
|
var (
|
||||||
|
current = last
|
||||||
|
headers []*types.Header
|
||||||
|
)
|
||||||
|
for {
|
||||||
|
parent := d.lightchain.GetHeaderByHash(current.ParentHash)
|
||||||
|
if parent == nil {
|
||||||
|
break // The chain is not continuous, or the chain is exhausted
|
||||||
|
}
|
||||||
|
headers = append(headers, parent)
|
||||||
|
if len(headers) >= count {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
current = parent
|
||||||
|
}
|
||||||
|
return headers
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue