#' Undesirable function linter
#'
#' Report the use of undesirable functions and suggest an alternative.
#'
#' @param fun Character vector of undesirable function names. Input can be any of three types:
#' - Unnamed entries must be a character string specifying an undesirable function.
#' - For named entries, the name specifies the undesirable function.
#' + If the entry is a character string, it is used as a description of
#' why a given function is undesirable
#' + Otherwise, entries should be missing (`NA`)
#' A generic message that the named function is undesirable is used if no
#' specific description is provided.
#' Input can also be a list of character strings for convenience.
#'
#' Defaults to [default_undesirable_functions]. To make small customizations to this list,
#' use [modify_defaults()].
#' @param symbol_is_undesirable Whether to consider the use of an undesirable function
#' name as a symbol undesirable or not.
#'
#' @examples
#' # defaults for which functions are considered undesirable
#' names(default_undesirable_functions)
#'
#' # will produce lints
#' lint(
#' text = "sapply(x, mean)",
#' linters = undesirable_function_linter()
#' )
#'
#' lint(
#' text = "log10(x)",
#' linters = undesirable_function_linter(fun = c("log10" = NA))
#' )
#'
#' lint(
#' text = "log10(x)",
#' linters = undesirable_function_linter(fun = c("log10" = "use log()"))
#' )
#'
#' lint(
#' text = 'dir <- "path/to/a/directory"',
#' linters = undesirable_function_linter(fun = c("dir" = NA))
#' )
#'
#'
#' lint(
#' text = 'dir <- "path/to/a/directory"',
#' linters = undesirable_function_linter(fun = "dir")
#' )
#'
#' # okay
#' lint(
#' text = "vapply(x, mean, FUN.VALUE = numeric(1))",
#' linters = undesirable_function_linter()
#' )
#'
#' lint(
#' text = "log(x, base = 10)",
#' linters = undesirable_function_linter(fun = c("log10" = "use log()"))
#' )
#'
#' lint(
#' text = 'dir <- "path/to/a/directory"',
#' linters = undesirable_function_linter(fun = c("dir" = NA), symbol_is_undesirable = FALSE)
#' )
#'
#' lint(
#' text = 'dir <- "path/to/a/directory"',
#' linters = undesirable_function_linter(fun = "dir", symbol_is_undesirable = FALSE)
#' )
#'
#' @evalRd rd_tags("undesirable_function_linter")
#' @seealso [linters] for a complete list of linters available in lintr.
#' @export
undesirable_function_linter <- function(fun = default_undesirable_functions,
symbol_is_undesirable = TRUE) {
if (is.list(fun)) fun <- unlist(fun)
stopifnot(
is.logical(symbol_is_undesirable),
# allow (uncoerced->implicitly logical) 'NA'
`\`fun\` should be a non-empty character vector` =
length(fun) > 0L && (is.character(fun) || all(is.na(fun)))
)
nm <- names2(fun)
implicit_idx <- !nzchar(nm)
if (any(implicit_idx)) {
names(fun)[implicit_idx] <- fun[implicit_idx]
is.na(fun) <- implicit_idx
}
if (anyNA(names(fun))) {
missing_idx <- which(is.na(names(fun))) # nolint: object_usage_linter. False positive.
cli_abort(paste(
"Unnamed elements of {.arg fun} must not be missing,",
"but {.val {missing_idx}} {qty(length(missing_idx))} {?is/are}."
))
}
xp_condition <- xp_and(
paste0(
"not(parent::expr/preceding-sibling::expr[last()][SYMBOL_FUNCTION_CALL[",
xp_text_in_table(c("library", "require")),
"]])"
),
"not(parent::expr[OP-DOLLAR or OP-AT])"
)
if (symbol_is_undesirable) {
symbol_xpath <- glue("//SYMBOL[({xp_text_in_table(names(fun))}) and {xp_condition}]")
}
xpath <- glue("SYMBOL_FUNCTION_CALL[{xp_condition}]")
Linter(linter_level = "expression", function(source_expression) {
xml <- source_expression$xml_parsed_content
xml_calls <- source_expression$xml_find_function_calls(names(fun))
matched_nodes <- xml_find_all(xml_calls, xpath)
if (symbol_is_undesirable) {
matched_nodes <- combine_nodesets(matched_nodes, xml_find_all(xml, symbol_xpath))
}
fun_names <- get_r_string(matched_nodes)
msgs <- vapply(
stats::setNames(nm = unique(fun_names)),
function(fun_name) {
msg <- sprintf('Avoid undesirable function "%s".', fun_name)
alternative <- fun[[fun_name]]
if (!is.na(alternative)) {
msg <- paste(msg, sprintf("As an alternative, %s.", alternative))
}
msg
},
character(1L)
)
xml_nodes_to_lints(
matched_nodes,
source_expression = source_expression,
lint_message = unname(msgs[fun_names])
)
})
}
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.