From e7ba686d23bd2c2245351a0979b851bc6c8348b0 Mon Sep 17 00:00:00 2001 From: amv-dev Date: Thu, 7 Mar 2024 23:47:11 +0500 Subject: [PATCH] fix: HeikinAshi --- src/methods/heikin_ashi.rs | 25 +++++++++++++++++++------ 1 file changed, 19 insertions(+), 6 deletions(-) diff --git a/src/methods/heikin_ashi.rs b/src/methods/heikin_ashi.rs index 2c2d4f6..7b3c6aa 100644 --- a/src/methods/heikin_ashi.rs +++ b/src/methods/heikin_ashi.rs @@ -1,4 +1,4 @@ -use crate::core::{Candle, Error, Method, OHLCV}; +use crate::core::{Candle, Error, Method, ValueType, OHLCV}; #[cfg(feature = "serde")] use serde::{Deserialize, Serialize}; @@ -7,7 +7,7 @@ use serde::{Deserialize, Serialize}; #[derive(Debug, Clone, Copy)] #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] pub struct HeikinAshi { - prev: Candle, + next_open: ValueType, } impl Method for HeikinAshi { @@ -16,16 +16,21 @@ impl Method for HeikinAshi { type Output = Candle; fn new((): Self::Params, value: &Self::Input) -> Result { + // It is not so obvious how we should correctly initialize first candle + // The reason why `ohlc4` is used is because it's the only way we can achieve stable results + // when constant input value is provided, which is one of requiremets for all the methods Ok(Self { - prev: Candle::from(value), + next_open: value.ohlc4(), }) } #[inline] fn next(&mut self, value: &Self::Input) -> Self::Output { - let open = (self.prev.open() + self.prev.close()) * 0.5; + let open = self.next_open; let close = value.ohlc4(); + self.next_open = (open + close) * 0.5; + Candle { open, high: value.high().max(open), @@ -39,7 +44,7 @@ impl Method for HeikinAshi { #[cfg(test)] mod tests { use super::{HeikinAshi, OHLCV}; - use crate::core::{Candle, Method}; + use crate::core::{Candle, Method, ValueType}; use crate::helpers::{assert_eq_float, RandomCandles}; #[test] @@ -50,7 +55,13 @@ mod tests { let first = candles.first(); let mut heikin_ashi = HeikinAshi::new((), &first).unwrap(); - let prev = candles.first(); + let mut prev = Candle { + open: first.ohlc4(), + high: ValueType::NAN, + low: ValueType::NAN, + close: first.ohlc4(), + volume: ValueType::NAN, + }; candles .take(100) @@ -66,6 +77,8 @@ mod tests { ..candle }; + prev = tested; + (tested, heikin_ashi.next(&candle)) }) .inspect(|(original, ha)| assert_eq_float(original.close(), ha.close()))