R/getDependencies.R

Defines functions getDependencies

Documented in getDependencies

#' Get package dependencies
#'
#' Get package dependencies
#'
#' This function recursively constructs the list of dependencies for the
#' packages given by \code{pkgs}.  By default, the dependency information is
#' extracted from both installed and available packages.  As a consequence, it
#' works both for local and CRAN packages.
#'
#' @param pkgs character vector of package names
#' @param dependencies character vector of dependency types to include.
#' Choices are "Depends", "Imports", "LinkingTo", "Suggests", and "Enhances".
#' Defaults to \code{c("Depends", "Imports", "LinkingTo")}.
#' @param installed Logical indicating whether to pull dependency information
#' from installed packages. Defaults to TRUE.
#' @param available Logical indicating whether to pull dependency information
#' from available packages.  Defaults to TRUE.
#' @param base Logical indicating whether to include dependencies on base
#' packages that are included in the R installation. Defaults to FALSE.
#' @param recommended Logical indicating whether to include dependencies on
#' recommended packages that are included in the R installation. Defaults to
#' FALSE.
#' @return A character vector of package names.
#' @note If \code{available=TRUE} R will attempt to access the currently
#' selected CRAN repository, prompting for one if necessary.
#' @author Gregory R. Warnes \email{greg@@warnes.net} based on the non exported
#' \code{utils:::getDependencies} and \code{utils:::.clean_up_dependencies2}.
#' @seealso \code{\link{installed.packages}}, \code{\link{available.packages}}
#' @keywords utilities
#' @examples
#'
#' \dontrun{
#' ## A locally installed package
#' #' getDependencies("MASS", installed = TRUE, available = FALSE)
#' 
#' ## A package on CRAN
#' getDependencies("gregmisc", installed = FALSE, available = TRUE)
#' 
#'
#' ## Show base and recommended dependencies
#' getDependencies("MASS", available = FALSE, base = TRUE, recommended = TRUE)
#' 
#' ## Download the set of packages necessary to support a local package
#' deps <- getDependencies("MyLocalPackage", available = FALSE)
#' download.packages(deps, destdir = "./R_Packages")
#' }
#'
#' @importFrom utils available.packages
#' @importFrom utils flush.console
#' @importFrom utils head
#' @importFrom utils installed.packages
#' @export
getDependencies <- function(pkgs,
                            dependencies = c("Depends", "Imports", "LinkingTo"),
                            installed = TRUE,
                            available = TRUE,
                            base = FALSE,
                            recommended = FALSE) {
  pkgs.in <- pkgs
  if (is.null(dependencies)) {
    return(unique(pkgs))
  }
  dep2 <- c("Depends", "Imports", "LinkingTo")


  if (installed && !available) {
    all.packages <- installed.packages()
  } else if (available && !installed) {
    all.packages <- available.packages()
  } else {
    all.packages <- as.matrix(smartbind( ##
      installed.packages(),
      available.packages()
    ))
  }
  rownames(all.packages) <- all.packages[, "Package"]

  p0 <- unique(pkgs)
  miss <- !p0 %in% row.names(all.packages)
  if (sum(miss)) {
    warning(sprintf(ngettext(
      sum(miss), "package %s is not available (for %s)",
      "packages %s are not available (for %s)"
    ), paste(sQuote(p0[miss]),
      collapse = ", "
    ), sub(" *\\(.*", "", R.version.string)),
    domain = NA, call. = FALSE
    )
    if (sum(miss) == 1L && !is.na(w <- match(
      tolower(p0[miss]),
      tolower(row.names(all.packages))
    ))) {
      warning(sprintf("Perhaps you meant %s ?", sQuote(row.names(all.packages)[w])),
        call. = FALSE, domain = NA
      )
    }
    flush.console()
  }

  ## Whether to exclude base and recommended packages
  if (!base || !recommended) {
    priority <- NULL
    if (!base) priority <- c("base", priority)
    if (!recommended) priority <- c("recommended", priority)
    installed <- installed.packages(priority = priority)
  }
  else {
    installed <- installed.packages()[FALSE, ]
  }

  p0 <- p0[!miss]

  p1 <- p0
  not_avail <- character()
  repeat {
    deps <- apply(
      all.packages[p1, dependencies, drop = FALSE],
      1L, function(x) paste(x[!is.na(x)], collapse = ", ")
    )

    res <- .clean_up_dependencies2(
      deps,
      installed = installed,
      all.packages
    )
    not_avail <- c(not_avail, res[[2L]])
    deps <- unique(res[[1L]])
    deps <- deps[!deps %in% c("R", pkgs)]
    if (!length(deps)) {
      break
    }
    pkgs <- c(deps, pkgs)
    p1 <- deps
    if (!is.null(dep2)) {
      dependencies <- dep2
      dep2 <- NULL
    }
  }
  if (length(not_avail)) {
    not_avail <- unique(not_avail)
    warning(sprintf(
      ngettext(
        length(not_avail),
        "dependency %s is not available",
        "dependencies %s are not available"
      ),
      paste(sQuote(not_avail),
        collapse = ", "
      )
    ),
    domain = NA,
    call. = FALSE,
    immediate. = TRUE
    )
    flush.console()
  }
  pkgs <- unique(pkgs)
  pkgs <- pkgs[pkgs %in% row.names(all.packages)]
  p0 <- pkgs


  p0[!p0 %in% pkgs.in]
}

Try the gtools package in your browser

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

gtools documentation built on Nov. 20, 2023, 5:07 p.m.