Skip to content

Commit

Permalink
Only rely on next page token for pagination (temporalio#1886)
Browse files Browse the repository at this point in the history
  • Loading branch information
Quinn-With-Two-Ns authored Oct 10, 2023
1 parent b083a70 commit 720a9cb
Show file tree
Hide file tree
Showing 2 changed files with 43 additions and 21 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -72,24 +72,27 @@ class ServiceWorkflowHistoryIterator implements WorkflowHistoryIterator {
}

// Returns true if more history events are available.
// Server can return page tokens that point to empty pages.
// We need to verify that page is valid before returning true.
// Otherwise next() method would throw NoSuchElementException after hasNext() returning true.
@Override
public boolean hasNext() {
if (current.hasNext()) {
return true;
}
if (nextPageToken.isEmpty()) {
return false;
}

GetWorkflowExecutionHistoryResponse response = queryWorkflowExecutionHistory();

current = response.getHistory().getEventsList().iterator();
nextPageToken = response.getNextPageToken();
while (!nextPageToken.isEmpty()) {
// Server can return page tokens that point to empty pages.
// We need to verify that page is valid before returning true.
// Otherwise, next() method would throw NoSuchElementException after hasNext() returning
// true.
GetWorkflowExecutionHistoryResponse response = queryWorkflowExecutionHistory();

return current.hasNext();
current = response.getHistory().getEventsList().iterator();
nextPageToken = response.getNextPageToken();
// Server can return an empty page, but a valid nextPageToken that contains
// more events.
if (current.hasNext()) {
return true;
}
}
return false;
}

@Override
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,19 +32,23 @@
import org.junit.Test;

public class ServiceWorkflowHistoryIteratorTest {

public static final ByteString EMPTY_PAGE_TOKEN =
ByteString.copyFrom("empty page token", Charset.defaultCharset());
public static final ByteString NEXT_PAGE_TOKEN =
ByteString.copyFrom("next token", Charset.defaultCharset());
public static final ByteString EMPTY_HISTORY_PAGE =
ByteString.copyFrom("empty history page token", Charset.defaultCharset());
public static final ByteString NEXT_NEXT_PAGE_TOKEN =
ByteString.copyFrom("next next token", Charset.defaultCharset());
public static final ByteString EMPTY_PAGE_TOKEN =
ByteString.copyFrom("empty page token", Charset.defaultCharset());

/*
This test Scenario verifies following things:
1. hasNext() method makes a call to the server to retrieve workflow history when current
history is empty and history token is available and cached the result.
2. next() method reuses cached history when possible.
3. hasNext() fetches an empty page and return false.
4. next() throws NoSuchElementException when neither history no history token is available.
3. hasNext() keeps fetching as long as the server returns a next page token.
4. hasNext() fetches an empty page and return false.
5. next() throws NoSuchElementException when neither history no history token is available.
*/
@Test
public void verifyHasNextIsFalseWhenHistoryIsEmpty() {
Expand All @@ -58,13 +62,22 @@ public void verifyHasNextIsFalseWhenHistoryIsEmpty() {
GetWorkflowExecutionHistoryResponse queryWorkflowExecutionHistory() {
timesCalledServer.incrementAndGet();
try {
History history = HistoryUtils.generateWorkflowTaskWithInitialHistory().getHistory();
if (EMPTY_PAGE_TOKEN.equals(nextPageToken)) {
return GetWorkflowExecutionHistoryResponse.newBuilder().build();
} else if (EMPTY_HISTORY_PAGE.equals(nextPageToken)) {
return GetWorkflowExecutionHistoryResponse.newBuilder()
.setNextPageToken(NEXT_NEXT_PAGE_TOKEN)
.build();
} else if (NEXT_NEXT_PAGE_TOKEN.equals(nextPageToken)) {
return GetWorkflowExecutionHistoryResponse.newBuilder()
.setHistory(history)
.setNextPageToken(EMPTY_PAGE_TOKEN)
.build();
}
History history = HistoryUtils.generateWorkflowTaskWithInitialHistory().getHistory();
return GetWorkflowExecutionHistoryResponse.newBuilder()
.setHistory(history)
.setNextPageToken(EMPTY_PAGE_TOKEN)
.setNextPageToken(EMPTY_HISTORY_PAGE)
.build();
} catch (Exception e) {
throw new RuntimeException(e);
Expand All @@ -80,9 +93,15 @@ GetWorkflowExecutionHistoryResponse queryWorkflowExecutionHistory() {
Assert.assertTrue(iterator.hasNext());
Assert.assertNotNull(iterator.next());
Assert.assertEquals(1, timesCalledServer.get());
Assert.assertTrue(iterator.hasNext());
Assert.assertEquals(3, timesCalledServer.get());
Assert.assertNotNull(iterator.next());
Assert.assertTrue(iterator.hasNext());
Assert.assertNotNull(iterator.next());
Assert.assertTrue(iterator.hasNext());
Assert.assertNotNull(iterator.next());
Assert.assertFalse(iterator.hasNext());
Assert.assertEquals(2, timesCalledServer.get());
Assert.assertThrows(NoSuchElementException.class, iterator::next);
Assert.assertEquals(2, timesCalledServer.get());
Assert.assertEquals(4, timesCalledServer.get());
}
}

0 comments on commit 720a9cb

Please sign in to comment.