Running Nextflow pipelines on Puhti
Nextflow is a scalable and reproducible scientific workflow management system that interacts with containerized applications to perform compute-intensive tasks. Nextflow provides built-in support for HPC-friendly containers such as Apptainer and Singularity. Although Nextflow pipelines allow us to choose Docker engine as an executor for running pipelines, please note that Docker containers can't be used on Puhti due to the lack of administrative privileges for regular users.
Strengths of Nextflow
- Easy installation
- Supports implicit parallelism
- Can handle complex dependencies and conditional execution
- Handles error recovery
Disadvantages of Nextflow
- Has limited MPI support
- Creates a lot of job steps and excessive I/O load
- Does not efficiently integrate with Slurm scheduler
Use Apptainer/Singularity containers with Nextflow
Containers can be smoothly integrated with Nextflow pipelines. No additional
modifications to Nextflow scripts are needed except enabling the
Singularity/Apptainer engine (instead of Docker) in the Nextflow configuration
file in the HPC environment. Nextflow is able to pull remote container images
stored in Singularity or Docker Hub registry. The remote container images are
usually specified in the Nextflow script or configuration file by simply
prefixing the image name with shub://
or docker://
. It is also possible to
specify a different Singularity image for each process definition in the
Nextflow pipeline script.
Here is a generic recipe for running a Nextflow pipeline on Puhti:
- 1. Login to Puhti supercomputer
- 2. Prepare your Apptainer/Singularity images
- 3. Load Nextflow module on Puhti
- 4. Set-up your Nextflow pipeline environment
- 5. Run your Nextflow pipeline as a batch job
- 6. Demonstration of nf-core Nextflow pipeline using HyperQueue executor (optional)
1. Login to Puhti supercomputer
SSH to the login node of Puhti supercomputer (more instructions here).
ssh <username>@puhti.csc.fi # replace <username> with your CSC username
2. Prepare your Apptainer/Singularity images
Most Nextflow pipelines pull the needed container images on the fly. However, when there are multiple images involved, it is a good idea to prepare the images locally first before launching your Nextflow pipeline.
Here are some options for preparing your Apptainer/Singularity image:
- Build a Singularity/Apptainer image on Puhti without
sudo
access using--fakeroot
flag. - Convert a Docker image to Apptainer on your local system and then copy it to Puhti.
- Convert a Docker image from a container registry on Puhti.
More information on these different options can be found in our documentation on creating containers.
Note
Singularity/Apptainer is installed on login and compute nodes and does not require loading a separate module on either Puhti, Mahti or LUMI.
3. Load Nextflow module on Puhti
Nextflow is available as a module on Puhti and can be loaded for example as below:
module load nextflow/22.10.1
Note
Please make sure to specify the correct version of the Nextflow module as some pipelines require a specific version of Nextflow.
4. Set-up your Nextflow pipeline environment
Running Nextflow pipelines can sometimes be quite compute-intensive and may require downloading large volumes of data such as databases and container images. This can take a while and may not even work successfully for the first time when downloading multiple Apptainer/Singularity images or databases.
You can do the following basic preparation steps before running your Nextflow pipeline:
- Copy Apptainer images from your local workstation to your project folder on
Puhti. Pay attention to the Apptainer cache directory (i.e.
$APPTAINER_CACHEDIR
) which is usually$HOME/.apptainer/cache
. Note that$HOME
directory quota is only 10 GB on Puhti, so it may fill up quickly. - Move all your raw data to your project directory (
/scratch/<project>
) on Puhti. - Clone the GitHub repository of your pipeline to your scratch directory and then run your pipeline.
5. Run your Nextflow pipeline as a batch job
Please follow our instructions for writing a batch job script for Puhti.
Although Nextflow comes with native Slurm support, one has to avoid launching large amounts of very short jobs using it. Instead, one can launch a Nextflow job as a regular batch job that co-executes all job tasks in the same job allocation. Below is a minimal example to get started with your Nextflow pipeline on Puhti:
#!/bin/bash
#SBATCH --time=00:15:00 # Change your runtime settings
#SBATCH --partition=test # Change partition as needed
#SBATCH --account=<project> # Add your project name here
#SBATCH --cpus-per-task=<value> # Change as needed
#SBATCH --mem-per-cpu=1G # Increase as needed
# Load Nextflow module
module load nextflow/22.10.1
# Actual Nextflow command here
nextflow run workflow.nf <options>
Note
If you are directly pulling multiple images on the fly, please set
$APPTAINER_TMPDIR
and $APPTAINER_CACHEDIR
to either local scratch
(i.e. $LOCAL_SCRATCH
) or to your scratch folder (/scratch/<project>
)
in the batch script. Otherwise $HOME
directory, the size of which is
only 10 GB, will be used. To avoid any disk quota errors while pulling
images, set $APPTAINER_TMPDIR
and $APPTAINER_CACHEDIR
in your batch
script as below:
export APPTAINER_TMPDIR=$LOCAL_SCRATCH
export APPTAINER_CACHEDIR=$LOCAL_SCRATCH
Note that this also requires requesting NVMe disk in the batch script by
adding #SBATCH --gres=nvme:<value in GB>
. For example, add
#SBATCH --gres=nvme:100
to request 100 GB of space on $LOCAL_SCRATCH
.
6. Demonstration of nf-core Nextflow pipeline using HyperQueue executor (optional)
In this example, let's use the HyperQueue meta-scheduler for executing a Nextflow pipeline. This executor can be used to scale up analysis across multiple nodes when needed.
Here is a batch script for running a nf-core pipeline on Puhti:
#!/bin/bash
#SBATCH --job-name=nextflowjob
#SBATCH --account=<project>
#SBATCH --partition=small
#SBATCH --time=01:00:00
#SBATCH --nodes=1
#SBATCH --ntasks-per-node=1
#SBATCH --cpus-per-task=40
#SBATCH --mem-per-cpu=2G
module load hyperqueue/0.16.0
module load nextflow/22.10.1
module load git
# create a directory on scratch folder for running nextflow pipeline
export SCRATCH=/scratch/<project>/$USER/nextflow
mkdir -p ${SCRATCH}
cd ${SCRATCH}
export APPTAINER_TMPDIR="${SCRATCH}"
export APPTAINER_CACHEDIR="${SCRATCH}"
unset XDG_RUNTIME_DIR
# Specify a location for the HyperQueue server
export HQ_SERVER_DIR=${PWD}/hq-server-${SLURM_JOB_ID}
mkdir -p "${HQ_SERVER_DIR}"
# Start the server in the background (&) and wait until it has started
hq server start &
until hq job list &>/dev/null ; do sleep 1 ; done
# Start the workers in the background and wait for them to start
srun --overlap --cpu-bind=none --mpi=none hq worker start --cpus=${SLURM_CPUS_PER_TASK} &
hq worker wait "${SLURM_NTASKS}"
# As an example, let's clone a nf-core pipeline and run a test sample
git clone https://github.com/nf-core/rnaseq.git -b 3.10
cd rnaseq
# Ensure Nextflow uses the right executor and knows how much it can submit
echo "executor {
queueSize = $(( 40*SLURM_NNODES ))
name = 'hq'
cpus = $(( 40*SLURM_NNODES ))
}" >> nextflow.config
nextflow run main.nf -profile test,singularity --outdir . -resume
# Wait for all jobs to finish, then shut down the workers and server
hq job wait all
hq worker stop all
hq server stop
Note
Please make sure that your nextflow configuration file (nextflow.config
)
has the correct executor name when using the HypeQueue executor. Also,
when multiple nodes are used, ensure that the executor knows how many jobs
it can submit using the parameter queueSize
under the executor
block.
The queueSize
can be limited as needed. Here is an example snippet that
you can use and modify as needed in your nextflow.config
file:
executor {
queueSize = 40*SLURM_NNODES
name = 'hq'
cpus = 40*SLURM_NNODES
}