R/generics.R

Defines functions as_dist_structure compose_systems substitute_component vesely_fussell_importance criticality_importance birnbaum_importance reliability structural_importance is_coherent dual system_censoring system_lifetime critical_states system_signature min_cuts min_paths phi component ncomponents

Documented in as_dist_structure birnbaum_importance component compose_systems criticality_importance critical_states dual is_coherent min_cuts min_paths ncomponents phi reliability structural_importance substitute_component system_censoring system_lifetime system_signature vesely_fussell_importance

# ==========================================================================
# S3 generic functions
# ==========================================================================


#' Number of components
#'
#' @param x A [dist_structure] object.
#' @return A positive integer `m`, the number of components.
#' @export
ncomponents <- function(x) UseMethod("ncomponents")


#' The j-th component as a dist
#'
#' Returns the j-th component as an `algebraic.dist::dist` object with
#' parameters baked in (so `sampler`, `surv`, etc. work directly on it).
#'
#' @param x A [dist_structure] object.
#' @param j Integer component index in `1:ncomponents(x)`.
#' @param ... Implementation-specific arguments (e.g., parameter overrides
#'   for lazily parameterized systems).
#' @return An object inheriting from `dist`.
#' @export
component <- function(x, j, ...) UseMethod("component")


#' Structure function
#'
#' Evaluate the coherent structure function `phi: {0, 1}^m -> {0, 1}` at a
#' component state vector `state`. By convention `state[j] = 1` means
#' component `j` is functioning; `phi(state) = 1` means the system is
#' functioning.
#'
#' `phi` and [min_paths()] are dual primitives: if a subclass provides
#' only `phi.<class>()`, the default [min_paths.dist_structure()]
#' enumerates minimal subsets via `phi`; if a subclass provides only
#' `min_paths.<class>()`, the default [phi.dist_structure()] checks
#' whether `state` covers any minimal path. Subclasses providing both
#' must keep them consistent.
#'
#' @param x A [dist_structure] object.
#' @param state Integer or logical vector of length `ncomponents(x)` in
#'   `{0, 1}`.
#' @return Integer scalar, `0` or `1`.
#' @seealso [min_paths()] for the dual primitive; [validate_dist_structure()]
#'   for construction-time checking.
#' @export
phi <- function(x, state) UseMethod("phi")


#' Minimal path sets
#'
#' Returns the minimal subsets of `1:m` whose joint functioning is
#' sufficient for the system to function. `min_paths` and [phi()] are
#' dual primitives in the `dist_structure` protocol: providing one is
#' enough, the other has an enumerative default.
#'
#' @param x A [dist_structure] object.
#' @return A list of integer vectors.
#' @seealso [phi()] for the dual primitive; [min_cuts()] for the dual
#'   topology query (minimal subsets whose joint failure causes system
#'   failure).
#' @export
min_paths <- function(x) UseMethod("min_paths")


#' Minimal cut sets
#'
#' @param x A [dist_structure] object.
#' @return A list of integer vectors.
#' @export
min_cuts <- function(x) UseMethod("min_cuts")


#' System signature
#'
#' Samaniego's signature: `s = (s_1, ..., s_m)` where `s_k` is the
#' probability the system fails at the k-th component failure under
#' i.i.d. absolutely-continuous component lifetimes. Only depends on the
#' structure (phi), not on the component distribution. A default method
#' on `dist_structure` enumerates the `m!` orderings; this is feasible
#' for `m` up to about 8 or 9. Specialized subclasses should override
#' with closed-form expressions.
#'
#' @param x A [dist_structure] object.
#' @return Numeric vector of length `ncomponents(x)` summing to 1.
#' @export
system_signature <- function(x) UseMethod("system_signature")


#' Critical states of a component
#'
#' @param x A [dist_structure] object.
#' @param j Component index.
#' @return Integer matrix with `m - 1` columns; rows are the states of the
#'   other components for which `j` is critical.
#' @export
critical_states <- function(x, j) UseMethod("critical_states")


#' System lifetime from component times
#'
#' @param x A [dist_structure] object.
#' @param times Non-negative numeric vector of length `ncomponents(x)`.
#' @return Scalar system lifetime.
#' @export
system_lifetime <- function(x, times) UseMethod("system_lifetime")


#' Per-component censoring from system observation
#'
#' @param x A [dist_structure] object.
#' @param times Non-negative numeric vector of length `ncomponents(x)`.
#' @return A list with `system_time` (scalar) and `component_status`
#'   (character vector of length `m`, values in `"exact"`, `"left"`,
#'   `"right"`).
#' @export
system_censoring <- function(x, times) UseMethod("system_censoring")


#' Dual structure
#'
#' The dual structure satisfies `phi_dual(state) = 1 - phi(1 - state)`. The
#' dual of a series system is parallel; the dual of k-out-of-n is
#' (n - k + 1)-out-of-n.
#'
#' @param x A [dist_structure] object.
#' @return A [dist_structure] object representing the dual.
#' @export
dual <- function(x) UseMethod("dual")


#' Coherence axiom check
#'
#' @param x A [dist_structure] object.
#' @return `TRUE` if monotone and every component relevant.
#' @export
is_coherent <- function(x) UseMethod("is_coherent")


#' Birnbaum structural importance
#'
#' Fraction of the `2^(m - 1)` states of the other components for which
#' component `j` is critical.
#'
#' @param x A [dist_structure] object.
#' @param j Component index.
#' @return Numeric scalar in `[0, 1]`.
#' @export
structural_importance <- function(x, j) UseMethod("structural_importance")


#' System reliability polynomial
#'
#' `R(p) = E[phi(X)]` where `X_j ~ Bernoulli(p_j)` independent. The
#' multilinear extension of `phi` to component reliabilities.
#'
#' @param x A [dist_structure] object.
#' @param p Numeric vector of length `ncomponents(x)` or a scalar recycled
#'   to all components; values in `[0, 1]`.
#' @return Numeric scalar in `[0, 1]`.
#' @export
reliability <- function(x, p) UseMethod("reliability")


#' Birnbaum reliability importance
#'
#' The Birnbaum importance of component `j` at component reliabilities
#' `p` is `dR/dp_j = R(p with p_j = 1) - R(p with p_j = 0)`. Measures
#' how much the system reliability changes if `j` moves between certain
#' failure and certain success (and, by monotonicity, the rate of
#' change at any intermediate `p_j`).
#'
#' @param x A [dist_structure] object.
#' @param j Component index.
#' @param p Numeric vector of length `ncomponents(x)` or a scalar in
#'   `[0, 1]`.
#' @return Numeric scalar in `[0, 1]`.
#' @export
birnbaum_importance <- function(x, j, p) UseMethod("birnbaum_importance")


#' Criticality (Fussell) importance at time t
#'
#' Probability that component `j` has failed AND is critical given
#' that the system has failed by time `t`. Equals
#' `I_B(j; S(t)) * F_j(t) / F_sys(t)`.
#'
#' @param x A [dist_structure] object.
#' @param j Component index.
#' @param t Scalar time.
#' @return Numeric scalar in `[0, 1]`.
#' @export
criticality_importance <- function(x, j, t) UseMethod("criticality_importance")


#' Vesely-Fussell importance at time t
#'
#' Probability that at least one minimal cut set containing `j` has all
#' its components failed, given the system has failed by time `t`.
#' Computed exactly via inclusion-exclusion over subsets of cuts that
#' contain `j`.
#'
#' @param x A [dist_structure] object.
#' @param j Component index.
#' @param t Scalar time.
#' @return Numeric scalar in `[0, 1]`.
#' @export
vesely_fussell_importance <- function(x, j, t) {
  UseMethod("vesely_fussell_importance")
}


#' Substitute a component
#'
#' Return a new [dist_structure] with the j-th component replaced by
#' `new_component`. Topology is preserved; the returned object is a
#' `coherent_dist` with the same min_paths and the modified component
#' list.
#'
#' @param x A [dist_structure] object.
#' @param j Component index.
#' @param new_component A dist-compatible object to install at position j.
#' @return A new [dist_structure] object.
#' @export
substitute_component <- function(x, j, new_component) {
  UseMethod("substitute_component")
}


#' Compose systems hierarchically
#'
#' Produce a new [dist_structure] by replacing each component of `outer`
#' with a sub-system (either a [dist_structure] or a plain dist). The
#' composed system's components are the flattened inner components; its
#' min_paths enumerate the Cartesian products of inner min_paths within
#' each outer min_path.
#'
#' @param outer A [dist_structure] object.
#' @param inner_list A list of length `ncomponents(outer)`. Each element
#'   is either a [dist_structure] or a dist (single-component sub-system).
#' @return A `coherent_dist` representing the composed system.
#' @export
compose_systems <- function(outer, inner_list) {
  UseMethod("compose_systems")
}


#' Coerce to dist_structure
#'
#' Wraps a plain `algebraic.dist` distribution in a 1-component [series_dist]
#' so that it satisfies the dist_structure protocol. Useful for
#' polymorphic code that operates on dist_structure inputs but is also
#' handed unwrapped distributions. A `dist_structure` is returned
#' unchanged.
#'
#' @param x A dist or dist_structure.
#' @param ... Reserved for methods.
#' @return A [dist_structure] object.
#' @export
as_dist_structure <- function(x, ...) UseMethod("as_dist_structure")

Try the dist.structure package in your browser

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

dist.structure documentation built on May 13, 2026, 1:07 a.m.