Introduction to testdat

testdat is a package designed to ease data validation, particularly for complex data processing, inspired by software unit testing. testdat extends the strong and flexible unit testing framework already provided by testthat with a family of functions and reporting tools focused on checking data frames.

Features include:

  • A fully fledged test framework so you can spend more time specifying tests and less time running them

  • A set of common methods for simply specifying data validation rules

  • Repeatability of data tests (avoid unintentionally breaking your data set!)

  • Data-focused reporting of test results

Getting started

As an extension of testthat, testdat uses the same basic testing framework. Before using testdat (and reading this documentation), make sure you’re familiar with the introduction to testthat in R packages.

The main addition provided by testdat is a set of expectations designed for testing data frames and an accompanying mechanism to globally specify a data set for testing.

Data expectations

Overview

In general, our approach to data testing is variable-centric - the majority of expectations perform a check on one or more variables in a given data frame.

The standard form of a data expectation is:

expect_*(var(s), ..., flt = TRUE, data = get_testdata())

testdat uses dplyr at its core, and thus supports tidy evaluation.

Variables

Most operations act on one or more variables. There are two variants of the variable argument:

test_that("multi-variable identifier is unique", {
  expect_unique(c(name, year, month, day, hour), data = storms)
})
#> ── Failure: multi-variable identifier is unique ────────────────────────────────
#> `storms` has 48 duplicate records on variable `name, year, month, day, hour`.
#> Filter: None
#> Error:
#> ! Test failed
  • var requires an unquoted variable name. This only applies to a small number of expectations.
test_that("hour values are valid", {
  expect_base(ts_diameter, year >= 2004)
})
#> ── Error: hour values are valid ────────────────────────────────────────────────
#> Error: A test data frame has not been specified. Use `set_testdata()` to set the data frame.
#> Backtrace:
#>     ▆
#>  1. └─testdat::expect_base(ts_diameter, year >= 2004)
#>  2.   ├─testthat::quasi_label(enquo(data))
#>  3.   │ └─rlang::eval_bare(expr, quo_get_env(quo))
#>  4.   └─testdat::get_testdata()
#> Error:
#> ! Test failed

Filter

The flt argument takes a logical predicate defined in terms of the variables in data using data masking. Only rows where the condition evaluates to TRUE are included in the test.

test_that("iris range checks", {
  expect_range(Petal.Width, 0, 1, data = iris)
})
#> ── Failure: iris range checks ──────────────────────────────────────────────────
#> `iris` has 93 records failing range check on variable `Petal.Width`.
#> Variable set: `Petal.Width`
#> Filter: None
#> Arguments: `min = 0, max = 1`
#> Error:
#> ! Test failed

test_that("iris range checks filtered", {
  # Test passes for setosa rows
  expect_range(Petal.Width, 0, 1, flt = Species == "setosa", data = iris)
  # Failures will provide the filter
  expect_range(Petal.Width, 0, 0.5, flt = Species == "setosa", data = iris)
})
#> ── Failure: iris range checks filtered ─────────────────────────────────────────
#> `iris` has 1 records failing range check on variable `Petal.Width`.
#> Variable set: `Petal.Width`
#> Filter: `Species == "setosa"`
#> Arguments: `min = 0, max = 0.5`
#> Error:
#> ! Test failed

Data

The data argument takes a data frame to test. To avoid redundant code the data argument defaults to a global test data set retrieved using get_testdata(). This can be used in two ways:

  • Setting the global test data using set_testdata().
set_testdata(iris)
identical(get_testdata(), iris)
#> [1] TRUE

test_that("Versicolor has sepal length greater than 5 - will fail", {
  expect_cond(Species %in% "versicolor", Sepal.Length >= 5)
})
#> ── Failure: Versicolor has sepal length greater than 5 - will fail ─────────────
#> get_testdata() failed consistency check. 1 cases have `Species %in% "versicolor"` but not `Sepal.Length >= 5`.
#> Error:
#> ! Test failed
  • Using the with_testdata() wrapper to temporarily set the global test data for a block of code.
set_testdata(mtcars)
identical(get_testdata(), mtcars)
#> [1] TRUE

with_testdata(iris, {
  test_that("Versicolor has sepal length greater than 5 - will fail", {
    expect_cond(Species %in% "versicolor", Sepal.Length >= 5)
  })
})
#> ── Failure: Versicolor has sepal length greater than 5 - will fail ─────────────
#> get_testdata() failed consistency check. 1 cases have `Species %in% "versicolor"` but not `Sepal.Length >= 5`.
#> Error:
#> ! Test failed

identical(get_testdata(), mtcars)
#> [1] TRUE

Both approaches are equivalent to:

test_that("Versicolor has sepal length greater than 5 - will fail", {
  expect_cond(Species %in% "versicolor", Sepal.Length >= 5, data = iris)
})
#> ── Failure: Versicolor has sepal length greater than 5 - will fail ─────────────
#> `iris` failed consistency check. 1 cases have `Species %in% "versicolor"` but not `Sepal.Length >= 5`.
#> Error:
#> ! Test failed

By default, set_testdata() stores a quosure with a reference to the provided data frame rather than the data itself, so changes made to the data frame will be reflected in the test results.

tmp_data <- tibble(x = c(1, 0), y = c(1, NA))

set_testdata(tmp_data)
print(get_testdata())
#> # A tibble: 2 × 2
#>       x     y
#>   <dbl> <dbl>
#> 1     1     1
#> 2     0    NA
expect_base(y, x == 1)

tmp_data$y <- 1
print(get_testdata())
#> # A tibble: 2 × 2
#>       x     y
#>   <dbl> <dbl>
#> 1     1     1
#> 2     0     1
expect_base(y, x == 1)
#> Error: get_testdata() has a base mismatch in variable `y`.
#> 0 cases have `x == 1` but `y` is missing.
#> 1 cases do not have `x == 1` but `y` is non missing.

...

Additional arguments are specific to the expectation. See the help page for the expectation function for details.

Categories

Data expectations fall into a few main classes.

Value

?`date-expectations`

?`label-expectations`
?`pattern-expectations`
?`proportion-expectations`
?`text-expectations`

?`value-expectations`

Value expectations test variable values for valid data. Tests include explicit value checks, pattern checks and others.

Relationships

?`exclusivity-expectations`

?`uniqueness-expectations`

Relationship expectations test for relationships among variables. Tests include uniqueness and exclusivity checks.

Conditional

?`conditional-expectations`

Conditional expectations check for the co-existence of multiple conditions.

Data frame comparison

?`datacomp-expectations`

Data frame comparison expectations test for consistency between data frames, for example ensuring similar frequencies between similar variables in different data frames.

Generic

?`generic-expectations`

Generic expectations allow for testing of a data frame using an arbitrary function. The function provided should take a single vector as its first argument and return a logical vector showing whether each element has passed or failed. Additional arguments to the checking function can be passed as a list using the args argument.

testdat includes a set of useful checking functions. See ?`chk-generic` for details. Several of the checking functions have a corresponding expectation. These are listed in ?`chk-expect`.

Using tests

Testing inside a script

The easiest way to use data testing is directly inside an R script. Expecations and test blocks throw an error if they fail, so it is very clear to the user that something needs to be checked, and the script will fail when sourced.

The example below shows how to adapt a script from exploratory “print and check” to a testing approach, using a simple data processing exercise.

testdat

library(testdat)
library(dplyr)

x <- tribble(
  ~id, ~pcode, ~state, ~nsw_only,
  1,   2000,   "NSW",  1,
  2,   3123,   "VIC",  NA,
  3,   2123,   "NSW",  3,
  4,   12345,  "VIC",  3
)

with_testdata(x, {
  test_that("id is unique", {
    expect_unique(id)
  })
  
  test_that("variable values are correct", {
    expect_values(pcode, 2000:2999, 3000:3999)
    expect_values(state, c("NSW", "VIC"))
    expect_values(nsw_only, 1:3) # by default expect_values allows NAs
  })
  
  test_that("filters applied correctly", {
    expect_base(nsw_only, state == "NSW")
  })
})
#> Test passed 🎉
#> ── Failure: variable values are correct ────────────────────────────────────────
#> get_testdata() has 1 records failing value check on variable `pcode`.
#> Variable set: `pcode`
#> Filter: None
#> Arguments: `<int: 2000L, 2001L, 2002L, 2003L, 2004L, ...>, <int: 3000L, 3001L, 3002L,`
#> get_testdata() has 1 records failing value check on variable `pcode`.
#> Variable set: `pcode`
#> Filter: None
#> Arguments: `  3003L, 3004L, ...>, miss = <chr: NA, "">`
#> Error:
#> ! Test failed

x <- x %>% mutate(market = case_when(pcode %in% 2000:2999 ~ 1,
                                     pcode %in% 3000:3999 ~ 2))

with_testdata(x, {
  test_that("market derived correctly", {
    expect_values(market, 1:2, miss = NULL) # miss = NULL excludes NAs from valid values
  })
})
#> ── Failure: market derived correctly ───────────────────────────────────────────
#> get_testdata() has 1 records failing value check on variable `market`.
#> Variable set: `market`
#> Filter: None
#> Arguments: `<int: 1L, 2L>, miss = NULL`
#> Error:
#> ! Test failed

Note that:

  • The global test data can be set with set_testdata() or with_testdata().

  • The test_that() wrapper is not strictly necessary, but it provides more informative error messaging and logically groups a set of expectations.

  • Each with_testdata() block will fail on the first test_that() block that fails. The “filters applied correctly” test block is not run in the example.

Using a test suite

Setting up a proper project test suite can be useful for automatically validating final processed data sets.

Setting up proper file based testing infrastructure is out of the scope of this vignette. See R packages for a brief introduction to testing infrastructure.

In testthat, related tests are grouped into files. When using testdat, each file should have a test data set specified with a call to set_testdata().

set_testdata(iris)

test_that("Variable format checks", {
  expect_regex(Species, "^[a-z]+$")
})
#> Test passed 🥳

test_that("Versicolor has sepal length greater than 5 - will fail", {
  expect_cond(Species %in% "versicolor", Sepal.Length >= 5)
})
#> ── Failure: Versicolor has sepal length greater than 5 - will fail ─────────────
#> get_testdata() failed consistency check. 1 cases have `Species %in% "versicolor"` but not `Sepal.Length >= 5`.
#> Error:
#> ! Test failed