inst/testme/test-value.R

#' @tags value
#' @tags list listenv environment
#' @tags sequential multisession multicore

library(future)
library(listenv)
options(future.debug = FALSE)

## https://stat.ethz.ch/pipermail/r-devel/2025-March/083892.html
## Fixed in R-devel (>= 4.6.0) /HB 2025-03-23
attr_on_primitive_is_error <- local({
  res <- NA
  function() {
    if (is.na(res)) {
      test <- tryCatch(structure(`+`, abc = TRUE), error = identity)
      res <<- inherits(test, "error")
      ## Undo
      if (!res) structure(`+`, abc = NULL)
    }
    res
  }
})
        
sleep <- function(n) Sys.sleep(0.01*n)

identical_envs <- function(x, y) {
  if (length(x) > 1) x <- x[order(names(x))]
  if (length(y) > 1) y <- y[order(names(y))]
  identical(x, y)
}

message("*** value() ...")

strategies <- supportedStrategies(cores = 2L)
for (strategy in strategies) {
  message(sprintf("- plan('%s') ...", strategy))
  plan(strategy)

  truth <- list(a = 1, b = 42L)


  for (container in c("list", "listenv", "environment")) {
    message(sprintf("  Container %s ...", container))
    
    coerce <- switch(container,
             list = as.list,
          listenv = as.listenv,
      environment = as.environment
    )
    equals <- switch(container,
             list = identical,
          listenv = identical,
      environment = identical_envs
    )
      
    for (lazy in c(FALSE, TRUE)) {
      message(sprintf("    lazy = %s ...", lazy))
      
      message(sprintf("      Future: resolve with value"))
      f <- future({
        truth
      }, lazy = lazy)
      v <- value(f)
      stopifnot(identical(v, truth))
  
      message(sprintf("      Future: resolve with an error"))
      f <- future({
        stop("Nah!")
      }, lazy = lazy)
      res <- value(f, signal = FALSE)
      stopifnot(inherits(res, "error"))
      res <- tryCatch(value(f), error = identity)
      stopifnot(inherits(res, "error"))

      message(sprintf("      Future: resolve with value and drop"))
      truth <- 42
      f <- future(42, lazy = TRUE)
      y <- value(f, drop = TRUE)
      stopifnot(equals(y, truth))
      res <- tryCatch(value(f), error = identity)
      stopifnot(
        inherits(res, "error"),
        inherits(res, "FutureError"),
        inherits(res, "FutureDroppedError")
      )

      message(sprintf("      Containers: an empty %s", container))
      truth <- list()
      x <- coerce(truth)
      y <- value(x)
      stopifnot(equals(y, truth))

      message(sprintf("      Containers: a named %s", container))
      truth <- list(a = 1, b = 2, c = 3)
      x <- coerce(truth)
      y <- value(x)
      stopifnot(equals(y, truth))

      message(sprintf("      Containers: reducing a named %s", container))
      truth <- list(a = 1, b = 2, c = 3)
      x <- coerce(truth)
      truth <- do.call(sum, args = truth)
      y <- value(x, reduce = `+`)
      stopifnot(equals(y, truth))

      message(sprintf("      Containers: a named %s mixed with values and futures", container))
      truth <- list(a = 1, b = 2, c = 3, d = 3, e = 42)
      x <- list(
        a = future({ sleep(1); 1 }, lazy = lazy),
        b = future(2, lazy = lazy),
        c = future(3, lazy = lazy),
        d = 3,
        e = 42
      )
      x <- coerce(x)
      y <- value(x)
      stopifnot(equals(y, truth))

      message(sprintf("      Containers: reducing a named %s mixed with values and futures", container))
      truth <- list(a = 1, b = 2, c = 3, d = 3, e = 42)
      x <- list(
        a = future({ sleep(1); 1 }, lazy = lazy),
        b = future(2, lazy = lazy),
        c = future(3, lazy = lazy),
        d = 3,
        e = 42
      )
      x <- coerce(x)
      truth <- do.call(sum, args = c(list(pi), truth))
      y <- value(x, reduce = structure(function(...) `+`(...), init = pi))
      stopifnot(equals(y, truth))

      message(sprintf("      Containers: reducing a named %s with 'init' attribute on primary function is an error", container))
      truth <- list(a = 1, b = 2, c = 3)
      x <- coerce(truth)
      y <- tryCatch({
        value(x, reduce = structure(`+`, init = 42))
      }, error = identity)
      stopifnot(inherits(y, "error"))
      if (!attr_on_primitive_is_error()) {
        void <- structure(`+`, init = NULL)  ## undo
      }

      message(sprintf("      Containers: workaround for reducing a named %s with 'init' attribute on primary function", container))
      truth <- list(a = 1, b = 2, c = 3)
      x <- coerce(truth)
      truth <- do.call(sum, args = c(list(pi), truth))
      y <- value(x, reduce = structure("+", init = pi))
      stopifnot(equals(y, truth))

      message(sprintf("      Containers: a named %s mixed with values and futures out of order", container))
      truth <- list(a = 3, b = 2, c = 1, d = data.frame(a = 1, b = 2), e = 42)
      x <- list(
        a = future({ sleep(2); 3 }, lazy = lazy),
        b = future({ sleep(1); 2 }, lazy = lazy),
        c = future({           1 }, lazy = lazy),
        d = data.frame(a = 1, b = 2),
        e = 42
      )
      x <- coerce(x)
      y <- value(x, inorder = FALSE)
      stopifnot(equals(y, truth))

      message(sprintf("      Containers: a named %s mixed with values and futures dropped", container))
      truth <- list(a = 3, b = 2, c = 1, d = data.frame(a = 1, b = 2), e = 42)
      x <- list(
        a = future({ sleep(2); 3 }, lazy = lazy),
        b = future({ sleep(1); 2 }, lazy = lazy),
        c = future({           1 }, lazy = lazy),
        d = data.frame(a = 1, b = 2),
        e = 42
      )
      x <- coerce(x)
      y <- value(x, drop = TRUE)
      stopifnot(equals(y, truth))

      message(sprintf("      Containers: reducing a named %s mixed with values and futures dropped out of order", container))
      truth <- list(a = 1, b = 2, c = 3, d = 3, e = 42)
      x <- list(
        a = future({ sleep(2); 3 }, lazy = lazy),
        b = future({ sleep(1); 2 }, lazy = lazy),
        c = future({           1 }, lazy = lazy),
        d = 3,
        e = 42
      )
      x <- coerce(x)
      truth <- do.call(sum, args = truth)
      y <- value(x, reduce = `+`, inorder = FALSE, drop = TRUE)
      stopifnot(equals(y, truth))

      res <- tryCatch(value(x), error = identity)
      stopifnot(
        inherits(res, "error"),
        inherits(res, "FutureError"),
        inherits(res, "FutureDroppedError")
      )

      message(sprintf("    lazy = %s ... DONE", lazy))
    } ## for (lazy ...)
    message(sprintf("  Container %s ... DONE", container))
  } ## for (container ...)
} ## for (strategy ...)

message("*** value() ... DONE")

Try the future package in your browser

Any scripts or data that you put into this service are public.

future documentation built on April 12, 2025, 1:25 a.m.