R/coherent_dist.R

Defines functions consecutive_k_dist bridge_dist phi.kofn_dist kofn_dist surv.parallel_dist min_paths.parallel_dist phi.parallel_dist parallel_dist surv.series_dist min_paths.series_dist phi.series_dist series_dist dual.coherent_dist min_paths.coherent_dist component.coherent_dist ncomponents.coherent_dist coherent_dist

Documented in bridge_dist coherent_dist consecutive_k_dist dual.coherent_dist kofn_dist parallel_dist series_dist

# ==========================================================================
# General coherent_dist + topology shortcut constructors
# ==========================================================================


#' Coherent system distribution from minimal path sets
#'
#' General-purpose constructor. Users supply a list of minimal path sets
#' (each a vector of component indices) and a list of component
#' distributions (each an `algebraic.dist::dist` object with parameters
#' baked in). The resulting object is a `dist_structure` and `dist`.
#'
#' @param min_paths List of integer vectors; each is a minimal path set.
#' @param components List of `dist` objects, length `m`. Each is a
#'   fully-parameterized component lifetime distribution.
#' @param m Optional integer. Inferred from `components` and `min_paths`
#'   if omitted.
#' @return An object of class
#'   `c("coherent_dist", "dist_structure", "univariate_dist", "dist")`.
#' @examples
#' # A bridge network with exponential components
#' sys <- coherent_dist(
#'   min_paths = list(c(1, 4), c(2, 5), c(1, 3, 5), c(2, 3, 4)),
#'   components = replicate(5, algebraic.dist::exponential(1), simplify = FALSE)
#' )
#' reliability(sys, 0.9)
#' @export
coherent_dist <- function(min_paths, components, m = NULL) {
  stopifnot(is.list(min_paths), is.list(components))
  is_dist <- vapply(components, function(d) inherits(d, "dist"),
                    logical(1L))
  if (!all(is_dist)) {
    bad <- which(!is_dist)
    stop(sprintf(
      "components must all inherit from 'dist'; offending positions: %s",
      paste(bad, collapse = ", ")
    ), call. = FALSE)
  }
  if (is.null(m)) {
    all_idx <- unlist(min_paths)
    m <- max(length(components),
             if (length(all_idx)) max(all_idx) else 0L)
  }
  stopifnot(length(components) == m)
  structure(
    list(
      min_paths = lapply(min_paths, as.integer),
      m = as.integer(m),
      components = components
    ),
    class = c("coherent_dist", "dist_structure",
              "univariate_dist", "continuous_dist", "dist")
  )
}


#' @export
ncomponents.coherent_dist <- function(x) x$m


#' @export
component.coherent_dist <- function(x, j, ...) {
  stopifnot(j >= 1L, j <= x$m)
  x$components[[j]]
}


#' @export
min_paths.coherent_dist <- function(x) x$min_paths


#' Dual of a coherent_dist: swap cuts and paths
#'
#' Overrides the lazy-wrapper default with a proper `coherent_dist`:
#' `min_paths(dual(x)) = min_cuts(x)`.
#'
#' @rdname dual
#' @export
dual.coherent_dist <- function(x) {
  coherent_dist(
    min_paths = min_cuts(x),
    components = x$components,
    m = x$m
  )
}


# ==========================================================================
# Topology shortcut constructors
# ==========================================================================


#' Series system distribution
#'
#' A series system fails if any single component fails. Equivalent to
#' `min(components)` as random variables; unlike `min` in the base algebra,
#' `series_dist` preserves the component decomposition so topology queries
#' work.
#'
#' @param components List of `dist` objects.
#' @return A `series_dist` inheriting from `coherent_dist`.
#' @examples
#' sys <- series_dist(replicate(3, algebraic.dist::exponential(1), simplify = FALSE))
#' algebraic.dist::surv(sys)(0.5)
#' @export
series_dist <- function(components) {
  stopifnot(is.list(components), length(components) >= 1L)
  m <- length(components)
  obj <- coherent_dist(
    min_paths = list(seq_len(m)),
    components = components,
    m = m
  )
  class(obj) <- c("series_dist", class(obj))
  obj
}


#' @export
phi.series_dist <- function(x, state) {
  stopifnot(length(state) == x$m)
  as.integer(all(state == 1L))
}

#' @export
min_paths.series_dist <- function(x) list(seq_len(x$m))


# Override the dist_structure default (which evaluates the 2^m-state
# reliability polynomial via min_path/min_cut enumeration) with the
# closed form S_sys(t) = prod_j S_j(t). This avoids the exponential blow-up
# for series of arbitrary components when no closed-form family-specific
# specialization (exp_series, wei_series, etc.) applies.
#' @export
surv.series_dist <- function(x, ...) {
  m <- ncomponents(x)
  comp_surv_fns <- lapply(seq_len(m), function(j) {
    algebraic.dist::surv(component(x, j))
  })
  function(t, ...) {
    vapply(t, function(ti) {
      prod(vapply(comp_surv_fns, function(S_j) S_j(ti), numeric(1L)))
    }, numeric(1L))
  }
}


#' Parallel system distribution
#'
#' A parallel system fails only when all components fail. Equivalent to
#' `max(components)` but preserves topology.
#'
#' @param components List of `dist` objects.
#' @return A `parallel_dist` inheriting from `coherent_dist`.
#' @export
parallel_dist <- function(components) {
  stopifnot(is.list(components), length(components) >= 1L)
  m <- length(components)
  obj <- coherent_dist(
    min_paths = as.list(seq_len(m)),
    components = components,
    m = m
  )
  class(obj) <- c("parallel_dist", class(obj))
  obj
}


#' @export
phi.parallel_dist <- function(x, state) {
  stopifnot(length(state) == x$m)
  as.integer(any(state == 1L))
}

#' @export
min_paths.parallel_dist <- function(x) as.list(seq_len(x$m))


# Closed-form override: S_sys(t) = 1 - prod_j F_j(t).  Avoids the 2^m
# reliability-polynomial enumeration for parallel systems whose component
# family is not covered by exp_parallel.
#' @export
surv.parallel_dist <- function(x, ...) {
  m <- ncomponents(x)
  comp_cdf_fns <- lapply(seq_len(m), function(j) {
    algebraic.dist::cdf(component(x, j))
  })
  function(t, ...) {
    vapply(t, function(ti) {
      1 - prod(vapply(comp_cdf_fns, function(F_j) F_j(ti), numeric(1L)))
    }, numeric(1L))
  }
}


#' k-out-of-n system distribution
#'
#' A k-out-of-n system functions if at least `k` of its `m` components
#' function. Equivalent to the `(m - k + 1)`-th order statistic of
#' component lifetimes.
#'
#' This constructor uses the **:G** convention: `k` is the number of
#' components that must remain **functioning** for the system to function.
#' `k = 1` is parallel; `k = m` is series. The companion `kofn` package
#' (which depends on `dist.structure`) uses the **:F** convention, where
#' `k` is the number of failures that trigger system failure; conversion
#' is `k_dist = m - k_kofn + 1`. When in doubt, draw a small example:
#' `kofn_dist(k = 2, ...)` for `m = 3` functions until two of the three
#' components have failed.
#'
#' @param k Minimum functioning components for system operation (:G).
#' @param components List of `dist` objects (length `m`).
#' @return A `kofn_dist` inheriting from `coherent_dist`.
#' @seealso [order_statistic()] for the closely-related order-statistic
#'   parameterization.
#' @export
kofn_dist <- function(k, components) {
  stopifnot(is.list(components), length(components) >= 1L)
  m <- length(components)
  stopifnot(k >= 1L, k <= m)
  paths <- utils::combn(m, k, simplify = FALSE)
  obj <- coherent_dist(
    min_paths = paths,
    components = components,
    m = m
  )
  obj$k <- as.integer(k)
  class(obj) <- c("kofn_dist", class(obj))
  obj
}


#' @export
phi.kofn_dist <- function(x, state) as.integer(sum(state) >= x$k)

# min_paths.kofn_dist is intentionally absent: kofn_dist() stores
# combn(m, k) as $min_paths at construction time, so the inherited
# min_paths.coherent_dist (which returns x$min_paths directly) is both
# correct and faster than recomputing combn on every call.


#' Bridge system distribution
#'
#' The classical 5-component bridge reliability network with minimal
#' path sets `{1,4}`, `{2,5}`, `{1,3,5}`, `{2,3,4}`. Components 1 and 2
#' are the input side, 4 and 5 the output side, and 3 the cross-link.
#' The bridge is a canonical non-series, non-parallel example used
#' throughout the reliability literature; see Barlow and Proschan (1975,
#' "Statistical Theory of Reliability and Life Testing") for the
#' standard treatment.
#'
#' @param components List of 5 `dist` objects.
#' @return A `bridge_dist` inheriting from `coherent_dist`.
#' @export
bridge_dist <- function(components) {
  stopifnot(is.list(components), length(components) == 5L)
  paths <- list(
    c(1L, 4L), c(2L, 5L),
    c(1L, 3L, 5L), c(2L, 3L, 4L)
  )
  obj <- coherent_dist(
    min_paths = paths,
    components = components,
    m = 5L
  )
  class(obj) <- c("bridge_dist", class(obj))
  obj
}


#' Consecutive-k-out-of-n system distribution (type G)
#'
#' The consecutive-k-out-of-n:G system functions when at least one block
#' of `k` consecutive components all function. Minimal path sets are the
#' `n - k + 1` consecutive blocks of size `k`. (Note: this is the :G
#' variant; the :F variant, "fails when any `k` consecutive fail", has
#' different minimal paths.)
#'
#' @param k Block size.
#' @param components List of `dist` objects (length `n`).
#' @return A `consecutive_k_dist` inheriting from `coherent_dist`.
#' @export
consecutive_k_dist <- function(k, components) {
  n <- length(components)
  stopifnot(k >= 1L, k <= n)
  paths <- lapply(seq_len(n - k + 1L), function(i) as.integer(i:(i + k - 1L)))
  obj <- coherent_dist(
    min_paths = paths,
    components = components,
    m = n
  )
  class(obj) <- c("consecutive_k_dist", class(obj))
  obj
}

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.