use crate::error::{Error, Result};
use crate::indicators::linreg::LinearRegression;
use crate::indicators::rvi::Rvi;
use crate::ohlcv::Candle;
use crate::traits::Indicator;
#[derive(Debug, Clone)]
pub struct Inertia {
rvi_period: usize,
linreg_period: usize,
rvi: Rvi,
linreg: LinearRegression,
}
impl Inertia {
pub fn new(rvi_period: usize, linreg_period: usize) -> Result<Self> {
if rvi_period == 0 || linreg_period == 0 {
return Err(Error::PeriodZero);
}
Ok(Self {
rvi_period,
linreg_period,
rvi: Rvi::new(rvi_period)?,
linreg: LinearRegression::new(linreg_period)?,
})
}
pub fn classic() -> Self {
Self::new(14, 20).expect("classic Inertia parameters are valid")
}
pub const fn periods(&self) -> (usize, usize) {
(self.rvi_period, self.linreg_period)
}
}
impl Indicator for Inertia {
type Input = Candle;
type Output = f64;
fn update(&mut self, candle: Candle) -> Option<f64> {
let rvi = self.rvi.update(candle)?;
self.linreg.update(rvi)
}
fn reset(&mut self) {
self.rvi.reset();
self.linreg.reset();
}
fn warmup_period(&self) -> usize {
self.rvi_period + self.linreg_period - 1
}
fn is_ready(&self) -> bool {
self.linreg.is_ready()
}
fn name(&self) -> &'static str {
"Inertia"
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::traits::BatchExt;
use approx::assert_relative_eq;
fn candle(open: f64, high: f64, low: f64, close: f64, ts: i64) -> Candle {
Candle::new(open, high, low, close, 1.0, ts).unwrap()
}
#[test]
fn rejects_zero_period() {
assert!(matches!(Inertia::new(0, 20), Err(Error::PeriodZero)));
assert!(matches!(Inertia::new(14, 0), Err(Error::PeriodZero)));
}
#[test]
fn accessors_and_metadata() {
let inertia = Inertia::classic();
assert_eq!(inertia.periods(), (14, 20));
assert_eq!(inertia.warmup_period(), 33);
assert_eq!(inertia.name(), "Inertia");
}
#[test]
fn classic_factory() {
assert_eq!(Inertia::classic().periods(), (14, 20));
}
#[test]
fn warmup_emits_first_value_at_warmup_period() {
let mut inertia = Inertia::new(3, 4).unwrap();
assert_eq!(inertia.warmup_period(), 6);
for i in 0..5 {
assert_eq!(inertia.update(candle(10.0, 11.0, 9.0, 10.5, i)), None);
}
assert!(inertia.update(candle(10.0, 11.0, 9.0, 10.5, 5)).is_some());
}
#[test]
fn constant_rvi_yields_constant_inertia() {
let mut inertia = Inertia::new(3, 4).unwrap();
let mut last = None;
for i in 0..40 {
last = inertia.update(candle(10.0, 11.0, 9.0, 10.5, i));
}
let v = last.unwrap();
assert_relative_eq!(v, 0.25, epsilon = 1e-12);
}
#[test]
fn batch_equals_streaming() {
let candles: Vec<Candle> = (0..80_i64)
.map(|i| {
let o = 100.0 + (i as f64 * 0.3).sin() * 5.0;
let c = o + (i as f64 * 0.1).cos();
candle(o, o.max(c) + 0.5, o.min(c) - 0.5, c, i)
})
.collect();
let batch = Inertia::classic().batch(&candles);
let mut b = Inertia::classic();
let streamed: Vec<_> = candles.iter().map(|c| b.update(*c)).collect();
assert_eq!(batch, streamed);
}
#[test]
fn reset_clears_state() {
let mut inertia = Inertia::classic();
for i in 0..50 {
inertia.update(candle(10.0, 11.0, 9.0, 10.5, i));
}
assert!(inertia.is_ready());
inertia.reset();
assert!(!inertia.is_ready());
assert_eq!(inertia.update(candle(10.0, 11.0, 9.0, 10.5, 0)), None);
}
}