wickra-core 0.4.2

Core streaming-first technical indicators engine for the Wickra library
Documentation
//! Rolling Jensen's Alpha (CAPM).

use std::collections::VecDeque;

use crate::error::{Error, Result};
use crate::traits::Indicator;

/// Rolling Jensen's Alpha.
///
/// Each `update` receives one `(asset_return, benchmark_return)` pair. Over
/// the trailing window of `period` pairs:
///
/// ```text
/// Beta  = cov(asset, bench) / var(bench)
/// Alpha = mean(asset) − ( risk_free + Beta · (mean(bench) − risk_free) )
/// ```
///
/// Alpha is the *risk-adjusted excess return* — the slice of the asset's
/// performance that cannot be explained by simple exposure to the
/// benchmark. A positive alpha indicates outperformance net of the market
/// premium implied by the asset's beta; negative alpha is the opposite.
///
/// Population covariance and variance are used (matching common
/// implementations in pandas-ta / quantstats); the rolling estimator stays
/// unbiased in the steady state for fixed `period`.
///
/// If the benchmark is flat (`var(bench) = 0`) the indicator falls back to
/// `alpha = mean(asset) − risk_free` — the asset's mean excess return, with
/// no market-risk adjustment, since the regression slope is undefined.
///
/// Each `update` is O(1).
#[derive(Debug, Clone)]
pub struct Alpha {
    period: usize,
    risk_free: f64,
    window: VecDeque<(f64, f64)>,
    sum_a: f64,
    sum_b: f64,
    sum_bb: f64,
    sum_ab: f64,
}

impl Alpha {
    /// Construct a new rolling Alpha.
    ///
    /// # Errors
    /// Returns [`Error::InvalidPeriod`] if `period < 2`.
    pub fn new(period: usize, risk_free: f64) -> Result<Self> {
        if period < 2 {
            return Err(Error::InvalidPeriod {
                message: "alpha needs period >= 2",
            });
        }
        Ok(Self {
            period,
            risk_free,
            window: VecDeque::with_capacity(period),
            sum_a: 0.0,
            sum_b: 0.0,
            sum_bb: 0.0,
            sum_ab: 0.0,
        })
    }

    /// Configured window length.
    pub const fn period(&self) -> usize {
        self.period
    }

    /// Configured per-period risk-free rate.
    pub const fn risk_free(&self) -> f64 {
        self.risk_free
    }
}

impl Indicator for Alpha {
    type Input = (f64, f64);
    type Output = f64;

    fn update(&mut self, input: (f64, f64)) -> Option<f64> {
        let (a, b) = input;
        if !a.is_finite() || !b.is_finite() {
            return None;
        }
        if self.window.len() == self.period {
            let (oa, ob) = self.window.pop_front().expect("non-empty");
            self.sum_a -= oa;
            self.sum_b -= ob;
            self.sum_bb -= ob * ob;
            self.sum_ab -= oa * ob;
        }
        self.window.push_back((a, b));
        self.sum_a += a;
        self.sum_b += b;
        self.sum_bb += b * b;
        self.sum_ab += a * b;
        if self.window.len() < self.period {
            return None;
        }
        let n = self.period as f64;
        let mean_a = self.sum_a / n;
        let mean_b = self.sum_b / n;
        let var_b = (self.sum_bb / n) - mean_b * mean_b;
        if var_b <= 0.0 {
            // Undefined beta: report unadjusted excess.
            return Some(mean_a - self.risk_free);
        }
        let cov_ab = (self.sum_ab / n) - mean_a * mean_b;
        let beta = cov_ab / var_b;
        Some(mean_a - (self.risk_free + beta * (mean_b - self.risk_free)))
    }

    fn reset(&mut self) {
        self.window.clear();
        self.sum_a = 0.0;
        self.sum_b = 0.0;
        self.sum_bb = 0.0;
        self.sum_ab = 0.0;
    }

    fn warmup_period(&self) -> usize {
        self.period
    }

    fn is_ready(&self) -> bool {
        self.window.len() == self.period
    }

    fn name(&self) -> &'static str {
        "Alpha"
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use crate::traits::BatchExt;
    use approx::assert_relative_eq;

    #[test]
    fn rejects_period_less_than_two() {
        assert!(matches!(
            Alpha::new(1, 0.0),
            Err(Error::InvalidPeriod { .. })
        ));
    }

    #[test]
    fn accessors_and_metadata() {
        let a = Alpha::new(20, 0.001).unwrap();
        assert_eq!(a.period(), 20);
        assert_relative_eq!(a.risk_free(), 0.001, epsilon = 1e-12);
        assert_eq!(a.name(), "Alpha");
        assert_eq!(a.warmup_period(), 20);
    }

    #[test]
    fn capm_perfect_fit_yields_zero_alpha() {
        // asset = 2 * bench - constant beta of 2, no alpha; with rf = 0 the
        // CAPM-implied return matches the asset's mean perfectly.
        let mut a = Alpha::new(20, 0.0).unwrap();
        let inputs: Vec<(f64, f64)> = (1..=20)
            .map(|i| (2.0 * f64::from(i) * 0.01, f64::from(i) * 0.01))
            .collect();
        let out = a.batch(&inputs);
        assert_relative_eq!(out[19].unwrap(), 0.0, epsilon = 1e-12);
    }

    #[test]
    fn constant_alpha_offset_recovered() {
        // asset = bench + 0.005 (additive alpha of 0.5%), beta == 1.
        // Expected alpha = 0.005.
        let mut a = Alpha::new(20, 0.0).unwrap();
        let inputs: Vec<(f64, f64)> = (1..=20)
            .map(|i| (f64::from(i) * 0.01 + 0.005, f64::from(i) * 0.01))
            .collect();
        let out = a.batch(&inputs);
        assert_relative_eq!(out[19].unwrap(), 0.005, epsilon = 1e-9);
    }

    #[test]
    fn flat_benchmark_falls_back_to_excess_return() {
        // Benchmark all 0 -> beta undefined -> alpha = mean_a - rf.
        let mut a = Alpha::new(4, 0.001).unwrap();
        let out = a.batch(&[(0.01, 0.0), (0.02, 0.0), (-0.01, 0.0), (0.04, 0.0)]);
        let mean = (0.01 + 0.02 - 0.01 + 0.04) / 4.0;
        assert_relative_eq!(out[3].unwrap(), mean - 0.001, epsilon = 1e-12);
    }

    #[test]
    fn ignores_non_finite_input() {
        let mut a = Alpha::new(3, 0.0).unwrap();
        assert_eq!(a.update((f64::NAN, 0.0)), None);
        assert_eq!(a.update((0.0, f64::INFINITY)), None);
    }

    #[test]
    fn reset_clears_state() {
        let mut a = Alpha::new(3, 0.0).unwrap();
        a.batch(&[(0.01, 0.005), (0.02, 0.01), (-0.01, -0.005)]);
        assert!(a.is_ready());
        a.reset();
        assert!(!a.is_ready());
        assert_eq!(a.update((0.01, 0.005)), None);
    }

    #[test]
    fn batch_equals_streaming() {
        let inputs: Vec<(f64, f64)> = (0..50)
            .map(|i| {
                let b = (f64::from(i) * 0.2).sin() * 0.01;
                (1.5 * b + 0.002, b)
            })
            .collect();
        let batch = Alpha::new(10, 0.0).unwrap().batch(&inputs);
        let mut s = Alpha::new(10, 0.0).unwrap();
        let streamed: Vec<_> = inputs.iter().map(|x| s.update(*x)).collect();
        assert_eq!(batch, streamed);
    }
}