This is an example workflow of what steps are needed to produce a singularity image (from a docker image) that works w/ our pipelines written in snakemake.
We have to start with the building of a docker image and for this, a Dockerfile is required that has the recipe for that particular image. I like to separete packages based on their general function in different files, so it's easier to edit them. The building script builds the actual image and pushes it to Docker Hub where it'll be hosted. From there, the image is pulled with singularity to a location on our cluster folder and can be used.
This has to be done on a local machine 'cause on HPC, docker won't be installed for security reasons
docker build -t name_of_the_image .
-t
is the --tag flag and it's the name of the docker image i'm building. .
means ofc the currrent directory, which is where the Dockerfile is. This tells docker to search for a docker file in the current dir.
These are the minimun steps required to push the image to Docker Hub.
- login into the account
docker login
You'll be prompted with username and password of your Docker Hub repo
- Tag the container
docker tag image_name tag/image:anything
This can be done with any tag. Tags are mutable names referred to docker images, they make it easier to pull and run images and also roll out updates automatically.
- Push the tagged container to the repo
docker push tag/image:anything
- To remove the tagged images (if needed)
docker rmi tagged/image
They're actually quite heavy (~ 8gb for ChIPseq and depending on the number of packages in it), so when building multiple images or different versions of them it's worth to keep an eye out on memory space.
- Pull the image to the cluster
singularity pull name_of_image.sif docker://username/image:tag
From any directory, pull the docker image. The command pull
takes a docker image and create a copy as a singularity image. .sif
is the extension of singularity images name_of_image
can be anything and can be != from the docker image name. docker://
tells singularity that we're downloading a docker image, otherwise i'll search for it on singularity repos.
- Notes on the workflow
This general workflow creates a fully functional image, however i'd advise some little changes here and there to make it quicker, more automated and less prone to errors.
- Building script
Create a bash script that builds the image, it'll be called like any other script with ./script.sh
. In this script, first we create a variable IMAGE_VERSION
that contains the day when the image is created, the building function with a couple of options, the tagging step that's skipped 'cause we add the username to the image directly here, the redirection of STDOUT and ERR to a file that can be inspected to troubleshoot the build and finally we push the newly created image to the repo. The script looks like this.
IMAGE_VERSION=$(date "+%m%d%y") # this sets the version
docker build --progress=plain --no-cache -t username/image:${IMAGE_VERSION} -t username/image:latest . 2>&1 | tee stdout.log && \
docker push username/image:${IMAGE_VERSION}
--progress=plain
it's similar to a verbose argument, but in reality extends the STDOUT that was shrinked with buildkit. Tho it creates a wave of text, i think it'll be easier to troubleshoot later on.
--no-cache
doesn't cache the build. I'd advise to use it only when building the final image or when conflicts between dependencies are spotted, se we can move around stuff and fix it. One reason to use this is if a particular/large package has dependecies required for other packages in the build and this is the limiting/painful step (i.e. when buiding this image i had trouble with spp
package from Cran that required Rsamtools
as dependency but for some reason wasn't able to download it. My workaround was to first download Bioconductor package ChIPseeker
that had the same dependency, so spp
could use it and was properly installed).
tee
allows to both write to a file (STDOUT and ERR 2>&1
) and to standard output.
This is the script i'd use when building the final image. In the same file there's a different version of the same script and it's the version i usually use when i'm first testing out that all packages/dependencies in the build have 0 conflicts.
- Dockerfile
The only difference is that I write packages in different files based on either function or repo used to download them. Then, w/ COPY
I pull them into the build. I also like to move those file in a new directory to be as tidy as possible.
- This is probably the smartest solution, aka opening a container and working directly in it to debug softwares incompatibilities. It's useful for testing if certian R/other packages can be installed succesfully. This will immediatly point out missing libs and other dependencies.