mirror of
https://github.com/freqtrade/freqtrade.git
synced 2026-03-01 15:52:43 +00:00
refactor: optimize performance and remove redundant logic in protections
This commit is contained in:
@@ -2423,7 +2423,8 @@ class FreqtradeBot(LoggingMixin):
|
||||
self.strategy.lock_pair(pair, datetime.now(UTC), reason="Auto lock", side=side)
|
||||
starting_balance = self.wallets.get_starting_balance()
|
||||
prot_trig = self.protections.stop_per_pair(
|
||||
pair, side=side, starting_balance=starting_balance)
|
||||
pair, side=side, starting_balance=starting_balance
|
||||
)
|
||||
if prot_trig:
|
||||
msg: RPCProtectionMsg = {
|
||||
"type": RPCMessageType.PROTECTION_TRIGGER,
|
||||
|
||||
@@ -66,8 +66,11 @@ class ProtectionManager:
|
||||
return result
|
||||
|
||||
def stop_per_pair(
|
||||
self, pair, now: datetime | None = None, side: LongShort = "long",
|
||||
starting_balance: float = 0.0
|
||||
self,
|
||||
pair,
|
||||
now: datetime | None = None,
|
||||
side: LongShort = "long",
|
||||
starting_balance: float = 0.0,
|
||||
) -> PairLock | None:
|
||||
if not now:
|
||||
now = datetime.now(UTC)
|
||||
|
||||
@@ -53,7 +53,7 @@ class CooldownPeriod(IProtection):
|
||||
return None
|
||||
|
||||
def global_stop(
|
||||
self, date_now: datetime, side: LongShort, starting_balance: float = 0.0
|
||||
self, date_now: datetime, side: LongShort, starting_balance: float
|
||||
) -> ProtectionReturn | None:
|
||||
"""
|
||||
Stops trading (position entering) for all pairs
|
||||
@@ -65,7 +65,7 @@ class CooldownPeriod(IProtection):
|
||||
return None
|
||||
|
||||
def stop_per_pair(
|
||||
self, pair: str, date_now: datetime, side: LongShort, starting_balance: float = 0.0
|
||||
self, pair: str, date_now: datetime, side: LongShort, starting_balance: float
|
||||
) -> ProtectionReturn | None:
|
||||
"""
|
||||
Stops trading (position entering) for this pair
|
||||
|
||||
@@ -103,7 +103,7 @@ class IProtection(LoggingMixin, ABC):
|
||||
|
||||
@abstractmethod
|
||||
def global_stop(
|
||||
self, date_now: datetime, side: LongShort, starting_balance: float = 0.0
|
||||
self, date_now: datetime, side: LongShort, starting_balance: float
|
||||
) -> ProtectionReturn | None:
|
||||
"""
|
||||
Stops trading (position entering) for all pairs
|
||||
@@ -112,7 +112,7 @@ class IProtection(LoggingMixin, ABC):
|
||||
|
||||
@abstractmethod
|
||||
def stop_per_pair(
|
||||
self, pair: str, date_now: datetime, side: LongShort, starting_balance: float = 0.0
|
||||
self, pair: str, date_now: datetime, side: LongShort, starting_balance: float
|
||||
) -> ProtectionReturn | None:
|
||||
"""
|
||||
Stops trading (position entering) for this pair
|
||||
|
||||
@@ -82,7 +82,7 @@ class LowProfitPairs(IProtection):
|
||||
return None
|
||||
|
||||
def global_stop(
|
||||
self, date_now: datetime, side: LongShort, starting_balance: float = 0.0
|
||||
self, date_now: datetime, side: LongShort, starting_balance: float
|
||||
) -> ProtectionReturn | None:
|
||||
"""
|
||||
Stops trading (position entering) for all pairs
|
||||
@@ -93,7 +93,7 @@ class LowProfitPairs(IProtection):
|
||||
return None
|
||||
|
||||
def stop_per_pair(
|
||||
self, pair: str, date_now: datetime, side: LongShort, starting_balance: float = 0.0
|
||||
self, pair: str, date_now: datetime, side: LongShort, starting_balance: float
|
||||
) -> ProtectionReturn | None:
|
||||
"""
|
||||
Stops trading (position entering) for this pair
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
import logging
|
||||
from datetime import UTC, datetime, timedelta
|
||||
from datetime import datetime, timedelta
|
||||
from typing import Any
|
||||
|
||||
import pandas as pd
|
||||
@@ -48,41 +48,50 @@ class MaxDrawdown(IProtection):
|
||||
"""
|
||||
look_back_until = date_now - timedelta(minutes=self._lookback_period)
|
||||
|
||||
# Get all closed trades to calculate balance at the start of the window
|
||||
all_closed_trades = Trade.get_trades_proxy(is_open=False)
|
||||
|
||||
trades_in_window = []
|
||||
profit_before_window = 0.0
|
||||
for trade in all_closed_trades:
|
||||
if trade.close_date:
|
||||
# Ensure close_date is aware for comparison
|
||||
close_date = (trade.close_date.replace(tzinfo=UTC)
|
||||
if trade.close_date.tzinfo is None else trade.close_date)
|
||||
if close_date > look_back_until:
|
||||
trades_in_window.append(trade)
|
||||
else:
|
||||
profit_before_window += (trade.close_profit_abs or 0.0)
|
||||
trades_in_window = Trade.get_trades_proxy(is_open=False, close_date=look_back_until)
|
||||
|
||||
if len(trades_in_window) < self._trade_limit:
|
||||
# Not enough trades in the relevant period
|
||||
return None
|
||||
|
||||
# Calculate actual balance at the start of the lookback window
|
||||
actual_starting_balance = starting_balance + profit_before_window
|
||||
# Get all trades to calculate cumulative profit before the window
|
||||
all_closed_trades = Trade.get_trades_proxy(is_open=False)
|
||||
profit_before_window = sum(
|
||||
trade.close_profit_abs or 0.0
|
||||
for trade in all_closed_trades
|
||||
if trade.close_date_utc <= look_back_until
|
||||
)
|
||||
|
||||
trades_df = pd.DataFrame([trade.to_json() for trade in trades_in_window])
|
||||
# Get calculation mode
|
||||
method = self._protection_config.get("method", "ratios")
|
||||
|
||||
# Drawdown is always positive
|
||||
try:
|
||||
# Use absolute profit calculation with the actual balance at window start.
|
||||
drawdown_obj = calculate_max_drawdown(
|
||||
trades_df,
|
||||
value_col="profit_abs",
|
||||
starting_balance=actual_starting_balance,
|
||||
relative=True
|
||||
)
|
||||
# Use relative drawdown to compare against max_allowed_drawdown percentage
|
||||
drawdown = drawdown_obj.relative_account_drawdown
|
||||
if method == "equity":
|
||||
# Standard equity-based drawdown
|
||||
trades_df = pd.DataFrame(
|
||||
[
|
||||
{"close_date": t.close_date_utc, "profit_abs": t.close_profit_abs}
|
||||
for t in trades_in_window
|
||||
]
|
||||
)
|
||||
actual_starting_balance = starting_balance + profit_before_window
|
||||
drawdown_obj = calculate_max_drawdown(
|
||||
trades_df,
|
||||
value_col="profit_abs",
|
||||
starting_balance=actual_starting_balance,
|
||||
relative=True,
|
||||
)
|
||||
drawdown = drawdown_obj.relative_account_drawdown
|
||||
else:
|
||||
# Legacy ratios-based calculation (default)
|
||||
trades_df = pd.DataFrame(
|
||||
[
|
||||
{"close_date": t.close_date_utc, "close_profit": t.close_profit}
|
||||
for t in trades_in_window
|
||||
]
|
||||
)
|
||||
drawdown_obj = calculate_max_drawdown(trades_df, value_col="close_profit")
|
||||
# In ratios mode, drawdown_abs is the cumulative ratio drop
|
||||
drawdown = drawdown_obj.drawdown_abs
|
||||
except ValueError:
|
||||
return None
|
||||
|
||||
@@ -104,7 +113,7 @@ class MaxDrawdown(IProtection):
|
||||
return None
|
||||
|
||||
def global_stop(
|
||||
self, date_now: datetime, side: LongShort, starting_balance: float = 0.0
|
||||
self, date_now: datetime, side: LongShort, starting_balance: float
|
||||
) -> ProtectionReturn | None:
|
||||
"""
|
||||
Stops trading (position entering) for all pairs
|
||||
@@ -115,7 +124,7 @@ class MaxDrawdown(IProtection):
|
||||
return self._max_drawdown(date_now, starting_balance)
|
||||
|
||||
def stop_per_pair(
|
||||
self, pair: str, date_now: datetime, side: LongShort, starting_balance: float = 0.0
|
||||
self, pair: str, date_now: datetime, side: LongShort, starting_balance: float
|
||||
) -> ProtectionReturn | None:
|
||||
"""
|
||||
Stops trading (position entering) for this pair
|
||||
|
||||
@@ -87,7 +87,7 @@ class StoplossGuard(IProtection):
|
||||
)
|
||||
|
||||
def global_stop(
|
||||
self, date_now: datetime, side: LongShort, starting_balance: float = 0.0
|
||||
self, date_now: datetime, side: LongShort, starting_balance: float
|
||||
) -> ProtectionReturn | None:
|
||||
"""
|
||||
Stops trading (position entering) for all pairs
|
||||
@@ -100,7 +100,7 @@ class StoplossGuard(IProtection):
|
||||
return self._stoploss_guard(date_now, None, side)
|
||||
|
||||
def stop_per_pair(
|
||||
self, pair: str, date_now: datetime, side: LongShort, starting_balance: float = 0.0
|
||||
self, pair: str, date_now: datetime, side: LongShort, starting_balance: float
|
||||
) -> ProtectionReturn | None:
|
||||
"""
|
||||
Stops trading (position entering) for this pair
|
||||
|
||||
Reference in New Issue
Block a user