|
|
|
@ -36,6 +36,9 @@ import tech.pegasys.pantheon.ethereum.mainnet.ProtocolSchedule; |
|
|
|
|
import tech.pegasys.pantheon.metrics.noop.NoOpMetricsSystem; |
|
|
|
|
import tech.pegasys.pantheon.util.uint.UInt256; |
|
|
|
|
|
|
|
|
|
import java.util.ArrayList; |
|
|
|
|
import java.util.List; |
|
|
|
|
import java.util.OptionalLong; |
|
|
|
|
import java.util.concurrent.CompletableFuture; |
|
|
|
|
import java.util.concurrent.atomic.AtomicInteger; |
|
|
|
|
|
|
|
|
@ -44,39 +47,27 @@ import org.junit.Test; |
|
|
|
|
|
|
|
|
|
public class FastSyncActionsTest { |
|
|
|
|
|
|
|
|
|
private final SynchronizerConfiguration syncConfig = |
|
|
|
|
new SynchronizerConfiguration.Builder() |
|
|
|
|
.syncMode(SyncMode.FAST) |
|
|
|
|
.fastSyncPivotDistance(1000) |
|
|
|
|
.build(); |
|
|
|
|
private final BlockchainSetupUtil<Void> blockchainSetupUtil = BlockchainSetupUtil.forTesting(); |
|
|
|
|
private final SynchronizerConfiguration.Builder syncConfigBuilder = |
|
|
|
|
new SynchronizerConfiguration.Builder().syncMode(SyncMode.FAST).fastSyncPivotDistance(1000); |
|
|
|
|
|
|
|
|
|
private final FastSyncStateStorage fastSyncStateStorage = mock(FastSyncStateStorage.class); |
|
|
|
|
private final AtomicInteger timeoutCount = new AtomicInteger(0); |
|
|
|
|
private SynchronizerConfiguration syncConfig = syncConfigBuilder.build(); |
|
|
|
|
private FastSyncActions<Void> fastSyncActions; |
|
|
|
|
private EthProtocolManager ethProtocolManager; |
|
|
|
|
private MutableBlockchain blockchain; |
|
|
|
|
|
|
|
|
|
@Before |
|
|
|
|
public void setUp() { |
|
|
|
|
final BlockchainSetupUtil<Void> blockchainSetupUtil = BlockchainSetupUtil.forTesting(); |
|
|
|
|
blockchainSetupUtil.importAllBlocks(); |
|
|
|
|
blockchain = blockchainSetupUtil.getBlockchain(); |
|
|
|
|
final ProtocolSchedule<Void> protocolSchedule = blockchainSetupUtil.getProtocolSchedule(); |
|
|
|
|
final ProtocolContext<Void> protocolContext = blockchainSetupUtil.getProtocolContext(); |
|
|
|
|
ethProtocolManager = |
|
|
|
|
EthProtocolManagerTestUtil.create( |
|
|
|
|
blockchain, |
|
|
|
|
blockchainSetupUtil.getWorldArchive(), |
|
|
|
|
() -> timeoutCount.getAndDecrement() > 0); |
|
|
|
|
final EthContext ethContext = ethProtocolManager.ethContext(); |
|
|
|
|
fastSyncActions = |
|
|
|
|
new FastSyncActions<>( |
|
|
|
|
syncConfig, |
|
|
|
|
protocolSchedule, |
|
|
|
|
protocolContext, |
|
|
|
|
ethContext, |
|
|
|
|
new SyncState(blockchain, ethContext.getEthPeers()), |
|
|
|
|
new NoOpMetricsSystem()); |
|
|
|
|
fastSyncActions = createFastSyncActions(syncConfig); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@ -116,6 +107,11 @@ public class FastSyncActionsTest { |
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
|
public void selectPivotBlockShouldSelectBlockPivotDistanceFromBestPeer() { |
|
|
|
|
final int minPeers = 1; |
|
|
|
|
syncConfigBuilder.fastSyncMinimumPeerCount(minPeers); |
|
|
|
|
syncConfig = syncConfigBuilder.build(); |
|
|
|
|
fastSyncActions = createFastSyncActions(syncConfig); |
|
|
|
|
|
|
|
|
|
EthProtocolManagerTestUtil.createPeer(ethProtocolManager, 5000); |
|
|
|
|
|
|
|
|
|
final CompletableFuture<FastSyncState> result = |
|
|
|
@ -126,6 +122,11 @@ public class FastSyncActionsTest { |
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
|
public void selectPivotBlockShouldConsiderTotalDifficultyWhenSelectingBestPeer() { |
|
|
|
|
final int minPeers = 1; |
|
|
|
|
syncConfigBuilder.fastSyncMinimumPeerCount(minPeers); |
|
|
|
|
syncConfig = syncConfigBuilder.build(); |
|
|
|
|
fastSyncActions = createFastSyncActions(syncConfig); |
|
|
|
|
|
|
|
|
|
EthProtocolManagerTestUtil.createPeer(ethProtocolManager, UInt256.of(1000), 5500); |
|
|
|
|
EthProtocolManagerTestUtil.createPeer(ethProtocolManager, UInt256.of(2000), 4000); |
|
|
|
|
|
|
|
|
@ -136,19 +137,124 @@ public class FastSyncActionsTest { |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
|
public void selectPivotBlockShouldWaitAndRetryIfNoPeersAreAvailable() { |
|
|
|
|
public void selectPivotBlockShouldWaitAndRetryUntilMinHeightEstimatesAreAvailable() { |
|
|
|
|
EthProtocolManagerTestUtil.disableEthSchedulerAutoRun(ethProtocolManager); |
|
|
|
|
final int minPeers = 2; |
|
|
|
|
syncConfigBuilder.fastSyncMinimumPeerCount(minPeers); |
|
|
|
|
syncConfig = syncConfigBuilder.build(); |
|
|
|
|
fastSyncActions = createFastSyncActions(syncConfig); |
|
|
|
|
|
|
|
|
|
final CompletableFuture<FastSyncState> result = |
|
|
|
|
fastSyncActions.selectPivotBlock(EMPTY_SYNC_STATE); |
|
|
|
|
EthProtocolManagerTestUtil.runPendingFutures(ethProtocolManager); |
|
|
|
|
assertThat(result).isNotDone(); |
|
|
|
|
|
|
|
|
|
// First peer is under the threshold, we should keep retrying
|
|
|
|
|
EthProtocolManagerTestUtil.createPeer(ethProtocolManager, 5000); |
|
|
|
|
EthProtocolManagerTestUtil.runPendingFutures(ethProtocolManager); |
|
|
|
|
assertThat(result).isNotDone(); |
|
|
|
|
|
|
|
|
|
// Second peers meets min peer threshold, we should select the pivot
|
|
|
|
|
EthProtocolManagerTestUtil.createPeer(ethProtocolManager, 5000); |
|
|
|
|
EthProtocolManagerTestUtil.runPendingFutures(ethProtocolManager); |
|
|
|
|
assertThat(result).isDone(); |
|
|
|
|
final FastSyncState expected = new FastSyncState(4000); |
|
|
|
|
assertThat(result).isCompletedWithValue(expected); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
|
public void selectPivotBlockShouldWaitAndRetryIfSufficientChainHeightEstimatesAreUnavailable() { |
|
|
|
|
final int minPeers = 3; |
|
|
|
|
syncConfigBuilder.fastSyncMinimumPeerCount(minPeers); |
|
|
|
|
syncConfig = syncConfigBuilder.build(); |
|
|
|
|
fastSyncActions = createFastSyncActions(syncConfig); |
|
|
|
|
final long minPivotHeight = syncConfig.getFastSyncPivotDistance() + 1L; |
|
|
|
|
EthProtocolManagerTestUtil.disableEthSchedulerAutoRun(ethProtocolManager); |
|
|
|
|
|
|
|
|
|
// Create peers without chain height estimates
|
|
|
|
|
List<RespondingEthPeer> peers = new ArrayList<>(); |
|
|
|
|
for (int i = 0; i < minPeers; i++) { |
|
|
|
|
final UInt256 td = UInt256.of(i); |
|
|
|
|
final OptionalLong height = OptionalLong.empty(); |
|
|
|
|
final RespondingEthPeer peer = |
|
|
|
|
EthProtocolManagerTestUtil.createPeer(ethProtocolManager, td, height); |
|
|
|
|
peers.add(peer); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// No pivot should be selected while peers do not have height estimates
|
|
|
|
|
final CompletableFuture<FastSyncState> result = |
|
|
|
|
fastSyncActions.selectPivotBlock(EMPTY_SYNC_STATE); |
|
|
|
|
assertThat(result).isNotDone(); |
|
|
|
|
EthProtocolManagerTestUtil.runPendingFutures(ethProtocolManager); |
|
|
|
|
assertThat(result).isNotDone(); |
|
|
|
|
|
|
|
|
|
// Set subset of heights
|
|
|
|
|
peers |
|
|
|
|
.subList(0, minPeers - 1) |
|
|
|
|
.forEach(p -> p.getEthPeer().chainState().updateHeightEstimate(minPivotHeight + 10)); |
|
|
|
|
|
|
|
|
|
// No pivot should be selected while only a subset of peers have height estimates
|
|
|
|
|
EthProtocolManagerTestUtil.runPendingFutures(ethProtocolManager); |
|
|
|
|
assertThat(result).isNotDone(); |
|
|
|
|
|
|
|
|
|
// Set final height
|
|
|
|
|
final long bestPeerHeight = minPivotHeight + 1; |
|
|
|
|
peers.get(minPeers - 1).getEthPeer().chainState().updateHeightEstimate(bestPeerHeight); |
|
|
|
|
final FastSyncState expected = |
|
|
|
|
new FastSyncState(bestPeerHeight - syncConfig.getFastSyncPivotDistance()); |
|
|
|
|
EthProtocolManagerTestUtil.runPendingFutures(ethProtocolManager); |
|
|
|
|
assertThat(result).isCompletedWithValue(expected); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
|
public void selectPivotBlockUsesBestPeerWithHeightEstimate() { |
|
|
|
|
final int minPeers = 3; |
|
|
|
|
final int peerCount = minPeers + 1; |
|
|
|
|
syncConfigBuilder.fastSyncMinimumPeerCount(minPeers); |
|
|
|
|
syncConfig = syncConfigBuilder.build(); |
|
|
|
|
fastSyncActions = createFastSyncActions(syncConfig); |
|
|
|
|
final long minPivotHeight = syncConfig.getFastSyncPivotDistance() + 1L; |
|
|
|
|
EthProtocolManagerTestUtil.disableEthSchedulerAutoRun(ethProtocolManager); |
|
|
|
|
|
|
|
|
|
// Create peers without chain height estimates
|
|
|
|
|
List<RespondingEthPeer> peers = new ArrayList<>(); |
|
|
|
|
for (int i = 0; i < peerCount; i++) { |
|
|
|
|
// Best peer by td is the first peer, td decreases as i increases
|
|
|
|
|
final UInt256 td = UInt256.of(peerCount - i); |
|
|
|
|
|
|
|
|
|
final OptionalLong height; |
|
|
|
|
if (i == 0) { |
|
|
|
|
// Don't set a height estimate for the best peer
|
|
|
|
|
height = OptionalLong.empty(); |
|
|
|
|
} else { |
|
|
|
|
// Height increases with i
|
|
|
|
|
height = OptionalLong.of(minPivotHeight + i); |
|
|
|
|
} |
|
|
|
|
final RespondingEthPeer peer = |
|
|
|
|
EthProtocolManagerTestUtil.createPeer(ethProtocolManager, td, height); |
|
|
|
|
peers.add(peer); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
final CompletableFuture<FastSyncState> result = |
|
|
|
|
fastSyncActions.selectPivotBlock(EMPTY_SYNC_STATE); |
|
|
|
|
EthProtocolManagerTestUtil.runPendingFutures(ethProtocolManager); |
|
|
|
|
|
|
|
|
|
final long expectedBestChainHeight = |
|
|
|
|
peers.get(1).getEthPeer().chainState().getEstimatedHeight(); |
|
|
|
|
final FastSyncState expected = |
|
|
|
|
new FastSyncState(expectedBestChainHeight - syncConfig.getFastSyncPivotDistance()); |
|
|
|
|
EthProtocolManagerTestUtil.runPendingFutures(ethProtocolManager); |
|
|
|
|
assertThat(result).isCompletedWithValue(expected); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
|
public void selectPivotBlockShouldWaitAndRetryIfBestPeerChainIsShorterThanPivotDistance() { |
|
|
|
|
final int minPeers = 1; |
|
|
|
|
syncConfigBuilder.fastSyncMinimumPeerCount(minPeers); |
|
|
|
|
syncConfig = syncConfigBuilder.build(); |
|
|
|
|
fastSyncActions = createFastSyncActions(syncConfig); |
|
|
|
|
final long pivotDistance = syncConfig.getFastSyncPivotDistance(); |
|
|
|
|
|
|
|
|
|
EthProtocolManagerTestUtil.disableEthSchedulerAutoRun(ethProtocolManager); |
|
|
|
|
EthProtocolManagerTestUtil.createPeer(ethProtocolManager, pivotDistance - 1); |
|
|
|
|
|
|
|
|
@ -169,7 +275,10 @@ public class FastSyncActionsTest { |
|
|
|
|
public void selectPivotBlockShouldRetryIfBestPeerChainIsEqualToPivotDistance() { |
|
|
|
|
final long pivotDistance = syncConfig.getFastSyncPivotDistance(); |
|
|
|
|
EthProtocolManagerTestUtil.disableEthSchedulerAutoRun(ethProtocolManager); |
|
|
|
|
EthProtocolManagerTestUtil.createPeer(ethProtocolManager, pivotDistance); |
|
|
|
|
// Create peers with chains that are too short
|
|
|
|
|
for (int i = 0; i < syncConfig.getFastSyncMinimumPeerCount(); i++) { |
|
|
|
|
EthProtocolManagerTestUtil.createPeer(ethProtocolManager, pivotDistance); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
final CompletableFuture<FastSyncState> result = |
|
|
|
|
fastSyncActions.selectPivotBlock(EMPTY_SYNC_STATE); |
|
|
|
@ -203,4 +312,17 @@ public class FastSyncActionsTest { |
|
|
|
|
|
|
|
|
|
assertThat(result).isCompletedWithValue(new FastSyncState(blockchain.getBlockHeader(1).get())); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
private FastSyncActions<Void> createFastSyncActions(final SynchronizerConfiguration syncConfig) { |
|
|
|
|
final ProtocolSchedule<Void> protocolSchedule = blockchainSetupUtil.getProtocolSchedule(); |
|
|
|
|
final ProtocolContext<Void> protocolContext = blockchainSetupUtil.getProtocolContext(); |
|
|
|
|
final EthContext ethContext = ethProtocolManager.ethContext(); |
|
|
|
|
return new FastSyncActions<>( |
|
|
|
|
syncConfig, |
|
|
|
|
protocolSchedule, |
|
|
|
|
protocolContext, |
|
|
|
|
ethContext, |
|
|
|
|
new SyncState(blockchain, ethContext.getEthPeers()), |
|
|
|
|
new NoOpMetricsSystem()); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|