R/utils_internal.R

Defines functions list_keep_or_drop args_as_lookup sanitize_path check_args_contain_illegal_flags error_illegal_flag flag_is_illegal concatenate_args warn_multimatched_arg find_multimatched_args count_matched_args convert_names true_to_empty convert_logical_to_empty drop_list_NULL drop_list_logicals drop_list_fun

utils::globalVariables(".")

#' Drops list members by function
#'
#' @param list any list object
#' @param fun a function that evaluates to boolean value, used to filter members of list
#'
#' @return the same list without any entries with logical vectors
#' 
#' 
#' @examples
#' myList <- list(a = 1, b = TRUE, c = FALSE)
#' # This will drop logicals from list
#' drop_list_fun(myList, fun = is.logical)
#' 
#' @noRd
drop_list_fun <- function(list, fun){
  testthat::expect_equal(class(fun), "function")
  list[!purrr::map_lgl(list, fun)]
}

#' Removes logical vectors from list objects
#'
#' @param list any list object
#'
#' @return the same list without any entries with logical vectors
#'
#' @examples
#' myList <- list(a = 1, b = TRUE, c = FALSE)
#' drop_list_logicals(myList)
#' @noRd
drop_list_logicals <- function(list){
  drop_list_fun(list, is.logical)
}

#' Removes logical vectors from list objects
#'
#' @param list any list object
#'
#' @return the same list without any entries with logical vectors
#'
#' @examples
#' myList <- list(a = 1, b = TRUE, c = FALSE, d = NULL)
#' drop_list_NULL(myList)
#' @noRd
drop_list_NULL <- function(list){
  drop_list_fun(list, is.null)
}


#' Replaces list entries of bool of specified value with empty strings
#'
#' @param list a list
#' @param bool whether to convert TRUE or FALSE to empty string (default TRUE)
#'
#' @return same list with entries replaced 
#' 
#'
#' @examples
#' myList <- list(a = TRUE, b = FALSE)
#' convert_logical_to_empty(myList)
#' @noRd
convert_logical_to_empty <- function(list, bool = TRUE){
  list <- purrr::map(list, ~{
    if (!is.logical(.x)) return(.x)

    if (.x == bool) return("")

    return(.x)
  })
  
}

#' Replace list entries with boolean values of TRUE with empty string
#'
#' @param list a list
#'
#' @return list where entries with TRUE are replaced for ""
#'
#' @examples
#' myList <- list(a = TRUE)
#' \dontrun{
#' true_to_empty(myList)
#' }
#' @noRd
true_to_empty <- function(list){
  list <- convert_logical_to_empty(list, TRUE)
  return(list)
}


#' converts names of object (obj) to values of named vector (dict)
#'
#' @param obj list or vector
#' @param dict named vector where names are converted to values
#'
#' @return
#'
#' @examples
#' dict <- c("long" = "l")
#' dots <- list("long" = 1, "a" = 1)
#' \dontrun{
#' convert_names(dots, dict)
#' }
#' @noRd
convert_names <- function(obj, dict){
  testthat::expect_named(obj)
  testthat::expect_named(dict)
  
  names(obj)[names(obj) %in% names(dict)] <- dict[names(obj)[names(obj) %in% names(dict)]]
  return(obj)
}

#' count number of times dots contains match to long or short version of flag
#'
#' @param value value from argsDict entry
#' @param name name of argsDict entry
#' @param dots list of dots from getDots()
#'
#' @return count of matches to value or name (should be 1 or greater if set, 0 if not)
#'
#' @importFrom magrittr %>%
#' 
#' @examples
#' dict <- c("name" = "value")
#' dots <- list("name" = 1, "value" = 1)
#' \dontrun{
#' count_matched_args(dict[1], names(dict)[1], dots)
#' }
#'
#' dict <- c("name" = "value")
#' dots <- list("name" = 1)
#' \dontrun{
#' count_matched_args(dict[1], names(dict)[1], dots)
#' }
#' @noRd
count_matched_args <- function(value, name, dots){
  names(dots) %in% c(value, name) %>% sum
}

#' returns list of named args with multiple definitions
#'
#' @param vec named vector output of count_matched_args
#'
#' @return names of args with matches > 1
#'
#' @examples
#' vec <- c("setArg" = 1, "unsetArg" = 0, "multiSetArg" = 3)
#' \dontrun{
#' find_multimatched_args(vec)
#' }
#' @noRd
find_multimatched_args <- function(vec){
  testthat::expect_named(vec)
  names(vec[vec > 1])
}

#' Prints message warning user of each argument that is defined multiple times in function call
#'
#' @param name name of arg
#'
#' @return
#'
#' @examples
#' \dontrun{
#' warn_multimatched_arg("arg")
#' }
#' @noRd
warn_multimatched_arg <- function(name){
  message(paste0(name, " is set multiple times in function call, ensure this is correct behavior."))
}

#' Combines key/value pairs from named vector by separator
#'
#' @param dict argsDict
#' @param sep separator
#'
#' @return
#'
#' @examples
#' dict <- c("name" = "val")
#' \dontrun{
#' concatenate_args(dict)
#' }
#' @noRd
concatenate_args <- function(dict, sep = "/"){
  paste(names(dict), dict, sep = sep)
}


#' Searches for illegal values in string
#'
#' @param flag string
#' @param illegal_chars vector of flags forbidden in string
#'     default values are: "&", "|", ";", "(", ")", "{", "}", "$", "\@", "/"
#' @return Boolean value for each flag
#'
#' @examples
#' \dontrun{
#' flag_is_illegal("&&echo")
#' }
#' @noRd
flag_is_illegal <- function(flag, 
                                  illegal_chars = c("&", "\\|", ";", "\\(", "\\)", "\\{", "\\}", "\\$", "\\@", "\\/", " ")){
  any(purrr::map_lgl(illegal_chars, grepl, flag))
}

#' Prints illegal flag warning for name
#'
#' @param name name containing illegal flag
#'
#' @return
#'
#' @examples
#' \dontrun{
#' error_illegal_flag("&&echo")
#' }
#' @noRd
error_illegal_flag <- function(name){
  stop(paste0(name, " is not a valid flag name. Contains illegal character."))
}

#' Checks dots for illegal flag names
#'
#' @param args list output of get*Args family function
#'
#' @return prints warning for each illegal flag
#'
#' @importFrom magrittr %>%
#' 
#' @examples
#' theFunction <- function(...) { cmd_args_dots() }
#' theDots <-  theFunction(validFlag = TRUE, `&illegalFlag` = "will error")
#' \dontrun{
#' theArgs <-  dotsToArgs(theDots)
#' }
#' @noRd
check_args_contain_illegal_flags <- function(args){
  is_illegal <- purrr::map_lgl(names(args), flag_is_illegal) %>% 
    purrr::set_names(names(args)) 
  
  illegals <- is_illegal[is_illegal]
  
  purrr::walk(names(illegals), error_illegal_flag)
}

#' Creates system-agnostic paths
#' 
#' Used to expand path shortcuts (like ~), and make system-agnostic calls.
#' In particular can be useful for trimming trailing slashes for path names.
#'
#' @param path file path
#'
#' @return sanitized file path
#'
#' @examples
#' path <- "~/bin/"
#' \dontrun{
#' sanitize_path(path)
#' }
#' @noRd
sanitize_path <- function(path){
  file.path(dirname(path), basename(path))
}

#' Converts list of arguments to named vector
#' 
#' This function is used when a lookup table isn't supplied for arguments converted to flags.
#' Used to say "all args defined should be kept as their original definition"
#'
#' @param args output of get*Args family function
#'
#' @return 
#'
#' @noRd
args_as_lookup <- function(args){
  flag_lookup <- names(args)
  names(flag_lookup) <- names(args)
  return(flag_lookup)
}


#' Drops or keeps items by name from list
#'
#' @param list a named list
#' @param keep names to keep
#' @param drop names to drop
#'
#' @return
#'
#' @noRd
list_keep_or_drop <- function(list, keep = NULL, drop = NULL){
  
  if (length(list) == 0){return(list)}
  
  testthat::expect_named(list)
  
  if (!is.null(keep) & !is.null(drop)) {
    stop("only one of keep or drop may be defined")
  }
  
  if (is.null(keep) & is.null(drop)) { return(list) }
  
  if (!is.null(keep)){
    testthat::expect_type(keep, "character")
    filteredList <- cmd_list_keep(list, keep)
  }
  
  if (!is.null(drop)){
    testthat::expect_type(drop, "character")
    filteredList <- cmd_list_drop(list, drop)
  }
  
  return(filteredList)
  
}
snystrom/dotargs documentation built on Oct. 18, 2020, 8:39 a.m.