Skip to main content

wickra_core/indicators/
funding_basis.rs

1//! Funding Basis — the perpetual mark's relative premium to the spot index.
2
3use crate::derivatives::DerivativesTick;
4use crate::traits::Indicator;
5
6/// Funding Basis — the relative basis between the perpetual mark price and the
7/// spot index it tracks.
8///
9/// ```text
10/// basis = (markPrice − indexPrice) / indexPrice
11/// ```
12///
13/// The basis is the spread that the funding mechanism continuously pulls toward
14/// zero: a positive basis (perpetual above spot) goes hand in hand with positive
15/// funding (longs pay), a negative basis with negative funding. Reading the
16/// instantaneous basis alongside the [funding rate] separates a genuine premium
17/// from a stale-funding artefact and sizes the carry available to a cash-and-carry
18/// or basis-arbitrage trade. The output is a fraction (e.g. `0.001` = 10 bps);
19/// multiply by `10_000` for basis points.
20///
21/// `Input = DerivativesTick`, `Output = f64`. Stateless; ready after the first
22/// tick.
23///
24/// [funding rate]: crate::FundingRate
25///
26/// # Example
27///
28/// ```
29/// use wickra_core::{DerivativesTick, FundingBasis, Indicator};
30///
31/// let mut fb = FundingBasis::new();
32/// // mark 100.5 vs index 100.0 -> (100.5 - 100.0) / 100.0 = 0.005.
33/// let tick = DerivativesTick::new(
34///     0.0, 100.5, 100.0, 100.5, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0,
35/// )
36/// .unwrap();
37/// assert!((fb.update(tick).unwrap() - 0.005).abs() < 1e-12);
38/// ```
39#[derive(Debug, Clone, Default)]
40pub struct FundingBasis {
41    has_emitted: bool,
42}
43
44impl FundingBasis {
45    /// Construct a new funding-basis indicator.
46    #[must_use]
47    pub const fn new() -> Self {
48        Self { has_emitted: false }
49    }
50}
51
52impl Indicator for FundingBasis {
53    type Input = DerivativesTick;
54    type Output = f64;
55
56    fn update(&mut self, tick: DerivativesTick) -> Option<f64> {
57        self.has_emitted = true;
58        Some((tick.mark_price - tick.index_price) / tick.index_price)
59    }
60
61    fn reset(&mut self) {
62        self.has_emitted = false;
63    }
64
65    fn warmup_period(&self) -> usize {
66        1
67    }
68
69    fn is_ready(&self) -> bool {
70        self.has_emitted
71    }
72
73    fn name(&self) -> &'static str {
74        "FundingBasis"
75    }
76}
77
78#[cfg(test)]
79mod tests {
80    use super::*;
81    use crate::traits::BatchExt;
82
83    fn tick(mark: f64, index: f64) -> DerivativesTick {
84        DerivativesTick::new_unchecked(0.0, mark, index, mark, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0)
85    }
86
87    #[test]
88    fn accessors_and_metadata() {
89        let fb = FundingBasis::new();
90        assert_eq!(fb.name(), "FundingBasis");
91        assert_eq!(fb.warmup_period(), 1);
92        assert!(!fb.is_ready());
93    }
94
95    #[test]
96    fn premium_is_positive() {
97        let mut fb = FundingBasis::new();
98        let out = fb.update(tick(100.5, 100.0)).unwrap();
99        assert!((out - 0.005).abs() < 1e-12);
100        assert!(fb.is_ready());
101    }
102
103    #[test]
104    fn discount_is_negative() {
105        let mut fb = FundingBasis::new();
106        let out = fb.update(tick(99.5, 100.0)).unwrap();
107        assert!((out + 0.005).abs() < 1e-12);
108    }
109
110    #[test]
111    fn at_par_is_zero() {
112        let mut fb = FundingBasis::new();
113        assert_eq!(fb.update(tick(100.0, 100.0)), Some(0.0));
114    }
115
116    #[test]
117    fn batch_equals_streaming() {
118        let ticks: Vec<DerivativesTick> = (0..20)
119            .map(|i| tick(100.0 + f64::from(i % 5) * 0.1, 100.0))
120            .collect();
121        let mut a = FundingBasis::new();
122        let mut b = FundingBasis::new();
123        assert_eq!(
124            a.batch(&ticks),
125            ticks.iter().map(|x| b.update(*x)).collect::<Vec<_>>()
126        );
127    }
128
129    #[test]
130    fn reset_clears_state() {
131        let mut fb = FundingBasis::new();
132        fb.update(tick(100.5, 100.0));
133        assert!(fb.is_ready());
134        fb.reset();
135        assert!(!fb.is_ready());
136    }
137}