tests/testthat/test-matching-utils-coverage.R

# ==============================================================================
# Coverage tests for matching_utils.R
# ==============================================================================

# ------------------------------------------------------------------------------
# validate_matching_inputs
# ------------------------------------------------------------------------------

test_that("validate_matching_inputs errors on non-data frame left", {
  expect_error(
    couplr:::validate_matching_inputs(list(x = 1:3), data.frame(x = 1:3)),
    "left must be a data frame"
  )
})

test_that("validate_matching_inputs errors on non-data frame right", {
  expect_error(
    couplr:::validate_matching_inputs(data.frame(x = 1:3), matrix(1:3)),
    "right must be a data frame"
  )
})

test_that("validate_matching_inputs errors on empty left", {
  expect_error(
    couplr:::validate_matching_inputs(data.frame(x = numeric(0)), data.frame(x = 1:3)),
    "left must have at least one row"
  )
})

test_that("validate_matching_inputs errors on empty right", {
  expect_error(
    couplr:::validate_matching_inputs(data.frame(x = 1:3), data.frame(x = numeric(0))),
    "right must have at least one row"
  )
})

test_that("validate_matching_inputs errors on missing left vars", {
  expect_error(
    couplr:::validate_matching_inputs(
      data.frame(x = 1:3),
      data.frame(x = 1:3, y = 1:3),
      vars = c("x", "y")
    ),
    "left is missing required variables"
  )
})

test_that("validate_matching_inputs errors on missing right vars", {
  expect_error(
    couplr:::validate_matching_inputs(
      data.frame(x = 1:3, y = 1:3),
      data.frame(x = 1:3),
      vars = c("x", "y")
    ),
    "right is missing required variables"
  )
})

test_that("validate_matching_inputs errors on non-numeric left var", {
  expect_error(
    couplr:::validate_matching_inputs(
      data.frame(x = letters[1:3]),
      data.frame(x = 1:3),
      vars = "x"
    ),
    "Variable 'x' in left must be numeric"
  )
})

test_that("validate_matching_inputs errors on non-numeric right var", {
  expect_error(
    couplr:::validate_matching_inputs(
      data.frame(x = 1:3),
      data.frame(x = letters[1:3]),
      vars = "x"
    ),
    "Variable 'x' in right must be numeric"
  )
})

test_that("validate_matching_inputs passes for valid inputs", {
  expect_true(
    couplr:::validate_matching_inputs(
      data.frame(x = 1:3, y = 4:6),
      data.frame(x = 7:9, y = 10:12),
      vars = c("x", "y")
    )
  )
})

test_that("validate_matching_inputs passes without vars", {
  expect_true(
    couplr:::validate_matching_inputs(
      data.frame(x = 1:3),
      data.frame(x = 7:9)
    )
  )
})

# ------------------------------------------------------------------------------
# extract_ids
# ------------------------------------------------------------------------------

test_that("extract_ids uses id column when present", {
  df <- data.frame(id = c("a", "b", "c"), x = 1:3)
  ids <- couplr:::extract_ids(df, prefix = "test")
  expect_equal(ids, c("a", "b", "c"))
})

test_that("extract_ids uses row names when meaningful", {
  df <- data.frame(x = 1:3)
  rownames(df) <- c("row_a", "row_b", "row_c")
  ids <- couplr:::extract_ids(df, prefix = "test")
  expect_equal(ids, c("row_a", "row_b", "row_c"))
})

test_that("extract_ids creates sequential IDs when needed", {
  df <- data.frame(x = 1:3)
  ids <- couplr:::extract_ids(df, prefix = "unit")
  expect_equal(ids, c("unit_1", "unit_2", "unit_3"))
})

# ------------------------------------------------------------------------------
# extract_matching_vars
# ------------------------------------------------------------------------------

test_that("extract_matching_vars returns matrix", {
  df <- data.frame(x = 1:3, y = 4:6)
  mat <- couplr:::extract_matching_vars(df, c("x", "y"))
  expect_true(is.matrix(mat))
  expect_equal(dim(mat), c(3, 2))
})

test_that("extract_matching_vars errors on NA", {
  df <- data.frame(x = c(1, NA, 3))
  expect_error(
    couplr:::extract_matching_vars(df, "x"),
    "Missing values"
  )
})

test_that("extract_matching_vars errors on NaN", {
  # NaN triggers "Missing values" first because is.na(NaN) is TRUE
  df <- data.frame(x = c(1, NaN, 3))
  expect_error(
    couplr:::extract_matching_vars(df, "x"),
    "Missing values"
  )
})

test_that("extract_matching_vars errors on Inf", {
  df <- data.frame(x = c(1, Inf, 3))
  expect_error(
    couplr:::extract_matching_vars(df, "x"),
    "Infinite values"
  )
})

# ------------------------------------------------------------------------------
# get_block_id_column
# ------------------------------------------------------------------------------

test_that("get_block_id_column finds block_id", {
  df <- data.frame(block_id = 1:3, x = 4:6)
  expect_equal(couplr:::get_block_id_column(df), "block_id")
})

test_that("get_block_id_column finds blockid", {
  df <- data.frame(blockid = 1:3, x = 4:6)
  expect_equal(couplr:::get_block_id_column(df), "blockid")
})

test_that("get_block_id_column finds block", {
  df <- data.frame(block = 1:3, x = 4:6)
  expect_equal(couplr:::get_block_id_column(df), "block")
})

test_that("get_block_id_column finds stratum", {
  df <- data.frame(stratum = 1:3, x = 4:6)
  expect_equal(couplr:::get_block_id_column(df), "stratum")
})

test_that("get_block_id_column finds stratum_id", {
  df <- data.frame(stratum_id = 1:3, x = 4:6)
  expect_equal(couplr:::get_block_id_column(df), "stratum_id")
})

test_that("get_block_id_column returns NULL when no block column", {
  df <- data.frame(x = 1:3, y = 4:6)
  expect_null(couplr:::get_block_id_column(df))
})

# ------------------------------------------------------------------------------
# has_blocks
# ------------------------------------------------------------------------------

test_that("has_blocks returns TRUE when block column exists", {
  df <- data.frame(block_id = 1:3, x = 4:6)
  expect_true(couplr:::has_blocks(df))
})

test_that("has_blocks returns FALSE when no block column", {
  df <- data.frame(x = 1:3, y = 4:6)
  expect_false(couplr:::has_blocks(df))
})

# ------------------------------------------------------------------------------
# validate_weights
# ------------------------------------------------------------------------------

test_that("validate_weights returns ones for NULL", {
  result <- couplr:::validate_weights(NULL, c("x", "y", "z"))
  expect_equal(result, c(1, 1, 1))
})

test_that("validate_weights accepts numeric vector", {
  result <- couplr:::validate_weights(c(2, 3), c("x", "y"))
  expect_equal(result, c(2, 3))
})

test_that("validate_weights errors on wrong length", {
  expect_error(
    couplr:::validate_weights(c(1, 2, 3), c("x", "y")),
    "weights must have length"
  )
})

test_that("validate_weights errors on negative weights", {
  expect_error(
    couplr:::validate_weights(c(1, -1), c("x", "y")),
    "weights must be non-negative"
  )
})

test_that("validate_weights accepts named weights", {
  result <- couplr:::validate_weights(c(x = 2, y = 3), c("x", "y"))
  expect_equal(as.numeric(result), c(2, 3))
})

test_that("validate_weights errors on unknown variable in list weights", {
  # Named list weights validate unknown variables
  expect_error(
    couplr:::validate_weights(list(x = 1, z = 2), c("x", "y")),
    "unknown variable"
  )
})

test_that("validate_weights accepts list weights", {
  result <- couplr:::validate_weights(list(x = 2, y = 3), c("x", "y"))
  expect_equal(as.numeric(result), c(2, 3))
})

test_that("validate_weights errors on invalid type", {
  expect_error(
    couplr:::validate_weights("invalid", c("x", "y")),
    "must be a numeric vector or named list"
  )
})

# ------------------------------------------------------------------------------
# validate_calipers
# ------------------------------------------------------------------------------

test_that("validate_calipers returns NULL for NULL", {
  expect_null(couplr:::validate_calipers(NULL, c("x", "y")))
})

test_that("validate_calipers accepts named vector", {
  result <- couplr:::validate_calipers(c(x = 0.1), c("x", "y"))
  expect_equal(result, c(x = 0.1))
})

test_that("validate_calipers accepts list", {
  result <- couplr:::validate_calipers(list(x = 0.1), c("x", "y"))
  expect_equal(result, list(x = 0.1))
})

test_that("validate_calipers errors on non-list non-numeric", {
  expect_error(
    couplr:::validate_calipers("invalid", c("x", "y")),
    "must be a named numeric vector or list"
  )
})

test_that("validate_calipers errors on unnamed", {
  expect_error(
    couplr:::validate_calipers(c(0.1, 0.2), c("x", "y")),
    "must be named"
  )
})

test_that("validate_calipers errors on unknown variable", {
  expect_error(
    couplr:::validate_calipers(c(z = 0.1), c("x", "y")),
    "unknown variables"
  )
})

test_that("validate_calipers errors on non-positive values", {
  expect_error(
    couplr:::validate_calipers(c(x = 0), c("x", "y")),
    "must be positive"
  )
})

test_that("validate_calipers errors on negative values", {
  expect_error(
    couplr:::validate_calipers(c(x = -0.1), c("x", "y")),
    "must be positive"
  )
})

Try the couplr package in your browser

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

couplr documentation built on Jan. 20, 2026, 5:07 p.m.