Skip to main content

wickra_core/indicators/
kelly_criterion.rs

1//! Rolling Kelly Criterion.
2
3use std::collections::VecDeque;
4
5use crate::error::{Error, Result};
6use crate::traits::Indicator;
7
8/// Rolling Kelly Criterion fraction.
9///
10/// Input is treated as a per-period (or per-trade) return. Over the trailing
11/// window the indicator estimates the optimal capital fraction to allocate
12/// using the **even-money** Kelly formula generalised by the payoff ratio:
13///
14/// ```text
15/// win_rate     = P(r > 0)                          over window
16/// avg_win      = mean(r for r > 0)
17/// avg_loss     = mean(−r for r < 0)
18/// payoff_ratio = avg_win / avg_loss
19/// Kelly        = win_rate − (1 − win_rate) / payoff_ratio
20/// ```
21///
22/// The output is the recommended **fraction** of capital to bet (typically
23/// `(0, 1)`; can go negative if the estimated edge is negative, in which
24/// case the position should be reversed or sized to zero). Most
25/// practitioners use a "half-Kelly" or "quarter-Kelly" multiplier in
26/// practice to reduce variance — Wickra reports raw Kelly and leaves the
27/// scaling to the caller.
28///
29/// Edge cases:
30///   * No winners and no losers ⇒ `0.0` (no information).
31///   * No losers (`payoff_ratio = ∞`) ⇒ Kelly collapses to the win rate.
32///   * No winners but losers present ⇒ Kelly = `−(1 − 0) / payoff = …`,
33///     which is negative — bet nothing (or short).
34///
35/// Each `update` is O(period).
36#[derive(Debug, Clone)]
37pub struct KellyCriterion {
38    period: usize,
39    window: VecDeque<f64>,
40}
41
42impl KellyCriterion {
43    /// Construct a new rolling Kelly Criterion.
44    ///
45    /// # Errors
46    /// Returns [`Error::PeriodZero`] if `period == 0`.
47    pub fn new(period: usize) -> Result<Self> {
48        if period == 0 {
49            return Err(Error::PeriodZero);
50        }
51        Ok(Self {
52            period,
53            window: VecDeque::with_capacity(period),
54        })
55    }
56
57    /// Configured window length.
58    pub const fn period(&self) -> usize {
59        self.period
60    }
61}
62
63impl Indicator for KellyCriterion {
64    type Input = f64;
65    type Output = f64;
66
67    fn update(&mut self, input: f64) -> Option<f64> {
68        if !input.is_finite() {
69            return None;
70        }
71        if self.window.len() == self.period {
72            self.window.pop_front();
73        }
74        self.window.push_back(input);
75        if self.window.len() < self.period {
76            return None;
77        }
78        let mut sum_win = 0.0_f64;
79        let mut n_win = 0_u32;
80        let mut sum_loss = 0.0_f64;
81        let mut n_loss = 0_u32;
82        for &r in &self.window {
83            if r > 0.0 {
84                sum_win += r;
85                n_win += 1;
86            } else if r < 0.0 {
87                sum_loss += -r;
88                n_loss += 1;
89            }
90        }
91        let n = self.period as f64;
92        let win_rate = f64::from(n_win) / n;
93        if n_loss == 0 {
94            // No losses in window: payoff ratio is infinite; Kelly collapses
95            // to the win rate (limit of w - (1-w)/r as r -> ∞).
96            return Some(win_rate);
97        }
98        let avg_loss = sum_loss / f64::from(n_loss);
99        if n_win == 0 {
100            // All losses: avg_win = 0 -> payoff = 0 -> -(1)/0 -> -inf.
101            // Bet nothing (or reverse); clamp to -1 for sanity.
102            return Some(-1.0);
103        }
104        let avg_win = sum_win / f64::from(n_win);
105        let payoff = avg_win / avg_loss;
106        Some(win_rate - (1.0 - win_rate) / payoff)
107    }
108
109    fn reset(&mut self) {
110        self.window.clear();
111    }
112
113    fn warmup_period(&self) -> usize {
114        self.period
115    }
116
117    fn is_ready(&self) -> bool {
118        self.window.len() == self.period
119    }
120
121    fn name(&self) -> &'static str {
122        "KellyCriterion"
123    }
124}
125
126#[cfg(test)]
127mod tests {
128    use super::*;
129    use crate::traits::BatchExt;
130    use approx::assert_relative_eq;
131
132    #[test]
133    fn rejects_zero_period() {
134        assert!(matches!(KellyCriterion::new(0), Err(Error::PeriodZero)));
135    }
136
137    #[test]
138    fn accessors_and_metadata() {
139        let k = KellyCriterion::new(10).unwrap();
140        assert_eq!(k.period(), 10);
141        assert_eq!(k.name(), "KellyCriterion");
142        assert_eq!(k.warmup_period(), 10);
143    }
144
145    #[test]
146    fn reference_value() {
147        // returns = [0.02, 0.04, -0.01, -0.02] (n=4).
148        // n_win=2, n_loss=2; win_rate = 0.5.
149        // avg_win=0.03, avg_loss=0.015, payoff=2.
150        // Kelly = 0.5 - (0.5/2) = 0.25.
151        let mut k = KellyCriterion::new(4).unwrap();
152        let out = k.batch(&[0.02, 0.04, -0.01, -0.02]);
153        assert_relative_eq!(out[3].unwrap(), 0.25, epsilon = 1e-9);
154    }
155
156    #[test]
157    fn all_winners_returns_win_rate() {
158        let mut k = KellyCriterion::new(3).unwrap();
159        let out = k.batch(&[0.01, 0.02, 0.03]);
160        assert_relative_eq!(out[2].unwrap(), 1.0, epsilon = 1e-12);
161    }
162
163    #[test]
164    fn all_losers_returns_negative_one() {
165        let mut k = KellyCriterion::new(3).unwrap();
166        let out = k.batch(&[-0.01, -0.02, -0.03]);
167        assert_relative_eq!(out[2].unwrap(), -1.0, epsilon = 1e-12);
168    }
169
170    #[test]
171    fn flat_window_yields_zero() {
172        let mut k = KellyCriterion::new(3).unwrap();
173        let out = k.batch(&[0.0_f64; 3]);
174        assert_eq!(out[2], Some(0.0));
175    }
176
177    #[test]
178    fn ignores_non_finite_input() {
179        let mut k = KellyCriterion::new(3).unwrap();
180        assert_eq!(k.update(f64::NAN), None);
181        assert_eq!(k.update(f64::INFINITY), None);
182    }
183
184    #[test]
185    fn reset_clears_state() {
186        let mut k = KellyCriterion::new(3).unwrap();
187        k.batch(&[0.01, -0.02, 0.03]);
188        assert!(k.is_ready());
189        k.reset();
190        assert!(!k.is_ready());
191        assert_eq!(k.update(0.01), None);
192    }
193
194    #[test]
195    fn batch_equals_streaming() {
196        let returns: Vec<f64> = (0..40).map(|i| (f64::from(i) * 0.3).sin() * 0.01).collect();
197        let batch = KellyCriterion::new(10).unwrap().batch(&returns);
198        let mut s = KellyCriterion::new(10).unwrap();
199        let streamed: Vec<_> = returns.iter().map(|r| s.update(*r)).collect();
200        assert_eq!(batch, streamed);
201    }
202}