1use std::collections::VecDeque;
4
5use crate::error::{Error, Result};
6use crate::ohlcv::Candle;
7use crate::traits::Indicator;
8
9#[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 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 pub const fn period(&self) -> usize {
61 self.period
62 }
63
64 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 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 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 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 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}