freqtrade_origin/freqtrade/plugins/protectionmanager.py

116 lines
4.5 KiB
Python
Raw Normal View History

"""
Protection manager class
"""
2024-05-12 14:38:10 +00:00
import logging
2020-10-15 06:07:09 +00:00
from datetime import datetime, timezone
from typing import Any
2022-09-18 11:31:52 +00:00
from freqtrade.constants import Config, LongShort
from freqtrade.exceptions import ConfigurationError
2020-10-15 06:07:09 +00:00
from freqtrade.persistence import PairLocks
from freqtrade.persistence.models import PairLock
from freqtrade.plugins.protections import IProtection
from freqtrade.resolvers import ProtectionResolver
2020-10-15 05:38:00 +00:00
logger = logging.getLogger(__name__)
class ProtectionManager:
def __init__(self, config: Config, protections: list) -> None:
self._config = config
self._protection_handlers: list[IProtection] = []
self.validate_protections(protections)
for protection_handler_config in protections:
protection_handler = ProtectionResolver.load_protection(
2024-05-12 14:38:10 +00:00
protection_handler_config["method"],
config=config,
protection_config=protection_handler_config,
)
self._protection_handlers.append(protection_handler)
if not self._protection_handlers:
2020-10-14 18:03:56 +00:00
logger.info("No protection Handlers defined.")
@property
def name_list(self) -> list[str]:
"""
Get list of loaded Protection Handler names
"""
return [p.name for p in self._protection_handlers]
def short_desc(self) -> list[dict]:
"""
List of short_desc for each Pairlist Handler
"""
2020-10-14 18:03:56 +00:00
return [{p.name: p.short_desc()} for p in self._protection_handlers]
def global_stop(self, now: datetime | None = None, side: LongShort = "long") -> PairLock | None:
2020-11-16 19:09:34 +00:00
if not now:
now = datetime.now(timezone.utc)
result = None
2020-10-14 18:03:56 +00:00
for protection_handler in self._protection_handlers:
if protection_handler.has_global_stop:
2022-04-24 08:29:19 +00:00
lock = protection_handler.global_stop(date_now=now, side=side)
if lock and lock.until:
if not PairLocks.is_global_lock(lock.until, side=lock.lock_side):
2022-04-24 09:23:26 +00:00
result = PairLocks.lock_pair(
2024-05-12 14:38:10 +00:00
"*", lock.until, lock.reason, now=now, side=lock.lock_side
)
return result
2020-10-24 14:52:26 +00:00
2024-05-12 14:38:10 +00:00
def stop_per_pair(
self, pair, now: datetime | None = None, side: LongShort = "long"
) -> PairLock | None:
2020-11-16 19:09:34 +00:00
if not now:
now = datetime.now(timezone.utc)
result = None
2020-10-24 14:52:26 +00:00
for protection_handler in self._protection_handlers:
if protection_handler.has_local_stop:
2024-05-12 14:38:10 +00:00
lock = protection_handler.stop_per_pair(pair=pair, date_now=now, side=side)
2022-04-24 08:29:19 +00:00
if lock and lock.until:
2022-04-24 09:23:26 +00:00
if not PairLocks.is_pair_locked(pair, lock.until, lock.lock_side):
result = PairLocks.lock_pair(
2024-05-12 14:38:10 +00:00
pair, lock.until, lock.reason, now=now, side=lock.lock_side
)
return result
@staticmethod
def validate_protections(protections: list[dict[str, Any]]) -> None:
"""
Validate protection setup validity
"""
for prot in protections:
parsed_unlock_at = None
if (config_unlock_at := prot.get("unlock_at")) is not None:
try:
parsed_unlock_at = datetime.strptime(config_unlock_at, "%H:%M")
except ValueError:
raise ConfigurationError(
f"Invalid date format for unlock_at: {config_unlock_at}."
)
if "stop_duration" in prot and "stop_duration_candles" in prot:
raise ConfigurationError(
"Protections must specify either `stop_duration` or `stop_duration_candles`.\n"
f"Please fix the protection {prot.get('method')}."
)
if "lookback_period" in prot and "lookback_period_candles" in prot:
raise ConfigurationError(
"Protections must specify either `lookback_period` or "
f"`lookback_period_candles`.\n Please fix the protection {prot.get('method')}."
)
if parsed_unlock_at is not None and (
"stop_duration" in prot or "stop_duration_candles" in prot
):
raise ConfigurationError(
"Protections must specify either `unlock_at`, `stop_duration` or "
"`stop_duration_candles`.\n"
f"Please fix the protection {prot.get('method')}."
)