R/utils.R

Defines functions .infinite_to_NA .has_infinite .assert_cor_value .assert_args .assert_positive .assert_no_NA .assert_is_numeric .assert_column_exists .get_columns .compute_and_format

Documented in .compute_and_format

#' Compute And Format
#' 
#' Compute effect size and do some pre-return tasks
#'
#' @param data Data frame to use
#' @param effsize_func Function used to compute the effect size
#' @param effsize_args Arguments needed for that function
#' @param col_names Vector of length 2 with the name of the columns 
#' @param append Logical. If append or not the effect sizes to the data
#'
#' @keywords internal
.compute_and_format <- function(
  effsize_func,
  effsize_args,
  data,
  col_names,
  append
) {
  checkmate::assert_choice(effsize_func, choices = c(
    ### lnRR ###
    ".simple_lnRR",
    ".main_lnRR_Nakagawa",
    ".main_lnRR_Morris",
    ".interaction_lnRR",
    ### lnVR ###
    ".simple_lnVR",
    ".main_lnVR",
    ".interaction_lnVR",
    ### lnCVR ###
    ".simple_lnCVR",
    ".main_lnCVR",
    ".interaction_lnCVR",
    ### SMD ###
    ".simple_SMD",
    ".main_SMD",
    ".interaction_SMD",
    ### Factor x Time Interactions ###
    ".time_interaction_lnRR",
    ".time_interaction_lnVR",
    ".time_interaction_lnCVR",
    ".time_interaction_SMD"
    )
  )
  df <- do.call(effsize_func, effsize_args)
  names(df) <- col_names

  if (.has_infinite(df)) {
    df <- .infinite_to_NA(df)
  }

  if (append) {
    df <- cbind(data, df)
  }

  return(df)
}


.get_columns <- function(columns_list, data) {
  # Find and extract the columns from columns_list in data

  # NAs are required columns missing
  if (any(is.na(names(columns_list)))) {
    fun_name <- as.character(sys.call(-1)[[1]])
    stop(sprintf("Some required numeric arguments are missing. See ?%s", fun_name),
         call. = FALSE)
  }
  
  return_cols <- list()

  for (arg in names(columns_list)) {
    col_name <- as.character(columns_list[[arg]])
  
    .assert_column_exists(col_name, data)
    .assert_is_numeric(col_name, data)
    .assert_no_NA(col_name, data)
    
    # If column is sample sizes (e.g., Ctrl_n)
    if (endsWith(arg, "_n")) {
      tryCatch(
        .assert_positive(col_name, data),
        error = function(e) stop(e$message, "\nSample sizes must be positive values.", call. = FALSE)
      )
    }

    return_cols[[arg]] <- data[[col_name]]
  }

  return(return_cols)
}


.assert_column_exists <- function(col_name, data) {
  if (!(col_name %in% names(data))) {
    stop(sprintf("the column %s doesn't exists.", col_name), call. = FALSE)
  }
}


.assert_is_numeric <- function(col_name, data) {
  column = data[[col_name]]
  if (!is.numeric(column)) {
    stop(sprintf("the column %s is not numeric.", col_name), call. = FALSE)
  }
}


.assert_no_NA <- function(col_name, data) {
  column = data[[col_name]]
  if (anyNA(column)) {
    stop(sprintf("the column %s has NAs.", col_name), call. = FALSE)
  }
}


.assert_positive <- function(col_name, data) {
  column = data[[col_name]]
  if (any(column <= 0)) {
    stop(sprintf("the column %s has zeros or negative values", col_name), call. = FALSE)
  }
}


.assert_args <- function(col_names, append, data) {
  checkmate::assert_character(col_names, len = 2)
  checkmate::assert_logical(append, len = 1)
  checkmate::assert_data_frame(data)
}


.assert_cor_value <- function(x, data) {
  # Check if x is a valid correlation value that ranges between -1 and 1
  if (!checkmate::test_numeric(x, lower = -1, upper = 1)) {
    stop(sprintf(
      "Correlation values must be between -1 and 1, but some values in %s are out of range.",
      deparse(substitute(x))
    ), call. = FALSE)
  }

  # Check if x is a number or a vector of length of data
  if (!(checkmate::test_numeric(x, len = nrow(data)) ||
        checkmate::test_numeric(x, len = 1))) {
    stop(sprintf(
      "length of %s must be 1 or equal to data, but is %d",
      deparse(substitute(x)), length(x)
    ), call. = FALSE)
  }
}


.has_infinite <- function(x) {
  return(any(is.infinite(unlist(x))))
}


.infinite_to_NA <- function(df) {
  df_replaced <- lapply(df, function(x) { 
           replace(x, is.infinite(x), NA)
  })
  warning("Some effect sizes resulted in infinite values (division by zero). These values were recoded as NA.",
          call. = FALSE)

  return(as.data.frame(df_replaced))
}

Try the minter package in your browser

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

minter documentation built on May 3, 2026, 5:06 p.m.