Using preventr with a data frame

This vignette shows how you can use a data frame with preventr. This has always been possible since the inception of preventr, but things became more convenient with the introduction of the use_dat and add_to_dat arguments for the functions estimate_risk() and its synonym est_risk() in version 0.11. Hereafter, this vignette will use est_risk(). This vignette also assumes the reader has already read the relevant documentation for est_risk(), especially pertaining to arguments use_dat and add_to_dat.

Using use_dat and add_to_dat

Let’s start by defining some data to use.

make_vignette_dat <- function(add_time_and_model = FALSE) {
  
  n <- 10
  
  dat <- dplyr::tibble(
    # I am specifying `age`, `sex`, `egfr`, and `bmi` manually while letting
    # other parameters vary via `sample()` to facilitate later aspects of this
    # vignette (to show identical results from approaches I show below).
    age = c(40, 55, 45, 51, 52, 58, 57, 36, 49, 47),
    sex = rep(c("female", "male"), 5),
    sbp = sample(90:180, n, replace = TRUE),
    bp_tx = sample(c(TRUE, FALSE), n, replace = TRUE),
    total_c = sample(130:320, n, replace = TRUE),
    hdl_c = sample(20:100, n, replace = TRUE),
    statin = sample(c(TRUE, FALSE), n, replace = TRUE),
    dm = sample(c(TRUE, FALSE), n, replace = TRUE),
    smoking = sample(c(TRUE, FALSE), n, replace = TRUE),
    egfr = c(73, 71, 80, 73, 77, 70, 86, 89, 78, 68),
    bmi = c(37.4, 32.9, 37.5, 28.6, 37.5, 36.0, 36.7, 28.6, 18.7, 38.6),
    hba1c = sample(
      # I want to ensure NAs are equally represented in the sample space,
      # hence the composition shown below.
      c(
        seq(4.5, 15, 0.1), 
        rep(NA_real_, length(seq(4.5, 15, 0.1)))
      ), 
      n, 
      replace = TRUE
    ),
    uacr = sample(
      c(
        seq(0.1, 25000, 0.1), 
        rep(NA_real_, length(seq(0.1, 25000, 0.1)))
      ), 
      n, 
      replace = TRUE
    ),
    zip = sample(
      # (random sample of valid zips)
      c(
        "01518", "33321", "85206", "98591", "29138", 
        "98101", "44124", "48708", "48206", "77642", 
        rep(NA_character_, n)
      ), 
      n, 
      replace = TRUE
    )
  )
  
  if(add_time_and_model) {
    dat <- dat |> 
      dplyr::mutate(
        # I use `rep("both", 2)` for `time` because I want that option to have a
        # higher chance of being selected for this example.
        time = sample(c("10yr", "30yr", rep("both", 2)), n, replace = TRUE),
        model = sample(c("base", "hba1c", "uacr", "sdi", "full"), n, replace = TRUE)
      )
  }
  
  dat
}

dat <- make_vignette_dat()

knitr::kable(dat)
age sex sbp bp_tx total_c hdl_c statin dm smoking egfr bmi hba1c uacr zip
40 female 96 FALSE 262 96 TRUE TRUE FALSE 73 37.4 NA 6465.5 98101
55 male 109 FALSE 249 63 TRUE FALSE FALSE 71 32.9 4.5 5240.8 29138
45 female 90 TRUE 223 38 FALSE TRUE TRUE 80 37.5 NA 14184.3 NA
51 male 153 FALSE 211 83 FALSE TRUE FALSE 73 28.6 4.5 11394.4 NA
52 female 104 TRUE 166 39 TRUE TRUE TRUE 77 37.5 NA NA NA
58 male 151 FALSE 150 49 TRUE FALSE TRUE 70 36.0 5.8 6678.5 NA
57 female 136 FALSE 182 29 FALSE FALSE FALSE 86 36.7 5.1 24448.9 44124
36 male 109 FALSE 194 34 FALSE TRUE FALSE 89 28.6 NA 16891.8 33321
49 female 150 FALSE 299 31 TRUE FALSE FALSE 78 18.7 13.7 NA NA
47 male 161 FALSE 132 64 FALSE FALSE TRUE 68 38.6 NA NA NA

Our first call using use_dat

In the call below to est_risk(), because (1) dat does not contain any columns for optional behavior variables and (2) the call omits optional behavior variable arguments aside from progress = FALSE, the optional behavior variables will take their default values.

Note also the first column in the return table is preventr_id. This is a unique identifier for each row of the input data frame passed to use_dat. In this case, each row of the return data frame has 2 rows; this is expected, because we did not pass anything to arguments time or model in the call to est_risk(). They thus take their default values of time = "both" and model = NULL, with the latter meaning est_risk() will automatically select the model based on the input.

res <- est_risk(use_dat = dat, progress = FALSE)

knitr::kable(res)
preventr_id age sex sbp bp_tx total_c hdl_c statin dm smoking egfr bmi hba1c uacr zip total_cvd ascvd heart_failure chd stroke model over_years input_problems
1 40 female 96 FALSE 262 96 TRUE TRUE FALSE 73 37.4 NA 6465.5 98101 0.026 0.015 0.048 0.005 0.010 full 10 NA
1 40 female 96 FALSE 262 96 TRUE TRUE FALSE 73 37.4 NA 6465.5 98101 0.131 0.070 0.235 0.027 0.044 full 30 NA
2 55 male 109 FALSE 249 63 TRUE FALSE FALSE 71 32.9 4.5 5240.8 29138 0.101 0.059 0.062 0.031 0.029 full 10 NA
2 55 male 109 FALSE 249 63 TRUE FALSE FALSE 71 32.9 4.5 5240.8 29138 0.336 0.203 0.244 0.117 0.109 full 30 NA
3 45 female 90 TRUE 223 38 FALSE TRUE TRUE 80 37.5 NA 14184.3 NA 0.353 0.173 0.337 0.122 0.082 uacr 10 NA
3 45 female 90 TRUE 223 38 FALSE TRUE TRUE 80 37.5 NA 14184.3 NA 0.679 0.410 0.683 0.316 0.215 uacr 30 NA
4 51 male 153 FALSE 211 83 FALSE TRUE FALSE 73 28.6 4.5 11394.4 NA 0.167 0.081 0.126 0.034 0.056 full 10 NA
4 51 male 153 FALSE 211 83 FALSE TRUE FALSE 73 28.6 4.5 11394.4 NA 0.419 0.223 0.383 0.103 0.160 full 30 NA
5 52 female 104 TRUE 166 39 TRUE TRUE TRUE 77 37.5 NA NA NA 0.124 0.067 0.107 0.039 0.033 base 10 NA
5 52 female 104 TRUE 166 39 TRUE TRUE TRUE 77 37.5 NA NA NA 0.448 0.257 0.414 0.164 0.134 base 30 NA
6 58 male 151 FALSE 150 49 TRUE FALSE TRUE 70 36.0 5.8 6678.5 NA 0.214 0.115 0.231 0.057 0.070 full 10 NA
6 58 male 151 FALSE 150 49 TRUE FALSE TRUE 70 36.0 5.8 6678.5 NA 0.450 0.264 0.459 0.144 0.172 full 30 NA
7 57 female 136 FALSE 182 29 FALSE FALSE FALSE 86 36.7 5.1 24448.9 44124 0.154 0.083 0.089 0.043 0.043 full 10 NA
7 57 female 136 FALSE 182 29 FALSE FALSE FALSE 86 36.7 5.1 24448.9 44124 0.438 0.261 0.311 0.155 0.145 full 30 NA
8 36 male 109 FALSE 194 34 FALSE TRUE FALSE 89 28.6 NA 16891.8 33321 0.060 0.034 0.023 0.026 0.009 full 10 NA
8 36 male 109 FALSE 194 34 FALSE TRUE FALSE 89 28.6 NA 16891.8 33321 0.223 0.125 0.114 0.101 0.036 full 30 NA
9 49 female 150 FALSE 299 31 TRUE FALSE FALSE 78 18.7 13.7 NA NA 0.221 0.173 0.056 0.119 0.077 hba1c 10 NA
9 49 female 150 FALSE 299 31 TRUE FALSE FALSE 78 18.7 13.7 NA NA 0.608 0.505 0.233 0.401 0.272 hba1c 30 NA
10 47 male 161 FALSE 132 64 FALSE FALSE TRUE 68 38.6 NA NA NA 0.058 0.031 0.063 0.012 0.025 base 10 NA
10 47 male 161 FALSE 132 64 FALSE FALSE TRUE 68 38.6 NA NA NA 0.285 0.155 0.319 0.065 0.126 base 30 NA

Progress bar

Because some data frames may be large, est_risk() now has a progress bar that will automatically show (unless progress = FALSE) when use_dat is a data frame. This is independent of the quiet argument. The progress bar requires the utils package, but this is part of the R distribution, so outside of unusual situations, you should not need to install it yourself. The code below will not actually execute in this vignette because the progress bar is intended for the console. I include it below simply for demonstration.

# The default for `progress` when `use_dat` is a data frame is `TRUE`, so this
# call would yield a progress bar during computation.
res_for_prog_bar <- est_risk(use_dat = dat)

Passing a data frame with a predictor variable located in a differently-named column

You can pass a data frame with a predictor variable located in a differently-named column, and the column can be specified as a character string or symbol.

dat_age_rename <- dat |> dplyr::rename(years_old = age)

res_age_rename_sym <- est_risk(
  use_dat = dat_age_rename, 
  age = years_old, 
  progress = FALSE
)

res_age_rename_chr <- est_risk(
  use_dat = dat_age_rename, 
  age = "years_old", 
  progress = FALSE
)

The following two tests will be FALSE, because the column names housing the age data differ.

identical(res, res_age_rename_sym) 
#> [1] FALSE

identical(res, res_age_rename_chr)
#> [1] FALSE

But all of these will be TRUE.

identical(
  res |> dplyr::select(-age),
  res_age_rename_sym |> dplyr::select(-years_old)
)
#> [1] TRUE

identical(
  res |> dplyr::select(-age),
  res_age_rename_chr |> dplyr::select(-years_old)
)
#> [1] TRUE

identical(res_age_rename_sym, res_age_rename_chr)
#> [1] TRUE

And thus, if we rename the years_old columns …

res_age_rename_sym <- res_age_rename_sym |> dplyr::rename(age = years_old)

res_age_rename_chr <- res_age_rename_chr |> dplyr::rename(age = years_old)

… everything will be identical.

identical(res, res_age_rename_sym) 
#> [1] TRUE

identical(res, res_age_rename_chr)
#> [1] TRUE

Optional behavior variables: Data frame vs. in the call

You can also pass optional behavior variables via the data frame. As a reminder, optional behavior variables may either be in the data frame passed to use_dat in a column with the same name as the argument or passed to the function call as usual. If an optional behavior variable is omitted from the call when a user passes a data frame to use_dat, the function will first look for a column with the name of the optional behavior variable in the data frame; if it does not find such a column, it will use the default behavior for the optional behavior variable. If the user includes an argument for an optional behavior variable in the call, the function will always use this, irrespective of any column in the data frame that might share the same name. Additionally, the following arguments are not passable via the data frame: collapse (ignored when use_dat is a data frame), use_dat (this would be self-referential), add_to_dat (again, essentially self-referential), and progress (this applies to the entire call when use_dat is a data frame).

In what follows, I will show time and model, but you could also pass, for example, optional_strict or quiet if you wanted. Passing optional behavior variables via the data frame is likely most useful when you wish for est_risk()’s behavior to vary across rows within the data frame passed to use_dat. Otherwise, if you desire the same behavior for the entirety of the call to est_risk(), it is likely easier to pass the optional behavior variables in the call to est_risk() (or pass nothing to those arguments if you just want the default behavior).

dat_time_model <- make_vignette_dat(add_time_and_model = TRUE)

res_time_model_in_dat <- est_risk(use_dat = dat_time_model, progress = FALSE)

knitr::kable(res_time_model_in_dat)
preventr_id age sex sbp bp_tx total_c hdl_c statin dm smoking egfr bmi hba1c uacr zip time model_input total_cvd ascvd heart_failure chd stroke model over_years input_problems
1 40 female 111 FALSE 220 47 TRUE FALSE TRUE 73 37.4 5.7 15873.2 NA 10yr hba1c 0.017 0.012 0.009 0.006 0.006 hba1c 10 NA
2 55 male 113 TRUE 312 43 FALSE FALSE TRUE 71 32.9 8.1 NA 77642 10yr full 0.092 0.079 0.039 0.056 0.026 full 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 80 37.5 11.9 NA 48206 both sdi 0.080 0.047 0.135 0.015 0.037 sdi 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 80 37.5 11.9 NA 48206 both sdi 0.340 0.198 0.513 0.072 0.153 sdi 30 NA
4 51 male 98 TRUE 282 73 TRUE FALSE TRUE 73 28.6 NA NA 85206 10yr uacr 0.050 0.033 0.020 0.018 0.014 uacr 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both base 0.288 0.221 0.142 0.167 0.096 base 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both base 0.689 0.563 0.479 0.474 0.307 base 30 NA
6 58 male 165 TRUE 262 55 FALSE TRUE FALSE 70 36.0 NA NA NA 30yr base 0.555 0.417 0.482 0.280 0.232 base 30 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 86 36.7 NA NA 98591 both uacr 0.067 0.046 0.033 0.026 0.021 uacr 10 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 86 36.7 NA NA 98591 both uacr 0.325 0.214 0.195 0.134 0.102 uacr 30 NA
8 36 male 169 TRUE 214 72 TRUE TRUE FALSE 89 28.6 NA NA NA both hba1c 0.057 0.033 0.034 0.013 0.024 hba1c 10 NA
8 36 male 169 TRUE 214 72 TRUE TRUE FALSE 89 28.6 NA NA NA both hba1c 0.321 0.184 0.236 0.079 0.133 hba1c 30 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 78 18.7 NA 5521.5 NA both base 0.175 0.128 0.042 0.086 0.056 base 10 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 78 18.7 NA 5521.5 NA both base 0.610 0.475 0.233 0.367 0.251 base 30 NA
10 47 male 109 TRUE 297 38 TRUE TRUE TRUE 68 38.6 NA NA 48708 both base 0.199 0.141 0.087 0.112 0.039 base 10 NA
10 47 male 109 TRUE 297 38 TRUE TRUE TRUE 68 38.6 NA NA 48708 both base 0.602 0.466 0.373 0.403 0.167 base 30 NA

Because the input data frame in the above call to est_risk() has a column named model, est_risk() will rename this column to model_input in the return data frame. This is to accommodate the return data frame also needing to specify the model used for any given row, as specified in the documentation of how use_dat works when model is in the data frame passed to use_dat (see the “Value” section of the documentation in particular). This is perhaps most useful when the model input consists of a request for both PREVENT and PCE models (examples involving the PCEs will appear later).

Remember, you can override any of the optional behavior variables by passing them as an argument. Note the different values for time and model in the data frame.

dat_time_model[["time"]] 
#>  [1] "10yr" "10yr" "both" "10yr" "both" "30yr" "both" "both" "both" "both"

dat_time_model[["model"]]
#>  [1] "hba1c" "full"  "sdi"   "uacr"  "base"  "base"  "uacr"  "hba1c" "base" 
#> [10] "base"

Despite that, we will only get 10-year estimates from the base model of the PREVENT equations with the following call.

res_time_and_model_in_call <- est_risk(
  use_dat = dat_time_model, 
  time = 10, 
  model = "base",
  progress = FALSE
) 

all.equal(unique(res_time_and_model_in_call[["over_years"]]), 10) 
#> [1] TRUE

all.equal(unique(res_time_and_model_in_call[["model"]]), "base")  
#> [1] TRUE

And likewise, the following call will invoke automatic model selection despite the model column being all "base", because we pass the argument model = NULL to the call.

res_time_and_model_in_call <- est_risk(
  use_dat = dat_time_model |> dplyr::mutate(model = "base"), 
  model = NULL,
  progress = FALSE
) 

all.equal(unique(res_time_and_model_in_call[["model_input"]]), "base") 
#> [1] TRUE
res_time_and_model_in_call[["model"]]
#>  [1] "full" "full" "full" "full" "sdi"  "full" "full" "base" "sdi"  "sdi" 
#> [11] "base" "base" "uacr" "uacr" "sdi"  "sdi"

Number of rows returned for a given input row = number of models requested

For any given input row in the data frame passed to use_dat, the row will be expanded to match the number of models requested. Stated more formally, each row in the input data frame will be assigned a preventr_id value. For the row with preventr_id x (hereafter, “row x”), if n represents the number of models requested for row x, then row x will be replicated n times in the return data frame to accommodate reporting the different model outputs for that row. Note also n is determined by what the function receives for both the model and time arguments (because, for example, if model = "base" and time = "both", this is a request for 2 models).

One can explore this further by examining the model and time values in an arbitrary row of the data frame passed to use_dat and the corresponding row(s) in the return data frame. In this example, there will only ever be a maximum of 2 rows per row of input, because the example here only considers the PREVENT equations and time horizons of 10 and 30 years. Phrased differently, the variability in number of rows per preventr_id for this example will be relatively small: 1 or 2 rows. However, if one were also considering the PCEs, then there could be a maximum of 4 rows per row of input (if one asked for both the original and revised PCEs and both the 10- and 30-year estimates, as this would yield a 10-year estimate from the PREVENT equations, the original PCEs, and the revised PCEs, and a 30-year estimate from the PREVENT equations).

show_random_row <- function(dat, res, n = 5) {
  
  rows <- seq_len(nrow(dat))
  already_seen <- vector("double", n)
  
  for(i in seq_len(n)) {
    
    random_row <- sample(rows, 1)
    while(random_row %in% already_seen) random_row <- sample(rows, 1)
    already_seen[[i]] <- random_row
    
    cat(paste0("\n", "--- `preventr_id` ", random_row, " ---", "\n\n"))
    
    print(
      list(
        # `model_input` has `unlist(..., recursive = FALSE)` because sometimes
        # column `model` will be a list column, so each item therein will be
        # enclosed in a list, and unlisting one level improves the appearance of
        # printing a bit in this case.
        model_input = unlist(dat[random_row, ][["model"]], recursive = FALSE),
        time_input = dat[random_row, ][["time"]],
        nrow_res = dplyr::filter(res, preventr_id == random_row) |> nrow()
      )
    )
  }
  
}

show_random_row(dat_time_model, res_time_model_in_dat)
#> 
#> --- `preventr_id` 6 ---
#> 
#> $model_input
#> [1] "base"
#> 
#> $time_input
#> [1] "30yr"
#> 
#> $nrow_res
#> [1] 1
#> 
#> 
#> --- `preventr_id` 9 ---
#> 
#> $model_input
#> [1] "base"
#> 
#> $time_input
#> [1] "both"
#> 
#> $nrow_res
#> [1] 2
#> 
#> 
#> --- `preventr_id` 8 ---
#> 
#> $model_input
#> [1] "hba1c"
#> 
#> $time_input
#> [1] "both"
#> 
#> $nrow_res
#> [1] 2
#> 
#> 
#> --- `preventr_id` 4 ---
#> 
#> $model_input
#> [1] "uacr"
#> 
#> $time_input
#> [1] "10yr"
#> 
#> $nrow_res
#> [1] 1
#> 
#> 
#> --- `preventr_id` 3 ---
#> 
#> $model_input
#> [1] "sdi"
#> 
#> $time_input
#> [1] "both"
#> 
#> $nrow_res
#> [1] 2

add_to_dat

If you do not want the risk estimation results to be added back to the data frame passed to use_dat, set add_to_dat = FALSE. The results will be returned as a data frame, and the first column will again be preventr_id to identify the row in the use_dat data frame that corresponds with the results shown in the return data frame. This column is perhaps of additional importance when add_to_dat = FALSE, as the return data frame will not have the columns contained in the data frame passed to use_dat, so preventr_id provides a ready way to reassociate the data frame passed to use_dat with the results in the return data frame.

res_without_dat <- est_risk(
  use_dat = dat_time_model, 
  add_to_dat = FALSE, 
  progress = FALSE
)

knitr::kable(res_without_dat)
preventr_id total_cvd ascvd heart_failure chd stroke model over_years input_problems
1 0.017 0.012 0.009 0.006 0.006 hba1c 10 NA
2 0.092 0.079 0.039 0.056 0.026 full 10 NA
3 0.080 0.047 0.135 0.015 0.037 sdi 10 NA
3 0.340 0.198 0.513 0.072 0.153 sdi 30 NA
4 0.050 0.033 0.020 0.018 0.014 uacr 10 NA
5 0.288 0.221 0.142 0.167 0.096 base 10 NA
5 0.689 0.563 0.479 0.474 0.307 base 30 NA
6 0.555 0.417 0.482 0.280 0.232 base 30 NA
7 0.067 0.046 0.033 0.026 0.021 uacr 10 NA
7 0.325 0.214 0.195 0.134 0.102 uacr 30 NA
8 0.057 0.033 0.034 0.013 0.024 hba1c 10 NA
8 0.321 0.184 0.236 0.079 0.133 hba1c 30 NA
9 0.175 0.128 0.042 0.086 0.056 base 10 NA
9 0.610 0.475 0.233 0.367 0.251 base 30 NA
10 0.199 0.141 0.087 0.112 0.039 base 10 NA
10 0.602 0.466 0.373 0.403 0.167 base 30 NA

The default for add_to_dat is TRUE when use_dat is a data frame (it is ignored when use_dat is anything other than a data frame). add_to_dat = TRUE essentially just triggers a left join behind the scenes. As we’ll see in the example immediately below, calls with add_to_dat = FALSE are still easy to reassociate with the data frame passed to use_dat, thanks to the preventr_id column in the return data frame.

res_with_dat <- est_risk(use_dat = dat_time_model, progress = FALSE)

# Now, let's check identicality of `res_with_dat` with a version we
# recreate using `dat_for_join` and `res_without_dat`.
dat_for_join <- dat_time_model |> 
  # First, add the `preventer_id` column ...
  dplyr::mutate(preventr_id = seq_len(nrow(dat_time_model))) |> 
  # ... and then move it to be the first column in the data frame.
  dplyr::relocate(preventr_id) 

# Now, do the left join.
res_with_dat_manual_join <- dat_for_join |> 
  dplyr::left_join(
    res_without_dat, 
    by = "preventr_id",
    # Because both data frames will have a column named `model`, I'll provide
    # suffixes to distinguish them. The suffixes below will result in the column 
    # `model` in `dat_for_join` being renamed to `model_input` and column
    # `model` in the data frame `res_without_dat` retaining the same name.
    suffix = c("_input", "")  
    )

# (You could also do all the above without a pipe sequence, of course.)

identical(res_with_dat, res_with_dat_manual_join)   
#> [1] TRUE

Type-stability of return

The return data frame will be of the same type as the data frame passed to use_dat.

dat_tbl <- dat |> dplyr::mutate(quiet = TRUE)       
dat_dt <- data.table::as.data.table(dat_tbl)
dat_df <- as.data.frame(dat_tbl)

class(dat_tbl)
#> [1] "tbl_df"     "tbl"        "data.frame"
class(dat_dt)
#> [1] "data.table" "data.frame"
class(dat_df)
#> [1] "data.frame"

res_tbl <- est_risk(use_dat = dat_tbl, progress = FALSE)  # Return: tibble
res_dt <- est_risk(use_dat = dat_dt, progress = FALSE)    # Return: data.table
res_df <- est_risk(use_dat = dat_df, progress = FALSE)    # Return: data frame

identical(class(dat_tbl), class(res_tbl))
#> [1] TRUE
identical(class(dat_dt), class(res_dt))
#> [1] TRUE
identical(class(dat_df), class(res_df))
#> [1] TRUE

# Other than the attributes, these are all equal (of course).
all.equal(res_tbl, res_dt, check.attributes = FALSE)
#> [1] TRUE
all.equal(res_tbl, res_df, check.attributes = FALSE)
#> [1] TRUE

What about the PCEs?

Yes, you can request the PCEs in conjunction with using use_dat. Because of how the model argument works when requesting the PCEs (namely, it expects a list in this case), the model column in the data frame being passed to use_dat needs to be a list column.

dat_with_pce_requests <- dat_time_model |> 
  # We'll start with the data in `dat_time_model` and then overwrite the `model`
  # column for this example.
  dplyr::mutate(
    # Base R `lapply()` is a convenient choice here, as it will return a list; 
    # however, this is not the only way to create list columns.
    model = lapply(
      seq_len(nrow(dat_time_model)),
      function(x) {
        # Let's make some rows just have `NA` (leading to automatic PREVENT
        # model selection and no risk estimation from the PCEs) and other rows
        # specify both the PREVENT and PCE models. This is just to demonstrate
        # flexibility. You could also just generate a basic list column, and
        # that would be less involved than what I do here.
        if(x %% 2 == 0) {
          NA
        } else { 
          list(
            # (We could also omit `main_model`, in which case the PREVENT model
            # will be selected automatically.)
            main_model = sample(c("base", "hba1c", "uacr", "sdi", "full"), 1),
            other_models = sample(c("pce_both", "pce_rev", "pce_orig"), 1),
            race_eth = sample(c("Black", "White", "Other"), 1)
          )
        }
      }
    )
  )

res_with_pce_requests <- est_risk(
  use_dat = dat_with_pce_requests, 
  progress = FALSE
)

knitr::kable(res_with_pce_requests)
preventr_id age sex sbp bp_tx total_c hdl_c statin dm smoking egfr bmi hba1c uacr zip time model_input total_cvd ascvd heart_failure chd stroke model over_years input_problems
1 40 female 111 FALSE 220 47 TRUE FALSE TRUE 73 37.4 5.7 15873.2 NA 10yr hba1c , pce_orig, White 0.017 0.012 0.009 0.006 0.006 hba1c 10 NA
1 40 female 111 FALSE 220 47 TRUE FALSE TRUE 73 37.4 5.7 15873.2 NA 10yr hba1c , pce_orig, White NA 0.030 NA NA NA pce_orig 10 NA
2 55 male 113 TRUE 312 43 FALSE FALSE TRUE 71 32.9 8.1 NA 77642 10yr NA 0.092 0.079 0.039 0.056 0.026 full 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 80 37.5 11.9 NA 48206 both base , pce_orig, Black 0.079 0.045 0.130 0.015 0.035 base 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 80 37.5 11.9 NA 48206 both base , pce_orig, Black NA 0.084 NA NA NA pce_orig 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 80 37.5 11.9 NA 48206 both base , pce_orig, Black 0.344 0.192 0.503 0.073 0.147 base 30 NA
4 51 male 98 TRUE 282 73 TRUE FALSE TRUE 73 28.6 NA NA 85206 10yr NA 0.070 0.040 0.029 0.023 0.018 sdi 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both full , pce_both, White 0.536 0.395 0.425 0.335 0.179 full 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both full , pce_both, White NA 0.381 NA NA NA pce_orig 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both full , pce_both, White NA 0.358 NA NA NA pce_rev 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both full , pce_both, White 0.798 0.656 0.718 0.594 0.373 full 30 NA
6 58 male 165 TRUE 262 55 FALSE TRUE FALSE 70 36.0 NA NA NA 30yr NA 0.555 0.417 0.482 0.280 0.232 base 30 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 86 36.7 NA NA 98591 both uacr , pce_both, Other 0.067 0.046 0.033 0.026 0.021 uacr 10 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 86 36.7 NA NA 98591 both uacr , pce_both, Other NA 0.107 NA NA NA pce_orig 10 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 86 36.7 NA NA 98591 both uacr , pce_both, Other NA 0.070 NA NA NA pce_rev 10 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 86 36.7 NA NA 98591 both uacr , pce_both, Other 0.325 0.214 0.195 0.134 0.102 uacr 30 NA
8 36 male 169 TRUE 214 72 TRUE TRUE FALSE 89 28.6 NA NA NA both NA 0.085 0.049 0.058 0.019 0.038 base 10 NA
8 36 male 169 TRUE 214 72 TRUE TRUE FALSE 89 28.6 NA NA NA both NA 0.390 0.233 0.310 0.101 0.180 base 30 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 78 18.7 NA 5521.5 NA both sdi , pce_both, Other 0.181 0.134 0.046 0.090 0.058 sdi 10 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 78 18.7 NA 5521.5 NA both sdi , pce_both, Other NA 0.133 NA NA NA pce_orig 10 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 78 18.7 NA 5521.5 NA both sdi , pce_both, Other NA 0.107 NA NA NA pce_rev 10 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 78 18.7 NA 5521.5 NA both sdi , pce_both, Other 0.611 0.482 0.241 0.378 0.258 sdi 30 NA
10 47 male 109 TRUE 297 38 TRUE TRUE TRUE 68 38.6 NA NA 48708 both NA 0.211 0.147 0.094 0.120 0.042 sdi 10 NA
10 47 male 109 TRUE 297 38 TRUE TRUE TRUE 68 38.6 NA NA 48708 both NA 0.616 0.476 0.388 0.417 0.176 sdi 30 NA

Those reviewing closely will notice printing the list column model_input results in some extra spaces and the list item names being dropped, but it nevertheless permits insight into what was contained in the list for model for each row of the input data frame passed to use_dat. And note this is solely a side effect of printing the list column via knitr::kable(). The column model_input in the return data frame is identical to the column model in the data frame passed to use_dat aside from the expansion behavior already described (see section “Number of rows returned for a given input row = number of models requested” in this vignette).

identical_cols <- vapply(
  seq_len(nrow(dat_with_pce_requests)),
  
  function(x) {
    n_row <- res_with_pce_requests |> dplyr::filter(preventr_id == x) |> nrow()
    
    identical(
      rep(dat_with_pce_requests[["model"]][x], n_row),
      
      res_with_pce_requests |> 
        dplyr::filter(preventr_id == x) |>
        dplyr::pull(model_input)
    )
  },
  
  logical(1)
)

all(identical_cols)
#> [1] TRUE

On the note of the expansion behavior, note the results here further demonstrate the concept of the number of rows in the return data frame equaling the number of models requested for a given row in the input data frame.

show_random_row(dat_with_pce_requests, res_with_pce_requests)
#> 
#> --- `preventr_id` 10 ---
#> 
#> $model_input
#> [1] NA
#> 
#> $time_input
#> [1] "both"
#> 
#> $nrow_res
#> [1] 2
#> 
#> 
#> --- `preventr_id` 9 ---
#> 
#> $model_input
#> $model_input$main_model
#> [1] "sdi"
#> 
#> $model_input$other_models
#> [1] "pce_both"
#> 
#> $model_input$race_eth
#> [1] "Other"
#> 
#> 
#> $time_input
#> [1] "both"
#> 
#> $nrow_res
#> [1] 4
#> 
#> 
#> --- `preventr_id` 3 ---
#> 
#> $model_input
#> $model_input$main_model
#> [1] "base"
#> 
#> $model_input$other_models
#> [1] "pce_orig"
#> 
#> $model_input$race_eth
#> [1] "Black"
#> 
#> 
#> $time_input
#> [1] "both"
#> 
#> $nrow_res
#> [1] 3
#> 
#> 
#> --- `preventr_id` 6 ---
#> 
#> $model_input
#> [1] NA
#> 
#> $time_input
#> [1] "30yr"
#> 
#> $nrow_res
#> [1] 1
#> 
#> 
#> --- `preventr_id` 1 ---
#> 
#> $model_input
#> $model_input$main_model
#> [1] "hba1c"
#> 
#> $model_input$other_models
#> [1] "pce_orig"
#> 
#> $model_input$race_eth
#> [1] "White"
#> 
#> 
#> $time_input
#> [1] "10yr"
#> 
#> $nrow_res
#> [1] 2

Again, specifying the model variable in the data frame is likely most helpful when one desires to request different behavior across different rows of the input data frame passed to use_dat (just like for any optional behavior variable). If instead you want the same model behavior across all rows, you can either (1) pass nothing to the model argument and not include a model column in the data frame passed to use_dat or (2) specify the desired model behavior in the model argument, which will also override anything that might exist in a column named model in the data frame passed to use_dat.

What about the convenience functions for eGFR and BMI?

Yes, you can use the convenience functions for eGFR and BMI when using use_dat and add_to_dat (see the documentation for est_risk()’s arguments egfr and bmi if you are not familiar with these convenience functions). The input data frame passed to use_dat will need to have the calls to the convenience functions in columns. Because these will be columns, they will need to be list columns.

dat_with_calls_basic <- dat_with_pce_requests |> 
  dplyr::mutate(
    egfr = lapply(
      seq_len(nrow(dat)),
      function(x) {
        # We can make some rows have calls to `calc_egfr` and some just have
        # values. This is just for demonstration, and one could instead have a
        # simple list column composed entirely of calls.
        if(x %% 2 == 0) {
          call("calc_egfr", cr = sample(seq(0.5, 1.5, 0.1), 1))
        } else {
          sample(45:90, 1)
        }
      }
    ),
    bmi = lapply(
      seq_len(nrow(dat)),
      function(x) {
        # The comment above for `egfr` applies here as well.
        if(x %% 2 == 0) {
          call(
            "calc_bmi", 
            height = sample(60:78, 1),
            weight = sample(110:200, 1)
          )
        } else {
          sample(20:38, 1)
        }
      }
    )
  )

res_with_calls_basic <- est_risk(
  use_dat = dat_with_calls_basic, 
  progress = FALSE
)

knitr::kable(res_with_calls_basic)
preventr_id age sex sbp bp_tx total_c hdl_c statin dm smoking egfr bmi hba1c uacr zip time model_input total_cvd ascvd heart_failure chd stroke model over_years input_problems
1 40 female 111 FALSE 220 47 TRUE FALSE TRUE 89 36 5.7 15873.2 NA 10yr hba1c , pce_orig, White 0.017 0.011 0.007 0.006 0.006 hba1c 10 NA
1 40 female 111 FALSE 220 47 TRUE FALSE TRUE 89 36 5.7 15873.2 NA 10yr hba1c , pce_orig, White NA 0.030 NA NA NA pce_orig 10 NA
2 55 male 113 TRUE 312 43 FALSE FALSE TRUE calc_egfr(cr = 0.5) calc_bmi(height = 71L, weight = 166L) 8.1 NA 77642 10yr NA 0.090 0.076 0.034 0.054 0.024 full 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 50 36 11.9 NA 48206 both base , pce_orig, Black 0.132 0.071 0.215 0.025 0.056 base 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 50 36 11.9 NA 48206 both base , pce_orig, Black NA 0.084 NA NA NA pce_orig 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 50 36 11.9 NA 48206 both base , pce_orig, Black 0.413 0.229 0.581 0.089 0.176 base 30 NA
4 51 male 98 TRUE 282 73 TRUE FALSE TRUE calc_egfr(cr = 0.6) calc_bmi(height = 76L, weight = 153L) NA NA 85206 10yr NA 0.068 0.039 0.032 0.022 0.017 sdi 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 90 23 9.6 7902.2 NA both full , pce_both, White 0.525 0.385 0.322 0.328 0.171 full 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 90 23 9.6 7902.2 NA both full , pce_both, White NA 0.381 NA NA NA pce_orig 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 90 23 9.6 7902.2 NA both full , pce_both, White NA 0.358 NA NA NA pce_rev 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 90 23 9.6 7902.2 NA both full , pce_both, White 0.789 0.644 0.608 0.584 0.360 full 30 NA
6 58 male 165 TRUE 262 55 FALSE TRUE FALSE calc_egfr(cr = 0.5) calc_bmi(height = 70L, weight = 194L) NA NA NA 30yr NA 0.507 0.371 0.348 0.245 0.201 base 30 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 49 35 NA NA 98591 both uacr , pce_both, Other 0.097 0.062 0.049 0.036 0.029 uacr 10 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 49 35 NA NA 98591 both uacr , pce_both, Other NA 0.107 NA NA NA pce_orig 10 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 49 35 NA NA 98591 both uacr , pce_both, Other NA 0.070 NA NA NA pce_rev 10 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 49 35 NA NA 98591 both uacr , pce_both, Other 0.361 0.231 0.221 0.146 0.112 uacr 30 NA
8 36 male 169 TRUE 214 72 TRUE TRUE FALSE calc_egfr(cr = 0.6) calc_bmi(height = 68L, weight = 146L) NA NA NA both NA 0.082 0.047 0.057 0.018 0.036 base 10 NA
8 36 male 169 TRUE 214 72 TRUE TRUE FALSE calc_egfr(cr = 0.6) calc_bmi(height = 68L, weight = 146L) NA NA NA both NA 0.353 0.206 0.255 0.088 0.159 base 30 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 86 34 NA 5521.5 NA both sdi , pce_both, Other 0.177 0.131 0.053 0.089 0.057 sdi 10 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 86 34 NA 5521.5 NA both sdi , pce_both, Other NA 0.133 NA NA NA pce_orig 10 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 86 34 NA 5521.5 NA both sdi , pce_both, Other NA 0.107 NA NA NA pce_rev 10 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 86 34 NA 5521.5 NA both sdi , pce_both, Other 0.604 0.474 0.295 0.373 0.251 sdi 30 NA
10 47 male 109 TRUE 297 38 TRUE TRUE TRUE calc_egfr(cr = 0.5) calc_bmi(height = 69L, weight = 174L) NA NA 48708 both NA 0.203 0.140 0.052 0.115 0.039 sdi 10 NA
10 47 male 109 TRUE 297 38 TRUE TRUE TRUE calc_egfr(cr = 0.5) calc_bmi(height = 69L, weight = 174L) NA NA 48708 both NA 0.560 0.418 0.224 0.366 0.143 sdi 30 NA

The above scenario might be less realistic than something like having columns in a data frame for creatinine in mg/dL or μmol/L, height in cm, and weight in kg, and wanting to construct the calls from those. No worries, this is also quite doable.

dat_with_cr_cm_kg <- dat_with_pce_requests |> 
  dplyr::mutate(
    # Let's use values for `cr` in mg/dL, `cm`, and `kg` that would yield the
    # values originally entered directly for `egfr` and `bmi` in
    # `make_vignette_dat()` to demonstrate identical results when using the
    # direct values for eGFR and BMI vs. using calls to the convenience
    # functions. This is why the function `make_vignette_dat()` specifies values
    # for `age`, `sex`, `egfr`, and `bmi` directly while letting others vary
    # randomly.
    cr = c(1, 1.2, 0.9, 1.2, 0.9, 1.2, 0.8, 1.1, 0.9, 1.3),
    cm = c(199, 182, 184, 197, 189, 187, 191, 163, 199, 171),
    kg = c(148, 109, 127, 111, 134, 126, 134, 76, 74, 113),
    # Now, we'll create new list columns containing calls to calculate eGFR and
    # BMI (and remember, `dat_with_pce_requests` will already have columns for
    # `egfr` and `bmi`).
    egfr_call = lapply(
      seq_len(nrow(dat_with_pce_requests)),
      function(x) {
        call("calc_egfr", cr = cr[[x]])
      }
    ),
    bmi_call = lapply(
      seq_len(nrow(dat_with_pce_requests)),
      function(x) {
        call("calc_bmi", height = cm[[x]], weight = kg[[x]], units = "metric")
      }
    )
  )

res_with_calls <- est_risk(
  use_dat = dat_with_cr_cm_kg, 
  # Instruct `est_risk()` to use the call columns, else it will default to
  # grabbing values from `egfr` and `bmi`, which have direct values in them.
  egfr = "egfr_call", # Again, can pass column names as a character string ...
  bmi = bmi_call,     # ... or as a symbol
  progress = FALSE
)

res_without_calls <- est_risk(
  use_dat = dat_with_cr_cm_kg,
  # If you don't specify the call columns, `est_risk()` will default to using
  # the columns `egfr` and `bmi`, which have the original, direct values for
  # eGFR and BMI
  progress = FALSE
)

knitr::kable(res_with_calls)
preventr_id age sex sbp bp_tx total_c hdl_c statin dm smoking egfr bmi hba1c uacr zip time model_input cr cm kg egfr_call bmi_call total_cvd ascvd heart_failure chd stroke model over_years input_problems
1 40 female 111 FALSE 220 47 TRUE FALSE TRUE 73 37.4 5.7 15873.2 NA 10yr hba1c , pce_orig, White 1.0 199 148 calc_egfr(cr = 1) calc_bmi(height = 199, weight = 148, units = “metric”) 0.017 0.012 0.009 0.006 0.006 hba1c 10 NA
1 40 female 111 FALSE 220 47 TRUE FALSE TRUE 73 37.4 5.7 15873.2 NA 10yr hba1c , pce_orig, White 1.0 199 148 calc_egfr(cr = 1) calc_bmi(height = 199, weight = 148, units = “metric”) NA 0.030 NA NA NA pce_orig 10 NA
2 55 male 113 TRUE 312 43 FALSE FALSE TRUE 71 32.9 8.1 NA 77642 10yr NA 1.2 182 109 calc_egfr(cr = 1.2) calc_bmi(height = 182, weight = 109, units = “metric”) 0.092 0.079 0.039 0.056 0.026 full 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 80 37.5 11.9 NA 48206 both base , pce_orig, Black 0.9 184 127 calc_egfr(cr = 0.9) calc_bmi(height = 184, weight = 127, units = “metric”) 0.079 0.045 0.130 0.015 0.035 base 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 80 37.5 11.9 NA 48206 both base , pce_orig, Black 0.9 184 127 calc_egfr(cr = 0.9) calc_bmi(height = 184, weight = 127, units = “metric”) NA 0.084 NA NA NA pce_orig 10 NA
3 45 female 151 TRUE 215 99 FALSE TRUE TRUE 80 37.5 11.9 NA 48206 both base , pce_orig, Black 0.9 184 127 calc_egfr(cr = 0.9) calc_bmi(height = 184, weight = 127, units = “metric”) 0.344 0.192 0.503 0.073 0.147 base 30 NA
4 51 male 98 TRUE 282 73 TRUE FALSE TRUE 73 28.6 NA NA 85206 10yr NA 1.2 197 111 calc_egfr(cr = 1.2) calc_bmi(height = 197, weight = 111, units = “metric”) 0.070 0.040 0.029 0.023 0.018 sdi 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both full , pce_both, White 0.9 189 134 calc_egfr(cr = 0.9) calc_bmi(height = 189, weight = 134, units = “metric”) 0.536 0.395 0.425 0.335 0.179 full 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both full , pce_both, White 0.9 189 134 calc_egfr(cr = 0.9) calc_bmi(height = 189, weight = 134, units = “metric”) NA 0.381 NA NA NA pce_orig 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both full , pce_both, White 0.9 189 134 calc_egfr(cr = 0.9) calc_bmi(height = 189, weight = 134, units = “metric”) NA 0.358 NA NA NA pce_rev 10 NA
5 52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both full , pce_both, White 0.9 189 134 calc_egfr(cr = 0.9) calc_bmi(height = 189, weight = 134, units = “metric”) 0.798 0.656 0.718 0.594 0.373 full 30 NA
6 58 male 165 TRUE 262 55 FALSE TRUE FALSE 70 36.0 NA NA NA 30yr NA 1.2 187 126 calc_egfr(cr = 1.2) calc_bmi(height = 187, weight = 126, units = “metric”) 0.555 0.417 0.482 0.280 0.232 base 30 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 86 36.7 NA NA 98591 both uacr , pce_both, Other 0.8 191 134 calc_egfr(cr = 0.8) calc_bmi(height = 191, weight = 134, units = “metric”) 0.067 0.046 0.033 0.026 0.021 uacr 10 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 86 36.7 NA NA 98591 both uacr , pce_both, Other 0.8 191 134 calc_egfr(cr = 0.8) calc_bmi(height = 191, weight = 134, units = “metric”) NA 0.107 NA NA NA pce_orig 10 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 86 36.7 NA NA 98591 both uacr , pce_both, Other 0.8 191 134 calc_egfr(cr = 0.8) calc_bmi(height = 191, weight = 134, units = “metric”) NA 0.070 NA NA NA pce_rev 10 NA
7 57 female 125 TRUE 262 45 TRUE FALSE TRUE 86 36.7 NA NA 98591 both uacr , pce_both, Other 0.8 191 134 calc_egfr(cr = 0.8) calc_bmi(height = 191, weight = 134, units = “metric”) 0.325 0.214 0.195 0.134 0.102 uacr 30 NA
8 36 male 169 TRUE 214 72 TRUE TRUE FALSE 89 28.6 NA NA NA both NA 1.1 163 76 calc_egfr(cr = 1.1) calc_bmi(height = 163, weight = 76, units = “metric”) 0.085 0.049 0.058 0.019 0.038 base 10 NA
8 36 male 169 TRUE 214 72 TRUE TRUE FALSE 89 28.6 NA NA NA both NA 1.1 163 76 calc_egfr(cr = 1.1) calc_bmi(height = 163, weight = 76, units = “metric”) 0.390 0.233 0.310 0.101 0.180 base 30 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 78 18.7 NA 5521.5 NA both sdi , pce_both, Other 0.9 199 74 calc_egfr(cr = 0.9) calc_bmi(height = 199, weight = 74, units = “metric”) 0.181 0.134 0.046 0.090 0.058 sdi 10 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 78 18.7 NA 5521.5 NA both sdi , pce_both, Other 0.9 199 74 calc_egfr(cr = 0.9) calc_bmi(height = 199, weight = 74, units = “metric”) NA 0.133 NA NA NA pce_orig 10 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 78 18.7 NA 5521.5 NA both sdi , pce_both, Other 0.9 199 74 calc_egfr(cr = 0.9) calc_bmi(height = 199, weight = 74, units = “metric”) NA 0.107 NA NA NA pce_rev 10 NA
9 49 female 150 FALSE 243 21 TRUE TRUE FALSE 78 18.7 NA 5521.5 NA both sdi , pce_both, Other 0.9 199 74 calc_egfr(cr = 0.9) calc_bmi(height = 199, weight = 74, units = “metric”) 0.611 0.482 0.241 0.378 0.258 sdi 30 NA
10 47 male 109 TRUE 297 38 TRUE TRUE TRUE 68 38.6 NA NA 48708 both NA 1.3 171 113 calc_egfr(cr = 1.3) calc_bmi(height = 171, weight = 113, units = “metric”) 0.211 0.147 0.094 0.120 0.042 sdi 10 NA
10 47 male 109 TRUE 297 38 TRUE TRUE TRUE 68 38.6 NA NA 48708 both NA 1.3 171 113 calc_egfr(cr = 1.3) calc_bmi(height = 171, weight = 113, units = “metric”) 0.616 0.476 0.388 0.417 0.176 sdi 30 NA

identical(res_with_calls, res_without_calls)  
#> [1] TRUE

I don’t want to use use_dat and add_to_dat

Then don’t!

Perhaps you’ve already created a workflow and don’t want to update it to incorporate use_dat or add_to_dat. Perhaps your use case requires some more nuanced intermediate steps that wouldn’t work in conjunction with this new functionality. Perhaps you just vehemently dislike this functionality for some reason (though, admittedly, I would be curious to hear your reasons if this is you; feel free to get in touch). Whatever the case may be, you can still use est_risk() in the same way as before. And this includes being able to use a data frame with est_risk(). The arguments use_dat and add_to_dat just likely make many workflows more streamlined. But for example, you can still use est_risk() with base R solutions like lapply() or (the perhaps lesser-known) Map(), or tidyverse solutions like those found in the purrr package.

Before we start, let’s remind ourselves what dat_with_cr_cm_kg looks like.

knitr::kable(head(dat_with_cr_cm_kg))
age sex sbp bp_tx total_c hdl_c statin dm smoking egfr bmi hba1c uacr zip time model cr cm kg egfr_call bmi_call
40 female 111 FALSE 220 47 TRUE FALSE TRUE 73 37.4 5.7 15873.2 NA 10yr hba1c , pce_orig, White 1.0 199 148 calc_egfr(cr = 1) calc_bmi(height = 199, weight = 148, units = “metric”)
55 male 113 TRUE 312 43 FALSE FALSE TRUE 71 32.9 8.1 NA 77642 10yr NA 1.2 182 109 calc_egfr(cr = 1.2) calc_bmi(height = 182, weight = 109, units = “metric”)
45 female 151 TRUE 215 99 FALSE TRUE TRUE 80 37.5 11.9 NA 48206 both base , pce_orig, Black 0.9 184 127 calc_egfr(cr = 0.9) calc_bmi(height = 184, weight = 127, units = “metric”)
51 male 98 TRUE 282 73 TRUE FALSE TRUE 73 28.6 NA NA 85206 10yr NA 1.2 197 111 calc_egfr(cr = 1.2) calc_bmi(height = 197, weight = 111, units = “metric”)
52 female 157 FALSE 280 23 FALSE TRUE TRUE 77 37.5 9.6 7902.2 NA both full , pce_both, White 0.9 189 134 calc_egfr(cr = 0.9) calc_bmi(height = 189, weight = 134, units = “metric”)
58 male 165 TRUE 262 55 FALSE TRUE FALSE 70 36.0 NA NA NA 30yr NA 1.2 187 126 calc_egfr(cr = 1.2) calc_bmi(height = 187, weight = 126, units = “metric”)

lapply()

Now, let’s use good old lapply() with dat_with_cr_cm_kg and est_risk().

(As a brief aside, you could also certainly use a good old for() loop here instead of lapply(). Although I do not give a direct demonstration of using a for() loop in this vignette, the examples using lapply() should make it clear how you could do this with a for() loop if you prefer that for some reason. Just make sure you’ve taken care of pre-allocation. This isn’t a requirement, but it’s good practice and helps with performance in general.)

# First, add `preventr_id` to data frame for joining later, then move it to the
# first position.
dat_with_cr_cm_kg <- dat_with_cr_cm_kg |> 
  dplyr::mutate(preventr_id = seq_len(nrow(dat))) |> 
  dplyr::relocate(preventr_id)

res_basic_lapply <- lapply(
  # Using the row numbers of `dat_with_cr_cm_kg` as `x` in `function(x)`...
  seq_len(nrow(dat_with_cr_cm_kg)),
  function(x) {
    # ... run `est_risk()` on each row of `dat_with_cr_cm_kg`
    est_risk(
      age = dat_with_cr_cm_kg[["age"]][[x]],
      sex = dat_with_cr_cm_kg[["sex"]][[x]],
      sbp = dat_with_cr_cm_kg[["sbp"]][[x]],
      bp_tx = dat_with_cr_cm_kg[["bp_tx"]][[x]],
      total_c = dat_with_cr_cm_kg[["total_c"]][[x]],
      hdl_c = dat_with_cr_cm_kg[["hdl_c"]][[x]],
      statin = dat_with_cr_cm_kg[["statin"]][[x]],
      dm = dat_with_cr_cm_kg[["dm"]][[x]],
      smoking = dat_with_cr_cm_kg[["smoking"]][[x]],
      egfr = dat_with_cr_cm_kg[["egfr"]][[x]],
      bmi = dat_with_cr_cm_kg[["bmi"]][[x]],
      hba1c = dat_with_cr_cm_kg[["hba1c"]][[x]],
      uacr = dat_with_cr_cm_kg[["uacr"]][[x]],
      zip = dat_with_cr_cm_kg[["zip"]][[x]],
      model = dat_with_cr_cm_kg[["model"]][[x]],
      time = dat_with_cr_cm_kg[["time"]][[x]],
      quiet = TRUE
    )  |>
      # Bind the rows of the return from `est_risk()` together.
      # (Side note: You can skip this step if you call `est_risk()` with
      # `collapse = TRUE`.)
      dplyr::bind_rows() |>
      # Add column `preventr_id` to facilitate reassociation with the input
      # data frame.
      dplyr::mutate(preventr_id = x)
  }
) |>
  # Bind all the results from the `lapply()` call together to make a
  # single data frame.
  dplyr::bind_rows() |> 
  # Finally, do a quick left join to match the results with their
  # corresponding input row in `dat_with_cr_cm_kg`.
  dplyr::left_join(
    x = dat_with_cr_cm_kg, 
    y = _, 
    by = "preventr_id",
    # Because both data frames will have a column named `model`, we'll provide
    # suffixes to distinguish them. The suffixes below will cause the column 
    # `model` in `dat_with_cr_cm_kg` to be renamed to `model_input` and column
    # `model` in the data frame from the pipe sequence (represented via `_`) 
    # retaining the same name.
    suffix = c("_input", "")  
  )

# If all has proceeded as it should've, `res_basic_lapply` should be identical
# to `res_without_calls` (and thus also to `res_with_calls`) from the above 
# example (spoiler, it will be).
identical(res_basic_lapply, res_without_calls)
#> [1] TRUE

As somewhat of a quick side note, repeatedly calling dat_with_cr_cm_kg[["{var}"]][[x]] in the lapply() call above is relatively inefficient vs. doing something like:

with(
  dat_with_cr_cm_kg[x, ],
  est_risk(
    age = age,
    sex = sex,
    ...
  )
)

However, dat_with_cr_cm_kg[["{var}"]][[x]] is at least unambiguous, whereas using with() requires understanding its data-masking feature.

I used the more verbose approach for the initial lapply() call for a reason, namely to show variations in approach that might be used. Subsequent lapply() calls will in fact make use of with().

In addition to reading the documentation for the with() function, you can read more about data masking here (from the rlang package) if interested.

To show some other variants using lapply(), I’m going to delve into the concept of metaprogramming just a wee bit. Hadley Wickham’s Advanced R has some excellent content about this in the metaprogramming section, and the tidyverse team has supported such efforts with the fantastic rlang package. However, I’m actually just going to use base R functions here.

In what follows, I define a function that allows us to alter (1) the first argument in the call to with() and (2) the arguments passed to the est_risk() call inside the lapply() call.

do_lapply_and_join <- function(dat, with_arg, ..., eval = TRUE) {
  
  dat <- substitute(dat)
  with_arg <- substitute(with_arg)
  dots <- eval(substitute(alist(...)))
  
  mini_cl <- bquote(
    {
      lapply(
        # Using the row numbers of `.(dat)` as `x` in `function(x)`...
        seq_len(nrow(.(dat))),
        function(x) {
          with(
            # With the data mask contained in `with_arg` ...
            .(with_arg),
              # ... run `est_risk()` with the arguments contained within `dots`.
              est_risk(..(dots)) 
          ) |>
            # The vast majority of the following is nearly identical to the
            # basic `lapply()` example; it does not make any further use of 
            # metaprogramming unless otherwise noted.
            dplyr::bind_rows() |>
            dplyr::mutate(preventr_id = x)
        }
      ) |>
        dplyr::bind_rows() |> 
        dplyr::left_join(
          x = .(dat),       # Note the use of `.(dat)` 
          y = _, 
          by = "preventr_id",
          suffix = c("_input", "")  
        )
    },
    splice = TRUE           # This tells `bquote()` to splice anything in `..()`
  )
  
  if(eval) eval(mini_cl, parent.frame()) else mini_cl
}

We can now use this “augmented” lapply() for further demonstrations.

# Let's start by showing results identical to `res_basic_lapply`.
res_aug_lapply <- do_lapply_and_join(
  dat = dat_with_cr_cm_kg,
  with_arg = dat_with_cr_cm_kg[x, ],
  age = age,
  sex = sex,
  sbp = sbp,
  bp_tx = bp_tx,
  total_c = total_c,
  hdl_c = hdl_c,
  statin = statin,
  dm = dm,
  smoking = smoking,
  egfr = egfr,
  bmi = bmi,
  hba1c = hba1c,
  uacr = uacr,
  zip = zip,
  # Because of the data mask passed via argument `with_arg`, the evaluation
  # environment will be row x of the data frame (where x is defined within the
  # `lapply()` call). Thus, `model` will still be a list column, so I need to
  # get that list item out of the list column before passing it to
  # `est_risk()`.
  #
  # For `model`, I could instead do `unlist()`, but given this vignette also
  # demonstrates list columns containing calls (where `unlist()` will not do),
  # I will use `[[1]]` here for consistency. Note I can be confident the list
  # item I need from the list column `model` is indeed the first (and only)
  # list item, and the list item I extract via `[[1]]` will then either be
  # `NA` or a list with list items `main_model`, `other_models`, and
  # `race_eth` given how I created `dat_with_cr_cm_kg`.
  model = model[[1]],
  time = time,
  quiet = TRUE
)

Before we look at the results in res_aug_lapply, let’s understand what the call to do_lapply_and_join() is doing. With the above call, if we had instead set eval = FALSE, we would have gotten the following:

lapply(
  seq_len(nrow(dat_with_cr_cm_kg)),  # `dat_with_cr_cm_kg` replaces `.(dat)`
  function(x) {
    with(
      dat_with_cr_cm_kg[x, ],        # `dat_with_cr_cm_kg[x, ]` replaces 
      est_risk(                      # `.(with_arg)`
        age = age,                   
        sex = sex,                   # The arguments appearing in `est_risk()`
        sbp = sbp,                   # were spliced into the call from `..(dots)`
        bp_tx = bp_tx, 
        total_c = total_c, 
        hdl_c = hdl_c, 
        statin = statin, 
        dm = dm, 
        smoking = smoking, 
        egfr = egfr, 
        bmi = bmi, 
        hba1c = hba1c,
        uacr = uacr, 
        zip = zip, 
        model = model[[1]], 
        time = time, 
        quiet = TRUE
      ) 
    ) |>
      dplyr::bind_rows() |>
      dplyr::mutate(preventr_id = x)
  }
) |>
  dplyr::bind_rows() |> 
  dplyr::left_join(
    x = dat_with_cr_cm_kg,            # `dat_with_cr_cm_kg` replaces `.(dat)`
    y = _, 
    by = "preventr_id",
    suffix = c("_input", "")  
  )

… but that’s only partly true, because the real return would have been much harder to read due to the automatic formatting of the call and how piping works (e.g., lots of nested calls). For the curious, the real return (though functionally identical to the above) would have actually been something like:

{
    dplyr::left_join(x = dat_with_cr_cm_kg, y =
        dplyr::bind_rows(lapply(seq_len(nrow(dat_with_cr_cm_kg)), 
        function(x) {
            dplyr::mutate(dplyr::bind_rows(with(dat_with_cr_cm_kg[x, 
                ], est_risk(age = age, sex = sex, sbp = sbp, 
                bp_tx = bp_tx, total_c = total_c, hdl_c = hdl_c, 
                statin = statin, dm = dm, smoking = smoking, 
                egfr = egfr, bmi = bmi, hba1c = hba1c, uacr = uacr, 
                zip = zip, model = model[[1]], time = time, quiet = TRUE))), 
                preventr_id = x)
        })), by = "preventr_id", suffix = c("_input", 
        ""))
}

The point here is not to give an exhaustive (or exhausting!) vignette on metaprogramming, so I’ll stop there. I just wanted to briefly show what happens with the above function. With that under our belt, let’s confirm identicality of results.

identical(res_aug_lapply, res_basic_lapply)
#> [1] TRUE

Let’s now look at some variations.

res_aug_lapply_variant <- do_lapply_and_join(
  dat = dat_with_cr_cm_kg,
  with_arg = dat_with_cr_cm_kg,
  age = age[[x]],
  sex = sex[[x]],
  sbp = sbp[[x]],
  bp_tx = bp_tx[[x]],
  total_c = total_c[[x]],
  hdl_c = hdl_c[[x]],
  statin = statin[[x]],
  dm = dm[[x]],
  smoking = smoking[[x]],
  egfr = egfr[[x]],
  bmi = bmi[[x]],
  hba1c = hba1c[[x]],
  uacr = uacr[[x]],
  zip = zip[[x]],
  model = model[[x]],
  time = time[[x]],
  quiet = TRUE
)

identical(res_aug_lapply_variant, res_basic_lapply)
#> [1] TRUE

Calls are fine as well.

res_aug_lapply_with_calls <- do_lapply_and_join(
  dat = dat_with_cr_cm_kg,
  with_arg = dat_with_cr_cm_kg[x, ],
  age = age,
  sex = sex,
  sbp = sbp,
  bp_tx = bp_tx,
  total_c = total_c,
  hdl_c = hdl_c,
  statin = statin,
  dm = dm,
  smoking = smoking,
  # If needed, review the comment associated with `res_aug_lapply` to understand
  # why arguments `egfr`, `bmi`, and `model` are specified like this.
  egfr = egfr_call[[1]],
  bmi = bmi_call[[1]],
  hba1c = hba1c,
  uacr = uacr,
  zip = zip,
  model = model[[1]],
  time = time,
  quiet = TRUE
)

identical(res_aug_lapply_with_calls, res_basic_lapply)
#> [1] TRUE

You can construct calls in a more “on the fly” way as well.

res_aug_lapply_with_calls_in_flight <- do_lapply_and_join(
  dat = dat_with_cr_cm_kg,
  with_arg = dat_with_cr_cm_kg[x, ],
  age = age,
  sex = sex,
  sbp = sbp,
  bp_tx = bp_tx,
  total_c = total_c,
  hdl_c = hdl_c,
  statin = statin,
  dm = dm,
  smoking = smoking,
  egfr = call("calc_egfr", cr = cr),
  bmi = call("calc_bmi", height = cm, weight = kg, units = "metric"),
  hba1c = hba1c,
  uacr = uacr,
  zip = zip,
  model = model[[1]],
  time = time,
  quiet = TRUE
)

identical(res_aug_lapply_with_calls_in_flight, res_basic_lapply)
#> [1] TRUE

And of course, you can also pass optional behavior variables in the call (again, this will override any column that might exist in the data frame by the same name).

res_auto_opts_in_call <- est_risk(
  use_dat = dat_with_cr_cm_kg,
  model = "base",
  time = "10yr",
  progress = FALSE
)

res_aug_lapply_opts_in_call <- do_lapply_and_join(
  dat = dat_with_cr_cm_kg,
  with_arg = dat_with_cr_cm_kg[x, ],
  age = age,
  sex = sex,
  sbp = sbp,
  bp_tx = bp_tx,
  total_c = total_c,
  hdl_c = hdl_c,
  statin = statin,
  dm = dm,
  smoking = smoking,
  egfr = egfr,
  bmi = bmi,
  hba1c = hba1c,
  uacr = uacr,
  zip = zip,
  model = "base",
  time = "10yr",
  quiet = TRUE
)

identical(res_auto_opts_in_call, res_aug_lapply_opts_in_call)
#> [1] TRUE

Map()

est_risk() also works with Map().

do_map_and_join <- function(dat, ...) {

  dat <- dat |> dplyr::mutate(preventr_id = seq_len(nrow(dat)))
  dots <- eval(substitute(alist(...)))
  
  res <- eval(
    bquote(
      # With the data mask introduced by `dat`, evaluate `Map()` with the
      # function `est_risk()` and the arguments contained in `dots`.
      # (In other words, call `est_risk()` with the arguments in `dots` for
      # each row of `dat`.) 
      with(dat, Map(est_risk, ..(dots))),
      splice = TRUE
    )
  )
  
  # `res` from the above call to `Map()` will be a list, and the items in
  # the list may also be a list (e.g., a list of data frames), as such, we'll
  # need to iterate through `res` and bind the data frames together. We'll also
  # need to add the `preventr_id` column.
  for(i in seq_along(res)) {
    res[[i]] <- res[[i]] |> 
      dplyr::bind_rows() |>
      dplyr::mutate(preventr_id = i) |> 
      dplyr::relocate(preventr_id)
  }
  
  # Now do the left join, detailed previously in this vignette.
  dplyr::left_join(
      x = dat, 
      y = dplyr::bind_rows(res), 
      by = "preventr_id",
      suffix = c("_input", "")  
    )
}

res_map <- do_map_and_join(
  dat_with_cr_cm_kg,
  age = age,
  sex = sex,
  sbp = sbp,
  bp_tx = bp_tx,
  total_c = total_c,
  hdl_c = hdl_c,
  statin = statin,
  dm = dm,
  smoking = smoking,
  egfr = egfr,
  bmi = bmi,
  hba1c = hba1c,
  uacr = uacr,
  zip = zip,
  model = "base",
  time = "10yr",
  quiet = TRUE
)

identical(res_auto_opts_in_call, res_map)
#> [1] TRUE

Let’s now look at some variants.

res_map_all_cols <- do_map_and_join(
  dat_with_cr_cm_kg,
  age = age,
  sex = sex,
  sbp = sbp,
  bp_tx = bp_tx,
  total_c = total_c,
  hdl_c = hdl_c,
  statin = statin,
  dm = dm,
  smoking = smoking,
  # Note I'm passing the call columns here, showing you can still use the
  # convenience functions (stored as calls in list columns) with `Map()`.
  egfr = egfr_call,
  bmi = bmi_call,
  hba1c = hba1c,
  uacr = uacr,
  zip = zip,
  model = model,
  time = time,
  quiet = TRUE
)

identical(res_map_all_cols, res_basic_lapply)
#> [1] TRUE

# You can also pass applicable optional behavior variables.
res_map_only_10yr_hba1c_not_quiet <- do_map_and_join(
  dat_with_cr_cm_kg,
  age = age,
  sex = sex,
  sbp = sbp,
  bp_tx = bp_tx,
  total_c = total_c,
  hdl_c = hdl_c,
  statin = statin,
  dm = dm,
  smoking = smoking,
  egfr = egfr_call,
  bmi = bmi_call,
  hba1c = hba1c,
  uacr = uacr,
  zip = zip,
  model = "hba1c",
  time = "10yr",
  quiet = FALSE
)
#> PREVENT estimates are from: Base model adding HbA1c.
#> PREVENT estimates are from: Base model adding HbA1c.
#> PREVENT estimates are from: Base model adding HbA1c.
#> PREVENT estimates are from: Base model adding HbA1c.
#> PREVENT estimates are from: Base model adding HbA1c.
#> PREVENT estimates are from: Base model adding HbA1c.
#> PREVENT estimates are from: Base model adding HbA1c.
#> PREVENT estimates are from: Base model adding HbA1c.
#> PREVENT estimates are from: Base model adding HbA1c.
#> PREVENT estimates are from: Base model adding HbA1c.

# Despite `dat_with_cr_cm_kg` having columns `time` and `model`, the `time` and
# `model` arguments in the call to `est_risk()` (via `Map()`) get priority.
dat_with_cr_cm_kg[["model"]]
#> [[1]]
#> [[1]]$main_model
#> [1] "hba1c"
#> 
#> [[1]]$other_models
#> [1] "pce_orig"
#> 
#> [[1]]$race_eth
#> [1] "White"
#> 
#> 
#> [[2]]
#> [1] NA
#> 
#> [[3]]
#> [[3]]$main_model
#> [1] "base"
#> 
#> [[3]]$other_models
#> [1] "pce_orig"
#> 
#> [[3]]$race_eth
#> [1] "Black"
#> 
#> 
#> [[4]]
#> [1] NA
#> 
#> [[5]]
#> [[5]]$main_model
#> [1] "full"
#> 
#> [[5]]$other_models
#> [1] "pce_both"
#> 
#> [[5]]$race_eth
#> [1] "White"
#> 
#> 
#> [[6]]
#> [1] NA
#> 
#> [[7]]
#> [[7]]$main_model
#> [1] "uacr"
#> 
#> [[7]]$other_models
#> [1] "pce_both"
#> 
#> [[7]]$race_eth
#> [1] "Other"
#> 
#> 
#> [[8]]
#> [1] NA
#> 
#> [[9]]
#> [[9]]$main_model
#> [1] "sdi"
#> 
#> [[9]]$other_models
#> [1] "pce_both"
#> 
#> [[9]]$race_eth
#> [1] "Other"
#> 
#> 
#> [[10]]
#> [1] NA
dat_with_cr_cm_kg[["time"]]
#>  [1] "10yr" "10yr" "both" "10yr" "both" "30yr" "both" "both" "both" "both"

all.equal(unique(res_map_only_10yr_hba1c_not_quiet[["over_years"]]), 10)
#> [1] TRUE
all.equal(unique(res_map_only_10yr_hba1c_not_quiet[["model"]]), "hba1c")
#> [1] TRUE

purrr::pmap()

When calling purrr::pmap() with a bare call to the function and the entire data frame, purrr::pmap() expects either (1) no unused arguments (i.e., each column in the data frame will match an argument in the function being mapped over), or (2) the unused columns are absorbed by a ... argument in the function over which you are mapping. As such, we’ll need to slightly adjust the data frame dat_with_cr_cm_kg to remove columns not corresponding to an argument in est_risk().

pmap_data_frame_approach <- 
  dat_with_cr_cm_kg |>
  # Remove columns not corresponding to an argument in `est_risk()`.
  dplyr::select(-c(preventr_id, cr, cm, kg, egfr_call, bmi_call)) |> 
  purrr::pmap(est_risk)
#> PREVENT estimates are from: Base model adding HbA1c.
#> PREVENT estimates are from: Base model adding HbA1c, SDI, and UACR (also referred to as the full model).
#> PREVENT estimates are from: Base model.
#> PREVENT estimates are from: Base model adding SDI.
#> PREVENT estimates are from: Base model adding HbA1c, SDI, and UACR (also referred to as the full model).
#> PREVENT estimates are from: Base model.
#> PREVENT estimates are from: Base model adding UACR.
#> PREVENT estimates are from: Base model.
#> PREVENT estimates are from: Base model adding SDI.
#> PREVENT estimates are from: Base model adding SDI.

# Very similar to the `Map()` examples above, we'll need to bind the results
# from `purrr::pmap()` together and do some other minor actions, so I've 
# converted that into a mini-function to avoid repetition in these examples.
combine_pmap_res_and_join <- function(pmap_res, dat) {
  for(i in seq_along(pmap_res)) {
    pmap_res[[i]] <- pmap_res[[i]] |> 
      dplyr::bind_rows() |>
      dplyr::mutate(preventr_id = i) |> 
      dplyr::relocate(preventr_id)
  }
  
  dplyr::left_join(
    x = dat, 
    y = dplyr::bind_rows(pmap_res), 
    by = "preventr_id",
    suffix = c("_input", "")  
  )
}

pmap_data_frame_approach <- 
  combine_pmap_res_and_join(pmap_data_frame_approach, dat_with_cr_cm_kg)

identical(pmap_data_frame_approach, res_basic_lapply)
#> [1] TRUE

As an alternative, you can leave the data frame alone, but pass purrr::pmap() a more explicitly-delineated list for argument .l.

pmap_list_approach <- purrr::pmap(
  with(
    dat_with_cr_cm_kg,
    list(
      age = age,
      sex = sex,
      sbp = sbp,
      bp_tx = bp_tx,
      total_c = total_c,
      hdl_c = hdl_c,
      statin = statin,
      dm = dm,
      smoking = smoking,
      egfr = egfr,
      bmi = bmi,
      hba1c = hba1c,
      uacr = uacr,
      zip = zip,
      model = model,
      time = time,
      # Note passing an explicitly-delineated list for argument `.l` allows us
      # to easily specify the `quiet` argument here
      quiet = TRUE
    )
  ),
  est_risk
)

pmap_list_approach <- 
  combine_pmap_res_and_join(pmap_list_approach, dat_with_cr_cm_kg)

identical(pmap_list_approach, res_basic_lapply)
#> [1] TRUE

Calls continue to be fine.

pmap_list_approach_with_call <- purrr::pmap(
  with(
    dat_with_cr_cm_kg,
    list(
      age = age,
      sex = sex,
      sbp = sbp,
      bp_tx = bp_tx,
      total_c = total_c,
      hdl_c = hdl_c,
      statin = statin,
      dm = dm,
      smoking = smoking,
      egfr = egfr_call,
      bmi = bmi_call,
      hba1c = hba1c,
      uacr = uacr,
      zip = zip,
      model = model,
      time = time,
      quiet = TRUE
    )
  ),
  est_risk
)

pmap_list_approach_with_call <- 
  combine_pmap_res_and_join(pmap_list_approach_with_call, dat_with_cr_cm_kg)

identical(pmap_list_approach_with_call, res_basic_lapply)
#> [1] TRUE

Wrap-up

If you’ve made it this far, kudos on reading through the whole thing. Admittedly, I didn’t anticipate this vignette getting this long when I starting writing it, but here we are. I hope this has been helpful.