use std::collections::VecDeque;
use crate::derivatives::DerivativesTick;
use crate::error::{Error, Result};
use crate::traits::Indicator;
#[derive(Debug, Clone)]
pub struct FundingRateMean {
window: usize,
history: VecDeque<f64>,
sum: f64,
}
impl FundingRateMean {
pub fn new(window: usize) -> Result<Self> {
if window == 0 {
return Err(Error::PeriodZero);
}
Ok(Self {
window,
history: VecDeque::with_capacity(window),
sum: 0.0,
})
}
#[must_use]
pub fn window(&self) -> usize {
self.window
}
}
impl Indicator for FundingRateMean {
type Input = DerivativesTick;
type Output = f64;
fn update(&mut self, tick: DerivativesTick) -> Option<f64> {
self.history.push_back(tick.funding_rate);
self.sum += tick.funding_rate;
if self.history.len() > self.window {
let old = self.history.pop_front().expect("window >= 1, len > window");
self.sum -= old;
}
if self.history.len() < self.window {
return None;
}
Some(self.sum / self.window as f64)
}
fn reset(&mut self) {
self.history.clear();
self.sum = 0.0;
}
fn warmup_period(&self) -> usize {
self.window
}
fn is_ready(&self) -> bool {
self.history.len() >= self.window
}
fn name(&self) -> &'static str {
"FundingRateMean"
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::traits::BatchExt;
fn tick(rate: f64) -> DerivativesTick {
DerivativesTick::new_unchecked(
rate, 100.0, 100.0, 100.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0,
)
}
#[test]
fn rejects_zero_window() {
assert!(matches!(FundingRateMean::new(0), Err(Error::PeriodZero)));
}
#[test]
fn accessors_and_metadata() {
let frm = FundingRateMean::new(5).unwrap();
assert_eq!(frm.name(), "FundingRateMean");
assert_eq!(frm.warmup_period(), 5);
assert_eq!(frm.window(), 5);
assert!(!frm.is_ready());
}
#[test]
fn warms_up_then_emits_mean() {
let mut frm = FundingRateMean::new(2).unwrap();
assert_eq!(frm.update(tick(0.001)), None);
assert!(!frm.is_ready());
assert_eq!(frm.update(tick(0.003)), Some(0.002));
assert!(frm.is_ready());
}
#[test]
fn rolls_off_old_values() {
let mut frm = FundingRateMean::new(2).unwrap();
frm.update(tick(0.001));
frm.update(tick(0.003)); let out = frm.update(tick(0.005)).unwrap(); assert!((out - 0.004).abs() < 1e-12);
}
#[test]
fn handles_negative_rates() {
let mut frm = FundingRateMean::new(2).unwrap();
frm.update(tick(-0.002));
let out = frm.update(tick(0.004)).unwrap();
assert!((out - 0.001).abs() < 1e-12);
}
#[test]
fn batch_equals_streaming() {
let ticks: Vec<DerivativesTick> = (0..30)
.map(|i| tick(0.0001 * f64::from(i % 7) - 0.0003))
.collect();
let mut a = FundingRateMean::new(5).unwrap();
let mut b = FundingRateMean::new(5).unwrap();
assert_eq!(
a.batch(&ticks),
ticks.iter().map(|x| b.update(*x)).collect::<Vec<_>>()
);
}
#[test]
fn reset_clears_state() {
let mut frm = FundingRateMean::new(2).unwrap();
frm.update(tick(0.001));
frm.update(tick(0.003));
assert!(frm.is_ready());
frm.reset();
assert!(!frm.is_ready());
assert_eq!(frm.update(tick(0.002)), None);
}
}