To build the initial container visit Syslabs and use the remote build option. The def file used to build the environment is provided below, so copy the contents into the builder and start the build.
Known bugs: persistent overlay is not integrated into the .sif
and you must run an interactive R
session using a sandbox
BootStrap: docker
From: ubuntu:20.04
%labels
Maintainer Dylan Ammons
R_Version 4.3.2
%apprun R
exec R "${@}"
%apprun Rscript
exec Rscript "${@}"
%runscript
exec R "${@}"
%post
# Software versions
export R_VERSION=4.3.2
echo "export R_VERSION=${R_VERSION}" >> $SINGULARITY_ENVIRONMENT
# Get dependencies
apt-get update
apt-get install -y --no-install-recommends \
locales
# Configure default locale
echo "en_US.UTF-8 UTF-8" >> /etc/locale.gen
locale-gen en_US.utf8
/usr/sbin/update-locale LANG=en_US.UTF-8
export LC_ALL=en_US.UTF-8
export LANG=en_US.UTF-8
# Install R
apt-get update
apt-get install -y --no-install-recommends \
software-properties-common \
dirmngr \
wget
wget -qO- https://cloud.r-project.org/bin/linux/ubuntu/marutter_pubkey.asc | \
tee -a /etc/apt/trusted.gpg.d/cran_ubuntu_key.asc
add-apt-repository \
"deb https://cloud.r-project.org/bin/linux/ubuntu $(lsb_release -cs)-cran40/"
apt-get install -y --no-install-recommends \
libharfbuzz-dev \
libfribidi-dev \
r-base-dev=${R_VERSION}* \
r-cran-lattice \
r-cran-mgcv \
r-cran-nlme \
r-cran-survival \
r-cran-mass \
r-cran-class \
r-cran-nnet \
r-cran-matrix \
r-recommended=${R_VERSION}* \
r-base=${R_VERSION}* \
r-base-core=${R_VERSION}* \
libcurl4-openssl-dev \
libssl-dev \
libxml2-dev \
libcairo2-dev \
libxt-dev \
libopenblas-dev \
gcc \
g++ \
libfreetype6-dev \
libglib2.0-dev \
libpng-dev \
libtiff5-dev \
libjpeg-dev \
libhdf5-dev \
cmake
# Add a default CRAN mirror
echo "options(repos = c(CRAN = 'https://cran.rstudio.com/'), download.file.method = 'libcurl')" >> /usr/lib/R/etc/Rprofile.site
# Add a directory to host R libraries
mkdir -p /library
echo "R_LIBS_SITE=/library:\${R_LIBS_SITE}" >> /usr/lib/R/etc/Renviron.site
# Clean up
rm -rf /var/lib/apt/lists/*
%test
R --quiet -e "stopifnot(getRversion() == '${R_VERSION}')"
To retrieve the new base container, navigate to the repository and find the singularity pull
command that indicates how to retrieve the container.
#should look like this - do not run yet
# singularity pull library://dyammons/r-env/r4.3.2-seuratv5
Once the singularity pull
command has been located, you can create a directory to work from where you can install the required R packages. If working on Alpine consider pulling the .sif
in the following directory:
mkdir -p /projects/$USER/software/singularity/
cd /projects/$USER/software/singularity/
While in the singularity
directory you will use your repository's specific singularity pull
command.
#pull down your repository, so modify the below command as needed
singularity pull library://dyammons/r-env/r4.3.2-seuratv5
Once the container is there you will see a .sif
file. In this example it is called r4.3.2-seuratv5_latest.sif
.
ls
#r4.3.2-seuratv5_latest.sif
Because the .sif
file is an immutable object we must convert the object to a format that enables the installation of additional software.
To do this we will convert the file into a writable sandbox
which is a version of the container that can be modified by adding/deleting files from the directory structure.
#first argument is the name of the sandbox, second argument is the name of the .sif to convert
singularity build --sandbox r4.3.2-seuratv5 r4.3.2-seuratv5_latest.sif
Now the sandbox should be built and you can see a new directory called r4.3.2-seuratv5
.
ls
#r4.3.2-seuratv5 r4.3.2-seuratv5_latest.sif
We can now launch the container through the sandbox and see that base R
is properly installed.
singularity run r4.3.2-seuratv5
This will launch a R
session and we will be using a R
session to install packages. However, if you try to install packages now using install.packages()
you will see it cannot write to the container to install the new packages. To fix this we will need to enter the container through use of writable shell session. So, let's exit the current R
...
#quit R
q()
#do not save session
n
...and instead launch a writable sessions.
#launch a writable session to install packages
singularity shell --writable r4.3.2-seuratv5
Once inside this shell we can install packages directly through the terminal with commands such as the following:
#example command; change packages to what is needed
R --quiet --slave -e "install.packages(c('Seurat','clustree'))"
Note: these commands could also be placed directly into the singularity def file when completing the remote build. However due to the number of packages and dependency conflicts, it is easier to installed the R
after building the initial container.
Click for exact steps used in r4.3.2-seuratv5 build
#install completed on 12.10.2023 - DA
#attempt to install all at once -- several failed, so reorder in future
R --quiet --slave -e "install.packages(c('Rcpp', 'ggforce', 'ggrepel', 'graphlayouts','sitmo','dqrng','uwot','devtools','lme4','tidyverse','clustree','stringr','remotes','patchwork','scales','cowplot','ggrepel','colorspace','BiocManager','pheatmap','RColorBrewer','viridis','reshape','lemon','msigdbr','ggpubr','ape','UpSetR','Seurat'))"
R --quiet --slave -e "install.packages(c('RSpectra', 'ggforce', 'ggraph','Seurat','clustree'))"
R --quiet --slave -e "install.packages(c('pbkrtest', 'car', 'rstatix','R.utils','ape','tidytree','circlize','RColorBrewer'))"
#step through one by one to catch dependency errors
R --quiet --slave -e "BiocManager::install(c('limma','DESeq2'))"
R --quiet --slave -e "BiocManager::install(c('beachmat','BiocSingular','SingleR','celldex','treeio','ggtree','enrichplot','clusterProfiler','slingshot','scRNAseq','scuttle','ComplexHeatmap'))"
R --quiet --slave -e "remotes::install_github('chris-mcginnis-ucsf/DoubletFinder')"
R --quiet --slave -e "remotes::install_github('mojaveazure/seurat-disk')"
R --quiet --slave -e "devtools::install_github('davidsjoberg/ggsankey')"
R --quiet --slave -e "devtools::install_github('rpolicastro/scProportionTest')"
R --quiet --slave -e "devtools::install_github('immunogenomics/presto')"
R --quiet --slave -e "devtools::install_github('jinworks/CellChat')"
R --quiet --slave -e "devtools::install_github('arc85/singleseqgset')"
R --quiet --slave -e "BiocManager::install(c('beachmat','BiocSingular','SingleR','scuttle'))"
R --quiet --slave -e "BiocManager::install('BiocSingular')"
R --quiet --slave -e "BiocManager::install('SingleR')"
R --quiet --slave -e "install.packages('harmony')"
Now that R
packages are installed within the sandbox, we can finalize the .sif
and pack it up to share internally or externally. To do this we simply use the singularity build
command again, but flip the order of the arguments.
#pack up the container - this could take some time
singularity build r4.3.2-seurat5.sif r4_3_2s5/
There are several ways to share/distribute the finalized .sif
. The following steps are designed to walk through the process of making the container accessible to anyone with access to the internet. If you do not wish to make the container available for download from Syslabs you can stop here and use your container however you desire.
Note: In theory .sif
file contains all the software information, so if you are done adding packages to the container you could delete the sandbox.
The packaged .sif
file can be sent up to the Syslabs repository for easy sharing. To do this, you will need to create a Syslabs access token. See the Syslabs documention for details.
In addition to the access key we will need to create a key pair which is a unique value that will allow us to sign the container and forever link the container to you. The following code blocks walk through the steps.
#check if you already have keys
singularity keys list
#if you need a new key, create a new key pair with this
singularity keys newpair
Now you can check that the key exists.
#check for keys
singularity keys list
#example output
# 0) U: John Doe (my key) <johndoe@sylabs.io>
# C: 2018-08-21 20:14:39 +0200 CEST
# F: D87FE3AF5C1F063FCBCC9B02F812842B5EEE5934
# L: 4096
With a key established locally you will now need to send it up to your Syslabs account to get it fully linked to your profile. This can be done by copying the value in line F: in the above output and using a singularity keys push
command.
#example - modify the key to match your value
# singularity keys push A02599B62A9EF391FDB42E7A404C99A945E84E08
With all of those details taken care of, you are now ready to sign and upload the container to Syslabs. To do this you can modify the path that was used to pull down the initial .sif
build. In the below example the .sif
is being uploaded to a repository called r-env
and the image will be called r4.3.2-seurat5
with a tag of v2
. The v2
tag is added to avoid overriding the original version and to keep track of container versions.
#example - sign the container
# singularity sign r4.3.2-seurat5.sif
#example - send the container to Syslabs
# singularity push r4.3.2-seurat5.sif library://dyammons/r-env/r4.3.2-seurat5:v2
Congratulations on creating and depositing the container! You and collaborators can now pull down the singularity image file to ensure everyone is working in an identical computing environment.
Execution of the container is discussed in more detail later on.
TO DO: write more....
This seemed to work:
singularity build --sandbox r4.3.2-seuratv5_v2 r4.3.2-seurat5.sif
singularity shell -B $PWD/../ --fakeroot r4.3.2-seuratv5_v2
This approach is likely a better one, but the above also works, just is more clunky (this gave error... so def not the best way; need to figure out overlays)
singularity shell -B $PWD/../ --fakeroot --writable-tmpfs r4.3.2-seurat5.sif