use crate::distribution::{Continuous, ContinuousCDF};
use crate::statistics::{Distribution, Max, Median, Min, Mode};
use std::f64;
#[derive(Copy, Clone, PartialEq, Debug)]
pub struct Laplace {
location: f64,
scale: f64,
}
#[derive(Copy, Clone, PartialEq, Eq, Debug, Hash)]
#[non_exhaustive]
pub enum LaplaceError {
LocationInvalid,
ScaleInvalid,
}
impl std::fmt::Display for LaplaceError {
#[cfg_attr(coverage_nightly, coverage(off))]
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
match self {
LaplaceError::LocationInvalid => write!(f, "Location is NaN"),
LaplaceError::ScaleInvalid => write!(f, "Scale is NaN, zero or less than zero"),
}
}
}
impl std::error::Error for LaplaceError {}
impl Laplace {
pub fn new(location: f64, scale: f64) -> Result<Laplace, LaplaceError> {
if location.is_nan() {
return Err(LaplaceError::LocationInvalid);
}
if scale.is_nan() || scale <= 0.0 {
return Err(LaplaceError::ScaleInvalid);
}
Ok(Laplace { location, scale })
}
pub fn location(&self) -> f64 {
self.location
}
pub fn scale(&self) -> f64 {
self.scale
}
}
impl std::fmt::Display for Laplace {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "Laplace({}, {})", self.location, self.scale)
}
}
#[cfg(feature = "rand")]
#[cfg_attr(docsrs, doc(cfg(feature = "rand")))]
impl ::rand::distributions::Distribution<f64> for Laplace {
fn sample<R: ::rand::Rng + ?Sized>(&self, rng: &mut R) -> f64 {
let x: f64 = rng.gen_range(-0.5..0.5);
self.location - self.scale * x.signum() * (1. - 2. * x.abs()).ln()
}
}
impl ContinuousCDF<f64, f64> for Laplace {
fn cdf(&self, x: f64) -> f64 {
let y = (-(x - self.location).abs() / self.scale).exp() / 2.;
if x >= self.location {
1. - y
} else {
y
}
}
fn sf(&self, x: f64) -> f64 {
let y = (-(x - self.location).abs() / self.scale).exp() / 2.;
if x >= self.location {
y
} else {
1. - y
}
}
fn inverse_cdf(&self, p: f64) -> f64 {
if p <= 0. || 1. <= p {
panic!("p must be in [0, 1]");
};
if p <= 0.5 {
self.location + self.scale * (2. * p).ln()
} else {
self.location - self.scale * (2. - 2. * p).ln()
}
}
}
impl Min<f64> for Laplace {
fn min(&self) -> f64 {
f64::NEG_INFINITY
}
}
impl Max<f64> for Laplace {
fn max(&self) -> f64 {
f64::INFINITY
}
}
impl Distribution<f64> for Laplace {
fn mean(&self) -> Option<f64> {
Some(self.location)
}
fn variance(&self) -> Option<f64> {
Some(2. * self.scale * self.scale)
}
fn entropy(&self) -> Option<f64> {
Some((2. * self.scale).ln() + 1.)
}
fn skewness(&self) -> Option<f64> {
Some(0.)
}
}
impl Median<f64> for Laplace {
fn median(&self) -> f64 {
self.location
}
}
impl Mode<Option<f64>> for Laplace {
fn mode(&self) -> Option<f64> {
Some(self.location)
}
}
impl Continuous<f64, f64> for Laplace {
fn pdf(&self, x: f64) -> f64 {
(-(x - self.location).abs() / self.scale).exp() / (2. * self.scale)
}
fn ln_pdf(&self, x: f64) -> f64 {
((-(x - self.location).abs() / self.scale).exp() / (2. * self.scale)).ln()
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::testing_boiler;
testing_boiler!(location: f64, scale: f64; Laplace; LaplaceError);
fn test_rel_close<F>(location: f64, scale: f64, expected: f64, rtol: f64, get_fn: F)
where
F: Fn(Laplace) -> f64,
{
let x = create_and_get(location, scale, get_fn);
assert_relative_eq!(expected, x, epsilon = 0.0, max_relative = rtol);
}
#[test]
fn test_create() {
create_ok(1.0, 2.0);
create_ok(f64::NEG_INFINITY, 0.1);
create_ok(-5.0 - 1.0, 1.0);
create_ok(0.0, 5.0);
create_ok(1.0, 7.0);
create_ok(5.0, 10.0);
create_ok(f64::INFINITY, f64::INFINITY);
}
#[test]
fn test_bad_create() {
test_create_err(2.0, -1.0, LaplaceError::ScaleInvalid);
test_create_err(f64::NAN, 1.0, LaplaceError::LocationInvalid);
create_err(f64::NAN, -1.0);
}
#[test]
fn test_mean() {
let mean = |x: Laplace| x.mean().unwrap();
test_exact(f64::NEG_INFINITY, 0.1, f64::NEG_INFINITY, mean);
test_exact(-5.0 - 1.0, 1.0, -6.0, mean);
test_exact(0.0, 5.0, 0.0, mean);
test_exact(1.0, 10.0, 1.0, mean);
test_exact(f64::INFINITY, f64::INFINITY, f64::INFINITY, mean);
}
#[test]
fn test_variance() {
let variance = |x: Laplace| x.variance().unwrap();
test_absolute(f64::NEG_INFINITY, 0.1, 0.02, 1E-12, variance);
test_absolute(-5.0 - 1.0, 1.0, 2.0, 1E-12, variance);
test_absolute(0.0, 5.0, 50.0, 1E-12, variance);
test_absolute(1.0, 7.0, 98.0, 1E-12, variance);
test_absolute(5.0, 10.0, 200.0, 1E-12, variance);
test_absolute(f64::INFINITY, f64::INFINITY, f64::INFINITY, 1E-12, variance);
}
#[test]
fn test_entropy() {
let entropy = |x: Laplace| x.entropy().unwrap();
test_absolute(
f64::NEG_INFINITY,
0.1,
(2.0 * f64::consts::E * 0.1).ln(),
1E-12,
entropy,
);
test_absolute(-6.0, 1.0, (2.0 * f64::consts::E).ln(), 1E-12, entropy);
test_absolute(1.0, 7.0, (2.0 * f64::consts::E * 7.0).ln(), 1E-12, entropy);
test_absolute(5., 10., (2. * f64::consts::E * 10.).ln(), 1E-12, entropy);
test_absolute(f64::INFINITY, f64::INFINITY, f64::INFINITY, 1E-12, entropy);
}
#[test]
fn test_skewness() {
let skewness = |x: Laplace| x.skewness().unwrap();
test_exact(f64::NEG_INFINITY, 0.1, 0.0, skewness);
test_exact(-6.0, 1.0, 0.0, skewness);
test_exact(1.0, 7.0, 0.0, skewness);
test_exact(5.0, 10.0, 0.0, skewness);
test_exact(f64::INFINITY, f64::INFINITY, 0.0, skewness);
}
#[test]
fn test_mode() {
let mode = |x: Laplace| x.mode().unwrap();
test_exact(f64::NEG_INFINITY, 0.1, f64::NEG_INFINITY, mode);
test_exact(-6.0, 1.0, -6.0, mode);
test_exact(1.0, 7.0, 1.0, mode);
test_exact(5.0, 10.0, 5.0, mode);
test_exact(f64::INFINITY, f64::INFINITY, f64::INFINITY, mode);
}
#[test]
fn test_median() {
let median = |x: Laplace| x.median();
test_exact(f64::NEG_INFINITY, 0.1, f64::NEG_INFINITY, median);
test_exact(-6.0, 1.0, -6.0, median);
test_exact(1.0, 7.0, 1.0, median);
test_exact(5.0, 10.0, 5.0, median);
test_exact(f64::INFINITY, f64::INFINITY, f64::INFINITY, median);
}
#[test]
fn test_min() {
test_exact(0.0, 1.0, f64::NEG_INFINITY, |l| l.min());
}
#[test]
fn test_max() {
test_exact(0.0, 1.0, f64::INFINITY, |l| l.max());
}
#[test]
fn test_density() {
let pdf = |arg: f64| move |x: Laplace| x.pdf(arg);
test_absolute(0.0, 0.1, 1.529511602509129e-06, 1E-12, pdf(1.5));
test_absolute(1.0, 0.1, 7.614989872356341e-08, 1E-12, pdf(2.8));
test_absolute(-1.0, 0.1, 3.8905661205668983e-19, 1E-12, pdf(-5.4));
test_absolute(5.0, 0.1, 5.056107463052243e-43, 1E-12, pdf(-4.9));
test_absolute(-5.0, 0.1, 1.9877248679543235e-30, 1E-12, pdf(2.0));
test_absolute(f64::INFINITY, 0.1, 0.0, 1E-12, pdf(5.5));
test_absolute(f64::NEG_INFINITY, 0.1, 0.0, 1E-12, pdf(-0.0));
test_absolute(0.0, 1.0, 0.0, 1E-12, pdf(f64::INFINITY));
test_absolute(1.0, 1.0, 0.00915781944436709, 1E-12, pdf(5.0));
test_absolute(-1.0, 1.0, 0.5, 1E-12, pdf(-1.0));
test_absolute(5.0, 1.0, 0.0012393760883331792, 1E-12, pdf(-1.0));
test_absolute(-5.0, 1.0, 0.0002765421850739168, 1E-12, pdf(2.5));
test_absolute(f64::INFINITY, 0.1, 0.0, 1E-12, pdf(2.0));
test_absolute(f64::NEG_INFINITY, 0.1, 0.0, 1E-12, pdf(15.0));
test_absolute(0.0, f64::INFINITY, 0.0, 1E-12, pdf(89.3));
test_absolute(1.0, f64::INFINITY, 0.0, 1E-12, pdf(-0.1));
test_absolute(-1.0, f64::INFINITY, 0.0, 1E-12, pdf(0.1));
test_absolute(5.0, f64::INFINITY, 0.0, 1E-12, pdf(-6.1));
test_absolute(-5.0, f64::INFINITY, 0.0, 1E-12, pdf(-10.0));
test_is_nan(f64::INFINITY, f64::INFINITY, pdf(2.0));
test_is_nan(f64::NEG_INFINITY, f64::INFINITY, pdf(-5.1));
}
#[test]
fn test_ln_density() {
let ln_pdf = |arg: f64| move |x: Laplace| x.ln_pdf(arg);
test_absolute(0.0, 0.1, -13.3905620875659, 1E-12, ln_pdf(1.5));
test_absolute(1.0, 0.1, -16.390562087565897, 1E-12, ln_pdf(2.8));
test_absolute(-1.0, 0.1, -42.39056208756591, 1E-12, ln_pdf(-5.4));
test_absolute(5.0, 0.1, -97.3905620875659, 1E-12, ln_pdf(-4.9));
test_absolute(-5.0, 0.1, -68.3905620875659, 1E-12, ln_pdf(2.0));
test_exact(f64::INFINITY, 0.1, f64::NEG_INFINITY, ln_pdf(5.5));
test_exact(f64::NEG_INFINITY, 0.1, f64::NEG_INFINITY, ln_pdf(-0.0));
test_exact(0.0, 1.0, f64::NEG_INFINITY, ln_pdf(f64::INFINITY));
test_absolute(1.0, 1.0, -4.693147180559945, 1E-12, ln_pdf(5.0));
test_absolute(-1.0, 1.0, -f64::consts::LN_2, 1E-12, ln_pdf(-1.0));
test_absolute(5.0, 1.0, -6.693147180559945, 1E-12, ln_pdf(-1.0));
test_absolute(-5.0, 1.0, -8.193147180559945, 1E-12, ln_pdf(2.5));
test_exact(f64::INFINITY, 0.1, f64::NEG_INFINITY, ln_pdf(2.0));
test_exact(f64::NEG_INFINITY, 0.1, f64::NEG_INFINITY, ln_pdf(15.0));
test_exact(0.0, f64::INFINITY, f64::NEG_INFINITY, ln_pdf(89.3));
test_exact(1.0, f64::INFINITY, f64::NEG_INFINITY, ln_pdf(-0.1));
test_exact(-1.0, f64::INFINITY, f64::NEG_INFINITY, ln_pdf(0.1));
test_exact(5.0, f64::INFINITY, f64::NEG_INFINITY, ln_pdf(-6.1));
test_exact(-5.0, f64::INFINITY, f64::NEG_INFINITY, ln_pdf(-10.0));
test_is_nan(f64::INFINITY, f64::INFINITY, ln_pdf(2.0));
test_is_nan(f64::NEG_INFINITY, f64::INFINITY, ln_pdf(-5.1));
}
#[test]
fn test_cdf() {
let cdf = |arg: f64| move |x: Laplace| x.cdf(arg);
let loc = 0.0f64;
let scale = 1.0f64;
let reltol = 1e-15f64;
let expected = 0.69673467014368328819810023250440977f64;
test_rel_close(loc, scale, expected, reltol, cdf(0.5));
let expected = 0.30326532985631671180189976749559023f64;
test_rel_close(loc, scale, expected, reltol, cdf(-0.5));
let expected = 1.8600379880104179814798479019315592e-44f64;
test_rel_close(loc, scale, expected, reltol, cdf(-100.0));
}
#[test]
fn test_sf() {
let sf = |arg: f64| move |x: Laplace| x.sf(arg);
let loc = 0.0f64;
let scale = 1.0f64;
let reltol = 1e-15f64;
let expected = 0.30326532985631671180189976749559022f64;
test_rel_close(loc, scale, expected, reltol, sf(0.5));
let expected = 0.69673467014368328819810023250440977f64;
test_rel_close(loc, scale, expected, reltol, sf(-0.5));
let expected = 1.86003798801041798147984790193155916e-44;
test_rel_close(loc, scale, expected, reltol, sf(100.0));
}
#[test]
fn test_inverse_cdf() {
let inverse_cdf = |arg: f64| move |x: Laplace| x.inverse_cdf(arg);
let loc = 0.0f64;
let scale = 1.0f64;
let reltol = 1e-15f64;
let expected = -22.3327037493805115307626824253854655f64;
test_rel_close(loc, scale, expected, reltol, inverse_cdf(1e-10));
let expected = -6.2146080984221917426367422425949161f64;
test_rel_close(loc, scale, expected, reltol, inverse_cdf(0.001));
let expected = 2.3025850929940456840179914546843642f64;
test_rel_close(loc, scale, expected, reltol, inverse_cdf(0.95));
}
#[cfg(feature = "rand")]
#[test]
fn test_sample() {
use ::rand::distributions::Distribution;
use ::rand::thread_rng;
let l = create_ok(0.1, 0.5);
l.sample(&mut thread_rng());
}
#[cfg(feature = "rand")]
#[test]
fn test_sample_distribution() {
use ::rand::distributions::Distribution;
use ::rand::rngs::StdRng;
use ::rand::SeedableRng;
let location = 0.0;
let scale = 1.0;
let n = create_ok(location, scale);
let trials = 10_000;
let tolerance = 250;
for seed in 0..10 {
let mut r: StdRng = SeedableRng::seed_from_u64(seed);
let result = (0..trials).map(|_| n.sample(&mut r)).fold(0, |sum, val| {
if val > 0.0 {
sum + 1
} else if val < 0.0 {
sum - 1
} else {
0
}
});
assert!(
result > -tolerance && result < tolerance,
"Balance is {result} for seed {seed}"
);
}
}
}