tests/test-cran/test-cache.R

library(testit)

assert('cache_exec() with disk path caches and retrieves', {
  d = tempfile(); dir.create(d)
  path = paste0(d, '/')
  r1 = cache_exec({Sys.time()}, path = path, id = 'disk-test')
  Sys.sleep(0.01)
  r2 = cache_exec({Sys.time()}, path = path, id = 'disk-test')
  (r1 %==% r2)
  unlink(d, recursive = TRUE)
})

assert('cache_exec() invalidates when hash changes', {
  x = 1
  r1 = cache_exec({x + 1}, path = ':memory:', id = 'hash-change-test')
  (r1 %==% 2)
  x = 10
  r2 = cache_exec({x + 1}, path = ':memory:', id = 'hash-change-test')
  (r2 %==% 11)
})

assert('cache_exec() with keep = FALSE always re-evaluates', {
  r1 = cache_exec({runif(1)}, path = ':memory:', id = 'keep-false-test', keep = FALSE)
  r2 = cache_exec({runif(1)}, path = ':memory:', id = 'keep-false-test', keep = FALSE)
  (r1 != r2)
})

assert('cache_rds() with hash = "auto" detects global variables', {
  d = tempfile(); dir.create(d)
  x = 100
  r1 = cache_rds(x * 2, file = 'auto-hash.rds', dir = paste0(d, '/'), hash = 'auto')
  (r1 %==% 200)
  r2 = cache_rds(x * 2, file = 'auto-hash.rds', dir = paste0(d, '/'), hash = 'auto')
  (r2 %==% 200)
  unlink(d, recursive = TRUE)
})

assert('cache_rds() with hash as list invalidates on change', {
  d = tempfile(); dir.create(d)
  r1 = cache_rds(1 + 1, file = 'hash-list.rds', dir = paste0(d, '/'), hash = list(a = 1))
  (r1 %==% 2)
  r2 = cache_rds(1 + 1, file = 'hash-list.rds', dir = paste0(d, '/'), hash = list(a = 2))
  (r2 %==% 2)
  # clean = TRUE removes old cache, so only one file exists
  (length(list.files(d, pattern = '[.]rds$')) %==% 1L)
  unlink(d, recursive = TRUE)
})

assert('cache_rds() errors on unsupported file extension', {
  d = tempfile(); dir.create(d)
  (has_error(cache_rds(1, file = 'test.xyz', dir = paste0(d, '/'))))
  unlink(d, recursive = TRUE)
})

assert('cache_dir() returns a character string', {
  d = cache_dir()
  (is.character(d))
  (nchar(d) > 0)
})

assert('find_locals() finds assigned variables', {
  (find_locals('x = 1; y = 2') %==% c('x', 'y'))
  (find_locals('for (i in 1:10) {}') %==% 'i')
  (find_locals('x = 1 +') %==% character(0))
})

assert('find_globals() handles language input', {
  e = new.env(parent = emptyenv())
  e$z = 99
  (find_globals(quote(y <- z + 1), e) %==% 'z')
})

assert('md5() returns consistent checksums', {
  x1 = 1; x2 = 1:10; x3 = seq(1, 10)
  m = md5(x1, x2, x3)
  (length(m) %==% 3L)
  (is.character(m))
  (nchar(m) %==% c(32L, 32L, 32L))
  # x2 and x3 are identical sequences
  (m[2] %==% m[3])
  # named arguments
  mn = md5(a = x1, b = x2)
  (names(mn) %==% c('a', 'b'))
})

assert('md5_one() produces a 32-character hex string', {
  h = md5_one(42)
  (is.character(h) && nchar(h) %==% 32L)
  # same object same hash
  (md5_one(42) %==% md5_one(42))
  # different objects different hashes
  (md5_one(42) != md5_one(43))
})

assert('cache_exec() with vars argument lazy-loads locals on cache hit', {
  # first run: populates cache
  r1 = cache_exec({
    local_var = 123
    local_var
  }, path = ':memory:', id = 'vars-lazy-test', vars = 'local_var')
  (r1 %==% 123)
  # second run: hits cache and lazy-loads local_var into the calling env
  r2 = cache_exec({
    local_var = 123
    local_var
  }, path = ':memory:', id = 'vars-lazy-test', vars = 'local_var')
  (r2 %==% 123)
  # local_var should have been lazy-loaded into the current environment
  (local_var %==% 123)
})

assert('cache_code() cleans up hash dict when path is NULL', {
  # seed the dict with a fake hash for 'x', then call cache_code with path = NULL
  assign('x', 'fakehash', envir = .cache_dict)
  (exists('x', envir = .cache_dict, inherits = FALSE))
  cache_code(quote(x <- 1), parent.frame(), config = list(path = NULL, vars = 'x'))
  # the stale entry should have been removed
  (!exists('x', envir = .cache_dict, inherits = FALSE))
})

assert('find_globals() returns character(0) when no globals exist', {
  e = new.env(parent = emptyenv())
  (find_globals('y = 1 + 1', e) %==% character(0))
})

Try the xfun package in your browser

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

xfun documentation built on June 19, 2026, 9:06 a.m.