tests/testthat/test-glance.R

# =============================================================================
# Tests for glance.MP and glance.AGGTEobj S3 methods
# =============================================================================

# Shared setup: generate MP and AGGTEobj results
set.seed(20260401)
sp <- did::reset.sim()
data_gl <- did::build_sim_dataset(sp)

mp_slow <- suppressWarnings(suppressMessages(
  att_gt(yname = "Y", xformla = ~X, data = data_gl, tname = "period",
         idname = "id", gname = "G", est_method = "dr",
         bstrap = FALSE, faster_mode = FALSE)
))

mp_fast <- suppressWarnings(suppressMessages(
  att_gt(yname = "Y", xformla = ~X, data = data_gl, tname = "period",
         idname = "id", gname = "G", est_method = "dr",
         bstrap = FALSE, faster_mode = TRUE)
))

agg_types <- c("simple", "dynamic", "group", "calendar")
agg_slow <- lapply(setNames(agg_types, agg_types), function(tp) {
  suppressWarnings(aggte(mp_slow, type = tp))
})
agg_fast <- lapply(setNames(agg_types, agg_types), function(tp) {
  suppressWarnings(aggte(mp_fast, type = tp))
})

# =============================================================================
# glance.MP tests
# =============================================================================

test_that("glance.MP returns a one-row data.frame", {
  gl <- glance(mp_slow)
  expect_equal(nrow(gl), 1)
  expect_s3_class(gl, "data.frame")
})

test_that("glance.MP has expected columns", {
  gl <- glance(mp_slow)
  expected_cols <- c("nobs", "ngroup", "ntime", "control.group", "est.method")
  expect_true(all(expected_cols %in% names(gl)))
})

test_that("glance.MP values are reasonable", {
  gl <- glance(mp_slow)
  expect_true(gl$nobs > 0)
  expect_true(gl$ngroup > 0)
  expect_true(gl$ntime > 0)
  expect_equal(gl$control.group, "nevertreated")
  expect_equal(gl$est.method, "dr")
})

test_that("glance.MP nobs matches nobs.MP", {
  gl <- glance(mp_slow)
  expect_equal(gl$nobs, nobs(mp_slow))
})

test_that("glance.MP works with faster_mode = TRUE", {
  gl <- glance(mp_fast)
  expect_equal(nrow(gl), 1)
  expect_true(gl$nobs > 0)
  expect_true(gl$ngroup > 0)
  expect_true(gl$ntime > 0)
})

test_that("glance.MP agrees between faster_mode settings", {
  gl_slow <- glance(mp_slow)
  gl_fast <- glance(mp_fast)
  expect_equal(gl_slow$nobs, gl_fast$nobs)
  expect_equal(gl_slow$ngroup, gl_fast$ngroup)
  expect_equal(gl_slow$ntime, gl_fast$ntime)
  expect_equal(gl_slow$control.group, gl_fast$control.group)
  expect_equal(gl_slow$est.method, gl_fast$est.method)
})

# =============================================================================
# glance.AGGTEobj tests
# =============================================================================

test_that("glance.AGGTEobj returns a one-row data.frame for all 4 types", {
  for (tp in agg_types) {
    gl <- glance(agg_slow[[tp]])
    expect_equal(nrow(gl), 1, label = paste("nrow for", tp))
    expect_s3_class(gl, "data.frame")
  }
})

test_that("glance.AGGTEobj has expected columns", {
  gl <- glance(agg_slow[["dynamic"]])
  expected_cols <- c("type", "nobs", "ngroup", "ntime", "control.group", "est.method")
  expect_true(all(expected_cols %in% names(gl)))
})

test_that("glance.AGGTEobj type column matches requested type", {
  for (tp in agg_types) {
    gl <- glance(agg_slow[[tp]])
    expect_equal(gl$type, tp, label = paste("type for", tp))
  }
})

test_that("glance.AGGTEobj values are not NULL or NA", {
  for (tp in agg_types) {
    gl <- glance(agg_slow[[tp]])
    expect_false(any(sapply(gl, is.null)), label = paste("no NULLs for", tp))
    expect_false(any(is.na(gl)), label = paste("no NAs for", tp))
  }
})

test_that("glance.AGGTEobj works with faster_mode = TRUE", {
  for (tp in agg_types) {
    gl <- glance(agg_fast[[tp]])
    expect_equal(nrow(gl), 1, label = paste("nrow for", tp, "fast"))
    expect_false(any(sapply(gl, is.null)), label = paste("no NULLs for", tp, "fast"))
  }
})

test_that("glance.MP and glance.AGGTEobj agree on nobs", {
  gl_mp <- glance(mp_slow)
  for (tp in agg_types) {
    gl_agg <- glance(agg_slow[[tp]])
    expect_equal(gl_mp$nobs, gl_agg$nobs, label = paste("nobs for", tp))
  }
})

# =============================================================================
# Backwards compatibility: DIDparams lacking faster_mode (pre-2.2 saved objects)
# =============================================================================

test_that("glance.MP works when DIDparams lacks faster_mode", {
  mp_old <- mp_slow
  mp_old$DIDparams$faster_mode <- NULL
  expect_identical(glance(mp_old), glance(mp_slow))
})

test_that("glance.AGGTEobj works when DIDparams lacks faster_mode", {
  agg_old <- agg_slow[["dynamic"]]
  agg_old$DIDparams$faster_mode <- NULL
  expect_identical(glance(agg_old), glance(agg_slow[["dynamic"]]))
})

test_that("nobs.AGGTEobj works when DIDparams lacks faster_mode", {
  agg_old <- agg_slow[["dynamic"]]
  agg_old$DIDparams$faster_mode <- NULL
  expect_identical(stats::nobs(agg_old), stats::nobs(agg_slow[["dynamic"]]))
})

# =============================================================================
# Custom est_method (user-supplied function): glance reports "custom"
# =============================================================================

mp_custom <- suppressWarnings(suppressMessages(
  att_gt(yname = "Y", xformla = ~X, data = data_gl, tname = "period",
         idname = "id", gname = "G",
         est_method = function(y1, y0, D, covariates, i.weights, inffunc, ...) {
           DRDID::drdid_panel(y1 = y1, y0 = y0, D = D, covariates = covariates,
                              i.weights = i.weights, boot = FALSE,
                              inffunc = inffunc)
         },
         bstrap = FALSE, faster_mode = FALSE)
))

test_that("glance.MP works with a custom est_method function", {
  gl <- glance(mp_custom)
  expect_s3_class(gl, "data.frame")
  expect_equal(nrow(gl), 1)
  expect_equal(gl$est.method, "custom")
})

test_that("glance.AGGTEobj works with a custom est_method function", {
  agg_custom <- suppressWarnings(aggte(mp_custom, type = "simple"))
  gl <- glance(agg_custom)
  expect_s3_class(gl, "data.frame")
  expect_equal(nrow(gl), 1)
  expect_equal(gl$est.method, "custom")
})

Try the did package in your browser

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

did documentation built on June 13, 2026, 5:07 p.m.