wickra_core/indicators/
detrended_std_dev.rs1use std::collections::VecDeque;
4
5use crate::error::{Error, Result};
6use crate::traits::Indicator;
7
8#[derive(Debug, Clone)]
51pub struct DetrendedStdDev {
52 period: usize,
53 window: VecDeque<f64>,
54 sum_x: f64,
55 denom: f64,
57 sum_y: f64,
58 sum_xy: f64,
59 sum_y_sq: f64,
60}
61
62impl DetrendedStdDev {
63 pub fn new(period: usize) -> Result<Self> {
69 if period < 2 {
70 return Err(Error::InvalidPeriod {
71 message: "detrended stddev needs period >= 2",
72 });
73 }
74 let n = period as f64;
75 let sum_x = n * (n - 1.0) / 2.0;
76 let sum_xx = (n - 1.0) * n * (2.0 * n - 1.0) / 6.0;
77 Ok(Self {
78 period,
79 window: VecDeque::with_capacity(period),
80 sum_x,
81 denom: n * sum_xx - sum_x * sum_x,
82 sum_y: 0.0,
83 sum_xy: 0.0,
84 sum_y_sq: 0.0,
85 })
86 }
87
88 pub const fn period(&self) -> usize {
90 self.period
91 }
92}
93
94impl Indicator for DetrendedStdDev {
95 type Input = f64;
96 type Output = f64;
97
98 fn update(&mut self, value: f64) -> Option<f64> {
99 if !value.is_finite() {
100 return None;
101 }
102 if self.window.len() == self.period {
103 let y0 = self.window.pop_front().expect("non-empty");
104 self.sum_xy = self.sum_xy - self.sum_y + y0;
105 self.sum_y -= y0;
106 self.sum_y_sq -= y0 * y0;
107 }
108 let k = self.window.len() as f64;
109 self.window.push_back(value);
110 self.sum_y += value;
111 self.sum_xy += k * value;
112 self.sum_y_sq += value * value;
113
114 if self.window.len() < self.period {
115 return None;
116 }
117 let n = self.period as f64;
118 let slope = (n * self.sum_xy - self.sum_x * self.sum_y) / self.denom;
119 let mean_y = self.sum_y / n;
120 let ss_total = self.sum_y_sq - n * mean_y * mean_y;
121 let s_xx = self.denom / n;
122 let rss = (ss_total - slope * slope * s_xx).max(0.0);
123 Some((rss / n).sqrt())
124 }
125
126 fn reset(&mut self) {
127 self.window.clear();
128 self.sum_y = 0.0;
129 self.sum_xy = 0.0;
130 self.sum_y_sq = 0.0;
131 }
132
133 fn warmup_period(&self) -> usize {
134 self.period
135 }
136
137 fn is_ready(&self) -> bool {
138 self.window.len() == self.period
139 }
140
141 fn name(&self) -> &'static str {
142 "DetrendedStdDev"
143 }
144}
145
146#[cfg(test)]
147mod tests {
148 use super::*;
149 use crate::traits::BatchExt;
150 use approx::assert_relative_eq;
151
152 #[test]
153 fn rejects_period_below_two() {
154 assert!(DetrendedStdDev::new(0).is_err());
155 assert!(DetrendedStdDev::new(1).is_err());
156 assert!(DetrendedStdDev::new(2).is_ok());
157 }
158
159 #[test]
160 fn accessors_and_metadata() {
161 let d = DetrendedStdDev::new(14).unwrap();
162 assert_eq!(d.period(), 14);
163 assert_eq!(d.warmup_period(), 14);
164 assert_eq!(d.name(), "DetrendedStdDev");
165 }
166
167 #[test]
168 fn perfect_line_has_zero_residual() {
169 let prices: Vec<f64> = (0..30).map(|i| 2.0 * f64::from(i) + 5.0).collect();
171 let mut d = DetrendedStdDev::new(10).unwrap();
172 for v in d.batch(&prices).into_iter().flatten() {
173 assert_relative_eq!(v, 0.0, epsilon = 1e-9);
174 }
175 }
176
177 #[test]
178 fn constant_series_yields_zero() {
179 let mut d = DetrendedStdDev::new(5).unwrap();
180 for v in d.batch(&[42.0; 20]).into_iter().flatten() {
181 assert_relative_eq!(v, 0.0, epsilon = 1e-9);
182 }
183 }
184
185 #[test]
186 fn never_exceeds_stddev() {
187 let prices: Vec<f64> = (0..60)
191 .map(|i| 50.0 + f64::from(i) * 0.5 + (f64::from(i) * 0.7).sin() * 4.0)
192 .collect();
193 let mut d = DetrendedStdDev::new(14).unwrap();
194 let mut sd = crate::StdDev::new(14).unwrap();
195 for &p in &prices {
196 let (dv, sv) = (d.update(p), sd.update(p));
197 assert_eq!(dv.is_some(), sv.is_some());
198 if let (Some(dv), Some(sv)) = (dv, sv) {
199 assert!(dv <= sv + 1e-9, "detrended {dv} should be <= stddev {sv}");
200 }
201 }
202 }
203
204 #[test]
205 fn reset_clears_state() {
206 let mut d = DetrendedStdDev::new(5).unwrap();
207 d.batch(&[1.0, 2.0, 3.0, 4.0, 5.0]);
208 assert!(d.is_ready());
209 d.reset();
210 assert!(!d.is_ready());
211 assert_eq!(d.update(1.0), None);
212 }
213
214 #[test]
215 fn batch_equals_streaming() {
216 let prices: Vec<f64> = (0..60)
217 .map(|i| 100.0 + (f64::from(i) * 0.4).sin() * 10.0)
218 .collect();
219 let batch = DetrendedStdDev::new(14).unwrap().batch(&prices);
220 let mut b = DetrendedStdDev::new(14).unwrap();
221 let streamed: Vec<_> = prices.iter().map(|p| b.update(*p)).collect();
222 assert_eq!(batch, streamed);
223 }
224}