|
| 1 | +# lme4::bootMer(...) => |
| 2 | +# |
| 3 | +# local({ |
| 4 | +# .progressr_steps <- nsim |
| 5 | +# .progressr_first <- TRUE |
| 6 | +# .progressr_progressor <- progressr::progressor(steps = .progressr_steps) |
| 7 | +# .progressr_FUN <- FUN |
| 8 | +# lme4::bootMer( |
| 9 | +# x = x, |
| 10 | +# FUN = function(...) { |
| 11 | +# if (.progressr_first) { |
| 12 | +# .progressr_first <<- FALSE |
| 13 | +# } else { |
| 14 | +# on.exit(.progressr_progressor()) |
| 15 | +# } |
| 16 | +# .progressr_FUN(...) |
| 17 | +# }, |
| 18 | +# nsim = .progressr_steps |
| 19 | +# ) |
| 20 | +# }) |
| 21 | +# |
| 22 | +progressify_lme4 <- local({ |
| 23 | + template_bootMer_outer <- bquote_compile(local({ |
| 24 | + .progressr_steps <- .(STEPS) |
| 25 | + .progressr_first <- TRUE |
| 26 | + .progressr_progressor <- progressr::progressor(steps = .progressr_steps) |
| 27 | + .progressr_FUN <- .(FUN) |
| 28 | + .(EXPR) |
| 29 | + })) |
| 30 | + |
| 31 | + template_bootMer_FUN <- quote(function(...) { |
| 32 | + if (.progressr_first) { |
| 33 | + .progressr_first <<- FALSE |
| 34 | + } else { |
| 35 | + on.exit(.progressr_progressor()) |
| 36 | + } |
| 37 | + .progressr_FUN(...) |
| 38 | + }) |
| 39 | + |
| 40 | + function(expr, fcn_name, fcn, ..., envir = parent.frame()) { |
| 41 | + names <- names(expr) |
| 42 | + if (is.null(names)) names <- rep("", length.out = length(expr)) |
| 43 | + names <- names[-1] |
| 44 | + target_names <- names(formals(fcn))[seq_along(names)] |
| 45 | + unnamed <- setdiff(target_names, names) |
| 46 | + ## Handle '...' in formals - only resolve positional args before it |
| 47 | + ddd <- which(unnamed == "...") |
| 48 | + if (length(ddd) > 0L) { |
| 49 | + stopifnot(length(ddd) == 1L) |
| 50 | + unnamed <- unnamed[seq_len(ddd - 1L)] |
| 51 | + } |
| 52 | + empty_idxs <- which(names == "") |
| 53 | + n <- min(length(empty_idxs), length(unnamed)) |
| 54 | + if (n > 0L) names[empty_idxs[seq_len(n)]] <- unnamed[seq_len(n)] |
| 55 | + names <- c("", names) |
| 56 | + |
| 57 | + idx_nsim <- which(names == "nsim") |
| 58 | + idx_FUN <- which(names == "FUN") |
| 59 | + |
| 60 | + parts <- as.list(expr) |
| 61 | + |
| 62 | + if (length(idx_nsim) == 1L) { |
| 63 | + steps <- parts[[idx_nsim]] |
| 64 | + parts[[idx_nsim]] <- quote(.progressr_steps) |
| 65 | + } else { |
| 66 | + steps <- 1L |
| 67 | + parts$nsim <- quote(.progressr_steps) |
| 68 | + } |
| 69 | + |
| 70 | + stopifnot(length(idx_FUN) == 1L) |
| 71 | + FUN <- expr[[idx_FUN]] |
| 72 | + parts[[idx_FUN]] <- template_bootMer_FUN |
| 73 | + |
| 74 | + bquote_apply(template_bootMer_outer, STEPS = steps, FUN = FUN, EXPR = as.call(parts)) |
| 75 | + } ## progressify_lme4() |
| 76 | +}) |
| 77 | + |
| 78 | + |
| 79 | +append_builtin_transpilers_for_lme4 <- local({ |
| 80 | + known_fcns <- list( |
| 81 | + bootMer = c |
| 82 | + ) |
| 83 | + |
| 84 | + template <- bquote_compile(function(expr, options) { |
| 85 | + ns <- getNamespace("lme4") |
| 86 | + fcn <- get(.(fcn_name), mode = "function", envir = ns) |
| 87 | + progressify_lme4(expr, fcn_name = .(fcn_name), fcn = fcn, envir = parent.frame()) |
| 88 | + }) |
| 89 | + |
| 90 | + make_transpiler <- function(fcn_name) { |
| 91 | + transpiler <- eval(bquote_apply(template)) |
| 92 | + eval(transpiler) |
| 93 | + } |
| 94 | + |
| 95 | + function() { |
| 96 | + ## lme4::bootMer() |
| 97 | + transpilers <- list() |
| 98 | + for (fcn_name in names(known_fcns)) { |
| 99 | + transpilers[[fcn_name]] <- list( |
| 100 | + label = sprintf("lme4::%s() transpiler", fcn_name), |
| 101 | + transpiler = make_transpiler(fcn_name) |
| 102 | + ) |
| 103 | + } ## for (fcn_name ...) |
| 104 | + transpilers <- list(lme4 = transpilers) |
| 105 | + |
| 106 | + append_transpilers("progressify::built-in", transpilers) |
| 107 | + |
| 108 | + ## Return required packages |
| 109 | + c("lme4", "progressr") |
| 110 | + } |
| 111 | +}) |
0 commit comments