| Title: | Averaged Prediction Models | 
| Version: | 0.1.1 | 
| Description: | In panel data settings, specifies set of candidate models, fits them to data from pre-treatment validation periods, and selects model as average over candidate models, weighting each by posterior probability of being most robust given its differential average prediction errors in pre-treatment validation periods. Subsequent estimation and inference of causal effect's bounds accounts for both model and sampling uncertainty, and calculates the robustness changepoint value at which bounds go from excluding to including 0. The package also includes a range of diagnostic plots, such as those illustrating models' differential average prediction errors and the posterior distribution of which model is most robust. | 
| License: | GPL-2 | GPL-3 [expanded from: GPL (≥ 2)] | 
| Depends: | R (≥ 3.5.0) | 
| Imports: | stats, ggplot2 (≥ 3.5.1), ggh4x (≥ 0.2.8), ggrepel (≥ 0.9.6), MASS, sandwich, pbapply (≥ 1.7-2), fwb (≥ 0.3.0), chk (≥ 0.10.0) | 
| Suggests: | parallel, knitr, rmarkdown | 
| Encoding: | UTF-8 | 
| RoxygenNote: | 7.3.2 | 
| LazyData: | true | 
| VignetteBuilder: | knitr | 
| URL: | https://github.com/tl2624/apm/, https://tl2624.github.io/apm/ | 
| BugReports: | https://github.com/tl2624/apm/issues | 
| NeedsCompilation: | no | 
| Packaged: | 2025-05-22 19:01:33 UTC; ThomasLeavitt | 
| Author: | Thomas Leavitt | 
| Maintainer: | Thomas Leavitt <thomas.leavitt@baruch.cuny.edu> | 
| Repository: | CRAN | 
| Date/Publication: | 2025-05-22 20:00:02 UTC | 
apm: Averaged Prediction Models
Description
In panel data settings, specifies set of candidate models, fits them to data from pre-treatment validation periods, and selects model as average over candidate models, weighting each by posterior probability of being most robust given its differential average prediction errors in pre-treatment validation periods. Subsequent estimation and inference of causal effect's bounds accounts for both model and sampling uncertainty, and calculates the robustness changepoint value at which bounds go from excluding to including 0. The package also includes a range of diagnostic plots, such as those illustrating models' differential average prediction errors and the posterior distribution of which model is most robust.
Author(s)
Maintainer: Thomas Leavitt thomas.leavitt@baruch.cuny.edu (ORCID)
Authors:
- Laura Hatfield (ORCID) 
- Noah Greifer ngreifer@iq.harvard.edu (ORCID) 
See Also
Useful links:
- Report bugs at https://github.com/tl2624/apm/issues 
Estimate ATTs from models fits
Description
apm_est() computes the ATTs from the models previously fit by apm_pre(), choosing the optimal one by minimizing the largest absolute average prediction error across validation times. Optionally, this process can be simulated to arrive at a distribution of ATTs that accounts for the uncertainty in selecting the optimal model. plot() plots the resulting ATT(s).
Usage
apm_est(
  fits,
  post_time,
  M = 0,
  R = 1000L,
  all_models = FALSE,
  cl = NULL,
  verbose = TRUE,
  ...
)
## S3 method for class 'apm_est'
summary(object, level = 0.95, M = NULL, ...)
## S3 method for class 'apm_est'
plot(x, label = TRUE, size.weights = TRUE, ...)
Arguments
| fits | an  | 
| post_time | the value of the time variable considered post-treatment, for which the ATT is to be estimated. | 
| M | the sensitivity parameter for set identification. For  | 
| R | the number of bootstrap iterations used to compute the sampling variance of the ATT. Default is 1000. More is better but takes longer. | 
| all_models | 
 | 
| cl | a cluster object created by  | 
| verbose | 
 | 
| ... | other arguments passed to  | 
| level | the desired confidence level. Set to 0 to ignore sampling variation in computing the interval bounds. Default is .95. | 
| x,object | an  | 
| label | 
 | 
| size.weights | 
 | 
Details
apm_est() estimates the ATT from each model and combines them to form the BMA-weighted estimate of the ATT. Uncertainty for the BMA-weighted ATT is computed by combining two variance components, one that account for sampling and another that accounts for model selection. The component due to sampling is computed by bootstrapping the process of fitting the outcome model for the post-treatment outcome identified by post_time and computing the difference between the observed outcome mean difference and the model-predicted outcome mean difference. The fractional weighted bootstrap as implemented in fwb::fwb() is used to ensure no units are dropped from the analysis. In each bootstrap sample, the BMA-weighted ATT estimate is computed as the weighted average of the ATTs computed from the models using the fixed BMA weights computed by apm_pre(), and the variance is computed as the empirical variance over the bootstrapped estimates. The variance component due to model selection is computed as the BMA-weighted variance of the original ATTs.
When M is greater than 0, bounds for set identification and their uncertainty are additionally computed. This involves bootstrapping the fitting of the pre-period models along with post-treatment models on order to compute the maximum absolute difference in average prediction errors for each model across validation periods. Each bootstrap sample produces a margin of error for each model computed as M \times \delta_m where \delta_m is the maximum absolute difference in average prediction errors for model m. Upper and lower bounds for the set-identified BMA-weighted ATT are computed as \text{ATT}_m \pm M \times \delta_m. The same procedure as above is then used to compute the variance of these bounds.
summary() displays the BMA-weighted ATT estimate, its standard error, and Wald confidence intervals. When M is greater than 0, bounds for the set-identified ATT are displayed in the confidence interval bound columns. The lower bound is computed as \text{LB} - \sigma_{LB}Z_{l} and the upper bound as \text{UB} + \sigma_{UB}Z_{l}, where \text{LB} and \text{UB} are the lower and upper bounds, \sigma_{LB} and \sigma_{UB} are their variances accounting for sampling and model selection, and Z_{l} is the critical Z-statistic for confidence level l. To display the set-identification bounds themselves, one should set level = 0.
Value
apm_est() returns an apm_est object, which contains the ATT estimates and their variance estimates. The following components are included:
- BMA_att
- the BMA-weighted ATT 
- atts
- a 1-column matrix containing the ATT estimates from each model (when - all_models = FALSE, only models with positive BMA weights are included)
- BMA_var
- the total variance estimate for the BMA-weighted ATT incorporating the variance due to sampling and due to model selection 
- BMA_var_b
- the bootstrap-based component of the variance estimate for the BMA-weighted ATT due to sampling 
- BMA_var_m
- the component of the variance estimate for the BMA-weighted ATT due to model selection 
- M
- the value of the sensitivity parameter - M
- post_time
- the value supplied to - post_time
- observed_means
- a matrix of the observed outcome means at each pre-treatment validation period 
- pred_errors
- an array containing the average prediction errors for each model and each pre-treatment validation period 
- pred_error_diffs
- a matrix containing the difference in average prediction errors between groups for each model and each pre-treatment validation period 
- BMA_weights
- the BMA weights computed by - apm_pre()(when- all_models = FALSE, only positive BMA weights are included)
- boot_out
- an - fwbobject containing the bootstrap results
plot() returns a ggplot object displaying the ATT for each model plotted against the maximum absolute difference in average prediction errors for that model. The model with the lowest maximum absolute difference in average prediction errors is displayed in red.
summary() produces a table with the BMA-weighted ATT, it's estimated standard error, and confidence interval limits. When M is greater than 0, additional rows for each value of M are included with the lower and upper bound. When level is greater than 0, these bounds include the uncertainty due to sampling and model selection; otherwise, they correspond to the set identification bounds for the ATT.
See Also
apm_pre() for computing the BMA weights; fwb::fwb() for the fractional weighted bootstrap.
Examples
data("ptpdata")
# Combination of 4 models: 2 time trends, 2 lags
models <- apm_mod(list(crude_rate ~ 1),
                  lag = 0:1,
                  time_trend = 0:1)
models
# Fit the models to data; unit_var must be supplied for
# fixed effects
fits <- apm_pre(models,
                data = ptpdata,
                group_var = "group",
                time_var = "year",
                val_times = 2004:2007,
                unit_var = "state",
                nsim = 100,
                verbose = FALSE)
est <- apm_est(fits,
               post_time = 2008,
               M = 1,
               R = 20,
               verbose = FALSE)
est
# ATT estimate and bounds for M = 1
summary(est)
# Bounds for other values of M
summary(est, M = c(.5, 1, 1.5, 2))
# Set-ID bounds without uncertainty
summary(est, level = 0)
plot(est)
Generate models used to fit outcomes
Description
apm_mod() generates a list of models characterized by a basic model formulas and other options (e.g., lags, families, etc.) that are supplied to apm_pre(). These values are completely crossed to create a grid of model specifications, and multiple sets of model specifications can be combined using c() (see Examples).
Usage
apm_mod(
  formula_list,
  family = "gaussian",
  lag = 0L,
  diff_k = 0L,
  log = FALSE,
  time_trend = 0L,
  fixef = FALSE,
  identiy_only_log = TRUE
)
Arguments
| formula_list | a list of model formulas with the outcome on the left side and predictions (or just an intercept) on the right side. | 
| family | a list of family specifications; see  | 
| lag | a vector of integers indicating the desired outcome lags to be used as predictors. For example, a  | 
| diff_k | a vector of integers indicating the desired outcome lag to be used a an offset For example, a  | 
| log | a logical vector indicating whether the outcome should be log-transformed. Default is  | 
| time_trend | a vector of integers indicating the desired powers to be included in a time trend. For example, a  | 
| fixef | a logical vector indicating whether unit fixed effects should be included as predictors. Default is  | 
| identiy_only_log | 
 | 
Value
An apm_models object, which is a list containing the full cross (less any omitted combinations) of the model features specified in the arguments, with each combination a list. These have a print() method and can be combined using c(). Each model is named automatically, but these can be set manually using names() as well. Models can be removed by setting their value to NULL; see Examples.
See Also
Examples
data("ptpdata")
# Combination of 8 models: 1 baseline formulas,
# 2 families, 2 lags, 2 time trends
models1 <- apm_mod(crude_rate ~ 1,
                   family = list("gaussian", "quasipoisson"),
                   time_trend = 0:1,
                   lag = 0:1, fixef = TRUE)
models1
# Add a single other model with a square time trend
models2 <- apm_mod(crude_rate ~ 1,
                   family = "gaussian",
                   time_trend = 2,
                   fixef = FALSE)
models2
(models <- c(models1, models2))
# Remove a model
models[[4]] <- NULL
models
Fit validation models to pre-treatment data
Description
apm_pre() fits models to the pre-treatment data to compute the observed prediction errors for each model in each period and compute the Bayesian model averaging (BMA) weights eventually used in apm_est() to estimate the treatment effect.
Usage
apm_pre(
  models,
  data,
  weights = NULL,
  group_var,
  time_var,
  val_times,
  unit_var,
  nsim = 1000L,
  cl = NULL,
  verbose = TRUE
)
## S3 method for class 'apm_pre_fits'
summary(object, order = NULL, ...)
Arguments
| models | an  | 
| data | a dataset containing all the variables named in the supplied models (i.e., the outcome and any predictors) as well as any variable named below. | 
| weights | an optional vector of weights (e.g., sampling weights) used to fit weighted regression models. | 
| group_var | string; the name of the treatment variable in  | 
| time_var | string; the name of the variable in  | 
| val_times | a numeric vector corresponding to the pre-treatment times that will be used as validation times when select the model with the optimal average expected prediction error. | 
| unit_var | string; the name of the unit ID variable in  | 
| nsim | the number of simulated draws from the joint posterior of the fitted models to use to compute the BMA weights. Default is 1000. More is better but takes longer. | 
| cl | a cluster object created by  | 
| verbose | 
 | 
| object | an  | 
| order | how to order the summary;  | 
| ... | ignored. | 
Details
apm_pre() creates a grid of all models and all time points and fits all corresponding models. For each validation time supplied to val_times, each model is fit using all previous times. For example, for a validation time of 5, a model is fit with data only from periods 1-4.
lm(), glm(), or MASS::glm.nb() are used to fit the given models. The joint covariance matrix of all the coefficients is computed using the SUEST method described in Mize et al. (2019, p164), which is also used by the STATA command suest. This is equivalent to the covariance matrix computed by stacking the score equations for the models and fitting them using M-estimation and yields the equivalent of the HC0 covariance matrix for all within-model covariances. The covariance is clustered by unit_id.
To compute the BMA weights, random variate drawn from a multivariate normal distribution nsim times with mean vector equal to the concatenated model coefficients and covariance equal to the joint covariance matrix described above. For each iteration, the absolute average prediction errors are calculated for each model and validation period. A model is considered the "winner" if it its largest absolute average prediction error across validation periods is the smallest among all models. The BMA weight for each model is equal to the proportion of iterations in which that model was the "winner".
Value
apm_pre() returns an apm_pre_fits object, which is a list containing the models supplied to models, a grid of all fitted models, a list of all model fit objects, a list of all estimated coefficients, the joint covariance of the coefficients, the dataset supplied to data, and other components supplied to apm_pre().
summary() produces a data frame containing the BMA weights and maximum absolute difference in mean prediction errors for each model, ordered according order. An asterisk appears next to the model with the smallest error.
See Also
lm(),glm(), and MASS::glm.nb() for the functions used to fit the models; apm_est() to compute the ATT and its uncertainty; plot.apm_pre_fits() for plotting an apm_pre_fits object.
Examples
data("ptpdata")
# Combination of 8 models: 2 baseline formulas,
# 2 families, 2 lags
models <- apm_mod(crude_rate ~ 1,
                  family = list("gaussian", "quasipoisson"),
                  time_trend = 0:1,
                  lag = 0:1)
models
# Fit the models to data
fits <- apm_pre(models, data = ptpdata,
                group_var = "group",
                time_var = "year",
                val_times = 1999:2007,
                unit_var = "state",
                nsim = 200,
                verbose = FALSE)
fits
summary(fits)
plot(fits, type = "weights")
plot(fits, type = "error")
Plot outputs of apm_pre()
Description
plot() displays the Bayesian model averaging (BMA) weights for each model (computed by apm_fit() as the posterior probability of selection) and the distribution of the difference in average prediction errors.
Usage
## S3 method for class 'apm_pre_fits'
plot(
  x,
  type = "weights",
  abs = TRUE,
  ncol = 4L,
  clip_at = 15,
  model = ".optimal",
  ...
)
Arguments
| x | an  | 
| type | which values to plot: allowable options include  | 
| abs | 
 | 
| ncol | when  | 
| clip_at | when  | 
| model | string; when  | 
| ... | ignored. | 
Details
When type = "weights", plot() displays a bar plot with a bar for each model with height equal to the BMA weight/posterior probability of selection for that model. (Note that the plot margins can sometimes cut off the model names; use theme(plot.margins =) after loading ggplot2 to extend the left margin of the plot to ensure all text is visible. Alternatively, the axis text can be rotated using theme(axis.text.x =).)
When type = "errors", plot() displays a lattice of bar plots with a plot for each model displaying the difference in average prediction errors for each validation period. The period with the largest difference in average prediction errors will be shaded black. The model with the smallest maximum absolute difference in average prediction errors will have a gray label.
When type = "predict", plot() displays a lattice of line plots with a plot for each model displaying the observed and predicted outcomes for each validation period under each model. The observed outcomes are displayed as points, while the predicted outcomes are displayed as lines.
When type = "corrected", plot() displays a lattice of line plots with a plot for each model displaying the observed and corrected predictions for the treated group for each validation period under each model. The observed outcomes are displayed as points, while the corrected predictions are displayed as lines. Corrected predictions are computed as the observed outcome in the treated group minus the prediction error in the treated group plus the prediction error in the control group.
Value
A ggplot object, which can be manipulated using ggplot2 syntax (after loading ggplot2).
See Also
apm_pre() to to compute the difference in average prediction errors and BMA weights; ggplot2::geom_col(), which is used to create the plots.
Examples
data("ptpdata")
# Combination of 8 models: 2 baseline formulas,
# 2 families, 2 lags
models <- apm_mod(crude_rate ~ 1,
                  family = "gaussian",
                  time_trend = 0:1,
                  lag = 0:1,
                  diff_k = 0:1)
models
# Fit the models to data
fits <- apm_pre(models, data = ptpdata,
                group_var = "group",
                time_var = "year",
                val_times = 1999:2007,
                unit_var = "state",
                nsim = 50,
                verbose = FALSE)
fits
plot(fits, type = "weights")
plot(fits, type = "error", ncol = 2)
plot(fits, type = "predict", model = ".optimal")
plot(fits, type = "corrected", model = ".optimal")
Dataset on Annual Homicide Rates
Description
A dataset of homicide rates across 9 states from 1994 to 2008. Missouri repealed its permit-to-purchase (PTP) law in 2007.
Usage
ptpdata
Format
A dataframe of 7 variables with 135 observations (9 states, 15 years).
Details
details for dataset2
- state
- the name of the state; states present include Arkansas, Illinois, Iowa, Kansas, Kentucky, Missouri, Nebraska, Oklahoma, and Tennessee. 
- year
- the year of each observation. Years range from 1994 to 2008. 
- deaths
- the number of homicide deaths in the corresponding state in the corresponding year. 
- crude_rate
- the homicide rate in the corresponding state in the corresponding year. 
- age_adj_rate
- the age-adjusted homicide rate in the corresponding state in the corresponding year. 
- group
- whether each observation belongs to the "to-be-treated" group; 1 for Missouri and 0 for all other states. 
- treat
- whether each observation is treated; 1 for Missouri in 2008 and 0 otherwise. 
Compute the robustness changepoint
Description
robustness_bound() computes the value of the sensitivity parameter M at which the robustness bounds change from excluding to including an ATT of 0.
Usage
robustness_bound(object, level = 0.95)
Arguments
| object | an  | 
| level | the desired confidence level. Set to 0 to ignore sampling variation in computing the interval bounds. Default is .95. | 
Value
A single number corresponding to the changepoint value of M. If there is no positive value of M for which the interval bounds cross 0, NA will be returned.
See Also
summary.apm_est() for examining the ATT and bounds for a given value of M; uniroot() for the function that finds the changepoint value of M.
Examples
data("ptpdata")
# Combination of 4 models: 2 time trends, 2 lags
models <- apm_mod(list(crude_rate ~ 1),
                  lag = 0:1,
                  time_trend = 0:1)
models
# Fit the models to data; unit_var must be supplied for
# fixed effects
fits <- apm_pre(models,
                data = ptpdata,
                group_var = "group",
                time_var = "year",
                val_times = 2004:2007,
                unit_var = "state",
                nsim = 100,
                verbose = FALSE)
est <- apm_est(fits,
               post_time = 2008,
               M = 1,
               R = 20,
               verbose = FALSE)
est
# ATT estimate and bounds for M = 1
summary(est)
#Changepoint value of M ignoring estimation uncertainty
(M <- robustness_bound(est, level = 0))
summary(est, level = 0, M = M)
#Changepoint value of M accounting for estimation uncertainty
(M <- robustness_bound(est, level = .95))
summary(est, level = .95, M = M)