15 Advanced testing techniques
15.1 Test fixtures
When it’s not practical to make your test entirely self-sufficient, prefer making the necessary object, logic, or conditions available in a structured, explicit way. There’s a pre-existing term for this in software engineering: a test fixture.
A test fixture is something used to consistently test some item, device, or piece of software. — Wikipedia
The main idea is that we need to make it as easy and obvious as possible to arrange the world into a state that is conducive for testing. We describe several specific solutions to this problem:
- Put repeated code in a constructor-type helper function. Memoise it, if construction is demonstrably slow.
- If the repeated code has side effects, write a custom
local_*()
function to do what’s needed and clean up afterwards. - If the above approaches are too slow or awkward and the thing you need is fairly stable, save it as a static file and load it.
15.1.1 Create useful_thing
s with a helper function
Is it fiddly to create a useful_thing
? Does it take several lines of code, but not much time or memory? In that case, write a helper function to create a useful_thing
on-demand:
new_useful_thing <- function() {
# your fiddly code to create a useful_thing goes here
}
and call that helper in the affected tests:
test_that("foofy() does this", {
useful_thing1 <- new_useful_thing()
expect_equal(foofy(useful_thing1, x = "this"), EXPECTED_FOOFY_OUTPUT)
})
test_that("foofy() does that", {
useful_thing2 <- new_useful_thing()
expect_equal(foofy(useful_thing2, x = "that"), EXPECTED_FOOFY_OUTPUT)
})
Where should the new_useful_thing()
helper be defined? This comes back to what we outlined in sec-tests-files-overview. Test helpers can be defined below R/
, just like any other internal utility in your package. Another popular location is in a test helper file, e.g. tests/testthat/helper.R
. A key feature of both options is that the helpers are made available to you during interactive maintenance via devtools::load_all()
.
If it’s fiddly AND costly to create a useful_thing
, your helper function could even use memoisation to avoid unnecessary re-computation. Once you have a helper like new_useful_thing()
, you often discover that it has uses beyond testing, e.g. behind-the-scenes in a vignette. Sometimes you even realize you should just define it below R/
and export and document it, so you can use it freely in documentation and tests.
15.1.2 Create (and destroy) a “local” useful_thing
So far, our example of a useful_thing
was a regular R object, which is cleaned-up automatically at the end of each test. What if the creation of a useful_thing
has a side effect on the local file system, on a remote resource, R session options, environment variables, or the like? Then your helper function should create a useful_thing
and clean up afterwards. Instead of a simple new_useful_thing()
constructor, you’ll write a customized function in the style of withr’s local_*()
functions:
local_useful_thing <- function(..., env = parent.frame()) {
# your fiddly code to create a useful_thing goes here
withr::defer(
# your fiddly code to clean up after a useful_thing goes here
envir = env
)
}
Use it in your tests like this:
test_that("foofy() does this", {
useful_thing1 <- local_useful_thing()
expect_equal(foofy(useful_thing1, x = "this"), EXPECTED_FOOFY_OUTPUT)
})
test_that("foofy() does that", {
useful_thing2 <- local_useful_thing()
expect_equal(foofy(useful_thing2, x = "that"), EXPECTED_FOOFY_OUTPUT)
})
Where should the local_useful_thing()
helper be defined? All the advice given above for new_useful_thing()
applies: define it below R/
or in a test helper file.
To learn more about writing custom helpers like local_useful_thing()
, see the testthat vignette on test fixtures.
15.1.3 Store a concrete useful_thing
persistently
If a useful_thing
is costly to create, in terms of time or memory, maybe you don’t actually need to re-create it for each test run. You could make the useful_thing
once, store it as a static test fixture, and load it in the tests that need it. Here’s a sketch of how this could look:
test_that("foofy() does this", {
useful_thing1 <- readRDS(test_path("fixtures", "useful_thing1.rds"))
expect_equal(foofy(useful_thing1, x = "this"), EXPECTED_FOOFY_OUTPUT)
})
test_that("foofy() does that", {
useful_thing2 <- readRDS(test_path("fixtures", "useful_thing2.rds"))
expect_equal(foofy(useful_thing2, x = "that"), EXPECTED_FOOFY_OUTPUT)
})
Now we can revisit a file listing from earlier, which addressed exactly this scenario:
.
├── ...
└── tests
├── testthat
│ ├── fixtures
│ │ ├── make-useful-things.R
│ │ ├── useful_thing1.rds
│ │ └── useful_thing2.rds
│ ├── helper.R
│ ├── setup.R
│ └── (all the test files)
└── testthat.R
This shows static test files stored in tests/testthat/fixtures/
, but also notice the companion R script, make-useful-things.R
. From data analysis, we all know there is no such thing as a script that is run only once. Refinement and iteration is inevitable. This also holds true for test objects like useful_thing1.rds
. We highly recommend saving the R code used to create your test objects, so that they can be re-created as needed.
15.2 Building your own testing tools
Let’s return to the topic of duplication in your test code. We’ve encouraged you to have a higher tolerance for repetition in test code, in the name of making your tests obvious. But there’s still a limit to how much repetition to tolerate. We’ve covered techniques such as loading static objects with test_path()
, writing a constructor like new_useful_thing()
, or implementing a test fixture like local_useful_thing()
. There are even more types of test helpers that can be useful in certain situations.
15.2.1 Helper defined inside a test
Consider this test for the str_trunc()
function in stringr:
# from stringr (hypothetically)
test_that("truncations work for all sides", {
expect_equal(
str_trunc("This string is moderately long", width = 20, side = "right"),
"This string is mo..."
)
expect_equal(
str_trunc("This string is moderately long", width = 20, side = "left"),
"...s moderately long"
)
expect_equal(
str_trunc("This string is moderately long", width = 20, side = "center"),
"This stri...ely long"
)
})
There’s a lot of repetition here, which increases the chance of copy / paste errors and generally makes your eyes glaze over. Sometimes it’s nice to create a hyper-local helper, inside the test. Here’s how the test actually looks in stringr
# from stringr (actually)
test_that("truncations work for all sides", {
trunc <- function(direction) str_trunc(
"This string is moderately long",
direction,
width = 20
)
expect_equal(trunc("right"), "This string is mo...")
expect_equal(trunc("left"), "...s moderately long")
expect_equal(trunc("center"), "This stri...ely long")
})
A hyper-local helper like trunc()
is particularly useful when it allows you to fit all the important business for each expectation on one line. Then your expectations can be read almost like a table of actual vs. expected, for a set of related use cases. Above, it’s very easy to watch the result change as we truncate the input from the right, left, and in the center.
Note that this technique should be used in extreme moderation. A helper like trunc()
is yet another place where you can introduce a bug, so it’s best to keep such helpers extremely short and simple.
15.2.2 Custom expectations
If a more complicated helper feels necessary, it’s a good time to reflect on why that is. If it’s fussy to get into position to test a function, that could be a sign that it’s also fussy to use that function. Do you need to refactor it? If the function seems sound, then you probably need to use a more formal helper, defined outside of any individual test, as described earlier.
One specific type of helper you might want to create is a custom expectation. Here are two very simple ones from usethis:
expect_usethis_error <- function(...) {
expect_error(..., class = "usethis_error")
}
expect_proj_file <- function(...) {
expect_true(file_exists(proj_path(...)))
}
expect_usethis_error()
checks that an error has the "usethis_error"
class. expect_proj_file()
is a simple wrapper around file_exists()
that searches for the file in the current project. These are very simple functions, but the sheer amount of repetition and the expressiveness of their names makes them feel justified.
It is somewhat involved to make a proper custom expectation, i.e. one that behaves like the expectations built into testthat. We refer you to the Custom expectations vignette if you wish to learn more about that.
Finally, it can be handy to know that testthat makes specific information available when it’s running:
-
The environment variable
TESTTHAT
is set to"true"
.testthat::is_testing()
is a shortcut:is_testing <- function() { Sys.getenv("TESTTHAT") }
-
The package-under-test is available as the environment variable
TESTTHAT_PKG
andtestthat::testing_package()
is a shortcut:testing_package <- function() { Sys.getenv("TESTTHAT_PKG") }
In some situations, you may want to exploit this information without taking a run-time dependency on testthat. In that case, just inline the source of these functions directly into your package.
15.3 When testing gets hard
Despite all the techniques we’ve covered so far, there remain situations where it still feels very difficult to write tests. In this section, we review more ways to deal with challenging situations:
- Skipping a test in certain situations
- Mocking an external service
- Dealing with secrets
15.3.1 Skipping a test
Sometimes it’s impossible to perform a test - you may not have an internet connection or you may not have access to the necessary credentials. Unfortunately, another likely reason follows from this simple rule: the more platforms you use to test your code, the more likely it is that you won’t be able to run all of your tests, all of the time. In short, there are times when, instead of getting a failure, you just want to skip a test.
15.3.1.1 testthat::skip()
Here we use testthat::skip()
to write a hypothetical custom skipper, skip_if_no_api()
:
skip_if_no_api()
is a yet another example of a test helper and the advice already given about where to define it applies here too.
skip()
s and the associated reasons are reported inline as tests are executed and are also indicated clearly in the summary:
devtools::test()
#> ℹ Loading abcde
#> ℹ Testing abcde
#> ✔ | F W S OK | Context
#> ✔ | 2 | blarg
#> ✔ | 1 2 | foofy
#> ────────────────────────────────────────────────────────────────────────────────
#> Skip (test-foofy.R:6:3): foo api returns bar when given baz
#> Reason: API not available
#> ────────────────────────────────────────────────────────────────────────────────
#> ✔ | 0 | yo
#> ══ Results ═════════════════════════════════════════════════════════════════════
#> ── Skipped tests ──────────────────────────────────────────────────────────────
#> • API not available (1)
#>
#> [ FAIL 0 | WARN 0 | SKIP 1 | PASS 4 ]
#>
#> 🥳
Something like skip_if_no_api()
is likely to appear many times in your test suite. This is another occasion where it is tempting to DRY things out, by hoisting the skip()
to the top-level of the file. However, we still lean towards calling skip_if_no_api()
in each test where it’s needed.
# we prefer this:
test_that("foo api returns bar when given baz", {
skip_if_no_api()
...
})
test_that("foo api returns an errors when given qux", {
skip_if_no_api()
...
})
# over this:
skip_if_no_api()
test_that("foo api returns bar when given baz", {...})
test_that("foo api returns an errors when given qux", {...})
Within the realm of top-level code in test files, having a skip()
at the very beginning of a test file is one of the more benign situations. But once a test file does not fit entirely on your screen, it creates an implicit yet easy-to-miss connection between the skip()
and individual tests.
15.3.1.2 Built-in skip()
functions
Similar to testthat’s built-in expectations, there is a family of skip()
functions that anticipate some common situations. These functions often relieve you of the need to write a custom skipper. Here are some examples of the most useful skip()
functions:
test_that("foo api returns bar when given baz", {
skip_if(api_unavailable(), "API not available")
...
})
test_that("foo api returns bar when given baz", {
skip_if_not(api_available(), "API not available")
...
})
skip_if_not_installed("sp")
skip_if_not_installed("stringi", "1.2.2")
skip_if_offline()
skip_on_cran()
skip_on_os("windows")
15.3.1.3 Dangers of skipping
One challenge with skips is that they are currently completely invisible in CI — if you automatically skip too many tests, it’s easy to fool yourself that all your tests are passing when in fact they’re just being skipped! In an ideal world, your CI/CD would make it easy to see how many tests are being skipped and how that changes over time.
It’s a good practice to regularly dig into the R CMD check
results, especially on CI, and make sure the skips are as you expect. But this tends to be something you have to learn through experience.
15.3.2 Mocking
The practice known as mocking is when we replace something that’s complicated or unreliable or out of our control with something simpler, that’s fully within our control. Usually we are mocking an external service, such as a REST API, or a function that reports something about session state, such as whether the session is interactive.
The classic application of mocking is in the context of a package that wraps an external API. In order to test your functions, technically you need to make a live call to that API to get a response, which you then process. But what if that API requires authentication or what if it’s somewhat flaky and has occasional downtime? It can be more productive to just pretend to call the API but, instead, to test the code under your control by processing a pre-recorded response from the actual API.
Our main advice about mocking is to avoid it if you can. This is not an indictment of mocking, but just a realistic assessment that mocking introduces new complexity that is not always justified by the payoffs.
Since most R packages do not need full-fledged mocking, we do not cover it here. Instead we’ll point you to the packages that represent the state-of-the-art for mocking in R today:
- mockery: https://github.com/r-lib/mockery
- mockr: https://krlmlr.github.io/mockr/
- httptest: https://enpiar.com/r/httptest/
- httptest2: https://enpiar.com/httptest2/
- webfakes: https://webfakes.r-lib.org
Note also that, at the time of writing, it seems likely that the testthat package will re-introduce some mocking capabilities (after previously getting out of the mocking business once already). Version v3.1.7 has two new experimental functions, testthat::with_mocked_bindings()
and testthat::local_mocked_bindings()
.
15.3.3 Secrets
Another common challenge for packages that wrap an external service is the need to manage credentials. Specifically, it is likely that you will need to provide a set of test credentials to fully test your package.
Our main advice here is to design your package so that large parts of it can be tested without live, authenticated access to the external service.
Of course, you will still want to be able to test your package against the actual service that it wraps, in environments that support secure environment variables. Since this is also a very specialized topic, we won’t go into more detail here. Instead we refer you to the Wrapping APIs vignette in the httr2 package, which offers substantial support for secret management.
15.4 Special considerations for CRAN packages
CRAN runs R CMD check
on all contributed packages, both upon submission and on a regular basis after acceptance. This check includes, but is not limited to, your testthat tests. We discuss the general challenge of preparing your package to face all of CRAN’s check “flavors” in sec-cran-flavors-services. Here we focus on CRAN-specific considerations for your test suite.
When a package runs afoul of the CRAN Repository Policy (https://cran.r-project.org/web/packages/policies.html), the test suite is very often the culprit (although not always). If your package is destined for CRAN, this should influence how you write your tests and how (or whether) they will be run on CRAN.
15.4.1 Skip a test
If a specific test simply isn’t appropriate to be run by CRAN, include skip_on_cran()
at the very start.
test_that("some long-running thing works", {
skip_on_cran()
# test code that can potentially take "a while" to run
})
Under the hood, skip_on_cran()
consults the NOT_CRAN
environment variable. Such a test will only run when NOT_CRAN
has been explicitly defined as "true"
. This variable is set by devtools and testthat, allowing those tests to run in environments where you expect success (and where you can tolerate and troubleshoot occasional failure).
In particular, the GitHub Actions workflows that we recommend in sec-sw-dev-practices-gha will run tests with NOT_CRAN = "true"
. For certain types of functionality, there is no practical way to test it on CRAN and your own checks, on GitHub Actions or an equivalent continuous integration service, are your best method of quality assurance.
There are even rare cases where it makes sense to maintain tests outside of your package altogether. The tidymodels team uses this strategy for integration-type tests of their whole ecosystem that would be impossible to host inside an individual CRAN package.
15.4.2 Speed
Your tests need to run relatively quickly - ideally, less than a minute, in total. Use skip_on_cran()
in a test that is unavoidably long-running.
15.4.3 Reproducibility
Be careful about testing things that are likely to be variable on CRAN machines. It’s risky to test how long something takes (because CRAN machines are often heavily loaded) or to test parallel code (because CRAN runs multiple package tests in parallel, multiple cores will not always be available). Numerical precision can also vary across platforms, so use expect_equal()
unless you have a specific reason for using expect_identical()
.
15.4.4 Flaky tests
Due to the scale at which CRAN checks packages, there is basically no latitude for a test that’s “just flaky”, i.e. sometimes fails for incidental reasons. CRAN does not process your package’s test results the way you do, where you can inspect each failure and exercise some human judgment about how concerning it is.
It’s probably a good idea to eliminate flaky tests, just for your own sake! But if you have valuable, well-written tests that are prone to occasional nuisance failure, definitely put skip_on_cran()
at the start.
The classic example is any test that accesses a website or web API. Given that any web resource in the world will experience occasional downtime, it’s best to not let such tests run on CRAN. The CRAN Repository Policy says:
Packages which use Internet resources should fail gracefully with an informative message if the resource is not available or has changed (and not give a check warning nor error).
Often making such a failure “graceful” would run counter to the behaviour you actually want in practice, i.e. you would want your user to get an error if their request fails. This is why it is usually more practical to test such functionality elsewhere.
Recall that snapshot tests (sec-testing-basics), by default, are also skipped on CRAN. You typically use such tests to monitor, e.g., how various informational messages look. Slight changes in message formatting are something you want to be alerted to, but do not indicate a major defect in your package. This is the motivation for the default skip_on_cran()
behaviour of snapshot tests.
Finally, flaky tests cause problems for the maintainers of your dependencies. When the packages you depend on are updated, CRAN runs R CMD check
on all reverse dependencies, including your package. If your package has flaky tests, your package can be the reason another package does not clear CRAN’s incoming checks and can delay its release.
15.4.5 Process and file system hygiene
In sec-tests-files-where-write, we urged you to only write into the session temp directory and to clean up after yourself. This practice makes your test suite much more maintainable and predictable. For packages that are (or aspire to be) on CRAN, this is absolutely required per the CRAN repository policy:
Packages should not write in the user’s home filespace (including clipboards), nor anywhere else on the file system apart from the R session’s temporary directory (or during installation in the location pointed to by TMPDIR: and such usage should be cleaned up)…. Limited exceptions may be allowed in interactive sessions if the package obtains confirmation from the user.
Similarly, you should make an effort to be hygienic with respect to any processes you launch:
Packages should not start external software (such as PDF viewers or browsers) during examples or tests unless that specific instance of the software is explicitly closed afterwards.
Accessing the clipboard is the perfect storm that potentially runs afoul of both of these guidelines, as the clipboard is considered part of the user’s home filespace and, on Linux, can launch an external process (e.g. xsel or xclip). Therefore it is best to turn off any clipboard functionality in your tests (and to ensure that, during authentic usage, your user is clearly opting-in to that).