DownloadHeaderSequenceTask.java
/*
* Copyright ConsenSys AG.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on
* an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the
* specific language governing permissions and limitations under the License.
*
* SPDX-License-Identifier: Apache-2.0
*/
package org.hyperledger.besu.ethereum.eth.sync.tasks;
import static com.google.common.base.Preconditions.checkArgument;
import static java.util.Arrays.asList;
import org.hyperledger.besu.datatypes.Hash;
import org.hyperledger.besu.ethereum.ProtocolContext;
import org.hyperledger.besu.ethereum.chain.BadBlockCause;
import org.hyperledger.besu.ethereum.chain.BadBlockManager;
import org.hyperledger.besu.ethereum.core.Block;
import org.hyperledger.besu.ethereum.core.BlockHeader;
import org.hyperledger.besu.ethereum.eth.manager.EthContext;
import org.hyperledger.besu.ethereum.eth.manager.EthPeer;
import org.hyperledger.besu.ethereum.eth.manager.task.AbstractGetHeadersFromPeerTask;
import org.hyperledger.besu.ethereum.eth.manager.task.AbstractPeerTask.PeerTaskResult;
import org.hyperledger.besu.ethereum.eth.manager.task.AbstractRetryingPeerTask;
import org.hyperledger.besu.ethereum.eth.manager.task.GetBodiesFromPeerTask;
import org.hyperledger.besu.ethereum.eth.manager.task.GetHeadersFromPeerByHashTask;
import org.hyperledger.besu.ethereum.eth.sync.ValidationPolicy;
import org.hyperledger.besu.ethereum.eth.sync.tasks.exceptions.InvalidBlockException;
import org.hyperledger.besu.ethereum.mainnet.BlockHeaderValidator;
import org.hyperledger.besu.ethereum.mainnet.HeaderValidationMode;
import org.hyperledger.besu.ethereum.mainnet.ProtocolSchedule;
import org.hyperledger.besu.ethereum.mainnet.ProtocolSpec;
import org.hyperledger.besu.ethereum.p2p.rlpx.wire.messages.DisconnectMessage.DisconnectReason;
import org.hyperledger.besu.plugin.services.MetricsSystem;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.CompletableFuture;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.primitives.Ints;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Retrieves a sequence of headers, sending out requests repeatedly until all headers are fulfilled.
* Validates headers as they are received.
*/
public class DownloadHeaderSequenceTask extends AbstractRetryingPeerTask<List<BlockHeader>> {
private static final Logger LOG = LoggerFactory.getLogger(DownloadHeaderSequenceTask.class);
private static final int DEFAULT_RETRIES = 4;
private final EthContext ethContext;
private final ProtocolContext protocolContext;
private final ProtocolSchedule protocolSchedule;
private final BlockHeader[] headers;
private final BlockHeader referenceHeader;
private final int segmentLength;
private final long startingBlockNumber;
private final ValidationPolicy validationPolicy;
private final MetricsSystem metricsSystem;
private int lastFilledHeaderIndex;
private DownloadHeaderSequenceTask(
final ProtocolSchedule protocolSchedule,
final ProtocolContext protocolContext,
final EthContext ethContext,
final BlockHeader referenceHeader,
final int segmentLength,
final int maxRetries,
final ValidationPolicy validationPolicy,
final MetricsSystem metricsSystem) {
super(ethContext, maxRetries, Collection::isEmpty, metricsSystem);
this.protocolSchedule = protocolSchedule;
this.protocolContext = protocolContext;
this.ethContext = ethContext;
this.referenceHeader = referenceHeader;
this.segmentLength = segmentLength;
this.validationPolicy = validationPolicy;
this.metricsSystem = metricsSystem;
checkArgument(segmentLength > 0, "Segment length must not be 0");
startingBlockNumber = referenceHeader.getNumber() - segmentLength;
headers = new BlockHeader[segmentLength];
lastFilledHeaderIndex = segmentLength;
}
public static DownloadHeaderSequenceTask endingAtHeader(
final ProtocolSchedule protocolSchedule,
final ProtocolContext protocolContext,
final EthContext ethContext,
final BlockHeader referenceHeader,
final int segmentLength,
final int maxRetries,
final ValidationPolicy validationPolicy,
final MetricsSystem metricsSystem) {
return new DownloadHeaderSequenceTask(
protocolSchedule,
protocolContext,
ethContext,
referenceHeader,
segmentLength,
maxRetries,
validationPolicy,
metricsSystem);
}
public static DownloadHeaderSequenceTask endingAtHeader(
final ProtocolSchedule protocolSchedule,
final ProtocolContext protocolContext,
final EthContext ethContext,
final BlockHeader referenceHeader,
final int segmentLength,
final ValidationPolicy validationPolicy,
final MetricsSystem metricsSystem) {
return new DownloadHeaderSequenceTask(
protocolSchedule,
protocolContext,
ethContext,
referenceHeader,
segmentLength,
DEFAULT_RETRIES,
validationPolicy,
metricsSystem);
}
@Override
protected CompletableFuture<List<BlockHeader>> executePeerTask(
final Optional<EthPeer> assignedPeer) {
LOG.debug(
"Downloading headers from {} to {}.", startingBlockNumber, referenceHeader.getNumber());
final CompletableFuture<List<BlockHeader>> task =
downloadHeaders(assignedPeer).thenCompose(this::processHeaders);
return task.whenComplete(
(r, t) -> {
// We're done if we've filled all requested headers
if (lastFilledHeaderIndex == 0) {
LOG.debug(
"Finished downloading headers from {} to {}.",
headers[0].getNumber(),
headers[segmentLength - 1].getNumber());
result.complete(Arrays.asList(headers));
}
});
}
private CompletableFuture<PeerTaskResult<List<BlockHeader>>> downloadHeaders(
final Optional<EthPeer> assignedPeer) {
// Figure out parameters for our headers request
final boolean partiallyFilled = lastFilledHeaderIndex < segmentLength;
final BlockHeader referenceHeaderForNextRequest =
partiallyFilled ? headers[lastFilledHeaderIndex] : referenceHeader;
final Hash referenceHash = referenceHeaderForNextRequest.getHash();
final int count = partiallyFilled ? lastFilledHeaderIndex : segmentLength;
return executeSubTask(
() -> {
// Ask for count + 1 because we'll retrieve the previous header as well
final AbstractGetHeadersFromPeerTask headersTask =
GetHeadersFromPeerByHashTask.endingAtHash(
protocolSchedule,
ethContext,
referenceHash,
referenceHeaderForNextRequest.getNumber(),
count + 1,
metricsSystem);
assignedPeer.ifPresent(headersTask::assignPeer);
return headersTask.run();
});
}
@VisibleForTesting
CompletableFuture<List<BlockHeader>> processHeaders(
final PeerTaskResult<List<BlockHeader>> headersResult) {
return executeWorkerSubTask(
ethContext.getScheduler(),
() -> {
final CompletableFuture<List<BlockHeader>> future = new CompletableFuture<>();
BlockHeader child = null;
boolean firstSkipped = false;
final int previousHeaderIndex = lastFilledHeaderIndex;
for (final BlockHeader header : headersResult.getResult()) {
final int headerIndex =
Ints.checkedCast(
segmentLength - (referenceHeader.getNumber() - header.getNumber()));
if (!firstSkipped) {
// Skip over reference header
firstSkipped = true;
continue;
}
if (child == null) {
child =
(headerIndex == segmentLength - 1) ? referenceHeader : headers[headerIndex + 1];
}
final boolean foundChild = child != null;
final boolean headerInRange = checkHeaderInRange(header);
final boolean headerInvalid = foundChild && !validateHeader(child, header);
if (!headerInRange || !foundChild || headerInvalid) {
final BlockHeader invalidHeader = child;
final CompletableFuture<?> badBlockHandled =
headerInvalid
? markBadBlock(invalidHeader, headersResult.getPeer())
: CompletableFuture.completedFuture(null);
badBlockHandled.whenComplete(
(res, err) -> {
LOG.debug(
"Received invalid headers from peer (BREACH_OF_PROTOCOL), disconnecting from: {}",
headersResult.getPeer());
headersResult
.getPeer()
.disconnect(DisconnectReason.BREACH_OF_PROTOCOL_INVALID_HEADERS);
final InvalidBlockException exception;
if (invalidHeader == null) {
final String msg =
String.format(
"Received misordered blocks. Missing child of %s",
header.toLogString());
exception = InvalidBlockException.create(msg);
} else {
final String errorMsg =
headerInvalid
? "Header failed validation"
: "Out-of-range header received from peer";
exception = InvalidBlockException.fromInvalidBlock(errorMsg, invalidHeader);
}
future.completeExceptionally(exception);
});
return future;
}
headers[headerIndex] = header;
lastFilledHeaderIndex = headerIndex;
child = header;
}
future.complete(asList(headers).subList(lastFilledHeaderIndex, previousHeaderIndex));
return future;
});
}
private CompletableFuture<?> markBadBlock(final BlockHeader badHeader, final EthPeer badPeer) {
// even though the header is known bad we are downloading the block body for the debug_badBlocks
// RPC
final BadBlockManager badBlockManager = protocolContext.getBadBlockManager();
return GetBodiesFromPeerTask.forHeaders(
protocolSchedule, ethContext, List.of(badHeader), metricsSystem)
.assignPeer(badPeer)
.run()
.whenComplete(
(blockPeerTaskResult, error) -> {
final HeaderValidationMode validationMode =
validationPolicy.getValidationModeForNextBlock();
final String description =
String.format("Failed header validation (%s)", validationMode);
final BadBlockCause cause = BadBlockCause.fromValidationFailure(description);
if (blockPeerTaskResult != null) {
final Optional<Block> block = blockPeerTaskResult.getResult().stream().findFirst();
block.ifPresentOrElse(
(b) -> badBlockManager.addBadBlock(b, cause),
() -> badBlockManager.addBadHeader(badHeader, cause));
} else {
badBlockManager.addBadHeader(badHeader, cause);
}
});
}
private boolean checkHeaderInRange(final BlockHeader header) {
final long finalBlockNumber = startingBlockNumber + segmentLength;
return header.getNumber() >= startingBlockNumber && header.getNumber() < finalBlockNumber;
}
private boolean validateHeader(final BlockHeader header, final BlockHeader parent) {
final ProtocolSpec protocolSpec = protocolSchedule.getByBlockHeader(header);
final BlockHeaderValidator blockHeaderValidator = protocolSpec.getBlockHeaderValidator();
return blockHeaderValidator.validateHeader(
header, parent, protocolContext, validationPolicy.getValidationModeForNextBlock());
}
}