use std::collections::VecDeque;
use crate::error::{Error, Result};
use crate::microstructure::Trade;
use crate::traits::Indicator;
#[derive(Debug, Clone)]
pub struct TradeSignAutocorrelation {
period: usize,
signs: VecDeque<f64>,
last: Option<f64>,
}
impl TradeSignAutocorrelation {
pub fn new(period: usize) -> Result<Self> {
if period < 2 {
return Err(Error::InvalidPeriod {
message: "trade-sign autocorrelation needs period >= 2",
});
}
Ok(Self {
period,
signs: VecDeque::with_capacity(period),
last: None,
})
}
pub const fn period(&self) -> usize {
self.period
}
pub const fn value(&self) -> Option<f64> {
self.last
}
}
impl Indicator for TradeSignAutocorrelation {
type Input = Trade;
type Output = f64;
fn update(&mut self, trade: Trade) -> Option<f64> {
if self.signs.len() == self.period {
self.signs.pop_front();
}
self.signs.push_back(trade.side.sign());
if self.signs.len() < self.period {
return None;
}
let mut product_sum = 0.0;
let mut prev: Option<f64> = None;
for &s in &self.signs {
if let Some(p) = prev {
product_sum += s * p;
}
prev = Some(s);
}
let rho = product_sum / (self.period as f64 - 1.0);
self.last = Some(rho);
Some(rho)
}
fn reset(&mut self) {
self.signs.clear();
self.last = None;
}
fn warmup_period(&self) -> usize {
self.period
}
fn is_ready(&self) -> bool {
self.last.is_some()
}
fn name(&self) -> &'static str {
"TradeSignAutocorrelation"
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::microstructure::Side;
use crate::traits::BatchExt;
use approx::assert_relative_eq;
fn buy() -> Trade {
Trade::new_unchecked(100.0, 1.0, Side::Buy, 0)
}
fn sell() -> Trade {
Trade::new_unchecked(100.0, 1.0, Side::Sell, 0)
}
#[test]
fn rejects_period_below_two() {
assert!(matches!(
TradeSignAutocorrelation::new(1),
Err(Error::InvalidPeriod { .. })
));
assert!(TradeSignAutocorrelation::new(2).is_ok());
}
#[test]
fn accessors_and_metadata() {
let t = TradeSignAutocorrelation::new(20).unwrap();
assert_eq!(t.period(), 20);
assert_eq!(t.warmup_period(), 20);
assert_eq!(t.name(), "TradeSignAutocorrelation");
assert!(!t.is_ready());
assert_eq!(t.value(), None);
}
#[test]
fn first_emission_at_warmup_period() {
let mut t = TradeSignAutocorrelation::new(4).unwrap();
let out = t.batch(&[buy(), buy(), buy(), buy(), buy()]);
for v in out.iter().take(3) {
assert!(v.is_none());
}
assert!(out[3].is_some());
}
#[test]
fn persistent_flow_is_one() {
let mut t = TradeSignAutocorrelation::new(10).unwrap();
let trades: Vec<Trade> = (0..20).map(|_| buy()).collect();
let last = t.batch(&trades).into_iter().flatten().last().unwrap();
assert_relative_eq!(last, 1.0, epsilon = 1e-12);
}
#[test]
fn alternating_flow_is_minus_one() {
let mut t = TradeSignAutocorrelation::new(10).unwrap();
let trades: Vec<Trade> = (0..20)
.map(|i| if i % 2 == 0 { buy() } else { sell() })
.collect();
let last = t.batch(&trades).into_iter().flatten().last().unwrap();
assert_relative_eq!(last, -1.0, epsilon = 1e-12);
}
#[test]
fn output_in_range() {
let mut t = TradeSignAutocorrelation::new(16).unwrap();
let trades: Vec<Trade> = (0..200)
.map(|i| if (i * 7 % 13) < 6 { buy() } else { sell() })
.collect();
for v in t.batch(&trades).into_iter().flatten() {
assert!((-1.0..=1.0).contains(&v));
}
}
#[test]
fn reset_clears_state() {
let mut t = TradeSignAutocorrelation::new(4).unwrap();
t.batch(&[buy(), buy(), buy(), buy()]);
assert!(t.is_ready());
t.reset();
assert!(!t.is_ready());
assert_eq!(t.value(), None);
assert_eq!(t.update(buy()), None);
}
#[test]
fn batch_equals_streaming() {
let trades: Vec<Trade> = (0..120)
.map(|i| if i % 3 == 0 { sell() } else { buy() })
.collect();
let batch = TradeSignAutocorrelation::new(16).unwrap().batch(&trades);
let mut b = TradeSignAutocorrelation::new(16).unwrap();
let streamed: Vec<_> = trades.iter().map(|x| b.update(*x)).collect();
assert_eq!(batch, streamed);
}
}