makeit packageThe makeit package provides a simple make-like utility to run R scripts if needed, based on the last modified time. It comes with no package dependencies, organizational overhead, or structural requirements.
The general idea is to run a workflow without repeating tasks that are already completed. A workflow consists of one or more R scripts, where each script generates output files.
The following tutorials come with the package and can be copied from
library/makeit/examples to a working directory, or
downloaded from GitHub.
This example consists of a script analysis.R that uses
input.dat to produce output.dat.
Before
analysis.R
input.datRun
make("analysis.R", "input.dat", "output.dat")
#> Running analysis.R
#> Sorting numbers ... estimated run time is 3 secondsTry running again:
Note how a make() call has the general form: script
x uses y to produce z.
After
analysis.R
input.dat
output.datThis example consists of three scripts, where one runs after the other.
The plot script produces files inside a plots folder and
the table script produces files inside a tables folder.
Before
01_model.R
02_plots.R
03_tables.R
data.datRun
make("01_model.R", "data.dat", "results.dat")
#> Running 01_model.R
make("02_plots.R", "results.dat", c("plots/A.png", "plots/B.png"))
#> Running 02_plots.R
make("03_tables.R", "results.dat", c("tables/A.csv", "tables/B.csv"))
#> Running 03_tables.RFor convenience, a _make.R file is provided, containing
these make() calls.
After
plots/A.png
plots/B.png
tables/A.csv
tables/B.csv
01_model.R
02_plots.R
03_tables.R
data.dat
results.datSimilar to the ‘sequential’ example above, but based on the four-minutes
tutorial that comes with targets package.
Before
data_raw.csv
fit_model.R
get_data.R
plot_model.RRun
make("get_data.R", "data_raw.csv", "data/data.csv")
#> Running get_data.R
make("fit_model.R", "data/data.csv", "output/coefs.dat")
#> Running fit_model.R
make("plot_model.R", c("data/data.csv", "output/coefs.dat"), "output/plot.pdf")
#> Running plot_model.R
#> Saving 3 x 3 in imageFor convenience, a _make.R file is provided, containing
these make() calls.
After
data/data.csv
output/coefs.dat
output/plot.pdf
data_raw.csv
fit_model.R
get_data.R
plot_model.RDAG example based on the diagram provided in the Wikipedia article on directed acyclic graph.
Each script produces a corresponding output file: a.R
produces out/a.dat, b.R produces
out/b.dat, etc.
Before
a.R
b.R
c.R
d.R
e.RRun
make("a.R", prereq=NULL, target="out/a.dat")
#> Running a.R
#> Writing out/a.dat
make("b.R", prereq="out/a.dat", target="out/b.dat")
#> Running b.R
#> Writing out/b.dat
make("c.R", prereq="out/a.dat", target="out/c.dat")
#> Running c.R
#> Writing out/c.dat
make("d.R", prereq=c("out/b.dat", "out/c.dat"), target="out/d.dat")
#> Running d.R
#> Writing out/d.dat
make("e.R", prereq="out/d.dat", target="out/e.dat")
#> Running e.R
#> Writing out/e.datFor convenience, a _make.R file is provided, containing
these make() calls.
After
out/a.dat
out/b.dat
out/c.dat
out/d.dat
out/e.dat
a.R
b.R
c.R
d.R
e.RDAG example based on the example from the targets user
manual.
The second_target depends on first_target
and outer_function, which in turn depends on
inner_function and global_object.
Before
first_target.R
global_object.R
inner_function.R
outer_function.R
second_target.RRun
make("first_target.R", NULL, "output/first_target.dat")
#> Running first_target.R
#> Writing output/first_target.dat
make("global_object.R", NULL, "output/global_object.dat")
#> Running global_object.R
#> Writing output/global_object
make("second_target.R",
     prereq=c("output/first_target.dat", "output/global_object.dat",
              "inner_function.R", "outer_function.R"),
     target="output/second_target.dat")
#> Running second_target.R
#> Writing second_target.datFor convenience, a _make.R file is provided, containing
these make() calls.
After
output/first_target.dat
output/global_object.dat
output/second_target.dat
first_target.R
global_object.R
inner_function.R
outer_function.R
second_target.RThe make() function is a tool that can be applied to
many types of workflows, consisting of one or many R scripts. It is
especially useful when the complete workflow takes many minutes or hours
to run. Changing one part of the analysis will then update the related
plots and tables, without rerunning every part of the analysis.
Most analyses resemble the sequential example above,
dividing the workflow into steps that run one after another. As an
introductory example, the sequential workflow consists of
only three steps: model, plots, and tables.
In practice, it is usually practical to divide a workflow into more steps than that, the first step being data preparation, such as importing, filtering, aggregating, and converting the data to the format that the model expects.
If the model is non-trivial, it can be practical to have an output.R step righter after model.R, extracting the results from the model-specific format to a more general format that is easy to browse, tabulate, and plot. This way, the model.R script can be very short, making it easy to see and understand the modelling approach and configuration. Separating the fundamental modelling step from the manual labor of data preparation and plotting can make an analysis more open and reproducible - for others to browse and reuse.
The paradigm of using small dedicated scripts with clear input and output files (read and write function calls near the beginning and end of each script) is usually a better workflow design than managing a large monolithic script where the user navigates between sections to run selected blocks of code.
The four_minutes and dag_targets examples
above provide an interesting comparison between the makeit
package and the targets package, for example.
makeit
The makeit package is script-based, where each step
passes the results to the next step as output files. The user organizes
their workflow by writing scripts that produce files.
The makeit package has no package dependencies, takes a
very short time to learn, and can be used to run any existing workflows,
as long as they are based on scripts with input and output files. The
scripts may include functions, but that is not a requirement.
The package consists of a single function that does one thing: run an R script if underlying files have changed, otherwise do nothing.
TAF
The TAF package contains a similar make()
function and is an ancestor of the makeit package. The
overall aim of TAF is to support and strengthen
reproducibility in science, as well as reviewability.
The TAF package provides a structured modular design
that divides a workflow into four main stages: data.R,
model.R, output.R, and report.R.
The initial data are declared in a DATA.bib file and an
optional SOFTWARE.bib file can be used to declare specific
versions of R packages and other software.
The package consists of many useful tools to support reproducible workflows for scientific analyses.
targets
The targets package is function-based, where each step
passes the results to the next step as objects in memory. The user
organizes their workflow by writing functions that produce objects. It
is the successor of the older drake package.
The targets package relies on many underlying packages,
takes some time to learn, and some work may be required to realign
existing workflows into functions. The functions may produce files, but
that is not a requirement.
The package consists of many useful tools to support workflow design and management.
Comparison
| Package | Paradigm | State | Package dependencies | Time to learn | Run existing workflow | Features | 
|---|---|---|---|---|---|---|
| makeit | Scripts | Files | None | Very short | Must be file-based | One | 
| TAF | Scripts | Files | None | Some | Must be file-based | Many | 
| targets | Functions | Memory | Many | Some | Must be function-based | Many | 
The CRAN task view for reproducible research provides an annotated list of packages related to pipeline toolkits and project workflows.
Magnusson, A. makeit: Run R Scripts if Needed.
https://cran.r-project.org/package=makeit
Magnusson, A. and C. Millar. TAF: Transparent Assessment Framework
for Reproducible Research.
https://cran.r-project.org/package=TAF
Landau, W.M. 2021. The targets R package: a dynamic Make-like
function-oriented pipeline toolkit for reproducibility and
high-performance computing. Journal of Open Source Software,
6(57), 2959.
https://doi.org/10.21105/joss.02959, https://cran.r-project.org/package=targets
Stallman, R.M. et al. An introduction to makefiles. Chapter
2 in the GNU Make manual.
https://www.gnu.org/software/make/manual/