Skip to main content

wickra_core/indicators/
rvi.rs

1//! Relative Vigor Index (RVI).
2
3use std::collections::VecDeque;
4
5use crate::error::{Error, Result};
6use crate::ohlcv::Candle;
7use crate::traits::Indicator;
8
9/// Relative Vigor Index — Donald Dorsey's ratio of intra-bar drive (close − open)
10/// to intra-bar range (high − low), averaged over a `period`-bar window.
11///
12/// The reading is `SMA(close − open, period) / SMA(high − low, period)`. A
13/// positive value means the average bar in the window closed above where it
14/// opened (bullish "vigor"); a negative value means the average closed below.
15/// The denominator's rolling-window SMA can fall to zero on a perfectly flat
16/// stretch, in which case the recurrence is undefined and the indicator holds
17/// its previous value.
18///
19/// # Example
20///
21/// ```
22/// use wickra_core::{Candle, Indicator, Rvi};
23///
24/// let mut rvi = Rvi::new(10).unwrap();
25/// let mut last = None;
26/// for i in 0..40 {
27///     let o = 100.0 + f64::from(i);
28///     let c = o + 0.5;
29///     let candle = Candle::new(o, c + 0.2, o - 0.2, c, 1.0, i64::from(i)).unwrap();
30///     last = rvi.update(candle);
31/// }
32/// assert!(last.is_some());
33/// ```
34#[derive(Debug, Clone)]
35pub struct Rvi {
36    period: usize,
37    window: VecDeque<(f64, f64)>,
38    sum_num: f64,
39    sum_den: f64,
40    current: Option<f64>,
41}
42
43impl Rvi {
44    /// # Errors
45    /// Returns [`Error::PeriodZero`] if `period == 0`.
46    pub fn new(period: usize) -> Result<Self> {
47        if period == 0 {
48            return Err(Error::PeriodZero);
49        }
50        Ok(Self {
51            period,
52            window: VecDeque::with_capacity(period),
53            sum_num: 0.0,
54            sum_den: 0.0,
55            current: None,
56        })
57    }
58
59    /// Configured period.
60    pub const fn period(&self) -> usize {
61        self.period
62    }
63
64    /// Current value if available.
65    pub const fn value(&self) -> Option<f64> {
66        self.current
67    }
68}
69
70impl Indicator for Rvi {
71    type Input = Candle;
72    type Output = f64;
73
74    fn update(&mut self, candle: Candle) -> Option<f64> {
75        let num = candle.close - candle.open;
76        let den = candle.high - candle.low;
77        if self.window.len() == self.period {
78            let (old_n, old_d) = self.window.pop_front().expect("window is non-empty");
79            self.sum_num -= old_n;
80            self.sum_den -= old_d;
81        }
82        self.window.push_back((num, den));
83        self.sum_num += num;
84        self.sum_den += den;
85        if self.window.len() < self.period {
86            return None;
87        }
88        if self.sum_den <= 0.0 {
89            // Window of perfectly flat (zero-range) bars: ratio undefined.
90            // Hold the previous value rather than emitting NaN / inf.
91            return self.current;
92        }
93        let value = self.sum_num / self.sum_den;
94        self.current = Some(value);
95        Some(value)
96    }
97
98    fn reset(&mut self) {
99        self.window.clear();
100        self.sum_num = 0.0;
101        self.sum_den = 0.0;
102        self.current = None;
103    }
104
105    fn warmup_period(&self) -> usize {
106        self.period
107    }
108
109    fn is_ready(&self) -> bool {
110        self.current.is_some()
111    }
112
113    fn name(&self) -> &'static str {
114        "RVI"
115    }
116}
117
118#[cfg(test)]
119mod tests {
120    use super::*;
121    use crate::traits::BatchExt;
122    use approx::assert_relative_eq;
123
124    fn candle(open: f64, high: f64, low: f64, close: f64, ts: i64) -> Candle {
125        Candle::new(open, high, low, close, 1.0, ts).unwrap()
126    }
127
128    #[test]
129    fn rejects_zero_period() {
130        assert!(matches!(Rvi::new(0), Err(Error::PeriodZero)));
131    }
132
133    #[test]
134    fn accessors_and_metadata() {
135        let mut r = Rvi::new(10).unwrap();
136        assert_eq!(r.period(), 10);
137        assert_eq!(r.warmup_period(), 10);
138        assert_eq!(r.name(), "RVI");
139        assert_eq!(r.value(), None);
140        for i in 0..10 {
141            r.update(candle(10.0, 11.0, 9.0, 10.5, i));
142        }
143        assert!(r.value().is_some());
144    }
145
146    #[test]
147    fn reference_value_period_2() {
148        // Two bars with (open, high, low, close) = (10, 11, 9, 10.5) and
149        // (10.5, 11.5, 10, 11). Per bar:
150        //   num1 = 0.5, num2 = 0.5; sum = 1.0
151        //   den1 = 2.0, den2 = 1.5; sum = 3.5
152        //   RVI = 1.0 / 3.5 ≈ 0.2857142857
153        let mut r = Rvi::new(2).unwrap();
154        assert_eq!(r.update(candle(10.0, 11.0, 9.0, 10.5, 0)), None);
155        let v = r.update(candle(10.5, 11.5, 10.0, 11.0, 1)).unwrap();
156        assert_relative_eq!(v, 1.0 / 3.5, epsilon = 1e-12);
157    }
158
159    #[test]
160    fn warmup_emits_first_value_at_period() {
161        let mut r = Rvi::new(3).unwrap();
162        for i in 0..2 {
163            assert_eq!(r.update(candle(10.0, 11.0, 9.0, 10.5, i)), None);
164        }
165        assert!(r.update(candle(10.5, 11.5, 10.0, 11.0, 2)).is_some());
166    }
167
168    #[test]
169    fn pure_uptrend_is_positive() {
170        // Every bar closes above its open and has a non-zero range: RVI > 0.
171        let mut r = Rvi::new(5).unwrap();
172        for i in 0..10 {
173            let o = 10.0 + f64::from(i);
174            let c = o + 0.5;
175            r.update(candle(o, c + 0.2, o - 0.2, c, i64::from(i)));
176        }
177        let v = r.value().unwrap();
178        assert!(v > 0.0, "uptrend RVI should be positive: {v}");
179    }
180
181    #[test]
182    fn zero_range_window_holds_value() {
183        // Window of perfectly flat bars (high == low): ratio undefined,
184        // indicator holds.
185        let mut r = Rvi::new(3).unwrap();
186        r.update(candle(10.0, 10.0, 10.0, 10.0, 0));
187        r.update(candle(10.0, 10.0, 10.0, 10.0, 1));
188        assert_eq!(r.update(candle(10.0, 10.0, 10.0, 10.0, 2)), None);
189    }
190
191    #[test]
192    fn batch_equals_streaming() {
193        let candles: Vec<Candle> = (0..40_i64)
194            .map(|i| {
195                let o = 100.0 + (i as f64 * 0.3).sin() * 5.0;
196                let c = o + (i as f64 * 0.1).cos();
197                candle(o, o.max(c) + 0.5, o.min(c) - 0.5, c, i)
198            })
199            .collect();
200        let batch = Rvi::new(10).unwrap().batch(&candles);
201        let mut b = Rvi::new(10).unwrap();
202        let streamed: Vec<_> = candles.iter().map(|c| b.update(*c)).collect();
203        assert_eq!(batch, streamed);
204    }
205
206    #[test]
207    fn reset_clears_state() {
208        let mut r = Rvi::new(5).unwrap();
209        for i in 0..10 {
210            r.update(candle(10.0, 11.0, 9.0, 10.5, i));
211        }
212        assert!(r.is_ready());
213        r.reset();
214        assert!(!r.is_ready());
215        assert_eq!(r.update(candle(10.0, 11.0, 9.0, 10.5, 0)), None);
216    }
217}