GQ1: Tell me in 10 lines how to use this package.
GA1: Get the dependency graph of several R packages on CRAN or Github at a specific snapshot date(time)
graph <- resolve(c("crsh/papaja", "rio"), snapshot_date = "2019-07-21")Dockerize the dependency graph to a directory
dockerize(graph, output_dir = "rangtest")You can build the Docker image either by the R package stevedore or Docker CLI client. We use the CLI client.
cd rangtest
docker build -t rangtest . ## might need sudoLaunch the container with the built image
docker run --rm --name "rangcontainer" -ti rangtestAnd the tenth line is not needed.
GQ2: For running resolve(), how do I know which packages are used in a project?
GA2: We recommend renv::dependencies().
Suppose in a directory called “project” there are two R files:
here::here()library(rio)
x <- import("hello.csv")Running this reveals
renv::dependencies("project")#> Finding R package dependencies ... Done!
#>          Source Package Require Version   Dev
#> 1 myproject/1.R    here                 FALSE
#> 2 myproject/2.R     rio                 FALSEYou may still need to manually review which packages are from Github.
GQ3: Why is the R script generated by dockerize() and export_rang() so strange/unidiomatic/inefficient/did you guys read fortunes::fortune("answer is parse")?
GA3: It is because we optimize the R code in rang.R for backward compatibility. We need to make sure that the code runs well in vanilla R environments since 2.1.0.
GQ4: Why doesn’t rang support reconstructing computational environments with R < 2.1.0 yet?
GA4: It is because installing source packages from within R was introduced in R 2.1.0. Before that one needed to install source packages with R CMD INSTALL. But we are working on supporting R in the 1.x series.
GQ5: Does rang.R (generated by export_rang() or dockerize()) run on non-Linux OSes?
GA5: Theoretically speaking, yes. But strongly not recommended. If the system requirements are fulfilled, rang.R should probably run fine on OS X if the R packages do not contain compiled code. C and Fortran compilers are needed if it is the case. See this entry in R Mac OS X FAQ. On Windows, installing Github packages requires properly set up PATH and tar. Similarly, R packages with compiled code require C / Fortran compilers. See this entry in R for Windows FAQ.
GQ6: What are the caveats of using rang?
GA6: Many
rang does not support reconstructing computational environments with R < 2.1.0 (i.e. snapshot_date < “2005-04-19 09:01”) yetdockerize() can only generate Debian/Ubuntu-based Docker imagesdockerize(cache = TRUE) does not cache R source code (yet) and System Requirements (in deb packages)query_sysreqs() (as well as resolve(query_sysreqs = TRUE)) queries for System Requirements based on the latest version of the packages on CRAN / Github. Therefore:
snapshot_date and the date of running resolve() might produce incorrect System Requirementsresolve() with R version < 3.1 and has at least one Github package must be dockerized with caching (i.e. dockerize(cache = TRUE)). It is because the outdated version of Debian cannot communicate with the Github APIdockerize(cache = TRUE)).GQ7: rang depends on R >= 3.5.0. Several of the dependencies depend on many modern R packages. How dare you claiming your package supports R >= 2.1.0?
GA7: To clarify, it is true that resolve() and dockerize() depend on many factors, including a modern version of R. But the reconstruction process (if with caching of R packages) depends only on the availability of Docker images from Docker Hub, availability of R source code on CRAN (R < 3.1.0), and deb packages from Ubuntu and Debian in the future. If you don’t believe in all of these, see also: DQ4.
GQ8: What are the data sources of resolve()?
GA8: Several
GQ9: I am not convinced by this package. What are the alternatives?
GA9: If you don’t consider the Dockerization part of rang, the date-based pinning of R packages can be done by:
library(pak)
options(repos = c(REPO_NAME = "https://packagemanager.rstudio.com/cran/2019-07-21"))
pkg_install("rio")
pkg_install("crsh/papaja")library(groundhog)
pkgs <- c("rio","crsh/papaja")
groundhog.library(pkgs, "2019-07-21")If you don’t consider the date-based pinning of R packages, the Dockerization can be done by:
library(containerit)
## combine with Package Manager to pin packages by date
install.packages("rio")
remotes::install_github("crsh/papaja")
library(rio)
library(papaja)
print(containerit::dockerfile(from = utils::sessionInfo()))library(dockerfiler)
my_dock <- Dockerfile$new()
## combine with Package Manager to pin packages by date
my_dock$RUN(r(install.packages(c("remotes", "rio"))))
my_dock$RUN(r(remotes::install_github("crsh/papaja")))
my_dockDQ1: Is Docker an overkill to simply ensure that a few lines of R code are reproducible?
DA1: It might be the case for recent R code, e.g. R >= 3.0 (or snapshot_date > “2013-04-03 09:10”). But we position rang as an archaeological tool to run really old R code (snapshot_date >= “2005-04-19 09:01”, but see GQ4). For this, Docker is essential because R in the 2.x (or even 1.x in future) series might not be installable anymore in a non-virtualized environment.
According to The Turing Way, a research compendium that aids computational reproducibility should contain a complete description of the computational environment. The directory exported by dockerize(), especially when materials_dir and cache were used, can be directly shared as a research compendium.
DQ2: How do I access bash instead of R?
DA2: By default, containers launched with the images generated by rang goes to R. One can override this by launching the container with an alternative entry point.
Suppose an image was built as per GA1.
docker run --rm --name "rangcontainer" --entrypoint bash -ti rangtestDQ3: How do I copy files from and to a launched container?
DA3: Again an image was built as per GA1 and launched as below
docker run --rm --name "rangcontainer" -ti rangtest# probably you need to run this from another terminal
docker cp rangcontainer:/rang.R rang2.R
docker cp rang2.R rangcontainer:/rang2.RWe want to emphasize here that launching a container with --name is useful because the name of the container is randomly generated when --name was not used to launch it. It is also important to remind you that a relaunched container goes back to the initial state. Any file generated inside the container previously will be removed. So use docker cp to copy any artifact if one wants to preserve any artifact.
DQ4: How do I back up an image?
DA4: If you don’t believe Docker Hub / Debian archives / Ubuntu archives would be available forever, you may back up the generated image.
docker save rangtest | gzip > rangtest.tar.gzYou can also share the back up gzipped tarball file (usually < 1G, depending on the size of materials_dir, thus shareable on Zenodo).
To restore the backup image:
docker load < rangtest.tar.gzAnd launch a container the same way
docker run --rm --name "rangcontainer" -ti rangtest