I want to run a step based on the status of another step. The docs on conditionals state:
Since if conditions are evaluated at the time of the pipeline upload, it’s not
possible to use the if attribute to conditionally run a step based on the result of another step.
To run a step based on the result of another step, upload a new pipeline based on the if condition set up in the command step like in the example below:
steps:
- label: "Validation check"
command: ./scripts/validation_tests.sh
key: "validation-check"
- label: "Run regression only if validation check is passed"
depends_on: "validation-check"
command: |
if [ $$(buildkite-agent step get "outcome" --step "validation-check") == "passed" ]; then
cat <<- YAML | buildkite-agent pipeline upload
steps:
- label: "Run Regression"
command: ./scripts/regression_tests.sh
YAML
fi
This works fine, but is more error-prone. We tried using this but because our IDE was unable to syntax-check the embedded YAML, we committed an error.
Suggestion: could we add something like a if-runtime or if-after-load that essentially does the above under the hood for us? E.g. the following (mind my syntax, may not be 100% correct)
- label: "Run regression only if validation check is passed"
depends_on: "validation-check"
if-runtime: $$(buildkite-agent step get "outcome" --step "validation-check") == "passed"
command: ./scripts/regression_tests.sh
will be converted to/interpreted by Buildkite as:
- label: "Run regression only if validation check is passed"
depends_on: "validation-check"
command: |
if [ $$(buildkite-agent step get "outcome" --step "validation-check") == "passed" ]; then
cat <<- YAML | buildkite-agent pipeline upload
steps:
- label: "Run Regression"
command: ./scripts/regression_tests.sh
YAML
fi
Hey @jaimeb,
Thank you for your suggestions to implement an if-runtime or if-after-load attributes to Buildkite. I will pass your feedback to the product team and we will keep you updated.
Thanks,
Hi @jaimeb ,
I’m from the product team at Buildkite. What you said makes sense and would be much neater. But I’m also wondering why you need to explicitly check the outcome of the step when you’ve already set your second step to depends_on the first step? Because of this, it already will only run when the first step passes anyway. Was there something different about your use case of checking the outcome?
Cheers,
Chris
We have several steps that do some work whenever a specific job fails (send a slack notification, annotate the job, to name a few). We only want to run this step if the step it depends on failed, and otherwise skip it altogether.
Here is an example of one of those jobs:
- label: ":slack: Check if we should notify on deploy failure"
depends_on: "deploy-service"
command: |
if [ $(buildkite-agent step get "outcome" --step "deploy-service") != "passed" ]; then
cat <<- YAML | buildkite-agent pipeline upload
steps:
- label: ":alert: Notify slack about failed deploy step :alert:"
soft_fail: true
command:
- echo "Notifying slack about the deploy-service step"
- exit 1
notify:
- slack:
channels:
- "#deploy-notifications"
message: "deploy-service step has failed"
YAML
fi
We are uploading a new step IFF a step fails. If we could merge these two steps (the uploader and the uploaded), it’d be great.
I totally see where you’re coming from, but unfortunately, we can’t merge the two. The conditionals for steps are evaluated when the pipeline or step is uploaded, so it makes sense to handle the step upload after the first conditional is in place.
Since a new step is created at runtime, there’s no way to merge an upload that’s already been processed with one that’s currently happening.
Is there a specific challenge you’re running into with this setup, or is it more about keeping the YAML shorter and cleaner?
This idea came to mind when debugging an issue with one of our pipelines where a syntax error we introduced into our embedded YAML went unnoticed during review, and would have been caught if our IDE had highlighted it as YAML. So this is more of a nice-to-have to avoid human error
Interesting! Yes, selectively running on failure isn’t handled with depends_on. I get where you’re coming from now.
Being able to specify “Run this job if step key ABC state != passed” would be much more convenient. I can also see it being useful to just have an if statement on the notify to handle the same situation. “Notify if job.state is failed”.
Thanks for sharing another example, I’ll add them to our product feedback so we’ll see them when we review next.