R/utils.R

Defines functions release unmanage managed manage connection connectionForResource manager resourceDescription .parseURI isURL

# Utilities ---------------------------------------------------------------

isURL <- function(uri) {
    if (!isSingleString(uri))
        return(FALSE)
    windowsDriveLetter <- .Platform$OS.type == "windows" &&
        grepl("^[A-Za-z]:[/\\]", uri)
    grepl("^[A-Za-z]+:", uri) && !windowsDriveLetter
}

.parseURI <- function(uri) {
    if (!isURL(uri)) {
        parsed <- list(scheme = "", path = uri)
    } else {
        parsed <- list(scheme = "", path = uri)
        protocols <- c("file", "http", "https", "ftp", "smtp")
        protocol <- protocols[startsWith(uri, paste0(protocols, ':'))]
        if (length(protocol)) {
            parsed$scheme <- protocol
            rem <- paste0(protocol, "://")
            if (protocol %in% protocols[-1]) {
                domain <- strsplit(
                    gsub("http://|https://|ftp://|smtp://|www\\.", "", uri), "/"
                )[[c(1, 1)]]
                parsed$path <- sub(paste0(rem, domain), "", uri)
            }
            else
                parsed$path <- sub(rem, "", uri)
        } else {
            parsed$scheme <- "file"
        }
    }
    parsed
}

resourceDescription <- function(x) {
    resource <- resource(x)
    if (is(resource, "connection"))
        summary(resource)$description
    else
        resource
}

.ConnectionManager <-
    setRefClass("ConnectionManager", fields = c(connections = "list"))

manager <- function() .ConnectionManager()

connectionForResource <- function(manager, x, open = "") {
    resource <- decompress(manager, x)
    if (is.character(resource)) {
        if (!nzchar(resource))
            stop("path cannot be an empty string")
        uri <- .parseURI(resource)
        if (uri$scheme != "")
            con <- url(resource)
        else
            con <- file(resource)
    } else {
        con <- resource
    }
    if (!isOpen(con) && nzchar(open)) {
        open(con, open)
        con <- manage(manager, con)
    }
    con
}

connection <- function(manager, x, open = "") {
    connectionForResource(manager, resource(x), open = open)
}

# Connection management (similar to memory management) --------------------

manage <- function(manager, con) {
    manager$connections <- unique(c(manager$connections, list(con)))
    attr(con, "manager") <- manager
    con
}

managed <- function(manager, con) {
    con %in% manager$connections
}

unmanage <- function(manager, con) {
    manager$connections <- setdiff(manager$connections, con)
    attr(con, "manager") <- NULL
    con
}

release <- function(manager, con) {
    if (managed(manager, con)) {
        unmanage(manager, con)
        close(con)
    }
    con
}
Bioconductor/BiocIO documentation built on April 28, 2024, 7:43 a.m.