Skip to content

Latest commit

 

History

History
121 lines (89 loc) · 3.36 KB

README.md

File metadata and controls

121 lines (89 loc) · 3.36 KB

context

R-CMD-check Codecov test coverage

Context manager is used to create a context: a code block that runs in the current R environment, where another object is temporarily created and safely destroyed after the code block finishes running or throws an error. A common use is to open a file, do stuff with it, then close the file connection. For example:

library(context)

with(open(filename) %as% f, {
  x <- do_stuff(f)
})

do_stuff2(x)

is equivalent to this Python code:

with open(filename) as f:
  x = do_stuff(f)
  
do_stuff2(x)

Installation

You can install the development version of context from GitHub with:

# install.packages("devtools")
devtools::install_github("igor-sb/context")

Implementation

ContextManager S3 object is created using create_context_manager() function, which takes 3 arguments:

  • on_enter: function to evaluate before code block. Return value of this function is stored in the variable on the right side of the %as% operator.

  • on_exit: function to evaluate after code block completes or stops.

  • args: arguments passed on the on_enter function.

This can be use to create custom context managers.

Alternative withs

Base R function base::with functions differently: it creates a new environment and has no support for setup and teardown code. Even its own documentation discourages its use (!).

context::with is much more similiar to withr::with_* functions from the withr package in that they both have the setup and teardown code. However, withr::with_* functions run the code block in their own environment and have a somewhat different purpose, which is to run code under modified global state. I find them clunky and inflexible to use for this purpose. For example, to modify a current environment variable, one needs even more code wrapping it (hence the motivation for this package) or use the <<- assignment operator (which sometimes binds a variable to a parent, and sometimes to a global environment).

Setting up custom context managers

Example: block-by-block reading of fastq files

Define on_enter and on_exit functions:

fastq_open <- function(file_name, n = 1e6) {
  create_context_manager(
    on_enter = function(file_name, n) ShortRead::FastqStreamer(file_name, n),
    on_exit = function(file_connection) close(file_connection),
    args = list(file_name = file_name, n = n)
  )
}

and a function to move the iterator each time a block is read:

read_block <- function(file_connection) {
  ShortRead::yield(file_connection)
}

Then we can iterate over a fastq file:

fastq_file <- read_test_fastq()
blocks <- list()
with(fastq_open(fastq_file) %as% fq, {
  while (read_block(fq) %as% block) {
    blocks <- c(blocks, ShortRead::sread(block))
  }
})

print(blocks)

#> [[1]]
#> DNAStringSet object of length 5:
#>     width seq
#> [1]    15 AGAATACTTCCGGCC
#> [2]    15 CGTNGANCGGCGATG
#> [3]    15 GACATTCTGCTGTNC
#> [4]    15 GCTATCCTCGACGGC
#> [5]    15 NGAAGGAGGTCTCCT