Run Buildkite pipelines on a Slurm cluster.
The basic idea is that each Buildkite job is run inside a Slurm job: the Slurm job runs the Buildkite agent with the --acquire-job
option, which ensures that only the specific Buildkite job is scheduled, and is terminated and exits once complete.
Our Slurm cluster is not web-accessible, so we are unable to use webhooks to schedule the Slurm jobs. Instead poll the Buildkite API (via bin/poll.py
) via a cron job running on the cluser login node (bin/cron.sh
at a regular interval (currently every minute). This does the following:
-
Get a list of the Buildkite jobs which are currently queued or running on the cluster via
squeue
. We check this by using a specific job name (buildkite
), and storing the Buildkite job id in the Slurm job comment. -
Query the Buildkite API to get a list of all builds for the organization that are currently scheduled. For each build, and for each job in the build, if the job is not already scheduled in Slurm, then schedule a new job to run
bin/slurmjob.sh
. -
Query the Buildkite API for a list of all builds that are cancelled. For each build, and each job in the build, cancel any Slurm jobs with the matching job id.
Unlike regular Buildkite builds, we don't run each job in an isolated environment, so the checkout only happens on the first job (usually the pipeline upload) and the state is shared between all jobs in the build.
Any options in the agent metadata block which are prefixed with slurm_
are passed to sbatch
: underscores _
are converted to hyphens, and the value can be left blank for options which don't have values. For example
agents:
queue: new-central
slurm_nodes: 1
slurm_tasks_per_node: 2
would pass the options --nodes=1 --tasks-per-node=2
.