Liking cljdoc? Tell your friends :D

{:title "Condition Recipes" :category :cookbook :index 20 :related [["cookbook/generic" "Generic Recipes"] ["conditions" "Conditions"] ["jobs" "Jobs"]]}

Conditions determine whether a job, or multiple jobs, need to be executed in a build or not. These can be static, depending on values provided at trigger time (e.g. which branch we're on), or dynamic, determined by the results of other jobs.

Applying conditions is not possible in YAML or the other static configuration languages. You have to use Clojure for this.

Branch Condition on a Container Job

Execute a container job only when on a certain branch:

(ns build
  (:require [monkey.ci.api :as m]))

(defn terraform-job [ctx]
  (when (= "terraform" (m/branch ctx))
    (-> (m/container-job "terraform")
        (m/image "docker.io/hashicorp/terraform:latest")
        (m/script ["terraform apply"]))))

Condition on Multiple Jobs

Execute a list of jobs when on a given branch. We use a function to generate the list of jobs in this case:

(ns build
  (:require [monkey.ci.api :as m]))

(defn mvn-job
  "Utility function that creates a basic maven container job with 
   given id and script"
  [id script]
  (-> (m/container-job id)
      (m/image "docker.io/maven:latest")
      (m/script script)))

(def unit-test
  (mvn-job "unit-test" ["mvn verify"]))

(def install
  (-> (mvn-job "install" ["mvn install"])
      (m/depends-on "unit-test")))

(def publish
  (-> (mvn-job "publish" ["mvn publish"])
      (m/depends-on "unit-test")))

(defn jobs [ctx]
  [unit-test     ; Always run this
   (when (m/main-branch? ctx)
     ;; Only run these when on main branch
     [install
      publish])])

Parameter Condition

Only run a job if a parameter has a specific value.

(ns build
  (:require [monkey.ci.api :as m]))

(defn param-job [ctx]
  ;; Only run if TEST_VAR=run!
  (when (= "run!" (get (m/params ctx) "TEST_VAR"))
    (-> (m/container-job "some-job")
        (m/image "docker.io/alpine:latest")
        (m/script ["echo 'I am running'"]))))

Run on Changed Files

Run a job if file matching a regex have changed.

(ns build
  (:require [monkey.ci.api :as m]))

(defn unit-test [ctx]
  ;; Run unit tests if any of the files in `java/` dir have changed
  (when (m/touched? ctx #"^java/.*$")
    (-> (m/container-job "unit-tests")
        (m/image "docker.io/maven:latest")
        ;; Run in the java/ dir
        (m/work-dir "java")
        (m/script ["mvn verify"]))))

Regular Expressions

Regular expressions in string matches are possible. For instance, when you only want to release if a tag matches the pattern v[0-9]+.[0-9]+, you can do somethink like this:

(ns build
  (:require [monkey.ci.api :as m]))

(defn release-job [ctx]
  (when (re-matches #"^v\d+\.\d+$" (m/tag ctx))
    (-> (m/container-job "unit-tests")
        (m/image "docker.io/maven:latest")
        (m/script ["mvn release"]))))

Note that with pattern groups you can even parse your tags and act accordingly. See the Java Pattern Documentation for more details on how to use regular expressions.

Composite Conditions

You can extract conditions into reusable functions, and also compose them into new conditions.

(ns build
  (:require [monkey.ci.api :as m]))

(defn release? [ctx]
  (re-matches #"^v\d+\.\d+$" (m/tag ctx)))
  
(def src-changed? (m/touched? #"src/.*"))

;; Run tests when either there is a release, OR the files in src/ have changed.
(def should-test? (some-fn release? src-changed?))

(defn unit-test [ctx]
  (when (should-test? ctx)
    (-> (m/container-job "unit-tests")
        (m/image "docker.io/maven:latest")
        (m/script ["mvn verify"]))))

We're using the built-in some-fn function to run the unit tests when either we're on a release tag, or when the source files have changed.

Can you improve this documentation?Edit on GitHub

cljdoc builds & hosts documentation for Clojure/Script libraries

Keyboard shortcuts
Ctrl+kJump to recent docs
Move to previous article
Move to next article
Ctrl+/Jump to the search field
× close