Skip to content

Commit

Permalink
Add warning to ForkScanner Javadoc describing potential issues when u…
Browse files Browse the repository at this point in the history
…sing it with multiple heads
  • Loading branch information
dwnusbaum committed Aug 29, 2023
1 parent d9baddd commit 5df1165
Show file tree
Hide file tree
Showing 2 changed files with 89 additions and 3 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -51,8 +51,22 @@
import java.util.stream.Collectors;

/**
* Scanner that will scan down all forks when we hit parallel blocks before continuing, but generally runs in linear order
* <p>Think of it as the opposite of {@link DepthFirstScanner}.
* Scanner that will scan down all forks when we hit parallel blocks before continuing (as opposed to {@link DepthFirstScanner}), but generally runs in linear order.
* <p><strong>Warning</strong> This scanner has various issues when iterating over incomplete builds, or more generally
* when multiple heads are passed to {@link #setup}, especially if the build contains nested parallelism.
* Consider using {@link DepthFirstScanner} instead in those cases to avoid these issues, employing its iteration order
* as needed to sort results (although its global iteration order is inconsistent, its relative iteration order among
* siblings at the same nesting level (e.g. parallel branches) is consistent and that is good enough for typical use
* cases such as producing a tree).
* In particular, the following known issues are possible when using this scanner with multiple heads:
* <ul>
* <li>Nodes may be visited more than once (mainly with nested parallelism, but also possible with bad timing while a parallel step is initializing heads for its branches).
* <li>Nodes that should be visited may be skipped (only known to happen for nested parallel steps with less than two branches).
* <li>The ordering of branches in parallel steps is inconsistent with respect to the Pipeline script and the order may change as the build progresses.
* <li>When using {@link #visitSimpleChunks}, some chunks may be skipped, and the boundaries of chunks may be incorrect. For example, in some cases, the start node in {@link SimpleChunkVisitor#parallelStart} may not actually be the start node for a parallel step.
* </ul>
*
* <p>For completed builds, or when a single head is passed to {@link #setup}, the above issues should not occur and the following description should be accurate:
*
* <p>This is a fairly efficient way to visit all FlowNodes, and provides four useful guarantees:
* <ul>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -54,10 +54,16 @@
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Objects;
import java.util.Set;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;

import org.junit.Ignore;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.contains;

/**
* Tests for internals of ForkScanner
Expand Down Expand Up @@ -219,7 +225,8 @@ private void sanityTestIterationAndVisiter(List<FlowNode> heads) {
test.assertNoIllegalNullsInEvents();
test.assertNoDupes();
int nodeCount = new DepthFirstScanner().allNodes(heads).size();
Assert.assertEquals(nodeCount,
Assert.assertEquals("ForkScanner should visit the same number of nodes as DepthFirstScanner",
nodeCount,
new ForkScanner().allNodes(heads).size());
test.assertMatchingParallelStartEnd();
test.assertAllNodesGotChunkEvents(new DepthFirstScanner().allNodes(heads));
Expand Down Expand Up @@ -1022,9 +1029,74 @@ public void inProgressParallelInParallel() throws Exception {
SemaphoreStep.waitForStart("innerA/1", b);
ForkScanner scanner = new ForkScanner();
sanityTestIterationAndVisiter(b.getExecution().getCurrentHeads());
assertBranchOrder(b.getExecution(), "innerB", "innerA", "outerC", "outerB", "outerA");
SemaphoreStep.success("outerA/1", null);
SemaphoreStep.success("innerA/1", null);
r.assertBuildStatusSuccess(r.waitForCompletion(b));
sanityTestIterationAndVisiter(b.getExecution().getCurrentHeads());
assertBranchOrder(b.getExecution(), "innerB", "innerA", "outerC", "outerB", "outerA");
}

@Ignore("outerA gets skipped when iterating, see warning in ForkScanner Javadoc")
@Test
public void inProgressParallelInParallelOneNestedBranch() throws Exception {
WorkflowJob p = r.createProject(WorkflowJob.class);
p.setDefinition(new CpsFlowDefinition(
"parallel(\n" + // 3
" 'outerA': {\n" + // 5
" parallel(\n" + // 7
" 'innerA': {\n" + // 8
" semaphore('innerA')\n" + // 10
" }\n" + // 11
" )\n" + // 12
" },\n" + // 13
" 'outerB': {\n" + // 6
" }\n" + // 9
")", true)); // 14
WorkflowRun b = p.scheduleBuild2(0).waitForStart();
SemaphoreStep.waitForStart("innerA/1", b);
sanityTestIterationAndVisiter(b.getExecution().getCurrentHeads());
// TODO: We would like for this order to match the order of the completed build, but that would require an
// alternative fix for the issue described in https://github.com/jenkinsci/workflow-api-plugin/pull/287.
assertBranchOrder(b.getExecution(), "innerA", "outerA", "outerB");
SemaphoreStep.success("innerA/1", null);
r.assertBuildStatusSuccess(r.waitForCompletion(b));
sanityTestIterationAndVisiter(b.getExecution().getCurrentHeads());
assertBranchOrder(b.getExecution(), "outerB", "innerA", "outerA");
}

@Ignore("Actual ordering is all complete branches and then all incomplete branches, see warning in ForkScanner Javadoc")
@Test
public void parallelBranchOrdering() throws Exception {
WorkflowJob p = r.createProject(WorkflowJob.class);
p.setDefinition(new CpsFlowDefinition(
"parallel(\n" +
" '1': { echo '1' },\n" +
" '2': { semaphore('2') },\n" +
" '3': { echo '3' },\n" +
" '4': { semaphore('4') },\n" +
" '5': { echo '5' },\n" +
")", true));
WorkflowRun b = p.scheduleBuild2(0).waitForStart();
SemaphoreStep.waitForStart("2/1", b);
SemaphoreStep.waitForStart("4/1", b);
sanityTestIterationAndVisiter(b.getExecution().getCurrentHeads());
assertBranchOrder(b.getExecution(), "5", "4", "3", "2", "1");
SemaphoreStep.success("2/1", null);
SemaphoreStep.success("4/1", null);
r.assertBuildStatusSuccess(r.waitForCompletion(b));
sanityTestIterationAndVisiter(b.getExecution().getCurrentHeads());
assertBranchOrder(b.getExecution(), "5", "4", "3", "2", "1");
}

private static void assertBranchOrder(FlowExecution execution, String... expectedBranchNames) {
ForkScanner scanner = new ForkScanner();
scanner.setup(execution.getCurrentHeads());
List<String> branches = StreamSupport.stream(scanner.spliterator(), false)
.map(n -> n.getPersistentAction(ThreadNameAction.class))
.filter(Objects::nonNull)
.map(ThreadNameAction::getThreadName)
.collect(Collectors.toList());
assertThat(branches, contains(expectedBranchNames));
}
}

0 comments on commit 5df1165

Please sign in to comment.