tests/testthat/test-trend-var.R

# This tests the various fitTrendVar() options.
# require(scran); require(testthat); source("test-trend-var.R")

set.seed(20000)
ncells <- 200
ngenes <- 1000
means <- 2^runif(ngenes, -1, 5)
dummy <- matrix(rnbinom(ngenes*ncells, mu=means, size=5), ncol=ncells, nrow=ngenes)

library(scuttle)
out <- normalizeCounts(dummy)

library(DelayedMatrixStats)
means <- rowMeans(out)
vars <- rowVars(out)

test_that("fitTrendVar works on a basic scenario", {
    out <- fitTrendVar(means, vars)
    expect_true(out$std.dev > 0)
    expect_is(out$trend, "function")

    # Hard to test it without copying the code, so I'll just check the limits.
    expect_equal(out$trend(0), 0)
    expect_equal(out$trend(1:10), sapply(1:10, out$trend)) # Checking we get consistent results with returned function.
    expect_equal(out$trend(100:1/20), sapply(100:1/20, out$trend))  # More checking, reversed order.

    # Parametric left edge limits work (almost) correctly (below 0.1).
    expect_equal(out$trend(0.01)*2, out$trend(0.02), tol=1e-4)
    expect_equal(out$trend(0.01)*5, out$trend(0.05), tol=1e-4)

    # Parametric right limits do not use rule=2.
    mx <- max(means)
    expect_false(isTRUE(all.equal(out$trend(mx), out$trend(mx+1))))
})

test_that("fitTrendVar works when parametric mode is turned off", {
    out <- fitTrendVar(means, vars, parametric=FALSE)
    expect_equal(out$trend(0), 0)
    expect_equal(out$trend(1:10), sapply(1:10, out$trend)) 
    expect_equal(out$trend(100:1/20), sapply(100:1/20, out$trend)) 

    # Parametric left edge limits work correctly (below 0.1).
    expect_equal(out$trend(0.01)*2, out$trend(0.02))
    expect_equal(out$trend(0.01)*5, out$trend(0.05))
    expect_equal(out$trend(0.01)*10, out$trend(0.1))

    # Right edge goes to rule=2.
    mx <- max(means)
    expect_equal(out$trend(mx), out$trend(mx+1))
})

test_that("fitTrendVar works when lowess mode is turned off", {
    out <- fitTrendVar(means, vars, lowess=FALSE)
    expect_equal(out$trend(0), 0)
    expect_equal(out$trend(1:10), sapply(1:10, out$trend)) 
    expect_equal(out$trend(100:1/20), sapply(100:1/20, out$trend)) 

    # Right limits do not use rule=2.
    mx <- max(means)
    expect_false(isTRUE(all.equal(out$trend(mx), out$trend(mx+1))))

    expect_error(fitTrendVar(means, vars, lowess=FALSE, parametric=FALSE), "at least one")
})

test_that("fitTrendVar works when density weights are turned off", {
    out <- fitTrendVar(means, vars, density.weights=FALSE)
    expect_equal(out$trend(0), 0)
    expect_equal(out$trend(1:10), sapply(1:10, out$trend)) 
    expect_equal(out$trend(100:1/20), sapply(100:1/20, out$trend)) 

    # Right limits do not use rule=2.
    mx <- max(means)
    expect_false(isTRUE(all.equal(out$trend(mx), out$trend(mx+1))))
})

set.seed(91919)
test_that("fitTrendVar handles nls errors gracefully",  {
    # This requires the set.seed(), as sometimes it *doesn't* fail. 
    # Damn the robustness of this algorithm!
    X <- runif(100)
    Y <- runif(100)
    expect_error(out <- fitTrendVar(X, Y), NA)
    expect_true("Aest" %in% ls(environment(environment(environment(out$trend)$FUN)$PARAMFUN)))

    expect_error(fitTrendVar(runif(2), runif(2)), "need at least 4") 
    expect_error(fitTrendVar(runif(1), runif(1)), "need at least 2") 
})
MarioniLab/scran documentation built on March 7, 2024, 1:45 p.m.