tests/testthat/test_version.R

context("version()")

inst_pkgs <- installed.packages()

test_that("version is package_version class", {
    expect_s3_class(version(), "package_version")
    expect_s3_class(version(), "numeric_version")
})

test_that("version has two components", {
    .skip_if_BiocVersion_not_available()
    verNums <- strsplit(as.character(version()), "\\.")[[1L]]
    expect_identical(length(verNums), 2L)
})

test_that(".version_validate() validates version", {
    skip_if_offline()

    .version_validate <- BiocManager:::.version_validate

    expect_error(
        .version_validate("2.0"),
        "Bioconductor version '2.0' requires R version '2.5'; .*"
    )

    expect_error(
        .version_validate("1.2.3"),
        "version '1.2.3' must have two components, e.g., '3.7'"
    )

    expect_error(
        .version_validate("100.1"),
        "unknown Bioconductor version '100.1'; .*"
    )

})

test_that(".version_recommend() recommends update", {
    skip_if_offline()
    expect_true(startsWith(
        .version_recommend("2.0"),
        "Bioconductor version '2.0' is out-of-date"
    ))
})

test_that(".version_validity_online_check() works", {
    ## environment variable
    withr::with_options(list(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=NULL), {
        withr::with_envvar(c(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=NA), {
            expect_identical(.version_validity_online_check(), TRUE)
        })

        withr::with_envvar(c(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=TRUE), {
            expect_identical(.version_validity_online_check(), TRUE)
        })

        withr::with_envvar(c(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=FALSE), {
            suppressWarnings({
                expect_identical(.version_validity_online_check(), FALSE)
            })
        })
    })

    ## options
    withr::with_envvar(c(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=NA), {
        withr::with_options(list(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=NULL), {
            expect_identical(.version_validity_online_check(), TRUE)
        })
    })

    withr::with_options(list(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=TRUE), {
        expect_identical(.version_validity_online_check(), TRUE)

        ## prefer option to env
        withr::with_envvar(c(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=FALSE), {
            expect_identical(.version_validity_online_check(), TRUE)
        })
    })

    withr::with_options(list(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=FALSE), {
        expect_identical(.version_validity_online_check(), FALSE)

        ## prefer option to env
        withr::with_envvar(c(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=TRUE), {
            expect_identical(.version_validity_online_check(), FALSE)
        })
    })
})

test_that(".version_validity('devel') works", {
    .skip_if_misconfigured()
    skip_if_offline()
    devel <- .version_bioc("devel")
    R_version <- getRversion()[,1:2]
    map <- .version_map()
    R_ok <- map$R[map$Bioc == devel]
    if (identical(version(), devel) || R_version %in% R_ok) {
        expect_true(.version_validity("devel"))
    } else {
        test <- paste0("Bioconductor version '", devel, "' requires R version")
        expect_true(startsWith(.version_validity("devel"), test))
    }
})

test_that(".version_validity(...) works", {
    .skip_if_misconfigured()
    skip_if_offline()

    .version_validity <- BiocManager:::.version_validity

    .get_R_ver = function(ver = "4.3.0") {
        rver <- package_version(ver)
        class(rver) <- c("R_system_version", class(rver))
        rver
    }

    ## proper mock map should include all four BiocStatus fields
    .ver_map <- data.frame(
        Bioc = package_version(c("3.9", "4.0", "4.1", "4.1")),
        R = package_version(c("4.2", "4.3", "4.4", "4.5")),
        BiocStatus = c("out-of-date", "release", "devel", "future")
    )

    expect_true(
        .version_validity("4.0", .ver_map, .get_R_ver())
    )

    expect_match(
        .version_validity("4.1", .ver_map, .get_R_ver()),
        "Bioconductor version '4.1'"
    )

    ## installing with future R versions supported
    expect_true(
        .version_validity("4.1", .ver_map, .get_R_ver("4.5.0"))
    )

    .par_map <- .ver_map[1:2, ]
    expect_match(
        .version_validity("4.1", .par_map, .get_R_ver()),
        "Bioconductor version map cannot be validated; is it misconfigured?"
    )
})

test_that(".version_bioc() works", {
    .skip_if_misconfigured()
    skip_if_offline()

    .ver_map <- data.frame(
        Bioc = package_version(c("4.0", "4.1", "4.1")),
        R = package_version(c("4.3", "4.4", "4.5")),
        BiocStatus = c("release", "devel", "future")
    )

    ## map misconfiguration
    with_mock(
        `BiocManager:::.version_map` = function(...) {
            .ver_map
        },
        expect_match(
            .version_bioc("release"),
            "Bioconductor version map cannot be validated; is it misconfigured?"
        )
    )

    .ver_map <- rbind.data.frame(c("3.9", "4.2", "out-of-date"), .ver_map)

    ## all good
    with_mock(
        `BiocManager:::.version_map` = function(...) {
            .ver_map
        },
        expect_identical(
            .version_bioc("release"), package_version("4.0")
        )
    )

    ## type misspecified
    type_miss <-
        "Bioconductor version cannot be validated; is type input misspecified?.*"
    with_mock(
        `BiocManager:::.version_map` = function(...) {
            .ver_map
        },
        expect_match(
            .version_bioc("foo"),
            type_miss
        )
    )
})

test_that(".version_R() works", {
    .skip_if_misconfigured()
    skip_if_offline()

    .ver_map <- data.frame(
        Bioc = package_version(c("4.0", "4.1", "4.1")),
        R = package_version(c("4.3", "4.4", "4.5")),
        BiocStatus = c("release", "devel", "future")
    )
    ## out-of-date is missing
    with_mock(
        `BiocManager:::.version_map` = function(...) {
            .ver_map
        },
        expect_match(
            .version_R("release"),
            "Bioconductor version map cannot be validated; is it misconfigured?.*"
        )
    )
    .ver_map <- rbind.data.frame(c("3.9", "4.2", "out-of-date"), .ver_map)
    ## all good
    with_mock(
        `BiocManager:::.version_map` = function(...) {
            .ver_map
        },
        expect_identical(
            .version_R("release"), package_version("4.3")
        )
    )
    ## type is misspecified
    with_mock(
        `BiocManager:::.version_map` = function(...) {
            .ver_map
        },
        expect_match(
            .version_R("foo"),
            "Bioconductor version cannot be validated; is type input misspecified?.*"
        )
    )
})

test_that(".version_validity() and BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS work",
{
    skip_if_offline()
    .skip_if_BiocVersion_not_available()
    withr::with_options(list(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=FALSE), {
        expect_match(
            .version_validity("1.2"),
            "unknown Bioconductor version '1.2'; .*"
        )
    })
})

test_that(".version_validate() and BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS work",{
    skip_if_offline()
    .skip_if_BiocVersion_not_available()
    withr::with_options(list(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=FALSE), {
        expect_error(
            .version_validate("1.2"),
            "unknown Bioconductor version '1.2'; .*"
        )
    })
})


test_that(".version_map() and BIOCONDUCTOR_CONFIG_FILE work", {
    config <- tempfile(fileext = ".yaml")
    file.create(config)
    expect_true(
        !length(
            .version_map_get_online_config(config)
        )
    )
    expect_true(
        is.character(
            .version_map_get_online_config(config)
        )
    )
    expect_warning(
            .version_map_get_online_config("./fake/address/path/file.yaml")
    )
    expect_identical(
        .version_map_get_online(config),
        .VERSION_MAP_SENTINEL
    )
    skip_if_offline()
    expect_true(
        is.data.frame(
            .version_map_get_online("https://bioconductor.org/config.yaml")
        )
    )
})

test_that(".version_map_get() and BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS work",{
    withr::with_options(list(BIOCONDUCTOR_ONLINE_VERSION_DIAGNOSIS=FALSE), {
        value <- .version_map_get()
        if ("BiocVersion" %in% rownames(inst_pkgs))
            expect_identical(packageVersion("BiocVersion")[, 1:2], value[1, 1])
        else
            expect_identical(value, .VERSION_MAP_SENTINEL)
    })
})

test_that(".version_map_get() falls back to http", {
    .VERSION_MAP$WARN_NO_ONLINE_CONFIG <- TRUE
    ## better test ideas welcome...
    url <- "https://httpbin.org/status/404"
    msgs <- list()
    result <- withCallingHandlers({
        .version_map_get(url)
    }, warning = function(w) {
        msgs <<- c(msgs, list(w))
        invokeRestart("muffleWarning")
    })
    ## did we generate warnings and eventually fail gracefully?
    expect_identical(length(msgs), 2L)
    expect_true(all(vapply(msgs, is, logical(1), "simpleWarning")))
    msgs <- vapply(msgs, conditionMessage, character(1))
    expect_identical(sum(grepl("https://httpbin.org", msgs)), 1L)
    expect_identical(sum(grepl("http://httpbin.org", msgs)), 1L)
    expect_identical(result, .VERSION_MAP_SENTINEL)
})

test_that(".VERSION_MAP_SENTINEL class and colnames", {
    expect_true(
        is.data.frame(.VERSION_MAP_SENTINEL)
    )
    ## check column names
    expect_identical(
        c("Bioc", "R", "BiocStatus"),
        names(.VERSION_MAP_SENTINEL)
    )
})

test_that("BiocVersion version matches with .version_map()", {
    .skip_if_misconfigured()
    skip_if_offline()

    if (!"BiocVersion" %in% rownames(installed.packages()))
        skip("BiocVersion not installed")

    R_version <- getRversion()
    bioc_version <- packageVersion("BiocVersion")[, 1:2]

    expect_version <-
        function(bioc_version, R_version)
    {
        map <- .version_map()
        map <- map[map$R == R_version[,1:2], ]
        failure_message <- paste0(
            "BiocVersion package version '", bioc_version, "' does not match ",
            "BiocManager::.version_map() '", paste(map$Bioc, collapse="', '"),
            "'. Check configuration."
        )
        expect(bioc_version %in% map$Bioc, failure_message)
    }
    expect_version(bioc_version, R_version)
})

test_that(".version_sentinel() works", {
    msg <- "this is a message"
    vs <- .version_sentinel(msg)
    expect_true(is.na(vs))
    expect_true(inherits(vs, "version_sentinel"))
    expect_true(inherits(vs, "package_version"))
    out <- capture.output(vs)
    expect_identical(
        paste0("unknown version: ", msg),
        out
    )
})

test_that(".version_BiocVersion returns .version_sentinel output", {
    with_mock(
        `BiocManager:::.version_BiocVersion_installed` = function(...) {
            FALSE
        },
        expect_identical(
            .version_BiocVersion(),
            .version_sentinel("BiocVersion is not installed")
        )
    )
})

test_that(".version_map_get_offline() works", {
    with_mock(
        `BiocManager:::.version_BiocVersion` = function(...) {
            .version_sentinel("BiocVersion is not installed")
        },
        expect_identical(
            .version_map_get_offline(),
            .VERSION_MAP_SENTINEL
        )
    )
    .skip_if_misconfigured()
    skip_if_offline()
    rver <- package_version("4.3")
    class(rver) <- c("R_system_version", class(rver))
    with_mock(
        `BiocManager:::.version_BiocVersion` = function(...) {
            package_version("3.14")
        },
        `BiocManager:::.version_R_version` = function(...) {
            rver <- package_version("4.3")
            class(rver) <- c("R_system_version", class(rver))
            rver
        },
        expect_identical(
            .version_map_get_offline(),
            rbind(.VERSION_MAP_SENTINEL, data.frame(
                Bioc = package_version('3.14'), R = rver,
                BiocStatus = factor(NA, levels = .VERSION_TAGS)
            ))
        )
    )
})

test_that("version chooses best", {
    target_version <-  structure(
        list(c(3L, 17L), class = c("package_version", "numeric_version"))
    )
    with_mock(
        `BiocManager:::.version_BiocVersion` = function(...) {
            .version_sentinel("BiocVersion is not installed")
        },
        `BiocManager:::.version_choose_best` = function(...) {
            target_version
        },
        expect_identical(
            .version_choose_best(),
            target_version
        )
    )
})

test_that(".version_force_version() works", {
    skip_if_offline()

    test_version <- "3.16"
    withr::with_envvar(
        list(R_BIOC_VERSION = ""),
        expect_false(identical(version(), package_version(test_version)))
    )
    withr::with_envvar(
        list(R_BIOC_VERSION = ""),
        expect_true(is.na(.version_force_version()))
    )
    withr::with_envvar(
        list(R_BIOC_VERSION = test_version),
        expect_identical(
            .version_force_version(), package_version(test_version)
        )
    )
    withr::with_envvar(
        list(R_BIOC_VERSION = test_version),
        expect_identical(version(), package_version(test_version))
    )
    withr::with_envvar(
        list(R_BIOC_VERSION = test_version),
        expect_identical(
            .version_validate(test_version),
            package_version(test_version)
        )
    )
    withr::with_envvar(
        list(R_BIOC_VERSION = test_version),
        expect_identical(grep(test_version, repositories()), 1:5)
    )

    new_lib <- tempfile(); dir.create(new_lib)
    withr::local_libpaths(new_lib)
    expect_error(packageVersion("BiocVersion", lib.loc = .libPaths()[1]))
    ## do *not* install BiocVersion
    withr::with_envvar(list(R_BIOC_VERSION = test_version), install())
    expect_error(packageVersion("BiocVersion", lib.loc = .libPaths()[1]))
    ## install BiocVersion for original version
    original_version <- version()
    install("BiocVersion", lib = .libPaths()[1])
    vers <- packageVersion("BiocVersion", lib.loc = .libPaths()[1])
    expect_identical(
        package_version(paste(vers$major, vers$minor, sep = ".")),
        original_version
    )
    ## BiocVersion ignored with R_BIOC_VERSION
    withr::with_envvar(
        list(R_BIOC_VERSION = test_version),
        expect_identical(version(), package_version(test_version))
    )
})

Try the BiocManager package in your browser

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

BiocManager documentation built on Sept. 11, 2024, 8:07 p.m.