diff --git a/docs/job-traces.md b/docs/job-traces.md
index 5d95d49ef..5fddb27e8 100644
--- a/docs/job-traces.md
+++ b/docs/job-traces.md
@@ -75,6 +75,7 @@ Attributes reported on the span of pipeline steps:
| jenkins.pipeline.step.name | Step name (user friendly) | String |
| jenkins.pipeline.step.type | Step name | String |
| jenkins.pipeline.step.id | Step id | String |
+| jenkins.pipeline.step.result | Step result | Enum (`ABORTED`, `FAILURE`, `NOT_EXECUTED`, `PAUSED_PENDING_INPUT`, `QUEUED`, `SUCCESS`, `UNSTABLE`; see [GenericStatus](https://javadoc.jenkins.io/plugin/pipeline-graph-analysis/org/jenkinsci/plugins/workflow/pipelinegraphanalysis/GenericStatus.html)) |
| jenkins.pipeline.step.plugin.name | Jenkins plugin for that particular step | String |
| jenkins.pipeline.step.plugin.version| Jenkins plugin version | String |
| jenkins.pipeline.step.agent.label | Labels attached to the agent | String |
diff --git a/pom.xml b/pom.xml
index a9a7c8297..b03f75f2d 100644
--- a/pom.xml
+++ b/pom.xml
@@ -255,6 +255,10 @@
org.jenkins-ci.plugins
pipeline-stage-step
+
+ org.jenkins-ci.plugins
+ pipeline-graph-analysis
+
org.jenkins-ci.plugins
scm-api
diff --git a/src/main/java/io/jenkins/plugins/opentelemetry/job/MonitoringPipelineListener.java b/src/main/java/io/jenkins/plugins/opentelemetry/job/MonitoringPipelineListener.java
index 9070badf1..7b5b0f54c 100644
--- a/src/main/java/io/jenkins/plugins/opentelemetry/job/MonitoringPipelineListener.java
+++ b/src/main/java/io/jenkins/plugins/opentelemetry/job/MonitoringPipelineListener.java
@@ -46,6 +46,8 @@
import org.jenkinsci.plugins.workflow.flow.StepListener;
import org.jenkinsci.plugins.workflow.graph.FlowNode;
import org.jenkinsci.plugins.workflow.job.WorkflowRun;
+import org.jenkinsci.plugins.workflow.pipelinegraphanalysis.GenericStatus;
+import org.jenkinsci.plugins.workflow.pipelinegraphanalysis.StatusAndTiming;
import org.jenkinsci.plugins.workflow.steps.*;
import edu.umd.cs.findbugs.annotations.NonNull;
@@ -129,7 +131,7 @@ public void onStartNodeStep(@NonNull StepStartNode stepStartNode, @Nullable Stri
@Override
public void onAfterStartNodeStep(@NonNull StepStartNode stepStartNode, @Nullable String nodeLabel, @NonNull WorkflowRun run) {
// end the JenkinsOtelSemanticAttributes.AGENT_ALLOCATE span
- endCurrentSpan(stepStartNode, run);
+ endCurrentSpan(stepStartNode, run, null);
}
@Override
@@ -156,13 +158,17 @@ public void onStartStageStep(@NonNull StepStartNode stepStartNode, @NonNull Stri
}
@Override
- public void onEndNodeStep(@NonNull StepEndNode node, @NonNull String nodeName, @NonNull WorkflowRun run) {
- endCurrentSpan(node, run);
+ public void onEndNodeStep(@NonNull StepEndNode node, @NonNull String nodeName, FlowNode nextNode, @NonNull WorkflowRun run) {
+ StepStartNode nodeStartNode = node.getStartNode();
+ GenericStatus nodeStatus = StatusAndTiming.computeChunkStatus2(run, null, nodeStartNode, node, nextNode);
+ endCurrentSpan(node, run, nodeStatus);
}
@Override
- public void onEndStageStep(@NonNull StepEndNode node, @NonNull String stageName, @NonNull WorkflowRun run) {
- endCurrentSpan(node, run);
+ public void onEndStageStep(@NonNull StepEndNode node, @NonNull String stageName, FlowNode nextNode, @NonNull WorkflowRun run) {
+ StepStartNode stageStartNode = node.getStartNode();
+ GenericStatus stageStatus = StatusAndTiming.computeChunkStatus2(run, null, stageStartNode, node, nextNode);
+ endCurrentSpan(node, run, stageStatus);
}
protected List getStepHandlers() {
@@ -212,12 +218,13 @@ public void onAtomicStep(@NonNull StepAtomNode node, @NonNull WorkflowRun run) {
}
@Override
- public void onAfterAtomicStep(@NonNull StepAtomNode node, @NonNull WorkflowRun run) {
+ public void onAfterAtomicStep(@NonNull StepAtomNode node, FlowNode nextNode, @NonNull WorkflowRun run) {
if (isIgnoredStep(node.getDescriptor())){
LOGGER.log(Level.FINE, () -> run.getFullDisplayName() + " - don't end span for step '" + node.getDisplayFunctionName() + "'");
return;
}
- endCurrentSpan(node, run);
+ GenericStatus stageStatus = StatusAndTiming.computeChunkStatus2(run, null, node, node, nextNode);
+ endCurrentSpan(node, run, stageStatus);
}
private boolean isIgnoredStep(@Nullable StepDescriptor stepDescriptor) {
@@ -289,17 +296,21 @@ public void onStartParallelStepBranch(@NonNull StepStartNode stepStartNode, @Non
}
@Override
- public void onEndParallelStepBranch(@NonNull StepEndNode node, @NonNull String branchName, @NonNull WorkflowRun run) {
- endCurrentSpan(node, run);
+ public void onEndParallelStepBranch(@NonNull StepEndNode node, @NonNull String branchName, FlowNode nextNode, @NonNull WorkflowRun run) {
+ StepStartNode parallelStartNode = node.getStartNode();
+ GenericStatus parallelStatus = StatusAndTiming.computeChunkStatus2(run, null, parallelStartNode, node, nextNode);
+ endCurrentSpan(node, run, parallelStatus);
}
- private void endCurrentSpan(FlowNode node, WorkflowRun run) {
+ private void endCurrentSpan(FlowNode node, WorkflowRun run, GenericStatus status) {
try (Scope ignored = setupContext(run, node)) {
verifyNotNull(ignored, "%s - No span found for node %s", run, node);
Span span = getTracerService().getSpan(run, node);
+
ErrorAction errorAction = node.getError();
if (errorAction == null) {
+ if (status == null) status = GenericStatus.SUCCESS;
span.setStatus(StatusCode.OK);
} else {
Throwable throwable = errorAction.getError();
@@ -307,6 +318,8 @@ private void endCurrentSpan(FlowNode node, WorkflowRun run) {
FlowInterruptedException interruptedException = (FlowInterruptedException) throwable;
List causesOfInterruption = interruptedException.getCauses();
+ if (status == null) status = GenericStatus.fromResult(interruptedException.getResult());
+
List causeDescriptions = causesOfInterruption.stream().map(cause -> cause.getClass().getSimpleName() + ": " + cause.getShortDescription()).collect(Collectors.toList());
span.setAttribute(JenkinsOtelSemanticAttributes.JENKINS_STEP_INTERRUPTION_CAUSES, causeDescriptions);
@@ -326,10 +339,17 @@ private void endCurrentSpan(FlowNode node, WorkflowRun run) {
span.setStatus(StatusCode.ERROR, statusDescription);
}
} else {
+ if (status == null) status = GenericStatus.FAILURE;
span.recordException(throwable);
span.setStatus(StatusCode.ERROR, throwable.getMessage());
}
}
+
+ if (status != null) {
+ status = StatusAndTiming.coerceStatusApi(status, StatusAndTiming.API_V2);
+ span.setAttribute(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT, status.toString());
+ }
+
span.end();
LOGGER.log(Level.FINE, () -> run.getFullDisplayName() + " - < " + node.getDisplayFunctionName() + " - end " + OtelUtils.toDebugString(span));
diff --git a/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/AbstractPipelineListener.java b/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/AbstractPipelineListener.java
index 0d6bcfa63..1bb89d91b 100644
--- a/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/AbstractPipelineListener.java
+++ b/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/AbstractPipelineListener.java
@@ -36,12 +36,12 @@ public void onAfterStartNodeStep(@NonNull StepStartNode stepStartNode, @Nullable
}
@Override
- public void onEndNodeStep(@NonNull StepEndNode nodeStepEndNode, @NonNull String nodeName, @NonNull WorkflowRun run) {
+ public void onEndNodeStep(@NonNull StepEndNode nodeStepEndNode, @NonNull String nodeName, FlowNode nextNode, @NonNull WorkflowRun run) {
}
@Override
- public void onEndStageStep(@NonNull StepEndNode stageStepEndNode, @NonNull String stageName, @NonNull WorkflowRun run) {
+ public void onEndStageStep(@NonNull StepEndNode stageStepEndNode, @NonNull String stageName, FlowNode nextNode, @NonNull WorkflowRun run) {
}
@@ -51,7 +51,7 @@ public void onAtomicStep(@NonNull StepAtomNode node, @NonNull WorkflowRun run) {
}
@Override
- public void onAfterAtomicStep(@NonNull StepAtomNode stepAtomNode, @NonNull WorkflowRun run) {
+ public void onAfterAtomicStep(@NonNull StepAtomNode stepAtomNode, FlowNode nextNode, @NonNull WorkflowRun run) {
}
@@ -61,7 +61,7 @@ public void onStartParallelStepBranch(@NonNull StepStartNode stepStartNode, @Non
}
@Override
- public void onEndParallelStepBranch(@NonNull StepEndNode stepStepNode, @NonNull String branchName, @NonNull WorkflowRun run) {
+ public void onEndParallelStepBranch(@NonNull StepEndNode stepStepNode, @NonNull String branchName, FlowNode nextNode, @NonNull WorkflowRun run) {
}
diff --git a/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/GraphListenerAdapterToPipelineListener.java b/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/GraphListenerAdapterToPipelineListener.java
index 6bb42a6dd..ed400dbcb 100644
--- a/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/GraphListenerAdapterToPipelineListener.java
+++ b/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/GraphListenerAdapterToPipelineListener.java
@@ -55,18 +55,18 @@ private void processPreviousNodes(FlowNode node, WorkflowRun run) {
log(Level.FINE, () -> run.getFullDisplayName() + " - Process previous node " + PipelineNodeUtil.getDetailedDebugString(previousNode) + " of node " + PipelineNodeUtil.getDetailedDebugString(node));
if (previousNode instanceof StepAtomNode) {
StepAtomNode stepAtomNode = (StepAtomNode) previousNode;
- fireOnAfterAtomicStep(stepAtomNode, run);
+ fireOnAfterAtomicStep(stepAtomNode, node, run);
} else if (isBeforeEndExecutorNodeStep(previousNode)) {
String nodeName = PipelineNodeUtil.getDisplayName(((StepEndNode) previousNode).getStartNode());
- fireOnAfterEndNodeStep((StepEndNode) previousNode, nodeName, run);
+ fireOnAfterEndNodeStep((StepEndNode) previousNode, nodeName, node, run);
} else if (isBeforeEndStageStep(previousNode)) {
String stageName = PipelineNodeUtil.getDisplayName(((StepEndNode) previousNode).getStartNode());
- fireOnAfterEndStageStep((StepEndNode) previousNode, stageName, run);
+ fireOnAfterEndStageStep((StepEndNode) previousNode, stageName, node, run);
} else if (isBeforeEndParallelBranch(previousNode)) {
StepEndNode endParallelBranchNode = (StepEndNode) previousNode;
StepStartNode beginParallelBranch = endParallelBranchNode.getStartNode();
ThreadNameAction persistentAction = verifyNotNull(beginParallelBranch.getPersistentAction(ThreadNameAction.class), "Null ThreadNameAction on %s", beginParallelBranch);
- fireOnAfterEndParallelStepBranch(endParallelBranchNode, persistentAction.getThreadName(), run);
+ fireOnAfterEndParallelStepBranch(endParallelBranchNode, persistentAction.getThreadName(), node, run);
} else {
log(Level.FINE, () -> "Ignore previous node " + PipelineNodeUtil.getDetailedDebugString(previousNode));
}
@@ -129,11 +129,11 @@ private void fireOnBeforeStartParallelStepBranch(@NonNull StepStartNode node, @N
}
}
- private void fireOnAfterEndParallelStepBranch(@NonNull StepEndNode node, @NonNull String branchName, @NonNull WorkflowRun run) {
+ private void fireOnAfterEndParallelStepBranch(@NonNull StepEndNode node, @NonNull String branchName, FlowNode nextNode, @NonNull WorkflowRun run) {
for (PipelineListener pipelineListener : PipelineListener.all()) {
log(Level.FINE, () -> "onAfterEndParallelStepBranch(branchName: " + branchName + ", node[name:" + node.getDisplayName() + ", id: " + node.getId() + "]): " + pipelineListener.toString());
try {
- pipelineListener.onEndParallelStepBranch(node, branchName, run);
+ pipelineListener.onEndParallelStepBranch(node, branchName, nextNode, run);
} catch (RuntimeException e) {
LOGGER.log(Level.WARNING, e, () -> "Exception invoking `onAfterEndParallelStepBranch` on " + pipelineListener);
}
@@ -157,11 +157,11 @@ private void logFlowNodeDetails(@NonNull FlowNode node, @NonNull WorkflowRun run
});
}
- public void fireOnAfterAtomicStep(@NonNull StepAtomNode stepAtomNode, @NonNull WorkflowRun run) {
+ public void fireOnAfterAtomicStep(@NonNull StepAtomNode stepAtomNode, FlowNode nextNode, @NonNull WorkflowRun run) {
for (PipelineListener pipelineListener : PipelineListener.all()) {
log(() -> "onAfterAtomicStep(" + stepAtomNode.getDisplayName() + "): " + pipelineListener.toString());
try {
- pipelineListener.onAfterAtomicStep(stepAtomNode, run);
+ pipelineListener.onAfterAtomicStep(stepAtomNode, nextNode, run);
} catch (RuntimeException e) {
LOGGER.log(Level.WARNING, e, () -> "Exception invoking `onAfterAtomicStep` on " + pipelineListener);
}
@@ -190,22 +190,22 @@ public void fireOnStartPipeline(@NonNull FlowStartNode node, @NonNull WorkflowRu
}
}
- public void fireOnAfterEndNodeStep(@NonNull StepEndNode node, @NonNull String nodeName, @NonNull WorkflowRun run) {
+ public void fireOnAfterEndNodeStep(@NonNull StepEndNode node, @NonNull String nodeName, FlowNode nextNode, @NonNull WorkflowRun run) {
for (PipelineListener pipelineListener : PipelineListener.all()) {
log(() -> "onAfterEndNodeStep(" + node.getDisplayName() + "): " + pipelineListener.toString() + (node.getError() != null ? ("error: " + node.getError().getError()) : ""));
try {
- pipelineListener.onEndNodeStep(node, nodeName, run);
+ pipelineListener.onEndNodeStep(node, nodeName, nextNode, run);
} catch (RuntimeException e) {
LOGGER.log(Level.WARNING, e, () -> "Exception invoking `onAfterEndNodeStep` on " + pipelineListener);
}
}
}
- public void fireOnAfterEndStageStep(@NonNull StepEndNode node, @NonNull String stageName, @NonNull WorkflowRun run) {
+ public void fireOnAfterEndStageStep(@NonNull StepEndNode node, @NonNull String stageName, FlowNode nextNode, @NonNull WorkflowRun run) {
for (PipelineListener pipelineListener : PipelineListener.all()) {
log(() -> "onAfterEndStageStep(" + node.getDisplayName() + "): " + pipelineListener.toString() + (node.getError() != null ? ("error: " + node.getError().getError()) : ""));
try {
- pipelineListener.onEndStageStep(node, stageName, run);
+ pipelineListener.onEndStageStep(node, stageName, nextNode, run);
} catch (RuntimeException e) {
LOGGER.log(Level.WARNING, e, () -> "Exception invoking `onAfterEndStageStep` on " + pipelineListener);
}
diff --git a/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/PipelineListener.java b/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/PipelineListener.java
index 5ba4e4863..ea8175011 100644
--- a/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/PipelineListener.java
+++ b/src/main/java/io/jenkins/plugins/opentelemetry/job/jenkins/PipelineListener.java
@@ -41,7 +41,7 @@ static List all() {
/**
* Just after the `node` step ends
*/
- void onEndNodeStep(@NonNull StepEndNode nodeStepEndNode, @NonNull String nodeName, @NonNull WorkflowRun run);
+ void onEndNodeStep(@NonNull StepEndNode nodeStepEndNode, @NonNull String nodeName, FlowNode nextNode, @NonNull WorkflowRun run);
/**
* Just before the `stage`step starts
@@ -51,7 +51,7 @@ static List all() {
/**
* Just after the `stage` step ends
*/
- void onEndStageStep(@NonNull StepEndNode stageStepEndNode, @NonNull String stageName, @NonNull WorkflowRun run);
+ void onEndStageStep(@NonNull StepEndNode stageStepEndNode, @NonNull String stageName, FlowNode nextNode, @NonNull WorkflowRun run);
/**
* Just before the `parallel` branch starts
@@ -61,7 +61,7 @@ static List all() {
/**
* Just before the `parallel` branch ends
*/
- void onEndParallelStepBranch(@NonNull StepEndNode stepStepNode, @NonNull String branchName, @NonNull WorkflowRun run);
+ void onEndParallelStepBranch(@NonNull StepEndNode stepStepNode, @NonNull String branchName, FlowNode nextNode, @NonNull WorkflowRun run);
/**
* Just before the atomic step starts
@@ -71,7 +71,7 @@ static List all() {
/**
* Just after the atomic step
*/
- void onAfterAtomicStep(@NonNull StepAtomNode stepAtomNode, @NonNull WorkflowRun run);
+ void onAfterAtomicStep(@NonNull StepAtomNode stepAtomNode, FlowNode nextNode, @NonNull WorkflowRun run);
/**
* Just after the pipeline ends
diff --git a/src/main/java/io/jenkins/plugins/opentelemetry/semconv/JenkinsOtelSemanticAttributes.java b/src/main/java/io/jenkins/plugins/opentelemetry/semconv/JenkinsOtelSemanticAttributes.java
index f8d770962..c1950a0dd 100644
--- a/src/main/java/io/jenkins/plugins/opentelemetry/semconv/JenkinsOtelSemanticAttributes.java
+++ b/src/main/java/io/jenkins/plugins/opentelemetry/semconv/JenkinsOtelSemanticAttributes.java
@@ -75,6 +75,11 @@ public final class JenkinsOtelSemanticAttributes {
* @see org.jenkinsci.plugins.workflow.graph.FlowNode#getId()
*/
public static final AttributeKey JENKINS_STEP_ID = AttributeKey.stringKey("jenkins.pipeline.step.id");
+ /**
+ * @see org.jenkinsci.plugins.workflow.pipelinegraphanalysis.GenericStatus
+ * @see org.jenkinsci.plugins.workflow.pipelinegraphanalysis.StatusAndTiming#computeChunkStatus2(org.jenkinsci.plugins.workflow.job.WorkflowRun,org.jenkinsci.plugins.workflow.graph.FlowNode,org.jenkinsci.plugins.workflow.graph.FlowNode,org.jenkinsci.plugins.workflow.graph.FlowNode,org.jenkinsci.plugins.workflow.graph.FlowNode)
+ */
+ public static final AttributeKey JENKINS_STEP_RESULT = AttributeKey.stringKey("jenkins.pipeline.step.result");
/**
* @see PluginWrapper#getShortName()
*/
@@ -97,7 +102,7 @@ public final class JenkinsOtelSemanticAttributes {
public static final String JENKINS = "jenkins";
/**
- * As {@link Jenkins.MasterComputer#getName()} return "", choose another name
+ * As {@link Jenkins.MasterComputer#getName()} returns "", choose another name
*
* @see Jenkins.MasterComputer#getName()
*/
diff --git a/src/test/java/io/jenkins/plugins/opentelemetry/job/step/StepResultTest.java b/src/test/java/io/jenkins/plugins/opentelemetry/job/step/StepResultTest.java
new file mode 100644
index 000000000..74835bd33
--- /dev/null
+++ b/src/test/java/io/jenkins/plugins/opentelemetry/job/step/StepResultTest.java
@@ -0,0 +1,139 @@
+/*
+ * Copyright The Original Author or Authors
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+package io.jenkins.plugins.opentelemetry.job.step;
+
+import com.github.rutledgepaulv.prune.Tree;
+import hudson.model.Node;
+import hudson.model.Result;
+import io.jenkins.plugins.opentelemetry.BaseIntegrationTest;
+import io.jenkins.plugins.opentelemetry.semconv.JenkinsOtelSemanticAttributes;
+import io.opentelemetry.api.common.AttributeKey;
+import io.opentelemetry.sdk.trace.data.SpanData;
+import org.apache.commons.lang3.SystemUtils;
+import org.hamcrest.CoreMatchers;
+import org.hamcrest.MatcherAssert;
+import org.jenkinsci.plugins.workflow.cps.CpsFlowDefinition;
+import org.jenkinsci.plugins.workflow.job.WorkflowJob;
+import org.jenkinsci.plugins.workflow.job.WorkflowRun;
+import org.junit.Test;
+
+import static org.junit.Assume.assumeFalse;
+
+public class StepResultTest extends BaseIntegrationTest {
+
+ @Test
+ public void testSimplePipelineWithWithStepResults() throws Exception {
+ assumeFalse(SystemUtils.IS_OS_WINDOWS);
+ // BEFORE
+
+ String pipelineScript = "def xsh(cmd) {if (isUnix()) {sh cmd} else {bat cmd}};\n" +
+ "node() {\n" +
+ " stage('build') {\n" +
+ " unstable('stage unstable')\n" +
+ " }\n" +
+ " stage('parallel') {\n" +
+ " catchError(stageResult: 'UNSTABLE') { // otherwise, the timeout stage would never run\n" +
+ " parallel (\n" +
+ " first: { xsh (label: 'parallel-first', script: 'exit 1') },\n" +
+ " second: { xsh (label: 'parallel-second', script: 'exit 0') },\n" +
+ " )\n" +
+ " }\n" +
+ " }\n" +
+ " stage('skipped') {\n" +
+ " org.jenkinsci.plugins.pipeline.modeldefinition.Utils.markStageSkippedForConditional('skipped');\n" +
+ " }\n" +
+ " stage('timeout') {\n" +
+ " timeout(time: 1, unit: 'MILLISECONDS') {\n" +
+ " xsh (label: 'sleep', script: 'sleep 1')\n" +
+ " }\n" +
+ " }\n" +
+ "}";
+ final Node agent = jenkinsRule.createOnlineSlave();
+
+ final String jobName = "test-simple-pipeline-with-step-results" + jobNameSuffix.incrementAndGet();
+ WorkflowJob pipeline = jenkinsRule.createProject(WorkflowJob.class, jobName);
+ pipeline.setDefinition(new CpsFlowDefinition(pipelineScript, true));
+ WorkflowRun build = jenkinsRule.assertBuildStatus(Result.ABORTED, pipeline.scheduleBuild2(0));
+
+ String rootSpanName = JenkinsOtelSemanticAttributes.CI_PIPELINE_RUN_ROOT_SPAN_NAME_PREFIX + jobName;
+
+ final Tree spans = getGeneratedSpans();
+
+ checkChainOfSpans(spans, "Phase: Start", rootSpanName);
+ checkChainOfSpans(spans, JenkinsOtelSemanticAttributes.AGENT_ALLOCATION_UI, JenkinsOtelSemanticAttributes.AGENT_UI, "Phase: Run", rootSpanName);
+ checkChainOfSpans(spans, "unstable", "Stage: build", JenkinsOtelSemanticAttributes.AGENT_UI, "Phase: Run", rootSpanName);
+ checkChainOfSpans(spans, "parallel-first", "Parallel branch: first", "Stage: parallel", JenkinsOtelSemanticAttributes.AGENT_UI, "Phase: Run", rootSpanName);
+ checkChainOfSpans(spans, "parallel-second", "Parallel branch: second", "Stage: parallel", JenkinsOtelSemanticAttributes.AGENT_UI, "Phase: Run", rootSpanName);
+ checkChainOfSpans(spans, "Stage: skipped", JenkinsOtelSemanticAttributes.AGENT_UI, "Phase: Run", rootSpanName);
+ checkChainOfSpans(spans, "Stage: timeout", JenkinsOtelSemanticAttributes.AGENT_UI, "Phase: Run", rootSpanName);
+ checkChainOfSpans(spans, "Phase: Finalise", rootSpanName);
+
+ // Note: pipeline root span is not a step/stage, so it does not get a JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT attribute (just like it doesn't get any JenkinsOtelSemanticAttributes at all at the moment)
+ // Neither are any of the 3 "Phase" spans.
+
+ { // node span: 'ABORTED' (because timeout stage is aborted)
+ SpanData actualSpanData = spans.breadthFirstStream().filter(sdw -> JenkinsOtelSemanticAttributes.AGENT_UI.equals(sdw.spanData.getName())).findFirst().get().spanData;
+ String actualStageResult = actualSpanData.getAttributes().get(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT);
+ MatcherAssert.assertThat(actualStageResult, CoreMatchers.is("ABORTED"));
+ }
+
+ { // node allocation span: 'SUCCESS'
+ SpanData actualSpanData = spans.breadthFirstStream().filter(sdw -> JenkinsOtelSemanticAttributes.AGENT_ALLOCATION_UI.equals(sdw.spanData.getName())).findFirst().get().spanData;
+ String actualStageResult = actualSpanData.getAttributes().get(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT);
+ MatcherAssert.assertThat(actualStageResult, CoreMatchers.is("SUCCESS"));
+ }
+
+ { // stage 'build': 'UNSTABLE'
+ SpanData actualSpanData = spans.breadthFirstStream().filter(sdw -> "Stage: build".equals(sdw.spanData.getName())).findFirst().get().spanData;
+ String actualStageResult = actualSpanData.getAttributes().get(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT);
+ MatcherAssert.assertThat(actualStageResult, CoreMatchers.is("UNSTABLE"));
+ }
+
+ { // stage 'parallel': 'UNSTABLE' (because catchError caught parallel-first's FAILURE and set stageResult to UNSTABLE)
+ SpanData actualSpanData = spans.breadthFirstStream().filter(sdw -> "Stage: parallel".equals(sdw.spanData.getName())).findFirst().get().spanData;
+ String actualStageResult = actualSpanData.getAttributes().get(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT);
+ MatcherAssert.assertThat(actualStageResult, CoreMatchers.is("UNSTABLE"));
+ }
+
+ { // parallel node 'first': 'FAILURE'
+ SpanData actualSpanData = spans.breadthFirstStream().filter(sdw -> "Parallel branch: first".equals(sdw.spanData.getName())).findFirst().get().spanData;
+ String actualStepResult = actualSpanData.getAttributes().get(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT);
+ MatcherAssert.assertThat(actualStepResult, CoreMatchers.is("FAILURE"));
+ }
+
+ { // xsh node 'parallel-first': 'FAILURE'
+ SpanData actualSpanData = spans.breadthFirstStream().filter(sdw -> "parallel-first".equals(sdw.spanData.getName())).findFirst().get().spanData;
+ String actualStepResult = actualSpanData.getAttributes().get(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT);
+ MatcherAssert.assertThat(actualStepResult, CoreMatchers.is("FAILURE"));
+ }
+
+ { // parallel node 'second': 'SUCCESS'
+ SpanData actualSpanData = spans.breadthFirstStream().filter(sdw -> "Parallel branch: second".equals(sdw.spanData.getName())).findFirst().get().spanData;
+ String actualStepResult = actualSpanData.getAttributes().get(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT);
+ MatcherAssert.assertThat(actualStepResult, CoreMatchers.is("SUCCESS"));
+ }
+
+ { // xsh node 'parallel-second': 'SUCCESS'
+ SpanData actualSpanData = spans.breadthFirstStream().filter(sdw -> "parallel-second".equals(sdw.spanData.getName())).findFirst().get().spanData;
+ String actualStepResult = actualSpanData.getAttributes().get(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT);
+ MatcherAssert.assertThat(actualStepResult, CoreMatchers.is("SUCCESS"));
+ }
+
+ { // stage 'skipped': 'NOT_EXECUTED'
+ SpanData actualSpanData = spans.breadthFirstStream().filter(sdw -> "Stage: skipped".equals(sdw.spanData.getName())).findFirst().get().spanData;
+ String actualStageResult = actualSpanData.getAttributes().get(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT);
+ MatcherAssert.assertThat(actualStageResult, CoreMatchers.is("NOT_EXECUTED"));
+ }
+
+ { // stage 'timeout': 'ABORTED'
+ SpanData actualSpanData = spans.breadthFirstStream().filter(sdw -> "Stage: timeout".equals(sdw.spanData.getName())).findFirst().get().spanData;
+ String actualStageResult = actualSpanData.getAttributes().get(JenkinsOtelSemanticAttributes.JENKINS_STEP_RESULT);
+ MatcherAssert.assertThat(actualStageResult, CoreMatchers.is("ABORTED"));
+ }
+
+ MatcherAssert.assertThat(spans.cardinality(), CoreMatchers.is(15L));
+ }
+}
\ No newline at end of file