R/cooks.distance.R

Defines functions .symmetric_ginv .cooks.distance_internal cooks.distance.brma

Documented in cooks.distance.brma

# ============================================================================ #
# brma.cooks.distance.R
# ============================================================================ #
#
# This file contains the cooks.distance method for brma model fits. Cook's
# distance measures the aggregate impact of an observation on all model
# coefficients.
#
# ============================================================================ #


#' @title Cook's Distance for brma Objects
#'
#' @description Computes Cook's distance for a fitted brma object. Cook's
#' distance measures the aggregate influence of each observation on the
#' model coefficients.
#'
#' @param model a fitted brma object.
#' @param ... additional arguments (currently ignored).
#'
#' @details
#' Cook's distance is computed as a PSIS leave-one-out deletion diagnostic. For
#' each observation \eqn{i}, normalized PSIS weights estimate the fitted values
#' under the leave-one-out posterior. The distance is the posterior Mahalanobis
#' distance between the full-data and leave-one-out fitted-value vectors:
#' \deqn{D_i = \frac{\Delta_i' V_\mu^+ \Delta_i}{P}}
#'
#' where \eqn{\Delta_i = \hat{\mu} - \hat{\mu}_{(-i)}}, \eqn{V_\mu^+} is the
#' generalized inverse of the full-posterior fitted-value covariance, and
#' \eqn{P} is the rank of the fixed-effect model matrix.
#'
#' @return A numeric vector of Cook's distance values, one for each observation.
#'
#' @examples \dontrun{
#' if (requireNamespace("metadat", quietly = TRUE)) {
#'   data(dat.lehmann2018, package = "metadat")
#'   fit <- bPET(yi = yi, vi = vi, data = dat.lehmann2018, measure = "SMD")
#'   fit <- add_loo(fit)
#'
#'   cooks.distance(fit)
#' }
#' }
#'
#' @seealso \code{\link{influence.brma}}, \code{\link{dffits.brma}}, \code{\link{hatvalues.brma}}
#' @importFrom stats cooks.distance
#' @exportS3Method
cooks.distance.brma <- function(model, ...) {

  # the function relies on hatvalues
  # as such it is sensible only sensible for normal models
  outcome_type       <- .outcome_type(model)
  is_weightfunction  <- .is_weightfunction(model)

  if (outcome_type != "norm") {
    stop("cooks.distance is only available for normal outcome models.", call. = FALSE)
  }
  if (is_weightfunction) {
    stop("cooks.distance is not available for selection models (weightfunction).", call. = FALSE)
  }

  fit_samples <- .influence_fit_samples(model)
  weights     <- .diagnostic_psis_weights(model)
  P           <- qr(.get_model_matrix(model))[["rank"]]
  d_vec       <- .cooks.distance_internal(fit_samples, weights, P)
  d_vec       <- .diagnostic_set_names(d_vec, model)

  return(d_vec)
}

.cooks.distance_internal <- function(fit_samples, weights, P) {

  summary <- .psis_fit_influence_summary(fit_samples, weights)
  delta   <- sweep(summary[["loo_fit"]], 2, summary[["full_fit"]], "-")
  delta   <- -delta

  vcov_fit <- stats::cov(fit_samples)
  vcov_inv <- .symmetric_ginv(vcov_fit)

  d_vec <- rowSums((delta %*% vcov_inv) * delta) / max(P, 1L)
  names(d_vec) <- colnames(fit_samples)

  return(d_vec)
}


# ---------------------------------------------------------------------------- #
# .symmetric_ginv
# ---------------------------------------------------------------------------- #
#
# Generalized inverse for symmetric positive semi-definite covariance matrices.
#
# ---------------------------------------------------------------------------- #
.symmetric_ginv <- function(x, tol = sqrt(.Machine$double.eps)) {

  x   <- (x + t(x)) / 2
  eig <- eigen(x, symmetric = TRUE)

  keep <- eig[["values"]] > max(abs(eig[["values"]]), 1) * tol
  if (!any(keep)) {
    return(matrix(0, nrow = nrow(x), ncol = ncol(x)))
  }

  vectors <- eig[["vectors"]][, keep, drop = FALSE]
  values  <- eig[["values"]][keep]

  return(vectors %*% diag(1 / values, nrow = length(values)) %*% t(vectors))
}

Try the RoBMA package in your browser

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

RoBMA documentation built on May 7, 2026, 5:08 p.m.