use std::collections::VecDeque;
use crate::error::{Error, Result};
use crate::ohlcv::Candle;
use crate::traits::Indicator;
#[derive(Debug, Clone)]
pub struct ParkinsonVolatility {
period: usize,
trading_periods: usize,
window: VecDeque<f64>,
sum_sq: f64,
last: Option<f64>,
}
const PARKINSON_FACTOR: f64 = 0.360_673_760_222_241_2;
impl ParkinsonVolatility {
pub fn new(period: usize, trading_periods: usize) -> Result<Self> {
if period == 0 || trading_periods == 0 {
return Err(Error::PeriodZero);
}
Ok(Self {
period,
trading_periods,
window: VecDeque::with_capacity(period),
sum_sq: 0.0,
last: None,
})
}
pub const fn periods(&self) -> (usize, usize) {
(self.period, self.trading_periods)
}
pub const fn value(&self) -> Option<f64> {
self.last
}
}
impl Indicator for ParkinsonVolatility {
type Input = Candle;
type Output = f64;
fn update(&mut self, candle: Candle) -> Option<f64> {
let log_hl = (candle.high / candle.low).ln();
let sample = log_hl * log_hl;
if self.window.len() == self.period {
let old = self.window.pop_front().expect("window is non-empty");
self.sum_sq -= old;
}
self.window.push_back(sample);
self.sum_sq += sample;
if self.window.len() < self.period {
return None;
}
let n = self.period as f64;
let variance = (PARKINSON_FACTOR * self.sum_sq / n).max(0.0);
let sigma = variance.sqrt();
let out = sigma * (self.trading_periods as f64).sqrt() * 100.0;
self.last = Some(out);
Some(out)
}
fn reset(&mut self) {
self.window.clear();
self.sum_sq = 0.0;
self.last = None;
}
fn warmup_period(&self) -> usize {
self.period
}
fn is_ready(&self) -> bool {
self.last.is_some()
}
fn name(&self) -> &'static str {
"ParkinsonVolatility"
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::traits::BatchExt;
use approx::assert_relative_eq;
fn candle(h: f64, l: f64, c: f64, ts: i64) -> Candle {
Candle::new(f64::midpoint(h, l), h, l, c, 1.0, ts).unwrap()
}
#[test]
fn rejects_zero_period() {
assert!(matches!(
ParkinsonVolatility::new(0, 252),
Err(Error::PeriodZero)
));
assert!(matches!(
ParkinsonVolatility::new(20, 0),
Err(Error::PeriodZero)
));
}
#[test]
fn accessors_and_metadata() {
let pv = ParkinsonVolatility::new(20, 252).unwrap();
assert_eq!(pv.periods(), (20, 252));
assert_eq!(pv.value(), None);
assert_eq!(pv.warmup_period(), 20);
assert_eq!(pv.name(), "ParkinsonVolatility");
assert!(!pv.is_ready());
}
#[test]
fn zero_range_yields_zero() {
let candles: Vec<Candle> = (0..30).map(|i| candle(10.0, 10.0, 10.0, i)).collect();
let mut pv = ParkinsonVolatility::new(14, 1).unwrap();
for v in pv.batch(&candles).into_iter().flatten() {
assert_relative_eq!(v, 0.0, epsilon = 1e-12);
}
}
#[test]
fn constant_range_yields_constant_sigma() {
let candles: Vec<Candle> = (0..30).map(|i| candle(11.0, 9.0, 10.0, i)).collect();
let mut pv = ParkinsonVolatility::new(10, 1).unwrap();
let out = pv.batch(&candles);
let k = (11.0_f64 / 9.0_f64).ln().powi(2);
let expected = (PARKINSON_FACTOR * k).sqrt() * 100.0;
for v in out.iter().skip(9).flatten() {
assert_relative_eq!(*v, expected, epsilon = 1e-9);
}
}
#[test]
fn output_is_non_negative() {
let mut pv = ParkinsonVolatility::new(14, 252).unwrap();
let candles: Vec<Candle> = (0..200)
.map(|i| {
let base = 100.0 + (f64::from(i) * 0.3).sin() * 12.0;
let half = 0.5 + (f64::from(i) * 0.13).cos().abs() * 1.5;
candle(base + half, base - half, base, i64::from(i))
})
.collect();
for v in pv.batch(&candles).into_iter().flatten() {
assert!(v >= 0.0, "Parkinson volatility must be non-negative: {v}");
}
}
#[test]
fn annualisation_scales_by_sqrt_trading_periods() {
let candles: Vec<Candle> = (0..40)
.map(|i| {
let base = 100.0 + (f64::from(i) * 0.3).sin() * 5.0;
let half = 1.0 + (f64::from(i) * 0.2).cos().abs();
candle(base + half, base - half, base, i64::from(i))
})
.collect();
let raw = ParkinsonVolatility::new(10, 1).unwrap().batch(&candles);
let annual = ParkinsonVolatility::new(10, 252).unwrap().batch(&candles);
let scale = (252.0_f64).sqrt();
for (r, a) in raw.iter().zip(annual.iter()) {
assert_eq!(r.is_some(), a.is_some(), "warmup mismatch");
if let (Some(r), Some(a)) = (r, a) {
assert_relative_eq!(*a, r * scale, epsilon = 1e-9);
}
}
}
#[test]
fn first_emission_at_warmup_period() {
let candles: Vec<Candle> = (0..20).map(|i| candle(11.0, 9.0, 10.0, i)).collect();
let mut pv = ParkinsonVolatility::new(5, 1).unwrap();
let out = pv.batch(&candles);
for v in out.iter().take(4) {
assert!(v.is_none());
}
assert!(out[4].is_some());
}
#[test]
fn batch_equals_streaming() {
let candles: Vec<Candle> = (0..80)
.map(|i| {
let base = 100.0 + (f64::from(i) * 0.25).sin() * 6.0;
let half = 1.0 + (f64::from(i) * 0.15).cos().abs();
candle(base + half, base - half, base, i64::from(i))
})
.collect();
let batch = ParkinsonVolatility::new(14, 252).unwrap().batch(&candles);
let mut streamer = ParkinsonVolatility::new(14, 252).unwrap();
let streamed: Vec<_> = candles.iter().map(|c| streamer.update(*c)).collect();
assert_eq!(batch, streamed);
}
#[test]
fn reset_clears_state() {
let candles: Vec<Candle> = (0..30).map(|i| candle(11.0, 9.0, 10.0, i)).collect();
let mut pv = ParkinsonVolatility::new(14, 252).unwrap();
pv.batch(&candles);
assert!(pv.is_ready());
pv.reset();
assert!(!pv.is_ready());
assert_eq!(pv.value(), None);
assert_eq!(pv.update(candles[0]), None);
}
}