use std::{collections::BTreeMap, fmt::Debug};
use nautilus_model::{orders::Order, position::Position};
use crate::Returns;
const IMPL_ERR: &str = "is not implemented for";
#[allow(unused_variables)]
pub trait PortfolioStatistic: Debug {
type Item;
fn name(&self) -> String;
fn calculate_from_returns(&self, returns: &Returns) -> Option<Self::Item> {
panic!("`calculate_from_returns` {IMPL_ERR} `{}`", self.name());
}
fn calculate_from_realized_pnls(&self, realized_pnls: &[f64]) -> Option<Self::Item> {
panic!(
"`calculate_from_realized_pnls` {IMPL_ERR} `{}`",
self.name()
);
}
#[allow(dead_code)]
fn calculate_from_orders(&self, orders: Vec<Box<dyn Order>>) -> Option<Self::Item> {
panic!("`calculate_from_orders` {IMPL_ERR} `{}`", self.name());
}
fn calculate_from_positions(&self, positions: &[Position]) -> Option<Self::Item> {
panic!("`calculate_from_positions` {IMPL_ERR} `{}`", self.name());
}
fn check_valid_returns(&self, returns: &Returns) -> bool {
!returns.is_empty()
}
fn downsample_to_daily_bins(&self, returns: &Returns) -> Returns {
let nanos_per_day = 86_400_000_000_000; let mut daily_bins = BTreeMap::new();
for (×tamp, &value) in returns {
let day_start = timestamp - (timestamp.as_u64() % nanos_per_day);
let entry = daily_bins.entry(day_start).or_insert(0.0_f64);
*entry = (1.0_f64 + *entry).mul_add(1.0_f64 + value, -1.0_f64);
}
daily_bins
}
fn calculate_std(&self, returns: &Returns) -> f64 {
let n = returns.len() as f64;
if n < 2.0 {
return f64::NAN;
}
let mean = returns.values().sum::<f64>() / n;
let variance = returns.values().map(|x| (x - mean).powi(2)).sum::<f64>() / (n - 1.0);
variance.sqrt()
}
}
#[cfg(test)]
mod tests {
use nautilus_core::{UnixNanos, approx_eq};
use rstest::rstest;
use super::*;
#[derive(Debug)]
struct DummyStat;
impl PortfolioStatistic for DummyStat {
type Item = f64;
fn name(&self) -> String {
"DummyStat".to_string()
}
}
const NANOS_PER_DAY: u64 = 86_400_000_000_000;
const BASE_NS: u64 = 1_600_000_000_000_000_000;
#[rstest]
fn test_downsample_compounds_intraday_returns() {
let stat = DummyStat;
let mut returns: Returns = BTreeMap::new();
returns.insert(UnixNanos::from(BASE_NS), 0.05);
returns.insert(UnixNanos::from(BASE_NS + 3_600_000_000_000), -0.05);
let daily = stat.downsample_to_daily_bins(&returns);
assert_eq!(daily.len(), 1);
let value = *daily.values().next().unwrap();
assert!(approx_eq!(f64, value, -0.0025, epsilon = 1e-12));
}
#[rstest]
fn test_downsample_daily_inputs_unchanged() {
let stat = DummyStat;
let mut returns: Returns = BTreeMap::new();
returns.insert(UnixNanos::from(BASE_NS), 0.01);
returns.insert(UnixNanos::from(BASE_NS + NANOS_PER_DAY), -0.02);
returns.insert(UnixNanos::from(BASE_NS + 2 * NANOS_PER_DAY), 0.015);
let daily = stat.downsample_to_daily_bins(&returns);
let values: Vec<f64> = daily.values().copied().collect();
assert_eq!(values.len(), 3);
assert!(approx_eq!(f64, values[0], 0.01, epsilon = 1e-15));
assert!(approx_eq!(f64, values[1], -0.02, epsilon = 1e-15));
assert!(approx_eq!(f64, values[2], 0.015, epsilon = 1e-15));
}
#[rstest]
fn test_downsample_chains_three_intraday_returns() {
let stat = DummyStat;
let mut returns: Returns = BTreeMap::new();
returns.insert(UnixNanos::from(BASE_NS), 0.01);
returns.insert(UnixNanos::from(BASE_NS + 3_600_000_000_000), 0.02);
returns.insert(UnixNanos::from(BASE_NS + 7_200_000_000_000), -0.01);
let daily = stat.downsample_to_daily_bins(&returns);
assert_eq!(daily.len(), 1);
let value = *daily.values().next().unwrap();
let expected = 1.01_f64 * 1.02 * 0.99 - 1.0;
assert!(approx_eq!(f64, value, expected, epsilon = 1e-12));
}
}