Skip to content

Commit

Permalink
Add pause unit test (#10152)
Browse files Browse the repository at this point in the history
Signed-off-by: Austin Littley <[email protected]>
  • Loading branch information
litt3 authored Nov 30, 2023
1 parent 7116c8c commit f2aac3a
Showing 1 changed file with 63 additions and 13 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
import static com.swirlds.common.test.fixtures.RandomUtils.getRandomPrintSeed;
import static com.swirlds.common.test.fixtures.RandomUtils.randomHash;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotEquals;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.doAnswer;
Expand Down Expand Up @@ -84,6 +85,10 @@ class OrphanBufferTests {
*/
private static final int MAX_GENERATION_STEP = 10;

private AtomicLong eventsExitedIntakePipeline;

private OrphanBuffer orphanBuffer;

/**
* Create a bootstrap event for a node. This is just a descriptor, and will never be received from intake.
*
Expand Down Expand Up @@ -223,7 +228,7 @@ private EventDescriptor chooseOtherParent(@NonNull final List<EventDescriptor> p
void setup() {
random = getRandomPrintSeed();

final ArrayList<EventDescriptor> parentCandidates = new ArrayList<>();
final List<EventDescriptor> parentCandidates = new ArrayList<>();
final Map<NodeId, EventDescriptor> tips = new HashMap<>();

intakeEvents = new ArrayList<>();
Expand All @@ -234,14 +239,10 @@ void setup() {
parentCandidates.add(newEvent.getDescriptor());
intakeEvents.add(newEvent);
}
}

@Test
@DisplayName("Test standard orphan buffer operation")
void standardOperation() {
final AtomicLong minimumGenerationNonAncient = new AtomicLong(0);
Collections.shuffle(intakeEvents, random);

final AtomicLong eventsExitedIntakePipeline = new AtomicLong(0);
eventsExitedIntakePipeline = new AtomicLong(0);
final IntakeEventCounter intakeEventCounter = mock(IntakeEventCounter.class);
doAnswer(invocation -> {
eventsExitedIntakePipeline.incrementAndGet();
Expand All @@ -250,8 +251,13 @@ void standardOperation() {
.when(intakeEventCounter)
.eventExitedIntakePipeline(any());

final OrphanBuffer orphanBuffer =
new OrphanBuffer(TestPlatformContextBuilder.create().build(), intakeEventCounter);
orphanBuffer = new OrphanBuffer(TestPlatformContextBuilder.create().build(), intakeEventCounter);
}

@Test
@DisplayName("Test standard orphan buffer operation")
void standardOperation() {
long minimumGenerationNonAncient = 0;

// increase minimum generation non-ancient at the approximate rate that event generations are increasing
// this means that roughly half of the events will be ancient before they are received from intake
Expand All @@ -260,7 +266,6 @@ void standardOperation() {
// events that have been emitted from the orphan buffer
final Collection<Hash> emittedEvents = new HashSet<>();

Collections.shuffle(intakeEvents, random);
for (final GossipEvent intakeEvent : intakeEvents) {
final List<GossipEvent> unorphanedEvents = new ArrayList<>();

Expand All @@ -269,12 +274,12 @@ void standardOperation() {
// add some randomness to step size, so minimumGenerationNonAncient doesn't always just increase by 1
final int stepRandomness = Math.round(random.nextFloat() * MAX_GENERATION_STEP);
if (random.nextFloat() < averageGenerationAdvancement / stepRandomness) {
minimumGenerationNonAncient.addAndGet(stepRandomness);
unorphanedEvents.addAll(orphanBuffer.setMinimumGenerationNonAncient(minimumGenerationNonAncient.get()));
minimumGenerationNonAncient += stepRandomness;
unorphanedEvents.addAll(orphanBuffer.setMinimumGenerationNonAncient(minimumGenerationNonAncient));
}

for (final GossipEvent unorphanedEvent : unorphanedEvents) {
assertValidParents(unorphanedEvent, minimumGenerationNonAncient.get(), emittedEvents);
assertValidParents(unorphanedEvent, minimumGenerationNonAncient, emittedEvents);
emittedEvents.add(unorphanedEvent.getHashedData().getHash());
}
}
Expand All @@ -284,4 +289,49 @@ void standardOperation() {
assertEquals(TEST_EVENT_COUNT, eventsExitedIntakePipeline.get() + emittedEvents.size());
assertEquals(0, orphanBuffer.getCurrentOrphanCount());
}

@Test
@DisplayName("Test pausing orphan buffer")
void pause() {
// cause the bootstrap events to become ancient
orphanBuffer.setMinimumGenerationNonAncient(1);

// events that have been emitted from the orphan buffer
final Collection<Hash> emittedEvents = new HashSet<>();

// handle half of the events. this will result in many events being in the orphan buffer
final int halfEventCount = intakeEvents.size() / 2;
for (int i = 0; i < halfEventCount; i++) {
final GossipEvent intakeEvent = intakeEvents.get(i);

final List<GossipEvent> unorphanedEvents = new ArrayList<>();

unorphanedEvents.addAll(orphanBuffer.handleEvent(intakeEvent));

for (final GossipEvent unorphanedEvent : unorphanedEvents) {
assertValidParents(unorphanedEvent, 1, emittedEvents);
emittedEvents.add(unorphanedEvent.getHashedData().getHash());
}
}

final long eventsExitedPipelineBeforePause = eventsExitedIntakePipeline.get();

orphanBuffer.setPaused(true);
final List<GossipEvent> noEvents = orphanBuffer.setMinimumGenerationNonAncient(maxGeneration);
assertEquals(0, noEvents.size(), "A paused orphan buffer shouldn't emit events upon generation change");
assertEquals(
eventsExitedPipelineBeforePause,
eventsExitedIntakePipeline.get(),
"A paused orphan buffer shouldn't cause any events to exit the intake pipeline upon generation change");
orphanBuffer.setPaused(false);

orphanBuffer.setMinimumGenerationNonAncient(maxGeneration + 1);
assertNotEquals(
eventsExitedPipelineBeforePause,
eventsExitedIntakePipeline.get(),
"An unpaused orphan buffer with large generation change should cause many events to exit the intake pipeline");

assertEquals(halfEventCount, eventsExitedIntakePipeline.get() + emittedEvents.size());
assertEquals(0, orphanBuffer.getCurrentOrphanCount());
}
}

0 comments on commit f2aac3a

Please sign in to comment.