2018-02-09 07:35:38 +00:00
|
|
|
# pragma pylint: disable=missing-docstring, W0212, too-many-arguments
|
2017-11-14 21:15:24 +00:00
|
|
|
|
2018-02-09 07:35:38 +00:00
|
|
|
"""
|
|
|
|
This module contains the backtesting logic
|
|
|
|
"""
|
2018-03-25 19:37:14 +00:00
|
|
|
import logging
|
2020-10-07 18:59:05 +00:00
|
|
|
from collections import defaultdict
|
2018-07-27 21:01:52 +00:00
|
|
|
from copy import deepcopy
|
2021-01-13 06:47:03 +00:00
|
|
|
from datetime import datetime, timedelta, timezone
|
2021-01-23 12:02:48 +00:00
|
|
|
from typing import Any, Dict, List, Optional, Tuple
|
2018-03-17 21:44:47 +00:00
|
|
|
|
2022-01-16 13:46:43 +00:00
|
|
|
from numpy import nan
|
Fix exception when few pairs with no data do not result in aborting backtest.
Exception is triggered by backtesting 20210301-20210501 range with BAKE/USDT pair (binance). Pair data starts on 2021-04-30 12:00:00 and after adjusting for startup candles pair dataframe is empty.
Solution: Since there are other pairs with enough data - skip pairs with no data and issue a warning.
Exception:
```
Traceback (most recent call last):
File "/home/rk/src/freqtrade/freqtrade/main.py", line 37, in main
return_code = args['func'](args)
File "/home/rk/src/freqtrade/freqtrade/commands/optimize_commands.py", line 53, in start_backtesting
backtesting.start()
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 502, in start
min_date, max_date = self.backtest_one_strategy(strat, data, timerange)
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 474, in backtest_one_strategy
results = self.backtest(
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 365, in backtest
data: Dict = self._get_ohlcv_as_lists(processed)
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 199, in _get_ohlcv_as_lists
pair_data.loc[:, 'buy'] = 0 # cleanup from previous run
File "/home/rk/src/freqtrade/venv/lib/python3.9/site-packages/pandas/core/indexing.py", line 692, in __setitem__
iloc._setitem_with_indexer(indexer, value, self.name)
File "/home/rk/src/freqtrade/venv/lib/python3.9/site-packages/pandas/core/indexing.py", line 1587, in _setitem_with_indexer
raise ValueError(
ValueError: cannot set a frame with no defined index and a scalar
```
2021-05-13 06:47:28 +00:00
|
|
|
from pandas import DataFrame
|
2017-09-28 21:26:28 +00:00
|
|
|
|
2022-01-18 09:00:51 +00:00
|
|
|
from freqtrade import constants
|
2021-09-13 18:00:22 +00:00
|
|
|
from freqtrade.configuration import TimeRange, validate_config_consistency
|
2023-01-17 19:05:18 +00:00
|
|
|
from freqtrade.constants import DATETIME_PRINT_FORMAT, Config, IntOrInf, LongShort
|
2018-12-13 05:34:10 +00:00
|
|
|
from freqtrade.data import history
|
2022-01-06 09:53:11 +00:00
|
|
|
from freqtrade.data.btanalysis import find_existing_backtest_stats, trade_list_to_dataframe
|
2021-07-13 17:36:15 +00:00
|
|
|
from freqtrade.data.converter import trim_dataframe, trim_dataframes
|
2019-03-24 14:24:47 +00:00
|
|
|
from freqtrade.data.dataprovider import DataProvider
|
2022-04-16 16:03:24 +00:00
|
|
|
from freqtrade.enums import (BacktestState, CandleType, ExitCheckTuple, ExitType, RunMode,
|
|
|
|
TradingMode)
|
2021-02-10 19:37:55 +00:00
|
|
|
from freqtrade.exceptions import DependencyException, OperationalException
|
2022-08-25 05:08:22 +00:00
|
|
|
from freqtrade.exchange import (amount_to_contract_precision, price_to_precision,
|
2024-02-03 07:14:17 +00:00
|
|
|
timeframe_to_seconds)
|
2023-06-09 04:45:34 +00:00
|
|
|
from freqtrade.exchange.exchange import Exchange
|
2020-12-07 14:45:02 +00:00
|
|
|
from freqtrade.mixins import LoggingMixin
|
2022-04-29 17:37:13 +00:00
|
|
|
from freqtrade.optimize.backtest_caching import get_strategy_run_id
|
2021-03-21 14:56:36 +00:00
|
|
|
from freqtrade.optimize.bt_progress import BTProgress
|
2023-04-28 14:14:16 +00:00
|
|
|
from freqtrade.optimize.optimize_reports import (generate_backtest_stats, generate_rejected_signals,
|
|
|
|
generate_trade_signal_candles,
|
|
|
|
show_backtest_results,
|
2023-03-19 13:56:41 +00:00
|
|
|
store_backtest_analysis_results,
|
2020-06-26 05:46:59 +00:00
|
|
|
store_backtest_stats)
|
2024-01-10 18:53:06 +00:00
|
|
|
from freqtrade.persistence import (LocalTrade, Order, PairLocks, Trade, disable_database_use,
|
|
|
|
enable_database_use)
|
2020-12-23 16:00:02 +00:00
|
|
|
from freqtrade.plugins.pairlistmanager import PairListManager
|
2020-11-16 19:17:47 +00:00
|
|
|
from freqtrade.plugins.protectionmanager import ProtectionManager
|
2019-03-06 18:55:34 +00:00
|
|
|
from freqtrade.resolvers import ExchangeResolver, StrategyResolver
|
2022-03-25 05:50:18 +00:00
|
|
|
from freqtrade.strategy.interface import IStrategy
|
2020-10-11 17:50:37 +00:00
|
|
|
from freqtrade.strategy.strategy_wrapper import strategy_safe_wrapper
|
2023-07-30 08:52:23 +00:00
|
|
|
from freqtrade.types import BacktestResultType, get_BacktestResultType_default
|
2024-01-04 15:25:40 +00:00
|
|
|
from freqtrade.util.migrations import migrate_data
|
2021-01-28 06:06:58 +00:00
|
|
|
from freqtrade.wallets import Wallets
|
2019-04-09 09:27:35 +00:00
|
|
|
|
2021-03-25 08:34:33 +00:00
|
|
|
|
2018-03-25 19:37:14 +00:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2020-10-18 15:16:57 +00:00
|
|
|
# Indexes for backtest tuples
|
|
|
|
DATE_IDX = 0
|
2021-08-23 19:15:56 +00:00
|
|
|
OPEN_IDX = 1
|
|
|
|
HIGH_IDX = 2
|
|
|
|
LOW_IDX = 3
|
|
|
|
CLOSE_IDX = 4
|
2021-08-24 04:54:55 +00:00
|
|
|
LONG_IDX = 5
|
|
|
|
ELONG_IDX = 6 # Exit long
|
2021-08-23 19:15:56 +00:00
|
|
|
SHORT_IDX = 7
|
2021-08-24 04:54:55 +00:00
|
|
|
ESHORT_IDX = 8 # Exit short
|
2021-09-25 17:31:06 +00:00
|
|
|
ENTER_TAG_IDX = 9
|
2021-11-06 14:24:52 +00:00
|
|
|
EXIT_TAG_IDX = 10
|
2020-10-18 15:16:57 +00:00
|
|
|
|
2022-04-27 11:53:11 +00:00
|
|
|
# Every change to this headers list must evaluate further usages of the resulting tuple
|
|
|
|
# and eventually change the constants for indexes at the top
|
|
|
|
HEADERS = ['date', 'open', 'high', 'low', 'close', 'enter_long', 'exit_long',
|
|
|
|
'enter_short', 'exit_short', 'enter_tag', 'exit_tag']
|
|
|
|
|
2018-03-25 19:37:14 +00:00
|
|
|
|
2019-09-12 01:39:52 +00:00
|
|
|
class Backtesting:
|
2017-11-14 22:14:01 +00:00
|
|
|
"""
|
2018-02-09 07:35:38 +00:00
|
|
|
Backtesting class, this class contains all the logic to run a backtest
|
|
|
|
|
|
|
|
To run a backtest:
|
|
|
|
backtesting = Backtesting(config)
|
|
|
|
backtesting.start()
|
2017-11-14 22:14:01 +00:00
|
|
|
"""
|
2018-07-28 05:00:58 +00:00
|
|
|
|
2023-06-09 04:45:34 +00:00
|
|
|
def __init__(self, config: Config, exchange: Optional[Exchange] = None) -> None:
|
2020-11-19 19:05:56 +00:00
|
|
|
|
|
|
|
LoggingMixin.show_output = False
|
2018-02-09 07:35:38 +00:00
|
|
|
self.config = config
|
2023-07-30 08:52:23 +00:00
|
|
|
self.results: BacktestResultType = get_BacktestResultType_default()
|
2022-01-22 13:11:33 +00:00
|
|
|
self.trade_id_counter: int = 0
|
|
|
|
self.order_id_counter: int = 0
|
2018-04-06 07:57:08 +00:00
|
|
|
|
2021-09-13 18:00:22 +00:00
|
|
|
config['dry_run'] = True
|
2022-01-18 09:00:51 +00:00
|
|
|
self.run_ids: Dict[str, str] = {}
|
2018-07-28 05:41:38 +00:00
|
|
|
self.strategylist: List[IStrategy] = []
|
2020-09-18 05:44:11 +00:00
|
|
|
self.all_results: Dict[str, Dict] = {}
|
2022-03-16 12:16:24 +00:00
|
|
|
self.processed_dfs: Dict[str, Dict] = {}
|
2022-12-05 15:34:31 +00:00
|
|
|
self.rejected_dict: Dict[str, List] = {}
|
|
|
|
self.rejected_df: Dict[str, Dict] = {}
|
2022-04-16 13:37:36 +00:00
|
|
|
|
2021-08-23 19:13:47 +00:00
|
|
|
self._exchange_name = self.config['exchange']['name']
|
2023-06-09 04:45:34 +00:00
|
|
|
if not exchange:
|
|
|
|
exchange = ExchangeResolver.load_exchange(self.config, load_leverage_tiers=True)
|
|
|
|
self.exchange = exchange
|
|
|
|
|
2021-11-22 07:27:33 +00:00
|
|
|
self.dataprovider = DataProvider(self.config, self.exchange)
|
2019-03-24 14:24:47 +00:00
|
|
|
|
2022-05-17 22:11:10 +00:00
|
|
|
if self.config.get('strategy_list'):
|
2022-08-13 07:24:04 +00:00
|
|
|
if self.config.get('freqai', {}).get('enabled', False):
|
2022-09-24 11:21:01 +00:00
|
|
|
logger.warning("Using --strategy-list with FreqAI REQUIRES all strategies "
|
2023-02-21 13:22:40 +00:00
|
|
|
"to have identical feature_engineering_* functions.")
|
2018-07-28 05:55:59 +00:00
|
|
|
for strat in list(self.config['strategy_list']):
|
2018-07-28 05:41:38 +00:00
|
|
|
stratconf = deepcopy(self.config)
|
|
|
|
stratconf['strategy'] = strat
|
2019-12-23 09:23:48 +00:00
|
|
|
self.strategylist.append(StrategyResolver.load_strategy(stratconf))
|
2019-11-23 14:49:46 +00:00
|
|
|
validate_config_consistency(stratconf)
|
2018-07-28 05:41:38 +00:00
|
|
|
|
|
|
|
else:
|
2019-06-09 23:08:54 +00:00
|
|
|
# No strategy list specified, only one strategy
|
2019-12-23 09:23:48 +00:00
|
|
|
self.strategylist.append(StrategyResolver.load_strategy(self.config))
|
2019-11-23 14:49:46 +00:00
|
|
|
validate_config_consistency(self.config)
|
2019-06-09 23:08:54 +00:00
|
|
|
|
2020-06-01 18:49:40 +00:00
|
|
|
if "timeframe" not in self.config:
|
2022-03-20 08:00:53 +00:00
|
|
|
raise OperationalException("Timeframe needs to be set in either "
|
2020-06-01 18:49:40 +00:00
|
|
|
"configuration or as cli argument `--timeframe 5m`")
|
|
|
|
self.timeframe = str(self.config.get('timeframe'))
|
2024-02-03 07:14:17 +00:00
|
|
|
self.timeframe_secs = timeframe_to_seconds(self.timeframe)
|
|
|
|
self.timeframe_min = self.timeframe_secs // 60
|
|
|
|
self.timeframe_td = timedelta(seconds=self.timeframe_secs)
|
2024-01-23 05:38:24 +00:00
|
|
|
self.disable_database_use()
|
2021-09-26 13:07:48 +00:00
|
|
|
self.init_backtest_detail()
|
2022-09-18 12:59:16 +00:00
|
|
|
self.pairlists = PairListManager(self.exchange, self.config, self.dataprovider)
|
2023-06-09 04:45:34 +00:00
|
|
|
self._validate_pairlists_for_backtesting()
|
2020-06-07 14:06:20 +00:00
|
|
|
|
2021-05-02 09:20:54 +00:00
|
|
|
self.dataprovider.add_pairlisthandler(self.pairlists)
|
2020-06-07 14:06:20 +00:00
|
|
|
self.pairlists.refresh_pairlist()
|
|
|
|
|
|
|
|
if len(self.pairlists.whitelist) == 0:
|
|
|
|
raise OperationalException("No pair in whitelist.")
|
|
|
|
|
2022-06-23 18:47:51 +00:00
|
|
|
if config.get('fee', None) is not None:
|
2020-06-07 14:06:20 +00:00
|
|
|
self.fee = config['fee']
|
|
|
|
else:
|
|
|
|
self.fee = self.exchange.get_fee(symbol=self.pairlists.whitelist[0])
|
2022-08-16 08:15:10 +00:00
|
|
|
self.precision_mode = self.exchange.precisionMode
|
2020-06-07 14:06:20 +00:00
|
|
|
|
2022-09-26 02:14:00 +00:00
|
|
|
if self.config.get('freqai_backtest_live_models', False):
|
2022-11-07 18:35:28 +00:00
|
|
|
from freqtrade.freqai.utils import get_timerange_backtest_live_models
|
|
|
|
self.config['timerange'] = get_timerange_backtest_live_models(self.config)
|
2022-09-26 02:14:00 +00:00
|
|
|
|
2021-07-13 17:36:15 +00:00
|
|
|
self.timerange = TimeRange.parse_timerange(
|
|
|
|
None if self.config.get('timerange') is None else str(self.config.get('timerange')))
|
|
|
|
|
2019-10-20 11:56:01 +00:00
|
|
|
# Get maximum required startup period
|
|
|
|
self.required_startup = max([strat.startup_candle_count for strat in self.strategylist])
|
2022-09-22 15:13:51 +00:00
|
|
|
self.exchange.validate_required_startup_candles(self.required_startup, self.timeframe)
|
|
|
|
|
2023-12-17 19:11:10 +00:00
|
|
|
# Add maximum startup candle count to configuration for informative pairs support
|
|
|
|
self.config['startup_candle_count'] = self.required_startup
|
|
|
|
|
2022-09-22 15:13:51 +00:00
|
|
|
if self.config.get('freqai', {}).get('enabled', False):
|
|
|
|
# For FreqAI, increase the required_startup to includes the training data
|
2023-12-17 19:11:10 +00:00
|
|
|
# This value should NOT be written to startup_candle_count
|
2023-12-17 16:44:15 +00:00
|
|
|
self.required_startup = self.dataprovider.get_required_startup(self.timeframe)
|
2022-09-22 15:13:51 +00:00
|
|
|
|
2022-02-21 18:19:12 +00:00
|
|
|
self.trading_mode: TradingMode = config.get('trading_mode', TradingMode.SPOT)
|
2022-03-11 18:37:45 +00:00
|
|
|
# strategies which define "can_short=True" will fail to load in Spot mode.
|
2021-11-21 18:28:53 +00:00
|
|
|
self._can_short = self.trading_mode != TradingMode.SPOT
|
2022-10-16 15:14:23 +00:00
|
|
|
self._position_stacking: bool = self.config.get('position_stacking', False)
|
2022-10-18 04:39:55 +00:00
|
|
|
self.enable_protections: bool = self.config.get('enable_protections', False)
|
2024-01-04 15:44:17 +00:00
|
|
|
migrate_data(config, self.exchange)
|
2021-09-22 18:36:03 +00:00
|
|
|
|
2021-09-26 13:07:48 +00:00
|
|
|
self.init_backtest()
|
2021-03-11 18:16:18 +00:00
|
|
|
|
2023-06-09 04:45:34 +00:00
|
|
|
def _validate_pairlists_for_backtesting(self):
|
|
|
|
if 'VolumePairList' in self.pairlists.name_list:
|
|
|
|
raise OperationalException("VolumePairList not allowed for backtesting. "
|
|
|
|
"Please use StaticPairList instead.")
|
|
|
|
if 'PerformanceFilter' in self.pairlists.name_list:
|
|
|
|
raise OperationalException("PerformanceFilter not allowed for backtesting.")
|
|
|
|
|
|
|
|
if len(self.strategylist) > 1 and 'PrecisionFilter' in self.pairlists.name_list:
|
|
|
|
raise OperationalException(
|
|
|
|
"PrecisionFilter not allowed for backtesting multiple strategies."
|
|
|
|
)
|
|
|
|
|
2022-02-24 05:25:21 +00:00
|
|
|
@staticmethod
|
|
|
|
def cleanup():
|
2020-11-25 08:53:13 +00:00
|
|
|
LoggingMixin.show_output = True
|
2024-01-10 18:53:06 +00:00
|
|
|
enable_database_use()
|
2020-11-25 08:53:13 +00:00
|
|
|
|
2022-11-14 19:56:35 +00:00
|
|
|
def init_backtest_detail(self) -> None:
|
2021-09-26 13:07:48 +00:00
|
|
|
# Load detail timeframe if specified
|
|
|
|
self.timeframe_detail = str(self.config.get('timeframe_detail', ''))
|
|
|
|
if self.timeframe_detail:
|
2024-02-03 07:14:17 +00:00
|
|
|
timeframe_detail_secs = timeframe_to_seconds(self.timeframe_detail)
|
|
|
|
self.timeframe_detail_td = timedelta(seconds=timeframe_detail_secs)
|
|
|
|
if self.timeframe_secs <= timeframe_detail_secs:
|
2021-09-26 13:07:48 +00:00
|
|
|
raise OperationalException(
|
|
|
|
"Detail timeframe must be smaller than strategy timeframe.")
|
|
|
|
|
|
|
|
else:
|
2024-02-02 06:03:44 +00:00
|
|
|
self.timeframe_detail_td = timedelta(seconds=0)
|
2021-09-26 13:07:48 +00:00
|
|
|
self.detail_data: Dict[str, DataFrame] = {}
|
2022-01-17 18:41:01 +00:00
|
|
|
self.futures_data: Dict[str, DataFrame] = {}
|
2021-09-26 13:07:48 +00:00
|
|
|
|
|
|
|
def init_backtest(self):
|
|
|
|
|
|
|
|
self.prepare_backtest(False)
|
|
|
|
|
2024-02-25 08:07:53 +00:00
|
|
|
self.wallets = Wallets(self.config, self.exchange, is_backtest=True)
|
2021-09-26 13:07:48 +00:00
|
|
|
|
|
|
|
self.progress = BTProgress()
|
|
|
|
self.abort = False
|
|
|
|
|
2021-02-26 18:48:06 +00:00
|
|
|
def _set_strategy(self, strategy: IStrategy):
|
2018-07-28 04:54:33 +00:00
|
|
|
"""
|
|
|
|
Load strategy into backtesting
|
|
|
|
"""
|
2020-10-18 15:16:57 +00:00
|
|
|
self.strategy: IStrategy = strategy
|
2021-05-03 06:47:58 +00:00
|
|
|
strategy.dp = self.dataprovider
|
2021-07-11 09:20:31 +00:00
|
|
|
# Attach Wallets to Strategy baseclass
|
2021-09-19 23:44:12 +00:00
|
|
|
strategy.wallets = self.wallets
|
2019-03-06 18:55:34 +00:00
|
|
|
# Set stoploss_on_exchange to false for backtesting,
|
2022-04-24 12:30:50 +00:00
|
|
|
# since a "perfect" stoploss-exit is assumed anyway
|
2019-03-06 18:55:34 +00:00
|
|
|
# And the regular "stoploss" function would not apply to that case
|
|
|
|
self.strategy.order_types['stoploss_on_exchange'] = False
|
2023-03-26 15:27:52 +00:00
|
|
|
# Update can_short flag
|
|
|
|
self._can_short = self.trading_mode != TradingMode.SPOT and strategy.can_short
|
2022-05-30 05:07:47 +00:00
|
|
|
|
2022-05-25 18:01:21 +00:00
|
|
|
self.strategy.ft_bot_start()
|
2021-08-03 04:38:15 +00:00
|
|
|
|
|
|
|
def _load_protections(self, strategy: IStrategy):
|
2021-05-07 14:28:06 +00:00
|
|
|
if self.config.get('enable_protections', False):
|
|
|
|
conf = self.config
|
|
|
|
if hasattr(strategy, 'protections'):
|
|
|
|
conf = deepcopy(conf)
|
|
|
|
conf['protections'] = strategy.protections
|
2021-06-17 19:01:22 +00:00
|
|
|
self.protections = ProtectionManager(self.config, strategy.protections)
|
2018-07-28 04:54:33 +00:00
|
|
|
|
2020-03-15 14:04:48 +00:00
|
|
|
def load_bt_data(self) -> Tuple[Dict[str, DataFrame], TimeRange]:
|
2020-09-18 05:45:47 +00:00
|
|
|
"""
|
|
|
|
Loads backtest data and returns the data combined with the timerange
|
|
|
|
as tuple.
|
|
|
|
"""
|
2021-03-21 14:56:36 +00:00
|
|
|
self.progress.init_step(BacktestState.DATALOAD, 1)
|
2021-03-11 18:16:18 +00:00
|
|
|
|
2019-10-23 18:13:43 +00:00
|
|
|
data = history.load_data(
|
2019-12-23 18:32:31 +00:00
|
|
|
datadir=self.config['datadir'],
|
2020-04-25 13:37:13 +00:00
|
|
|
pairs=self.pairlists.whitelist,
|
2019-11-02 19:26:26 +00:00
|
|
|
timeframe=self.timeframe,
|
2021-07-13 17:36:15 +00:00
|
|
|
timerange=self.timerange,
|
2023-12-17 19:11:10 +00:00
|
|
|
startup_candles=self.required_startup,
|
2019-10-23 18:13:43 +00:00
|
|
|
fail_without_data=True,
|
2023-07-12 16:27:46 +00:00
|
|
|
data_format=self.config['dataformat_ohlcv'],
|
2021-12-27 18:46:05 +00:00
|
|
|
candle_type=self.config.get('candle_type_def', CandleType.SPOT)
|
2019-10-23 18:13:43 +00:00
|
|
|
)
|
|
|
|
|
2019-12-17 22:06:03 +00:00
|
|
|
min_date, max_date = history.get_timerange(data)
|
2019-10-23 18:13:43 +00:00
|
|
|
|
2020-06-09 06:07:34 +00:00
|
|
|
logger.info(f'Loading data from {min_date.strftime(DATETIME_PRINT_FORMAT)} '
|
|
|
|
f'up to {max_date.strftime(DATETIME_PRINT_FORMAT)} '
|
2021-05-06 18:49:48 +00:00
|
|
|
f'({(max_date - min_date).days} days).')
|
2020-06-09 06:07:34 +00:00
|
|
|
|
2019-10-23 18:13:43 +00:00
|
|
|
# Adjust startts forward if not enough data is available
|
2021-07-13 17:36:15 +00:00
|
|
|
self.timerange.adjust_start_if_necessary(timeframe_to_seconds(self.timeframe),
|
|
|
|
self.required_startup, min_date)
|
2019-10-23 18:13:43 +00:00
|
|
|
|
2021-03-21 14:56:36 +00:00
|
|
|
self.progress.set_new_value(1)
|
2021-07-13 17:36:15 +00:00
|
|
|
return data, self.timerange
|
2019-10-23 18:13:43 +00:00
|
|
|
|
2021-08-14 13:34:43 +00:00
|
|
|
def load_bt_data_detail(self) -> None:
|
|
|
|
"""
|
|
|
|
Loads backtest detail data (smaller timeframe) if necessary.
|
|
|
|
"""
|
|
|
|
if self.timeframe_detail:
|
|
|
|
self.detail_data = history.load_data(
|
|
|
|
datadir=self.config['datadir'],
|
|
|
|
pairs=self.pairlists.whitelist,
|
|
|
|
timeframe=self.timeframe_detail,
|
|
|
|
timerange=self.timerange,
|
|
|
|
startup_candles=0,
|
|
|
|
fail_without_data=True,
|
2023-07-12 16:27:46 +00:00
|
|
|
data_format=self.config['dataformat_ohlcv'],
|
2021-12-27 18:46:05 +00:00
|
|
|
candle_type=self.config.get('candle_type_def', CandleType.SPOT)
|
2021-08-14 13:34:43 +00:00
|
|
|
)
|
|
|
|
else:
|
|
|
|
self.detail_data = {}
|
2022-01-08 14:07:20 +00:00
|
|
|
if self.trading_mode == TradingMode.FUTURES:
|
2024-01-04 14:30:06 +00:00
|
|
|
self.funding_fee_timeframe: str = self.exchange.get_option('funding_fee_timeframe')
|
2023-12-10 12:23:40 +00:00
|
|
|
self.funding_fee_timeframe_secs: int = timeframe_to_seconds(self.funding_fee_timeframe)
|
2024-01-04 14:30:06 +00:00
|
|
|
mark_timeframe: str = self.exchange.get_option('mark_ohlcv_timeframe')
|
|
|
|
|
2022-01-08 14:07:20 +00:00
|
|
|
# Load additional futures data.
|
2022-01-17 18:41:01 +00:00
|
|
|
funding_rates_dict = history.load_data(
|
2022-01-08 14:07:20 +00:00
|
|
|
datadir=self.config['datadir'],
|
|
|
|
pairs=self.pairlists.whitelist,
|
2023-12-10 12:23:40 +00:00
|
|
|
timeframe=self.funding_fee_timeframe,
|
2022-01-08 14:07:20 +00:00
|
|
|
timerange=self.timerange,
|
|
|
|
startup_candles=0,
|
|
|
|
fail_without_data=True,
|
2023-07-12 16:27:46 +00:00
|
|
|
data_format=self.config['dataformat_ohlcv'],
|
2022-01-08 14:07:20 +00:00
|
|
|
candle_type=CandleType.FUNDING_RATE
|
|
|
|
)
|
|
|
|
|
|
|
|
# For simplicity, assign to CandleType.Mark (might contian index candles!)
|
2022-01-17 18:41:01 +00:00
|
|
|
mark_rates_dict = history.load_data(
|
2022-01-08 14:07:20 +00:00
|
|
|
datadir=self.config['datadir'],
|
|
|
|
pairs=self.pairlists.whitelist,
|
2024-01-04 14:30:06 +00:00
|
|
|
timeframe=mark_timeframe,
|
2022-01-08 14:07:20 +00:00
|
|
|
timerange=self.timerange,
|
|
|
|
startup_candles=0,
|
|
|
|
fail_without_data=True,
|
2023-07-12 16:27:46 +00:00
|
|
|
data_format=self.config['dataformat_ohlcv'],
|
2022-08-21 15:48:13 +00:00
|
|
|
candle_type=CandleType.from_string(self.exchange.get_option("mark_ohlcv_price"))
|
2022-01-08 14:07:20 +00:00
|
|
|
)
|
2022-01-17 18:41:01 +00:00
|
|
|
# Combine data to avoid combining the data per trade.
|
2022-04-28 18:05:19 +00:00
|
|
|
unavailable_pairs = []
|
2022-01-17 18:41:01 +00:00
|
|
|
for pair in self.pairlists.whitelist:
|
2022-04-28 18:05:19 +00:00
|
|
|
if pair not in self.exchange._leverage_tiers:
|
|
|
|
unavailable_pairs.append(pair)
|
|
|
|
continue
|
2022-05-17 21:05:33 +00:00
|
|
|
|
2022-05-21 06:50:39 +00:00
|
|
|
self.futures_data[pair] = self.exchange.combine_funding_and_mark(
|
|
|
|
funding_rates=funding_rates_dict[pair],
|
|
|
|
mark_rates=mark_rates_dict[pair],
|
2022-05-21 15:35:49 +00:00
|
|
|
futures_funding_rate=self.config.get('futures_funding_rate', None),
|
2022-05-21 06:50:39 +00:00
|
|
|
)
|
2022-01-08 14:07:20 +00:00
|
|
|
|
2022-04-28 18:05:19 +00:00
|
|
|
if unavailable_pairs:
|
|
|
|
raise OperationalException(
|
2022-07-31 12:19:04 +00:00
|
|
|
f"Pairs {', '.join(unavailable_pairs)} got no leverage tiers available. "
|
|
|
|
"It is therefore impossible to backtest with this pair at the moment.")
|
2022-01-08 14:07:20 +00:00
|
|
|
else:
|
|
|
|
self.futures_data = {}
|
2019-10-23 18:13:43 +00:00
|
|
|
|
2023-09-23 16:20:19 +00:00
|
|
|
def disable_database_use(self):
|
2024-01-10 18:53:06 +00:00
|
|
|
disable_database_use(self.timeframe)
|
2023-09-23 16:20:19 +00:00
|
|
|
|
2020-11-27 16:38:15 +00:00
|
|
|
def prepare_backtest(self, enable_protections):
|
|
|
|
"""
|
|
|
|
Backtesting setup method - called once for every call to "backtest()".
|
|
|
|
"""
|
2023-09-23 16:20:19 +00:00
|
|
|
self.disable_database_use()
|
2021-03-08 18:40:29 +00:00
|
|
|
PairLocks.reset_locks()
|
|
|
|
Trade.reset_trades()
|
2021-05-23 07:36:02 +00:00
|
|
|
self.rejected_trades = 0
|
2022-02-07 17:49:30 +00:00
|
|
|
self.timedout_entry_orders = 0
|
|
|
|
self.timedout_exit_orders = 0
|
2022-05-16 22:41:01 +00:00
|
|
|
self.canceled_trade_entries = 0
|
2022-05-17 11:07:02 +00:00
|
|
|
self.canceled_entry_orders = 0
|
|
|
|
self.replaced_entry_orders = 0
|
2021-05-02 09:20:54 +00:00
|
|
|
self.dataprovider.clear_cache()
|
2021-09-26 13:07:48 +00:00
|
|
|
if enable_protections:
|
|
|
|
self._load_protections(self.strategy)
|
2020-11-27 16:38:15 +00:00
|
|
|
|
2021-07-06 04:28:47 +00:00
|
|
|
def check_abort(self):
|
|
|
|
"""
|
|
|
|
Check if abort was requested, raise DependencyException if that's the case
|
|
|
|
Only applies to Interactive backtest mode (webserver mode)
|
|
|
|
"""
|
|
|
|
if self.abort:
|
|
|
|
self.abort = False
|
|
|
|
raise DependencyException("Stop requested")
|
|
|
|
|
2020-10-18 15:16:57 +00:00
|
|
|
def _get_ohlcv_as_lists(self, processed: Dict[str, DataFrame]) -> Dict[str, Tuple]:
|
2019-03-23 14:00:07 +00:00
|
|
|
"""
|
2020-03-08 10:35:31 +00:00
|
|
|
Helper function to convert a processed dataframes into lists for performance reasons.
|
2019-03-23 14:00:07 +00:00
|
|
|
|
|
|
|
Used by backtest() - so keep this optimized for performance.
|
2021-12-29 10:09:01 +00:00
|
|
|
|
|
|
|
:param processed: a processed dictionary with format {pair, data}, which gets cleared to
|
|
|
|
optimize memory usage!
|
2019-03-23 14:00:07 +00:00
|
|
|
"""
|
2022-04-27 11:53:11 +00:00
|
|
|
|
2020-03-08 10:35:31 +00:00
|
|
|
data: Dict = {}
|
2021-03-21 14:56:36 +00:00
|
|
|
self.progress.init_step(BacktestState.CONVERT, len(processed))
|
|
|
|
|
2020-03-08 10:35:31 +00:00
|
|
|
# Create dict with data
|
2021-12-29 10:09:01 +00:00
|
|
|
for pair in processed.keys():
|
|
|
|
pair_data = processed[pair]
|
2021-07-06 04:28:47 +00:00
|
|
|
self.check_abort()
|
2021-03-21 14:56:36 +00:00
|
|
|
self.progress.increment()
|
2021-08-24 04:28:16 +00:00
|
|
|
|
2021-05-20 04:49:25 +00:00
|
|
|
if not pair_data.empty:
|
2021-08-23 19:15:56 +00:00
|
|
|
# Cleanup from prior runs
|
2022-04-27 11:53:11 +00:00
|
|
|
pair_data.drop(HEADERS[5:] + ['buy', 'sell'], axis=1, errors='ignore')
|
2023-07-21 18:27:52 +00:00
|
|
|
df_analyzed = self.strategy.ft_advise_signals(pair_data, {'pair': pair})
|
2022-01-27 16:09:19 +00:00
|
|
|
# Update dataprovider cache
|
2022-02-11 16:02:04 +00:00
|
|
|
self.dataprovider._set_cached_df(
|
|
|
|
pair, self.timeframe, df_analyzed, self.config['candle_type_def'])
|
2022-01-27 16:09:19 +00:00
|
|
|
|
2023-08-15 17:43:04 +00:00
|
|
|
# Trim startup period from analyzed dataframe
|
|
|
|
df_analyzed = processed[pair] = pair_data = trim_dataframe(
|
|
|
|
df_analyzed, self.timerange, startup_candles=self.required_startup)
|
|
|
|
|
2022-04-24 12:28:15 +00:00
|
|
|
# Create a copy of the dataframe before shifting, that way the entry signal/tag
|
2022-01-27 16:09:19 +00:00
|
|
|
# remains on the correct candle for callbacks.
|
|
|
|
df_analyzed = df_analyzed.copy()
|
|
|
|
|
2022-04-24 12:28:15 +00:00
|
|
|
# To avoid using data from future, we use entry/exit signals shifted
|
2020-03-08 10:35:31 +00:00
|
|
|
# from the previous candle
|
2022-04-27 11:53:11 +00:00
|
|
|
for col in HEADERS[5:]:
|
2022-01-16 13:46:43 +00:00
|
|
|
tag_col = col in ('enter_tag', 'exit_tag')
|
2021-09-26 13:39:34 +00:00
|
|
|
if col in df_analyzed.columns:
|
2022-09-27 08:10:58 +00:00
|
|
|
df_analyzed[col] = df_analyzed.loc[:, col].replace(
|
2022-01-16 13:46:43 +00:00
|
|
|
[nan], [0 if not tag_col else None]).shift(1)
|
2022-04-02 13:36:25 +00:00
|
|
|
elif not df_analyzed.empty:
|
2022-09-27 08:10:58 +00:00
|
|
|
df_analyzed[col] = 0 if not tag_col else None
|
2019-03-23 14:00:07 +00:00
|
|
|
|
2021-08-10 05:09:38 +00:00
|
|
|
df_analyzed = df_analyzed.drop(df_analyzed.head(1).index)
|
|
|
|
|
2019-03-23 14:00:07 +00:00
|
|
|
# Convert from Pandas to list for performance reasons
|
|
|
|
# (Looping Pandas is slow.)
|
2022-04-27 11:53:11 +00:00
|
|
|
data[pair] = df_analyzed[HEADERS].values.tolist() if not df_analyzed.empty else []
|
2020-03-08 10:35:31 +00:00
|
|
|
return data
|
2019-03-23 14:00:07 +00:00
|
|
|
|
2022-04-24 12:30:50 +00:00
|
|
|
def _get_close_rate(self, row: Tuple, trade: LocalTrade, exit: ExitCheckTuple,
|
2020-02-02 04:00:40 +00:00
|
|
|
trade_dur: int) -> float:
|
2019-12-07 14:28:56 +00:00
|
|
|
"""
|
|
|
|
Get close rate for backtesting result
|
|
|
|
"""
|
2019-12-07 13:30:14 +00:00
|
|
|
# Special handling if high or low hit STOP_LOSS or ROI
|
2022-07-28 18:35:23 +00:00
|
|
|
if exit.exit_type in (
|
|
|
|
ExitType.STOP_LOSS, ExitType.TRAILING_STOP_LOSS, ExitType.LIQUIDATION):
|
2022-04-24 12:30:50 +00:00
|
|
|
return self._get_close_rate_for_stoploss(row, trade, exit, trade_dur)
|
|
|
|
elif exit.exit_type == (ExitType.ROI):
|
|
|
|
return self._get_close_rate_for_roi(row, trade, exit, trade_dur)
|
2022-03-14 10:38:44 +00:00
|
|
|
else:
|
2022-04-07 17:43:34 +00:00
|
|
|
return row[OPEN_IDX]
|
2022-03-14 10:38:44 +00:00
|
|
|
|
2022-04-24 12:30:50 +00:00
|
|
|
def _get_close_rate_for_stoploss(self, row: Tuple, trade: LocalTrade, exit: ExitCheckTuple,
|
2022-03-15 04:04:02 +00:00
|
|
|
trade_dur: int) -> float:
|
|
|
|
# our stoploss was already lower than candle high,
|
|
|
|
# possibly due to a cancelled trade exit.
|
2022-04-24 12:30:50 +00:00
|
|
|
# exit at open price.
|
2022-03-15 04:04:02 +00:00
|
|
|
is_short = trade.is_short or False
|
2022-03-14 10:38:44 +00:00
|
|
|
leverage = trade.leverage or 1.0
|
2022-03-15 04:04:02 +00:00
|
|
|
side_1 = -1 if is_short else 1
|
2022-07-28 18:35:23 +00:00
|
|
|
if exit.exit_type == ExitType.LIQUIDATION and trade.liquidation_price:
|
|
|
|
stoploss_value = trade.liquidation_price
|
|
|
|
else:
|
|
|
|
stoploss_value = trade.stop_loss
|
|
|
|
|
2022-03-15 04:04:02 +00:00
|
|
|
if is_short:
|
2022-07-28 18:35:23 +00:00
|
|
|
if stoploss_value < row[LOW_IDX]:
|
2022-04-07 17:43:34 +00:00
|
|
|
return row[OPEN_IDX]
|
2022-03-15 04:04:02 +00:00
|
|
|
else:
|
2022-07-28 18:35:23 +00:00
|
|
|
if stoploss_value > row[HIGH_IDX]:
|
2022-04-07 17:43:34 +00:00
|
|
|
return row[OPEN_IDX]
|
2021-05-15 06:38:51 +00:00
|
|
|
|
2022-03-15 04:04:02 +00:00
|
|
|
# Special case: trailing triggers within same candle as trade opened. Assume most
|
|
|
|
# pessimistic price movement, which is moving just enough to arm stoploss and
|
|
|
|
# immediately going down to stop price.
|
2022-04-24 12:30:50 +00:00
|
|
|
if exit.exit_type == ExitType.TRAILING_STOP_LOSS and trade_dur == 0:
|
2022-03-15 04:04:02 +00:00
|
|
|
if (
|
|
|
|
not self.strategy.use_custom_stoploss and self.strategy.trailing_stop
|
|
|
|
and self.strategy.trailing_only_offset_is_reached
|
|
|
|
and self.strategy.trailing_stop_positive_offset is not None
|
|
|
|
and self.strategy.trailing_stop_positive
|
|
|
|
):
|
|
|
|
# Worst case: price reaches stop_positive_offset and dives down.
|
2022-04-07 17:43:34 +00:00
|
|
|
stop_rate = (row[OPEN_IDX] *
|
2022-03-15 04:23:59 +00:00
|
|
|
(1 + side_1 * abs(self.strategy.trailing_stop_positive_offset) -
|
|
|
|
side_1 * abs(self.strategy.trailing_stop_positive / leverage)))
|
2022-03-15 04:04:02 +00:00
|
|
|
else:
|
|
|
|
# Worst case: price ticks tiny bit above open and dives down.
|
2023-03-01 05:40:07 +00:00
|
|
|
stop_rate = row[OPEN_IDX] * (1 - side_1 * abs(
|
|
|
|
(trade.stop_loss_pct or 0.0) / leverage))
|
2022-03-09 12:00:06 +00:00
|
|
|
|
2022-04-24 12:30:50 +00:00
|
|
|
# Limit lower-end to candle low to avoid exits below the low.
|
2022-03-15 04:04:02 +00:00
|
|
|
# This still remains "worst case" - but "worst realistic case".
|
|
|
|
if is_short:
|
2022-04-07 17:43:34 +00:00
|
|
|
return min(row[HIGH_IDX], stop_rate)
|
2022-03-14 10:38:44 +00:00
|
|
|
else:
|
2022-04-07 17:43:34 +00:00
|
|
|
return max(row[LOW_IDX], stop_rate)
|
2022-03-14 10:38:44 +00:00
|
|
|
|
2022-03-15 04:04:02 +00:00
|
|
|
# Set close_rate to stoploss
|
2022-07-28 18:35:23 +00:00
|
|
|
return stoploss_value
|
2022-03-15 04:04:02 +00:00
|
|
|
|
2022-04-24 12:30:50 +00:00
|
|
|
def _get_close_rate_for_roi(self, row: Tuple, trade: LocalTrade, exit: ExitCheckTuple,
|
2022-03-15 04:04:02 +00:00
|
|
|
trade_dur: int) -> float:
|
|
|
|
is_short = trade.is_short or False
|
2022-03-14 10:38:44 +00:00
|
|
|
leverage = trade.leverage or 1.0
|
2022-03-15 04:04:02 +00:00
|
|
|
side_1 = -1 if is_short else 1
|
|
|
|
roi_entry, roi = self.strategy.min_roi_reached_entry(trade_dur)
|
|
|
|
if roi is not None and roi_entry is not None:
|
|
|
|
if roi == -1 and roi_entry % self.timeframe_min == 0:
|
2022-04-10 13:56:29 +00:00
|
|
|
# When force_exiting with ROI=-1, the roi time will always be equal to trade_dur.
|
2022-03-15 04:04:02 +00:00
|
|
|
# If that entry is a multiple of the timeframe (so on candle open)
|
|
|
|
# - we'll use open instead of close
|
2022-04-07 17:43:34 +00:00
|
|
|
return row[OPEN_IDX]
|
2022-03-14 10:38:44 +00:00
|
|
|
|
2022-03-15 04:04:02 +00:00
|
|
|
# - (Expected abs profit - open_rate - open_fee) / (fee_close -1)
|
|
|
|
roi_rate = trade.open_rate * roi / leverage
|
|
|
|
open_fee_rate = side_1 * trade.open_rate * (1 + side_1 * trade.fee_open)
|
2023-03-01 05:40:07 +00:00
|
|
|
close_rate = -(roi_rate + open_fee_rate) / ((trade.fee_close or 0.0) - side_1 * 1)
|
2022-03-15 04:04:02 +00:00
|
|
|
if is_short:
|
2022-04-07 17:43:34 +00:00
|
|
|
is_new_roi = row[OPEN_IDX] < close_rate
|
2022-03-15 04:04:02 +00:00
|
|
|
else:
|
2022-04-07 17:43:34 +00:00
|
|
|
is_new_roi = row[OPEN_IDX] > close_rate
|
2022-03-15 04:04:02 +00:00
|
|
|
if (trade_dur > 0 and trade_dur == roi_entry
|
|
|
|
and roi_entry % self.timeframe_min == 0
|
|
|
|
and is_new_roi):
|
|
|
|
# new ROI entry came into effect.
|
2022-04-24 12:30:50 +00:00
|
|
|
# use Open rate if open_rate > calculated exit rate
|
2022-04-07 17:43:34 +00:00
|
|
|
return row[OPEN_IDX]
|
2021-06-12 07:16:30 +00:00
|
|
|
|
2022-03-15 04:04:02 +00:00
|
|
|
if (trade_dur == 0 and (
|
|
|
|
(
|
|
|
|
is_short
|
2022-03-15 05:39:07 +00:00
|
|
|
# Red candle (for longs)
|
2022-04-07 17:43:34 +00:00
|
|
|
and row[OPEN_IDX] < row[CLOSE_IDX] # Red candle
|
|
|
|
and trade.open_rate > row[OPEN_IDX] # trade-open above open_rate
|
|
|
|
and close_rate < row[CLOSE_IDX] # closes below close
|
2022-03-15 04:04:02 +00:00
|
|
|
)
|
|
|
|
or
|
|
|
|
(
|
|
|
|
not is_short
|
2022-03-15 05:39:07 +00:00
|
|
|
# green candle (for shorts)
|
2022-04-07 17:43:34 +00:00
|
|
|
and row[OPEN_IDX] > row[CLOSE_IDX] # green candle
|
|
|
|
and trade.open_rate < row[OPEN_IDX] # trade-open below open_rate
|
|
|
|
and close_rate > row[CLOSE_IDX] # closes above close
|
2022-03-15 04:04:02 +00:00
|
|
|
)
|
|
|
|
)):
|
|
|
|
# ROI on opening candles with custom pricing can only
|
2022-03-15 05:39:07 +00:00
|
|
|
# trigger if the entry was at Open or lower wick.
|
2022-03-15 04:04:02 +00:00
|
|
|
# details: https: // github.com/freqtrade/freqtrade/issues/6261
|
2022-04-24 12:30:50 +00:00
|
|
|
# If open_rate is < open, only allow exits below the close on red candles.
|
2022-03-15 04:04:02 +00:00
|
|
|
raise ValueError("Opening candle ROI on red candles.")
|
|
|
|
|
|
|
|
# Use the maximum between close_rate and low as we
|
2022-04-24 12:30:50 +00:00
|
|
|
# cannot exit outside of a candle.
|
2022-03-15 04:04:02 +00:00
|
|
|
# Applies when a new ROI setting comes in place and the whole candle is above that.
|
2022-04-07 17:43:34 +00:00
|
|
|
return min(max(close_rate, row[LOW_IDX]), row[HIGH_IDX])
|
2019-12-07 14:18:12 +00:00
|
|
|
|
2019-12-07 13:30:14 +00:00
|
|
|
else:
|
2022-03-15 04:04:02 +00:00
|
|
|
# This should not be reached...
|
2022-04-07 17:43:34 +00:00
|
|
|
return row[OPEN_IDX]
|
2019-12-07 13:30:14 +00:00
|
|
|
|
2023-10-11 17:45:00 +00:00
|
|
|
def _get_adjust_trade_entry_for_candle(
|
|
|
|
self, trade: LocalTrade, row: Tuple, current_time: datetime
|
|
|
|
) -> LocalTrade:
|
2023-12-29 18:31:54 +00:00
|
|
|
current_rate: float = row[OPEN_IDX]
|
2022-07-31 12:19:04 +00:00
|
|
|
current_profit = trade.calc_profit_ratio(current_rate)
|
|
|
|
min_stake = self.exchange.get_min_pair_stake_amount(trade.pair, current_rate, -0.1)
|
|
|
|
max_stake = self.exchange.get_max_pair_stake_amount(trade.pair, current_rate)
|
2022-02-02 02:39:22 +00:00
|
|
|
stake_available = self.wallets.get_available_stake_amount()
|
2024-01-28 19:21:47 +00:00
|
|
|
stake_amount, order_tag = self.strategy._adjust_trade_position_internal(
|
2022-05-07 12:53:51 +00:00
|
|
|
trade=trade, # type: ignore[arg-type]
|
2023-10-11 17:45:00 +00:00
|
|
|
current_time=current_time, current_rate=current_rate,
|
2022-02-02 02:39:22 +00:00
|
|
|
current_profit=current_profit, min_stake=min_stake,
|
2022-07-31 12:19:04 +00:00
|
|
|
max_stake=min(max_stake, stake_available),
|
|
|
|
current_entry_rate=current_rate, current_exit_rate=current_rate,
|
2024-01-28 19:21:47 +00:00
|
|
|
current_entry_profit=current_profit, current_exit_profit=current_profit
|
|
|
|
)
|
2021-12-09 21:21:35 +00:00
|
|
|
|
|
|
|
# Check if we should increase our position
|
2021-12-10 18:42:24 +00:00
|
|
|
if stake_amount is not None and stake_amount > 0.0:
|
2022-08-25 18:36:17 +00:00
|
|
|
check_adjust_entry = True
|
|
|
|
if self.strategy.max_entry_position_adjustment > -1:
|
|
|
|
entry_count = trade.nr_of_successful_entries
|
|
|
|
check_adjust_entry = (entry_count <= self.strategy.max_entry_position_adjustment)
|
|
|
|
if check_adjust_entry:
|
|
|
|
pos_trade = self._enter_trade(
|
2024-01-28 22:05:19 +00:00
|
|
|
trade.pair, row, 'short' if trade.is_short else 'long', stake_amount, trade,
|
|
|
|
entry_tag1=order_tag)
|
2022-08-25 18:36:17 +00:00
|
|
|
if pos_trade is not None:
|
|
|
|
self.wallets.update()
|
|
|
|
return pos_trade
|
2021-12-09 21:21:35 +00:00
|
|
|
|
2022-07-31 12:19:04 +00:00
|
|
|
if stake_amount is not None and stake_amount < 0.0:
|
2022-09-07 04:43:08 +00:00
|
|
|
amount = amount_to_contract_precision(
|
2022-09-27 17:53:55 +00:00
|
|
|
abs(stake_amount * trade.leverage) / current_rate, trade.amount_precision,
|
2022-09-07 04:43:08 +00:00
|
|
|
self.precision_mode, trade.contract_size)
|
|
|
|
if amount == 0.0:
|
|
|
|
return trade
|
2022-07-31 12:19:04 +00:00
|
|
|
remaining = (trade.amount - amount) * current_rate
|
2023-12-29 18:31:54 +00:00
|
|
|
if min_stake and remaining != 0 and remaining < min_stake:
|
2022-07-31 12:19:04 +00:00
|
|
|
# Remaining stake is too low to be sold.
|
|
|
|
return trade
|
2024-01-28 19:05:40 +00:00
|
|
|
exit_ = ExitCheckTuple(ExitType.PARTIAL_EXIT, order_tag)
|
2023-10-11 17:45:00 +00:00
|
|
|
pos_trade = self._get_exit_for_signal(trade, row, exit_, current_time, amount)
|
2022-07-31 12:19:04 +00:00
|
|
|
if pos_trade is not None:
|
|
|
|
order = pos_trade.orders[-1]
|
2023-10-11 17:45:00 +00:00
|
|
|
if self._try_close_open_order(order, trade, current_time, row):
|
2022-07-31 12:19:04 +00:00
|
|
|
trade.recalc_trade_from_orders()
|
|
|
|
self.wallets.update()
|
|
|
|
return pos_trade
|
|
|
|
|
2021-12-09 21:21:35 +00:00
|
|
|
return trade
|
|
|
|
|
2022-01-16 18:39:42 +00:00
|
|
|
def _get_order_filled(self, rate: float, row: Tuple) -> bool:
|
|
|
|
""" Rate is within candle, therefore filled"""
|
2022-01-30 14:27:18 +00:00
|
|
|
return row[LOW_IDX] <= rate <= row[HIGH_IDX]
|
2022-01-16 18:39:42 +00:00
|
|
|
|
2023-08-14 14:46:25 +00:00
|
|
|
def _call_adjust_stop(self, current_date: datetime, trade: LocalTrade, current_rate: float):
|
2023-08-14 14:00:33 +00:00
|
|
|
profit = trade.calc_profit_ratio(current_rate)
|
2023-08-14 14:46:25 +00:00
|
|
|
self.strategy.ft_stoploss_adjust(current_rate, trade, # type: ignore
|
|
|
|
current_date, profit, 0, after_fill=True)
|
2023-08-14 14:00:33 +00:00
|
|
|
|
2023-08-13 14:47:26 +00:00
|
|
|
def _try_close_open_order(
|
|
|
|
self, order: Optional[Order], trade: LocalTrade, current_date: datetime,
|
|
|
|
row: Tuple) -> bool:
|
|
|
|
"""
|
|
|
|
Check if an order is open and if it should've filled.
|
|
|
|
:return: True if the order filled.
|
|
|
|
"""
|
|
|
|
if order and self._get_order_filled(order.ft_price, row):
|
|
|
|
order.close_bt_order(current_date, trade)
|
2023-12-10 12:47:43 +00:00
|
|
|
self._run_funding_fees(trade, current_date, force=True)
|
|
|
|
|
2023-08-14 14:00:33 +00:00
|
|
|
if not (order.ft_order_side == trade.exit_side and order.safe_amount == trade.amount):
|
2023-09-27 04:30:03 +00:00
|
|
|
# trade is still open
|
|
|
|
trade.set_liquidation_price(self.exchange.get_liquidation_price(
|
|
|
|
pair=trade.pair,
|
|
|
|
open_rate=trade.open_rate,
|
|
|
|
is_short=trade.is_short,
|
|
|
|
amount=trade.amount,
|
|
|
|
stake_amount=trade.stake_amount,
|
|
|
|
leverage=trade.leverage,
|
|
|
|
wallet_balance=trade.stake_amount,
|
|
|
|
))
|
2023-08-14 14:00:33 +00:00
|
|
|
self._call_adjust_stop(current_date, trade, order.ft_price)
|
|
|
|
# pass
|
2023-08-13 14:47:26 +00:00
|
|
|
return True
|
|
|
|
return False
|
2023-08-13 14:35:27 +00:00
|
|
|
|
2022-08-27 06:50:09 +00:00
|
|
|
def _get_exit_for_signal(
|
|
|
|
self, trade: LocalTrade, row: Tuple, exit_: ExitCheckTuple,
|
2023-10-11 17:45:00 +00:00
|
|
|
current_time: datetime,
|
2022-08-27 06:50:09 +00:00
|
|
|
amount: Optional[float] = None) -> Optional[LocalTrade]:
|
2020-10-07 18:59:05 +00:00
|
|
|
|
2022-04-24 12:30:50 +00:00
|
|
|
if exit_.exit_flag:
|
2023-10-11 17:45:00 +00:00
|
|
|
trade.close_date = current_time
|
2022-05-15 15:41:50 +00:00
|
|
|
exit_reason = exit_.exit_reason
|
2022-08-27 06:50:09 +00:00
|
|
|
amount_ = amount if amount is not None else trade.amount
|
2022-03-09 16:34:59 +00:00
|
|
|
trade_dur = int((trade.close_date_utc - trade.open_date_utc).total_seconds() // 60)
|
2022-02-14 19:02:38 +00:00
|
|
|
try:
|
2022-05-13 15:49:40 +00:00
|
|
|
close_rate = self._get_close_rate(row, trade, exit_, trade_dur)
|
2022-02-14 19:02:38 +00:00
|
|
|
except ValueError:
|
|
|
|
return None
|
2022-05-13 15:49:40 +00:00
|
|
|
# call the custom exit price,with default value as previous close_rate
|
|
|
|
current_profit = trade.calc_profit_ratio(close_rate)
|
2022-03-07 19:32:16 +00:00
|
|
|
order_type = self.strategy.order_types['exit']
|
2022-08-27 06:50:09 +00:00
|
|
|
if exit_.exit_type in (ExitType.EXIT_SIGNAL, ExitType.CUSTOM_EXIT,
|
|
|
|
ExitType.PARTIAL_EXIT):
|
2022-05-15 15:41:50 +00:00
|
|
|
# Checks and adds an exit tag, after checking that the length of the
|
|
|
|
# row has the length for an exit tag column
|
2022-08-01 04:43:59 +00:00
|
|
|
if (
|
2022-05-15 15:41:50 +00:00
|
|
|
len(row) > EXIT_TAG_IDX
|
|
|
|
and row[EXIT_TAG_IDX] is not None
|
|
|
|
and len(row[EXIT_TAG_IDX]) > 0
|
|
|
|
and exit_.exit_type in (ExitType.EXIT_SIGNAL,)
|
|
|
|
):
|
|
|
|
exit_reason = row[EXIT_TAG_IDX]
|
2022-04-24 12:30:50 +00:00
|
|
|
# Custom exit pricing only for exit-signals
|
2022-02-08 06:10:54 +00:00
|
|
|
if order_type == 'limit':
|
2022-10-13 17:30:59 +00:00
|
|
|
rate = strategy_safe_wrapper(self.strategy.custom_exit_price,
|
|
|
|
default_retval=close_rate)(
|
2022-05-07 12:53:51 +00:00
|
|
|
pair=trade.pair,
|
|
|
|
trade=trade, # type: ignore[arg-type]
|
2023-10-11 17:45:00 +00:00
|
|
|
current_time=current_time,
|
2022-05-13 15:49:40 +00:00
|
|
|
proposed_rate=close_rate, current_profit=current_profit,
|
2022-05-15 15:41:50 +00:00
|
|
|
exit_tag=exit_reason)
|
2023-10-19 20:06:21 +00:00
|
|
|
if rate is not None and rate != close_rate:
|
2022-10-13 17:30:59 +00:00
|
|
|
close_rate = price_to_precision(rate, trade.price_precision,
|
|
|
|
self.precision_mode)
|
2022-02-08 18:10:29 +00:00
|
|
|
# We can't place orders lower than current low.
|
|
|
|
# freqtrade does not support this in live, and the order would fill immediately
|
2022-03-16 19:08:36 +00:00
|
|
|
if trade.is_short:
|
2022-05-13 15:49:40 +00:00
|
|
|
close_rate = min(close_rate, row[HIGH_IDX])
|
2022-03-16 19:08:36 +00:00
|
|
|
else:
|
2022-05-13 15:49:40 +00:00
|
|
|
close_rate = max(close_rate, row[LOW_IDX])
|
2021-03-25 08:25:25 +00:00
|
|
|
# Confirm trade exit:
|
2022-03-07 06:09:01 +00:00
|
|
|
time_in_force = self.strategy.order_time_in_force['exit']
|
2022-02-08 06:10:54 +00:00
|
|
|
|
2022-08-27 06:50:09 +00:00
|
|
|
if (exit_.exit_type not in (ExitType.LIQUIDATION, ExitType.PARTIAL_EXIT)
|
|
|
|
and not strategy_safe_wrapper(
|
|
|
|
self.strategy.confirm_trade_exit, default_retval=True)(
|
|
|
|
pair=trade.pair,
|
|
|
|
trade=trade, # type: ignore[arg-type]
|
|
|
|
order_type=order_type,
|
|
|
|
amount=amount_,
|
|
|
|
rate=close_rate,
|
|
|
|
time_in_force=time_in_force,
|
|
|
|
sell_reason=exit_reason, # deprecated
|
|
|
|
exit_reason=exit_reason,
|
2023-10-11 17:45:00 +00:00
|
|
|
current_time=current_time)):
|
2021-03-25 08:25:25 +00:00
|
|
|
return None
|
|
|
|
|
2022-05-15 15:41:50 +00:00
|
|
|
trade.exit_reason = exit_reason
|
2021-11-08 05:44:55 +00:00
|
|
|
|
2024-01-28 18:56:30 +00:00
|
|
|
return self._exit_trade(trade, row, close_rate, amount_, exit_reason)
|
2020-10-08 18:10:00 +00:00
|
|
|
return None
|
|
|
|
|
2024-01-28 18:56:30 +00:00
|
|
|
def _exit_trade(self, trade: LocalTrade, sell_row: Tuple, close_rate: float,
|
|
|
|
amount: float, exit_reason: Optional[str]) -> Optional[LocalTrade]:
|
2022-07-31 12:19:04 +00:00
|
|
|
self.order_id_counter += 1
|
|
|
|
exit_candle_time = sell_row[DATE_IDX].to_pydatetime()
|
|
|
|
order_type = self.strategy.order_types['exit']
|
2022-08-24 18:36:08 +00:00
|
|
|
# amount = amount or trade.amount
|
2022-08-25 05:08:22 +00:00
|
|
|
amount = amount_to_contract_precision(amount or trade.amount, trade.amount_precision,
|
|
|
|
self.precision_mode, trade.contract_size)
|
2022-07-31 12:19:04 +00:00
|
|
|
order = Order(
|
|
|
|
id=self.order_id_counter,
|
|
|
|
ft_trade_id=trade.id,
|
|
|
|
order_date=exit_candle_time,
|
|
|
|
order_update_date=exit_candle_time,
|
|
|
|
ft_is_open=True,
|
|
|
|
ft_pair=trade.pair,
|
|
|
|
order_id=str(self.order_id_counter),
|
|
|
|
symbol=trade.pair,
|
|
|
|
ft_order_side=trade.exit_side,
|
|
|
|
side=trade.exit_side,
|
|
|
|
order_type=order_type,
|
|
|
|
status="open",
|
2023-02-19 19:23:04 +00:00
|
|
|
ft_price=close_rate,
|
2022-10-13 17:30:59 +00:00
|
|
|
price=close_rate,
|
|
|
|
average=close_rate,
|
2022-07-31 12:19:04 +00:00
|
|
|
amount=amount,
|
|
|
|
filled=0,
|
|
|
|
remaining=amount,
|
2022-10-13 17:30:59 +00:00
|
|
|
cost=amount * close_rate,
|
2024-01-28 18:56:30 +00:00
|
|
|
ft_order_tag=exit_reason,
|
2022-07-31 12:19:04 +00:00
|
|
|
)
|
2023-07-15 08:27:10 +00:00
|
|
|
order._trade_bt = trade
|
2022-07-31 12:19:04 +00:00
|
|
|
trade.orders.append(order)
|
|
|
|
return trade
|
|
|
|
|
2023-10-11 17:45:00 +00:00
|
|
|
def _check_trade_exit(
|
|
|
|
self, trade: LocalTrade, row: Tuple, current_time: datetime
|
|
|
|
) -> Optional[LocalTrade]:
|
2022-01-08 14:07:20 +00:00
|
|
|
|
2023-12-10 10:52:40 +00:00
|
|
|
self._run_funding_fees(trade, current_time)
|
2022-01-08 14:07:20 +00:00
|
|
|
|
2023-01-21 17:00:57 +00:00
|
|
|
# Check if we need to adjust our current positions
|
|
|
|
if self.strategy.position_adjustment_enable:
|
2023-10-11 17:45:00 +00:00
|
|
|
trade = self._get_adjust_trade_entry_for_candle(trade, row, current_time)
|
2023-01-21 17:00:57 +00:00
|
|
|
|
|
|
|
enter = row[SHORT_IDX] if trade.is_short else row[LONG_IDX]
|
|
|
|
exit_sig = row[ESHORT_IDX] if trade.is_short else row[ELONG_IDX]
|
|
|
|
exits = self.strategy.should_exit(
|
|
|
|
trade, row[OPEN_IDX], row[DATE_IDX].to_pydatetime(), # type: ignore
|
|
|
|
enter=enter, exit_=exit_sig,
|
|
|
|
low=row[LOW_IDX], high=row[HIGH_IDX]
|
|
|
|
)
|
|
|
|
for exit_ in exits:
|
2023-10-11 17:45:00 +00:00
|
|
|
t = self._get_exit_for_signal(trade, row, exit_, current_time)
|
2023-01-21 17:00:57 +00:00
|
|
|
if t:
|
|
|
|
return t
|
|
|
|
return None
|
2021-08-09 13:45:02 +00:00
|
|
|
|
2023-12-11 18:12:08 +00:00
|
|
|
def _run_funding_fees(self, trade: LocalTrade, current_time: datetime, force: bool = False):
|
2023-12-10 10:52:40 +00:00
|
|
|
"""
|
|
|
|
Calculate funding fees if necessary and add them to the trade.
|
|
|
|
"""
|
|
|
|
if self.trading_mode == TradingMode.FUTURES:
|
2023-12-10 12:23:40 +00:00
|
|
|
|
|
|
|
if (
|
|
|
|
force
|
|
|
|
or (current_time.timestamp() % self.funding_fee_timeframe_secs) == 0
|
|
|
|
):
|
|
|
|
# Funding fee interval.
|
|
|
|
trade.set_funding_fees(
|
|
|
|
self.exchange.calculate_funding_fees(
|
|
|
|
self.futures_data[trade.pair],
|
|
|
|
amount=trade.amount,
|
|
|
|
is_short=trade.is_short,
|
|
|
|
open_date=trade.date_last_filled_utc,
|
|
|
|
close_date=current_time
|
|
|
|
)
|
2023-12-10 10:52:40 +00:00
|
|
|
)
|
|
|
|
|
2022-03-19 14:42:29 +00:00
|
|
|
def get_valid_price_and_stake(
|
2022-05-24 17:13:35 +00:00
|
|
|
self, pair: str, row: Tuple, propose_rate: float, stake_amount: float,
|
2022-04-04 14:51:57 +00:00
|
|
|
direction: LongShort, current_time: datetime, entry_tag: Optional[str],
|
2022-10-13 17:30:59 +00:00
|
|
|
trade: Optional[LocalTrade], order_type: str, price_precision: Optional[float]
|
2022-03-19 14:42:29 +00:00
|
|
|
) -> Tuple[float, float, float, float]:
|
2021-12-24 10:38:43 +00:00
|
|
|
|
2022-01-22 13:11:33 +00:00
|
|
|
if order_type == 'limit':
|
2022-10-13 17:30:59 +00:00
|
|
|
new_rate = strategy_safe_wrapper(self.strategy.custom_entry_price,
|
|
|
|
default_retval=propose_rate)(
|
2023-09-16 17:58:59 +00:00
|
|
|
pair=pair,
|
|
|
|
trade=trade, # type: ignore[arg-type]
|
|
|
|
current_time=current_time,
|
2022-04-04 14:48:27 +00:00
|
|
|
proposed_rate=propose_rate, entry_tag=entry_tag,
|
|
|
|
side=direction,
|
|
|
|
) # default value is the open rate
|
2022-05-01 09:10:11 +00:00
|
|
|
# We can't place orders higher than current high (otherwise it'd be a stop limit entry)
|
2022-02-08 18:10:29 +00:00
|
|
|
# which freqtrade does not support in live.
|
2023-10-19 20:06:21 +00:00
|
|
|
if new_rate is not None and new_rate != propose_rate:
|
2022-10-13 17:30:59 +00:00
|
|
|
propose_rate = price_to_precision(new_rate, price_precision,
|
|
|
|
self.precision_mode)
|
2022-03-09 12:00:06 +00:00
|
|
|
if direction == "short":
|
|
|
|
propose_rate = max(propose_rate, row[LOW_IDX])
|
|
|
|
else:
|
|
|
|
propose_rate = min(propose_rate, row[HIGH_IDX])
|
2021-12-03 16:44:53 +00:00
|
|
|
|
2022-04-30 10:38:17 +00:00
|
|
|
pos_adjust = trade is not None
|
2022-03-19 14:42:29 +00:00
|
|
|
leverage = trade.leverage if trade else 1.0
|
2022-03-19 14:27:06 +00:00
|
|
|
if not pos_adjust:
|
2022-03-19 14:42:29 +00:00
|
|
|
try:
|
2023-12-12 21:43:46 +00:00
|
|
|
stake_amount = self.wallets.get_trade_stake_amount(
|
|
|
|
pair, self.strategy.max_open_trades, update=False)
|
2022-03-19 14:42:29 +00:00
|
|
|
except DependencyException:
|
|
|
|
return 0, 0, 0, 0
|
|
|
|
|
2022-03-19 14:27:06 +00:00
|
|
|
max_leverage = self.exchange.get_max_leverage(pair, stake_amount)
|
|
|
|
leverage = strategy_safe_wrapper(self.strategy.leverage, default_retval=1.0)(
|
|
|
|
pair=pair,
|
|
|
|
current_time=current_time,
|
|
|
|
current_rate=row[OPEN_IDX],
|
|
|
|
proposed_leverage=1.0,
|
|
|
|
max_leverage=max_leverage,
|
2022-06-05 08:21:06 +00:00
|
|
|
side=direction, entry_tag=entry_tag,
|
2023-03-26 14:17:10 +00:00
|
|
|
) if self.trading_mode != TradingMode.SPOT else 1.0
|
2022-03-19 14:27:06 +00:00
|
|
|
# Cap leverage between 1.0 and max_leverage.
|
|
|
|
leverage = min(max(leverage, 1.0), max_leverage)
|
|
|
|
|
|
|
|
min_stake_amount = self.exchange.get_min_pair_stake_amount(
|
2023-03-14 07:14:01 +00:00
|
|
|
pair, propose_rate, -0.05 if not pos_adjust else 0.0, leverage=leverage) or 0
|
2022-03-19 14:27:06 +00:00
|
|
|
max_stake_amount = self.exchange.get_max_pair_stake_amount(
|
|
|
|
pair, propose_rate, leverage=leverage)
|
2022-02-02 02:39:22 +00:00
|
|
|
stake_available = self.wallets.get_available_stake_amount()
|
2021-07-11 09:20:31 +00:00
|
|
|
|
2022-01-13 18:24:21 +00:00
|
|
|
if not pos_adjust:
|
2021-12-24 10:38:43 +00:00
|
|
|
stake_amount = strategy_safe_wrapper(self.strategy.custom_stake_amount,
|
|
|
|
default_retval=stake_amount)(
|
2022-01-22 16:25:21 +00:00
|
|
|
pair=pair, current_time=current_time, current_rate=propose_rate,
|
2022-02-02 02:39:22 +00:00
|
|
|
proposed_stake=stake_amount, min_stake=min_stake_amount,
|
|
|
|
max_stake=min(stake_available, max_stake_amount),
|
2022-07-08 17:44:20 +00:00
|
|
|
leverage=leverage, entry_tag=entry_tag, side=direction)
|
2021-12-24 10:38:43 +00:00
|
|
|
|
2022-03-19 14:42:29 +00:00
|
|
|
stake_amount_val = self.wallets.validate_stake_amount(
|
2022-02-04 01:48:54 +00:00
|
|
|
pair=pair,
|
|
|
|
stake_amount=stake_amount,
|
|
|
|
min_stake_amount=min_stake_amount,
|
|
|
|
max_stake_amount=max_stake_amount,
|
2022-12-23 15:09:35 +00:00
|
|
|
trade_amount=trade.stake_amount if trade else None
|
2022-02-04 01:48:54 +00:00
|
|
|
)
|
2021-07-11 09:20:31 +00:00
|
|
|
|
2022-03-19 14:42:29 +00:00
|
|
|
return propose_rate, stake_amount_val, leverage, min_stake_amount
|
|
|
|
|
2022-04-04 14:51:57 +00:00
|
|
|
def _enter_trade(self, pair: str, row: Tuple, direction: LongShort,
|
2022-03-19 14:42:29 +00:00
|
|
|
stake_amount: Optional[float] = None,
|
2022-04-16 12:07:18 +00:00
|
|
|
trade: Optional[LocalTrade] = None,
|
2022-04-30 10:38:17 +00:00
|
|
|
requested_rate: Optional[float] = None,
|
2024-01-28 22:05:19 +00:00
|
|
|
requested_stake: Optional[float] = None,
|
|
|
|
entry_tag1: Optional[str] = None
|
|
|
|
) -> Optional[LocalTrade]:
|
2023-01-29 10:02:31 +00:00
|
|
|
"""
|
|
|
|
:param trade: Trade to adjust - initial entry if None
|
|
|
|
:param requested_rate: Adjusted entry rate
|
|
|
|
:param requested_stake: Stake amount for adjusted orders (`adjust_entry_price`).
|
|
|
|
"""
|
2022-03-19 14:42:29 +00:00
|
|
|
|
|
|
|
current_time = row[DATE_IDX].to_pydatetime()
|
2024-01-28 22:05:19 +00:00
|
|
|
entry_tag = entry_tag1 or (row[ENTER_TAG_IDX] if len(row) >= ENTER_TAG_IDX + 1 else None)
|
2022-03-19 14:42:29 +00:00
|
|
|
# let's call the custom entry price, using the open price as default price
|
|
|
|
order_type = self.strategy.order_types['entry']
|
2022-04-30 10:38:17 +00:00
|
|
|
pos_adjust = trade is not None and requested_rate is None
|
2022-03-19 14:42:29 +00:00
|
|
|
|
2022-05-24 17:13:35 +00:00
|
|
|
stake_amount_ = stake_amount or (trade.stake_amount if trade else 0.0)
|
2022-10-13 17:30:59 +00:00
|
|
|
precision_price = self.exchange.get_precision_price(pair)
|
|
|
|
|
2022-03-19 14:42:29 +00:00
|
|
|
propose_rate, stake_amount, leverage, min_stake_amount = self.get_valid_price_and_stake(
|
2022-05-24 17:13:35 +00:00
|
|
|
pair, row, row[OPEN_IDX], stake_amount_, direction, current_time, entry_tag, trade,
|
2022-10-13 17:30:59 +00:00
|
|
|
order_type, precision_price,
|
2022-03-19 14:42:29 +00:00
|
|
|
)
|
|
|
|
|
2022-04-30 10:38:17 +00:00
|
|
|
# replace proposed rate if another rate was requested
|
|
|
|
propose_rate = requested_rate if requested_rate else propose_rate
|
|
|
|
stake_amount = requested_stake if requested_stake else stake_amount
|
|
|
|
|
2021-07-11 09:20:31 +00:00
|
|
|
if not stake_amount:
|
2021-12-24 10:38:43 +00:00
|
|
|
# In case of pos adjust, still return the original trade
|
|
|
|
# If not pos adjust, trade is None
|
|
|
|
return trade
|
2022-03-07 06:09:01 +00:00
|
|
|
time_in_force = self.strategy.order_time_in_force['entry']
|
2021-03-23 08:09:41 +00:00
|
|
|
|
2023-01-29 11:47:16 +00:00
|
|
|
if stake_amount and (not min_stake_amount or stake_amount >= min_stake_amount):
|
2022-01-22 13:11:33 +00:00
|
|
|
self.order_id_counter += 1
|
2022-04-09 14:42:18 +00:00
|
|
|
base_currency = self.exchange.get_pair_base_currency(pair)
|
2022-08-20 09:24:49 +00:00
|
|
|
amount_p = (stake_amount / propose_rate) * leverage
|
2022-08-24 18:36:08 +00:00
|
|
|
|
2022-08-22 18:48:02 +00:00
|
|
|
contract_size = self.exchange.get_contract_size(pair)
|
|
|
|
precision_amount = self.exchange.get_precision_amount(pair)
|
2022-08-25 05:08:22 +00:00
|
|
|
amount = amount_to_contract_precision(amount_p, precision_amount, self.precision_mode,
|
|
|
|
contract_size)
|
2022-08-20 09:24:49 +00:00
|
|
|
# Backcalculate actual stake amount.
|
|
|
|
stake_amount = amount * propose_rate / leverage
|
|
|
|
|
2022-09-16 13:24:20 +00:00
|
|
|
if not pos_adjust:
|
|
|
|
# Confirm trade entry:
|
|
|
|
if not strategy_safe_wrapper(
|
|
|
|
self.strategy.confirm_trade_entry, default_retval=True)(
|
|
|
|
pair=pair, order_type=order_type, amount=amount, rate=propose_rate,
|
|
|
|
time_in_force=time_in_force, current_time=current_time,
|
|
|
|
entry_tag=entry_tag, side=direction):
|
|
|
|
return trade
|
|
|
|
|
2022-02-27 18:08:39 +00:00
|
|
|
is_short = (direction == 'short')
|
2022-02-28 18:45:15 +00:00
|
|
|
# Necessary for Margin trading. Disabled until support is enabled.
|
|
|
|
# interest_rate = self.exchange.get_interest_rate()
|
|
|
|
|
2021-12-24 10:38:43 +00:00
|
|
|
if trade is None:
|
|
|
|
# Enter trade
|
2022-01-22 13:11:33 +00:00
|
|
|
self.trade_id_counter += 1
|
2021-12-24 10:38:43 +00:00
|
|
|
trade = LocalTrade(
|
2022-01-22 13:11:33 +00:00
|
|
|
id=self.trade_id_counter,
|
2021-12-24 10:38:43 +00:00
|
|
|
pair=pair,
|
2022-04-09 14:42:18 +00:00
|
|
|
base_currency=base_currency,
|
|
|
|
stake_currency=self.config['stake_currency'],
|
2021-12-24 10:38:43 +00:00
|
|
|
open_rate=propose_rate,
|
2022-01-19 09:42:24 +00:00
|
|
|
open_rate_requested=propose_rate,
|
2022-01-22 16:25:21 +00:00
|
|
|
open_date=current_time,
|
2021-12-24 10:38:43 +00:00
|
|
|
stake_amount=stake_amount,
|
2022-01-08 14:07:20 +00:00
|
|
|
amount=amount,
|
2022-01-22 13:11:33 +00:00
|
|
|
amount_requested=amount,
|
2021-12-24 10:38:43 +00:00
|
|
|
fee_open=self.fee,
|
|
|
|
fee_close=self.fee,
|
|
|
|
is_open=True,
|
2022-01-29 13:19:30 +00:00
|
|
|
enter_tag=entry_tag,
|
2022-01-22 16:25:21 +00:00
|
|
|
exchange=self._exchange_name,
|
2022-02-26 16:07:00 +00:00
|
|
|
is_short=is_short,
|
2022-01-08 14:07:20 +00:00
|
|
|
trading_mode=self.trading_mode,
|
2022-01-22 16:25:21 +00:00
|
|
|
leverage=leverage,
|
2022-02-28 18:45:15 +00:00
|
|
|
# interest_rate=interest_rate,
|
2022-08-22 18:48:02 +00:00
|
|
|
amount_precision=precision_amount,
|
2022-08-24 18:36:08 +00:00
|
|
|
price_precision=precision_price,
|
2022-08-16 08:59:43 +00:00
|
|
|
precision_mode=self.precision_mode,
|
2022-08-22 18:48:02 +00:00
|
|
|
contract_size=contract_size,
|
2022-02-26 16:07:00 +00:00
|
|
|
orders=[],
|
2021-12-24 10:38:43 +00:00
|
|
|
)
|
2022-01-30 14:27:18 +00:00
|
|
|
|
2022-01-22 14:03:12 +00:00
|
|
|
trade.adjust_stop_loss(trade.open_rate, self.strategy.stoploss, initial=True)
|
2021-12-24 10:38:43 +00:00
|
|
|
|
2021-12-09 21:21:35 +00:00
|
|
|
order = Order(
|
2022-01-22 13:11:33 +00:00
|
|
|
id=self.order_id_counter,
|
|
|
|
ft_trade_id=trade.id,
|
2022-01-22 14:44:33 +00:00
|
|
|
ft_is_open=True,
|
2021-12-09 21:21:35 +00:00
|
|
|
ft_pair=trade.pair,
|
2022-01-22 13:11:33 +00:00
|
|
|
order_id=str(self.order_id_counter),
|
2021-12-09 21:21:35 +00:00
|
|
|
symbol=trade.pair,
|
2022-04-06 01:02:13 +00:00
|
|
|
ft_order_side=trade.entry_side,
|
|
|
|
side=trade.entry_side,
|
2022-01-22 13:11:33 +00:00
|
|
|
order_type=order_type,
|
|
|
|
status="open",
|
2022-01-23 18:58:25 +00:00
|
|
|
order_date=current_time,
|
|
|
|
order_filled_date=current_time,
|
|
|
|
order_update_date=current_time,
|
2023-02-19 19:23:04 +00:00
|
|
|
ft_price=propose_rate,
|
2021-12-24 10:38:43 +00:00
|
|
|
price=propose_rate,
|
|
|
|
average=propose_rate,
|
|
|
|
amount=amount,
|
2022-01-22 13:11:33 +00:00
|
|
|
filled=0,
|
|
|
|
remaining=amount,
|
2023-07-15 07:02:17 +00:00
|
|
|
cost=amount * propose_rate + trade.fee_open,
|
2024-01-28 18:56:30 +00:00
|
|
|
ft_order_tag=entry_tag,
|
2021-02-10 19:37:55 +00:00
|
|
|
)
|
2023-07-15 08:27:10 +00:00
|
|
|
order._trade_bt = trade
|
2022-05-15 13:30:57 +00:00
|
|
|
trade.orders.append(order)
|
2023-08-25 05:08:24 +00:00
|
|
|
self._try_close_open_order(order, trade, current_time, row)
|
2022-01-22 13:11:33 +00:00
|
|
|
trade.recalc_trade_from_orders()
|
2021-12-24 10:38:43 +00:00
|
|
|
|
|
|
|
return trade
|
2021-02-10 19:37:55 +00:00
|
|
|
|
2021-02-20 18:29:04 +00:00
|
|
|
def handle_left_open(self, open_trades: Dict[str, List[LocalTrade]],
|
2022-10-15 10:07:22 +00:00
|
|
|
data: Dict[str, List[Tuple]]) -> None:
|
2020-10-08 18:10:00 +00:00
|
|
|
"""
|
|
|
|
Handling of left open trades at the end of backtesting
|
|
|
|
"""
|
|
|
|
for pair in open_trades.keys():
|
2022-10-16 11:34:58 +00:00
|
|
|
for trade in list(open_trades[pair]):
|
2023-08-25 05:08:24 +00:00
|
|
|
if trade.has_open_orders and trade.nr_of_successful_entries == 0:
|
2022-10-15 10:08:58 +00:00
|
|
|
# Ignore trade if entry-order did not fill yet
|
|
|
|
continue
|
|
|
|
exit_row = data[pair][-1]
|
2024-01-28 18:56:30 +00:00
|
|
|
self._exit_trade(trade, exit_row, exit_row[OPEN_IDX], trade.amount,
|
|
|
|
ExitType.FORCE_EXIT.value)
|
2022-10-15 10:08:58 +00:00
|
|
|
trade.orders[-1].close_bt_order(exit_row[DATE_IDX].to_pydatetime(), trade)
|
|
|
|
|
|
|
|
trade.close_date = exit_row[DATE_IDX].to_pydatetime()
|
|
|
|
trade.exit_reason = ExitType.FORCE_EXIT.value
|
|
|
|
trade.close(exit_row[OPEN_IDX], show_msg=False)
|
|
|
|
LocalTrade.close_bt_trade(trade)
|
2018-02-09 07:35:38 +00:00
|
|
|
|
2023-01-08 11:39:39 +00:00
|
|
|
def trade_slot_available(self, open_trade_count: int) -> bool:
|
2021-05-23 07:46:51 +00:00
|
|
|
# Always allow trades when max_open_trades is enabled.
|
2023-12-11 18:52:49 +00:00
|
|
|
max_open_trades: IntOrInf = self.strategy.max_open_trades
|
2021-05-23 07:46:51 +00:00
|
|
|
if max_open_trades <= 0 or open_trade_count < max_open_trades:
|
2021-05-23 07:36:02 +00:00
|
|
|
return True
|
|
|
|
# Rejected trade
|
|
|
|
self.rejected_trades += 1
|
|
|
|
return False
|
|
|
|
|
2022-04-04 14:51:57 +00:00
|
|
|
def check_for_trade_entry(self, row) -> Optional[LongShort]:
|
2021-08-24 04:54:55 +00:00
|
|
|
enter_long = row[LONG_IDX] == 1
|
|
|
|
exit_long = row[ELONG_IDX] == 1
|
2021-09-22 18:36:03 +00:00
|
|
|
enter_short = self._can_short and row[SHORT_IDX] == 1
|
|
|
|
exit_short = self._can_short and row[ESHORT_IDX] == 1
|
2021-08-23 19:15:56 +00:00
|
|
|
|
|
|
|
if enter_long == 1 and not any([exit_long, enter_short]):
|
|
|
|
# Long
|
|
|
|
return 'long'
|
|
|
|
if enter_short == 1 and not any([exit_short, enter_long]):
|
|
|
|
# Short
|
|
|
|
return 'short'
|
|
|
|
return None
|
|
|
|
|
2022-10-18 04:39:55 +00:00
|
|
|
def run_protections(self, pair: str, current_time: datetime, side: LongShort):
|
|
|
|
if self.enable_protections:
|
2022-04-23 17:58:20 +00:00
|
|
|
self.protections.stop_per_pair(pair, current_time, side)
|
|
|
|
self.protections.global_stop(current_time, side)
|
2022-01-30 16:17:03 +00:00
|
|
|
|
2022-05-31 18:08:34 +00:00
|
|
|
def manage_open_orders(self, trade: LocalTrade, current_time: datetime, row: Tuple) -> bool:
|
2022-04-16 12:07:18 +00:00
|
|
|
"""
|
2022-05-01 15:06:20 +00:00
|
|
|
Check if any open order needs to be cancelled or replaced.
|
2022-04-30 10:38:17 +00:00
|
|
|
Returns True if the trade should be deleted.
|
2022-04-16 12:07:18 +00:00
|
|
|
"""
|
|
|
|
for order in [o for o in trade.orders if o.ft_is_open]:
|
2022-05-31 18:08:34 +00:00
|
|
|
oc = self.check_order_cancel(trade, order, current_time)
|
|
|
|
if oc:
|
2022-05-01 15:06:20 +00:00
|
|
|
# delete trade due to order timeout
|
|
|
|
return True
|
2022-05-31 18:08:34 +00:00
|
|
|
elif oc is None and self.check_order_replace(trade, order, current_time, row):
|
2022-05-01 15:06:20 +00:00
|
|
|
# delete trade due to user request
|
2022-05-16 22:41:01 +00:00
|
|
|
self.canceled_trade_entries += 1
|
2022-05-01 15:06:20 +00:00
|
|
|
return True
|
|
|
|
# default maintain trade
|
2022-04-30 10:38:17 +00:00
|
|
|
return False
|
2022-04-16 12:07:18 +00:00
|
|
|
|
2022-05-31 18:08:34 +00:00
|
|
|
def check_order_cancel(
|
|
|
|
self, trade: LocalTrade, order: Order, current_time: datetime) -> Optional[bool]:
|
2022-01-30 16:39:23 +00:00
|
|
|
"""
|
2022-05-01 15:06:20 +00:00
|
|
|
Check if current analyzed order has to be canceled.
|
2022-05-31 18:08:34 +00:00
|
|
|
Returns True if the trade should be Deleted (initial order was canceled),
|
|
|
|
False if it's Canceled
|
|
|
|
None if the order is still active.
|
2022-01-30 16:39:23 +00:00
|
|
|
"""
|
2022-05-07 12:53:51 +00:00
|
|
|
timedout = self.strategy.ft_check_timed_out(
|
|
|
|
trade, # type: ignore[arg-type]
|
|
|
|
order, current_time)
|
2022-05-01 15:06:20 +00:00
|
|
|
if timedout:
|
|
|
|
if order.side == trade.entry_side:
|
|
|
|
self.timedout_entry_orders += 1
|
|
|
|
if trade.nr_of_successful_entries == 0:
|
|
|
|
# Remove trade due to entry timeout expiration.
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
# Close additional entry order
|
2022-01-30 16:39:23 +00:00
|
|
|
del trade.orders[trade.orders.index(order)]
|
2022-05-31 18:08:34 +00:00
|
|
|
return False
|
2022-05-01 15:06:20 +00:00
|
|
|
if order.side == trade.exit_side:
|
|
|
|
self.timedout_exit_orders += 1
|
|
|
|
# Close exit order and retry exiting on next signal.
|
|
|
|
del trade.orders[trade.orders.index(order)]
|
2022-05-31 18:08:34 +00:00
|
|
|
return False
|
|
|
|
return None
|
2022-01-30 16:39:23 +00:00
|
|
|
|
2022-05-01 15:06:20 +00:00
|
|
|
def check_order_replace(self, trade: LocalTrade, order: Order, current_time,
|
|
|
|
row: Tuple) -> bool:
|
|
|
|
"""
|
|
|
|
Check if current analyzed entry order has to be replaced and do so.
|
|
|
|
If user requested cancellation and there are no filled orders in the trade will
|
|
|
|
instruct caller to delete the trade.
|
|
|
|
Returns True if the trade should be deleted.
|
|
|
|
"""
|
|
|
|
# only check on new candles for open entry orders
|
|
|
|
if order.side == trade.entry_side and current_time > order.order_date_utc:
|
|
|
|
requested_rate = strategy_safe_wrapper(self.strategy.adjust_entry_price,
|
2023-02-19 19:23:04 +00:00
|
|
|
default_retval=order.ft_price)(
|
2022-05-07 12:53:51 +00:00
|
|
|
trade=trade, # type: ignore[arg-type]
|
|
|
|
order=order, pair=trade.pair, current_time=current_time,
|
2023-02-19 19:23:04 +00:00
|
|
|
proposed_rate=row[OPEN_IDX], current_order_rate=order.ft_price,
|
2022-05-01 15:06:20 +00:00
|
|
|
entry_tag=trade.enter_tag, side=trade.trade_direction
|
|
|
|
) # default value is current order price
|
|
|
|
|
|
|
|
# cancel existing order whenever a new rate is requested (or None)
|
2023-02-19 19:23:04 +00:00
|
|
|
if requested_rate == order.ft_price:
|
2022-05-01 15:06:20 +00:00
|
|
|
# assumption: there can't be multiple open entry orders at any given time
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
del trade.orders[trade.orders.index(order)]
|
2022-05-17 11:07:02 +00:00
|
|
|
self.canceled_entry_orders += 1
|
2022-05-01 15:06:20 +00:00
|
|
|
|
2022-05-07 15:47:37 +00:00
|
|
|
# place new order if result was not None
|
2022-05-01 15:06:20 +00:00
|
|
|
if requested_rate:
|
|
|
|
self._enter_trade(pair=trade.pair, row=row, trade=trade,
|
|
|
|
requested_rate=requested_rate,
|
2023-02-19 19:23:04 +00:00
|
|
|
requested_stake=(
|
|
|
|
order.safe_remaining * order.ft_price / trade.leverage),
|
2022-05-01 15:06:20 +00:00
|
|
|
direction='short' if trade.is_short else 'long')
|
2023-03-26 14:46:41 +00:00
|
|
|
# Delete trade if no successful entries happened (if placing the new order failed)
|
2023-08-25 05:08:24 +00:00
|
|
|
if not trade.has_open_orders and trade.nr_of_successful_entries == 0:
|
2023-03-26 14:46:41 +00:00
|
|
|
return True
|
2022-05-17 11:07:02 +00:00
|
|
|
self.replaced_entry_orders += 1
|
2022-05-01 15:06:20 +00:00
|
|
|
else:
|
|
|
|
# assumption: there can't be multiple open entry orders at any given time
|
|
|
|
return (trade.nr_of_successful_entries == 0)
|
|
|
|
return False
|
|
|
|
|
2022-01-30 19:00:11 +00:00
|
|
|
def validate_row(
|
|
|
|
self, data: Dict, pair: str, row_index: int, current_time: datetime) -> Optional[Tuple]:
|
|
|
|
try:
|
|
|
|
# Row is treated as "current incomplete candle".
|
2022-04-24 12:28:15 +00:00
|
|
|
# entry / exit signals are shifted by 1 to compensate for this.
|
2022-01-30 19:00:11 +00:00
|
|
|
row = data[pair][row_index]
|
|
|
|
except IndexError:
|
|
|
|
# missing Data for one pair at the end.
|
|
|
|
# Warnings for this are shown during data loading
|
|
|
|
return None
|
|
|
|
|
|
|
|
# Waits until the time-counter reaches the start of the data for this pair.
|
|
|
|
if row[DATE_IDX] > current_time:
|
|
|
|
return None
|
|
|
|
return row
|
|
|
|
|
2022-12-05 15:34:31 +00:00
|
|
|
def _collate_rejected(self, pair, row):
|
|
|
|
"""
|
2022-12-08 18:47:09 +00:00
|
|
|
Temporarily store rejected signal information for downstream use in backtesting_analysis
|
2022-12-05 15:34:31 +00:00
|
|
|
"""
|
|
|
|
# It could be fun to enable hyperopt mode to write
|
|
|
|
# a loss function to reduce rejected signals
|
|
|
|
if (self.config.get('export', 'none') == 'signals' and
|
|
|
|
self.dataprovider.runmode == RunMode.BACKTEST):
|
|
|
|
if pair not in self.rejected_dict:
|
|
|
|
self.rejected_dict[pair] = []
|
|
|
|
self.rejected_dict[pair].append([row[DATE_IDX], row[ENTER_TAG_IDX]])
|
|
|
|
|
2022-10-16 15:12:44 +00:00
|
|
|
def backtest_loop(
|
|
|
|
self, row: Tuple, pair: str, current_time: datetime, end_date: datetime,
|
2023-01-11 17:55:57 +00:00
|
|
|
open_trade_count_start: int, trade_dir: Optional[LongShort],
|
2023-01-05 09:14:58 +00:00
|
|
|
is_first: bool = True) -> int:
|
2022-10-16 15:12:44 +00:00
|
|
|
"""
|
|
|
|
NOTE: This method is used by Hyperopt at each iteration. Please keep it optimized.
|
|
|
|
|
2022-10-16 15:14:23 +00:00
|
|
|
Backtesting processing for one candle/pair.
|
2022-10-16 15:12:44 +00:00
|
|
|
"""
|
|
|
|
for t in list(LocalTrade.bt_trades_open_pp[pair]):
|
|
|
|
# 1. Manage currently open orders of active trades
|
|
|
|
if self.manage_open_orders(t, current_time, row):
|
|
|
|
# Close trade
|
|
|
|
open_trade_count_start -= 1
|
|
|
|
LocalTrade.remove_bt_trade(t)
|
|
|
|
self.wallets.update()
|
|
|
|
|
|
|
|
# 2. Process entries.
|
|
|
|
# without positionstacking, we can only have one open trade per pair.
|
|
|
|
# max_open_trades must be respected
|
|
|
|
# don't open on the last row
|
2022-11-05 19:03:20 +00:00
|
|
|
# We only open trades on the main candle, not on detail candles
|
2022-10-16 15:12:44 +00:00
|
|
|
if (
|
2022-10-16 15:14:23 +00:00
|
|
|
(self._position_stacking or len(LocalTrade.bt_trades_open_pp[pair]) == 0)
|
2022-10-22 14:22:55 +00:00
|
|
|
and is_first
|
2022-10-16 15:12:44 +00:00
|
|
|
and current_time != end_date
|
|
|
|
and trade_dir is not None
|
|
|
|
and not PairLocks.is_pair_locked(pair, row[DATE_IDX], trade_dir)
|
|
|
|
):
|
2023-03-19 14:00:20 +00:00
|
|
|
if (self.trade_slot_available(open_trade_count_start)):
|
2022-12-05 15:34:31 +00:00
|
|
|
trade = self._enter_trade(pair, row, trade_dir)
|
|
|
|
if trade:
|
|
|
|
# TODO: hacky workaround to avoid opening > max_open_trades
|
|
|
|
# This emulates previous behavior - not sure if this is correct
|
|
|
|
# Prevents entering if the trade-slot was freed in this candle
|
|
|
|
open_trade_count_start += 1
|
|
|
|
# logger.debug(f"{pair} - Emulate creation of new trade: {trade}.")
|
|
|
|
LocalTrade.add_bt_trade(trade)
|
|
|
|
self.wallets.update()
|
|
|
|
else:
|
|
|
|
self._collate_rejected(pair, row)
|
2022-10-16 15:12:44 +00:00
|
|
|
|
|
|
|
for trade in list(LocalTrade.bt_trades_open_pp[pair]):
|
|
|
|
# 3. Process entry orders.
|
|
|
|
order = trade.select_order(trade.entry_side, is_open=True)
|
2023-08-13 14:47:26 +00:00
|
|
|
if self._try_close_open_order(order, trade, current_time, row):
|
2022-10-16 15:12:44 +00:00
|
|
|
self.wallets.update()
|
|
|
|
|
2023-08-13 14:16:13 +00:00
|
|
|
# 4. Create exit orders (if any)
|
2023-08-25 05:08:24 +00:00
|
|
|
if not trade.has_open_orders:
|
2023-10-11 17:45:00 +00:00
|
|
|
self._check_trade_exit(trade, row, current_time) # Place exit order if necessary
|
2022-10-16 15:12:44 +00:00
|
|
|
|
2023-08-13 14:16:13 +00:00
|
|
|
# 5. Process exit orders.
|
2022-10-16 15:12:44 +00:00
|
|
|
order = trade.select_order(trade.exit_side, is_open=True)
|
2023-08-14 05:51:25 +00:00
|
|
|
if order and self._try_close_open_order(order, trade, current_time, row):
|
2022-10-16 15:12:44 +00:00
|
|
|
sub_trade = order.safe_amount_after_fee != trade.amount
|
|
|
|
if sub_trade:
|
|
|
|
trade.recalc_trade_from_orders()
|
|
|
|
else:
|
|
|
|
trade.close_date = current_time
|
2023-02-19 19:23:04 +00:00
|
|
|
trade.close(order.ft_price, show_msg=False)
|
2022-10-16 15:12:44 +00:00
|
|
|
|
|
|
|
# logger.debug(f"{pair} - Backtesting exit {trade}")
|
|
|
|
LocalTrade.close_bt_trade(trade)
|
|
|
|
self.wallets.update()
|
2022-10-18 04:39:55 +00:00
|
|
|
self.run_protections(pair, current_time, trade.trade_direction)
|
2022-10-16 15:12:44 +00:00
|
|
|
return open_trade_count_start
|
|
|
|
|
2022-10-16 15:14:23 +00:00
|
|
|
def backtest(self, processed: Dict,
|
2023-01-08 11:39:39 +00:00
|
|
|
start_date: datetime, end_date: datetime) -> Dict[str, Any]:
|
2018-02-09 07:35:38 +00:00
|
|
|
"""
|
2019-12-13 23:12:16 +00:00
|
|
|
Implement backtesting functionality
|
2018-02-09 07:35:38 +00:00
|
|
|
|
|
|
|
NOTE: This method is used by Hyperopt at each iteration. Please keep it optimized.
|
|
|
|
Of course try to not have ugly code. By some accessor are sometime slower than functions.
|
2019-12-13 23:12:16 +00:00
|
|
|
Avoid extensive logging in this method and functions it calls.
|
|
|
|
|
2021-12-29 10:09:01 +00:00
|
|
|
:param processed: a processed dictionary with format {pair, data}, which gets cleared to
|
|
|
|
optimize memory usage!
|
2019-12-13 23:12:16 +00:00
|
|
|
:param start_date: backtesting timerange start datetime
|
|
|
|
:param end_date: backtesting timerange end datetime
|
|
|
|
:return: DataFrame with trades (results of backtesting)
|
2018-02-09 07:35:38 +00:00
|
|
|
"""
|
2022-10-18 04:39:55 +00:00
|
|
|
self.prepare_backtest(self.enable_protections)
|
2022-02-10 18:40:36 +00:00
|
|
|
# Ensure wallets are uptodate (important for --strategy-list)
|
|
|
|
self.wallets.update()
|
2020-03-08 10:35:31 +00:00
|
|
|
# Use dict of lists with data for performance
|
|
|
|
# (looping lists is a lot faster than pandas DataFrames)
|
|
|
|
data: Dict = self._get_ohlcv_as_lists(processed)
|
2018-10-15 20:02:23 +00:00
|
|
|
|
2019-04-04 18:23:10 +00:00
|
|
|
# Indexes per pair, so some pairs are allowed to have a missing start.
|
2021-04-24 17:15:09 +00:00
|
|
|
indexes: Dict = defaultdict(int)
|
2024-01-24 16:57:15 +00:00
|
|
|
current_time = start_date + self.timeframe_td
|
2019-03-20 17:38:10 +00:00
|
|
|
|
2021-03-21 14:56:36 +00:00
|
|
|
self.progress.init_step(BacktestState.BACKTEST, int(
|
2024-01-24 16:57:15 +00:00
|
|
|
(end_date - start_date) / self.timeframe_td))
|
2019-04-04 18:23:10 +00:00
|
|
|
# Loop timerange and get candle for each pair at that point in time
|
2022-01-22 13:11:33 +00:00
|
|
|
while current_time <= end_date:
|
2022-10-16 11:34:58 +00:00
|
|
|
open_trade_count_start = LocalTrade.bt_open_open_trade_count
|
2021-07-06 04:28:47 +00:00
|
|
|
self.check_abort()
|
2023-03-26 09:21:18 +00:00
|
|
|
strategy_safe_wrapper(self.strategy.bot_loop_start, supress_error=True)(
|
|
|
|
current_time=current_time)
|
2020-03-08 10:35:31 +00:00
|
|
|
for i, pair in enumerate(data):
|
2021-05-08 13:06:19 +00:00
|
|
|
row_index = indexes[pair]
|
2022-01-30 19:00:11 +00:00
|
|
|
row = self.validate_row(data, pair, row_index, current_time)
|
|
|
|
if not row:
|
2019-03-20 17:38:10 +00:00
|
|
|
continue
|
2021-05-08 13:06:19 +00:00
|
|
|
|
|
|
|
row_index += 1
|
|
|
|
indexes[pair] = row_index
|
2023-08-15 17:43:04 +00:00
|
|
|
self.dataprovider._set_dataframe_max_index(self.required_startup + row_index)
|
|
|
|
self.dataprovider._set_dataframe_max_date(current_time)
|
2022-10-22 14:22:55 +00:00
|
|
|
current_detail_time: datetime = row[DATE_IDX].to_pydatetime()
|
2023-01-05 09:14:58 +00:00
|
|
|
trade_dir: Optional[LongShort] = self.check_for_trade_entry(row)
|
|
|
|
|
|
|
|
if (
|
|
|
|
(trade_dir is not None or len(LocalTrade.bt_trades_open_pp[pair]) > 0)
|
|
|
|
and self.timeframe_detail and pair in self.detail_data
|
|
|
|
):
|
|
|
|
# Spread out into detail timeframe.
|
|
|
|
# Should only happen when we are either in a trade for this pair
|
|
|
|
# or when we got the signal for a new trade.
|
2024-01-24 16:57:15 +00:00
|
|
|
exit_candle_end = current_detail_time + self.timeframe_td
|
2022-10-22 14:22:55 +00:00
|
|
|
|
|
|
|
detail_data = self.detail_data[pair]
|
|
|
|
detail_data = detail_data.loc[
|
|
|
|
(detail_data['date'] >= current_detail_time) &
|
|
|
|
(detail_data['date'] < exit_candle_end)
|
|
|
|
].copy()
|
|
|
|
if len(detail_data) == 0:
|
|
|
|
# Fall back to "regular" data if no detail data was found for this candle
|
|
|
|
open_trade_count_start = self.backtest_loop(
|
2023-01-08 11:39:39 +00:00
|
|
|
row, pair, current_time, end_date,
|
2023-01-05 09:14:58 +00:00
|
|
|
open_trade_count_start, trade_dir)
|
2023-01-04 17:08:45 +00:00
|
|
|
continue
|
2022-10-22 14:22:55 +00:00
|
|
|
detail_data.loc[:, 'enter_long'] = row[LONG_IDX]
|
|
|
|
detail_data.loc[:, 'exit_long'] = row[ELONG_IDX]
|
|
|
|
detail_data.loc[:, 'enter_short'] = row[SHORT_IDX]
|
|
|
|
detail_data.loc[:, 'exit_short'] = row[ESHORT_IDX]
|
|
|
|
detail_data.loc[:, 'enter_tag'] = row[ENTER_TAG_IDX]
|
|
|
|
detail_data.loc[:, 'exit_tag'] = row[EXIT_TAG_IDX]
|
|
|
|
is_first = True
|
2022-11-04 06:07:56 +00:00
|
|
|
current_time_det = current_time
|
2022-10-22 14:22:55 +00:00
|
|
|
for det_row in detail_data[HEADERS].values.tolist():
|
2023-08-15 15:48:07 +00:00
|
|
|
self.dataprovider._set_dataframe_max_date(current_time_det)
|
2022-10-22 14:22:55 +00:00
|
|
|
open_trade_count_start = self.backtest_loop(
|
2023-01-08 11:39:39 +00:00
|
|
|
det_row, pair, current_time_det, end_date,
|
2023-01-05 09:14:58 +00:00
|
|
|
open_trade_count_start, trade_dir, is_first)
|
2024-02-02 06:03:44 +00:00
|
|
|
current_time_det += self.timeframe_detail_td
|
2022-10-22 14:22:55 +00:00
|
|
|
is_first = False
|
|
|
|
else:
|
2023-08-15 15:48:07 +00:00
|
|
|
self.dataprovider._set_dataframe_max_date(current_time)
|
2022-10-22 14:22:55 +00:00
|
|
|
open_trade_count_start = self.backtest_loop(
|
2023-01-11 17:55:57 +00:00
|
|
|
row, pair, current_time, end_date,
|
2023-01-05 09:14:58 +00:00
|
|
|
open_trade_count_start, trade_dir)
|
2022-01-22 13:11:33 +00:00
|
|
|
|
2020-10-07 18:59:05 +00:00
|
|
|
# Move time one configured time_interval ahead.
|
2021-03-21 14:56:36 +00:00
|
|
|
self.progress.increment()
|
2024-01-24 16:57:15 +00:00
|
|
|
current_time += self.timeframe_td
|
2019-04-04 17:44:03 +00:00
|
|
|
|
2022-10-16 11:34:58 +00:00
|
|
|
self.handle_left_open(LocalTrade.bt_trades_open_pp, data=data)
|
2021-02-17 19:07:27 +00:00
|
|
|
self.wallets.update()
|
2018-06-09 19:44:20 +00:00
|
|
|
|
2022-10-15 10:07:22 +00:00
|
|
|
results = trade_list_to_dataframe(LocalTrade.trades)
|
2021-04-30 05:31:57 +00:00
|
|
|
return {
|
|
|
|
'results': results,
|
|
|
|
'config': self.strategy.config,
|
|
|
|
'locks': PairLocks.get_all_locks(),
|
2021-05-23 07:46:51 +00:00
|
|
|
'rejected_signals': self.rejected_trades,
|
2022-02-07 17:49:30 +00:00
|
|
|
'timedout_entry_orders': self.timedout_entry_orders,
|
|
|
|
'timedout_exit_orders': self.timedout_exit_orders,
|
2022-05-16 22:41:01 +00:00
|
|
|
'canceled_trade_entries': self.canceled_trade_entries,
|
2022-05-17 11:07:02 +00:00
|
|
|
'canceled_entry_orders': self.canceled_entry_orders,
|
|
|
|
'replaced_entry_orders': self.replaced_entry_orders,
|
2021-04-30 05:31:57 +00:00
|
|
|
'final_balance': self.wallets.get_total(self.strategy.config['stake_currency']),
|
|
|
|
}
|
2018-02-09 07:35:38 +00:00
|
|
|
|
2021-08-09 12:53:18 +00:00
|
|
|
def backtest_one_strategy(self, strat: IStrategy, data: Dict[str, DataFrame],
|
|
|
|
timerange: TimeRange):
|
2021-03-21 14:56:36 +00:00
|
|
|
self.progress.init_step(BacktestState.ANALYZE, 0)
|
2023-04-28 14:09:09 +00:00
|
|
|
strategy_name = strat.get_strategy_name()
|
|
|
|
logger.info(f"Running backtesting for Strategy {strategy_name}")
|
2021-01-13 06:47:03 +00:00
|
|
|
backtest_start_time = datetime.now(timezone.utc)
|
2020-09-18 05:44:11 +00:00
|
|
|
self._set_strategy(strat)
|
|
|
|
|
|
|
|
# Use max_open_trades in backtesting, except --disable-max-market-positions is set
|
2023-01-08 11:39:39 +00:00
|
|
|
if not self.config.get('use_max_market_positions', True):
|
2020-09-18 05:44:11 +00:00
|
|
|
logger.info(
|
|
|
|
'Ignoring max_open_trades (--disable-max-market-positions was used) ...')
|
2023-01-15 10:44:10 +00:00
|
|
|
self.strategy.max_open_trades = float('inf')
|
|
|
|
self.config.update({'max_open_trades': self.strategy.max_open_trades})
|
2020-09-18 05:44:11 +00:00
|
|
|
|
|
|
|
# need to reprocess data every time to populate signals
|
2021-08-09 12:53:18 +00:00
|
|
|
preprocessed = self.strategy.advise_all_indicators(data)
|
2020-09-18 05:44:11 +00:00
|
|
|
|
|
|
|
# Trim startup period from analyzed dataframe
|
2023-07-21 18:19:58 +00:00
|
|
|
# This only used to determine if trimming would result in an empty dataframe
|
2021-07-13 17:36:15 +00:00
|
|
|
preprocessed_tmp = trim_dataframes(preprocessed, timerange, self.required_startup)
|
Fix exception when few pairs with no data do not result in aborting backtest.
Exception is triggered by backtesting 20210301-20210501 range with BAKE/USDT pair (binance). Pair data starts on 2021-04-30 12:00:00 and after adjusting for startup candles pair dataframe is empty.
Solution: Since there are other pairs with enough data - skip pairs with no data and issue a warning.
Exception:
```
Traceback (most recent call last):
File "/home/rk/src/freqtrade/freqtrade/main.py", line 37, in main
return_code = args['func'](args)
File "/home/rk/src/freqtrade/freqtrade/commands/optimize_commands.py", line 53, in start_backtesting
backtesting.start()
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 502, in start
min_date, max_date = self.backtest_one_strategy(strat, data, timerange)
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 474, in backtest_one_strategy
results = self.backtest(
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 365, in backtest
data: Dict = self._get_ohlcv_as_lists(processed)
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 199, in _get_ohlcv_as_lists
pair_data.loc[:, 'buy'] = 0 # cleanup from previous run
File "/home/rk/src/freqtrade/venv/lib/python3.9/site-packages/pandas/core/indexing.py", line 692, in __setitem__
iloc._setitem_with_indexer(indexer, value, self.name)
File "/home/rk/src/freqtrade/venv/lib/python3.9/site-packages/pandas/core/indexing.py", line 1587, in _setitem_with_indexer
raise ValueError(
ValueError: cannot set a frame with no defined index and a scalar
```
2021-05-13 06:47:28 +00:00
|
|
|
|
2021-07-13 17:36:15 +00:00
|
|
|
if not preprocessed_tmp:
|
2021-05-06 18:49:48 +00:00
|
|
|
raise OperationalException(
|
Fix exception when few pairs with no data do not result in aborting backtest.
Exception is triggered by backtesting 20210301-20210501 range with BAKE/USDT pair (binance). Pair data starts on 2021-04-30 12:00:00 and after adjusting for startup candles pair dataframe is empty.
Solution: Since there are other pairs with enough data - skip pairs with no data and issue a warning.
Exception:
```
Traceback (most recent call last):
File "/home/rk/src/freqtrade/freqtrade/main.py", line 37, in main
return_code = args['func'](args)
File "/home/rk/src/freqtrade/freqtrade/commands/optimize_commands.py", line 53, in start_backtesting
backtesting.start()
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 502, in start
min_date, max_date = self.backtest_one_strategy(strat, data, timerange)
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 474, in backtest_one_strategy
results = self.backtest(
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 365, in backtest
data: Dict = self._get_ohlcv_as_lists(processed)
File "/home/rk/src/freqtrade/freqtrade/optimize/backtesting.py", line 199, in _get_ohlcv_as_lists
pair_data.loc[:, 'buy'] = 0 # cleanup from previous run
File "/home/rk/src/freqtrade/venv/lib/python3.9/site-packages/pandas/core/indexing.py", line 692, in __setitem__
iloc._setitem_with_indexer(indexer, value, self.name)
File "/home/rk/src/freqtrade/venv/lib/python3.9/site-packages/pandas/core/indexing.py", line 1587, in _setitem_with_indexer
raise ValueError(
ValueError: cannot set a frame with no defined index and a scalar
```
2021-05-13 06:47:28 +00:00
|
|
|
"No data left after adjusting for startup candles.")
|
|
|
|
|
2021-07-13 17:36:15 +00:00
|
|
|
# Use preprocessed_tmp for date generation (the trimmed dataframe).
|
2022-04-24 12:28:15 +00:00
|
|
|
# Backtesting will re-trim the dataframes after entry/exit signal generation.
|
2021-07-13 17:36:15 +00:00
|
|
|
min_date, max_date = history.get_timerange(preprocessed_tmp)
|
2020-09-18 05:44:11 +00:00
|
|
|
logger.info(f'Backtesting with data from {min_date.strftime(DATETIME_PRINT_FORMAT)} '
|
|
|
|
f'up to {max_date.strftime(DATETIME_PRINT_FORMAT)} '
|
2021-05-06 18:49:48 +00:00
|
|
|
f'({(max_date - min_date).days} days).')
|
2020-09-18 05:44:11 +00:00
|
|
|
# Execute backtest and store results
|
|
|
|
results = self.backtest(
|
|
|
|
processed=preprocessed,
|
2021-05-06 17:34:10 +00:00
|
|
|
start_date=min_date,
|
|
|
|
end_date=max_date,
|
2020-09-18 05:44:11 +00:00
|
|
|
)
|
2021-01-13 06:47:03 +00:00
|
|
|
backtest_end_time = datetime.now(timezone.utc)
|
2021-04-30 05:31:57 +00:00
|
|
|
results.update({
|
2023-04-28 14:09:09 +00:00
|
|
|
'run_id': self.run_ids.get(strategy_name, ''),
|
2021-01-13 06:47:03 +00:00
|
|
|
'backtest_start_time': int(backtest_start_time.timestamp()),
|
|
|
|
'backtest_end_time': int(backtest_end_time.timestamp()),
|
2021-04-30 05:31:57 +00:00
|
|
|
})
|
2023-04-28 14:09:09 +00:00
|
|
|
self.all_results[strategy_name] = results
|
2021-04-30 05:31:57 +00:00
|
|
|
|
2022-04-23 06:51:52 +00:00
|
|
|
if (self.config.get('export', 'none') == 'signals' and
|
|
|
|
self.dataprovider.runmode == RunMode.BACKTEST):
|
2023-04-28 14:14:16 +00:00
|
|
|
self.processed_dfs[strategy_name] = generate_trade_signal_candles(
|
2023-04-28 14:09:09 +00:00
|
|
|
preprocessed_tmp, results)
|
2023-04-28 14:14:16 +00:00
|
|
|
self.rejected_df[strategy_name] = generate_rejected_signals(
|
2023-04-28 14:09:09 +00:00
|
|
|
preprocessed_tmp, self.rejected_dict)
|
2022-03-16 12:16:24 +00:00
|
|
|
|
2020-09-18 05:44:11 +00:00
|
|
|
return min_date, max_date
|
|
|
|
|
2022-01-18 09:00:51 +00:00
|
|
|
def _get_min_cached_backtest_date(self):
|
|
|
|
min_backtest_date = None
|
|
|
|
backtest_cache_age = self.config.get('backtest_cache', constants.BACKTEST_CACHE_DEFAULT)
|
2022-11-10 17:26:14 +00:00
|
|
|
if self.timerange.stopts == 0 or self.timerange.stopdt > datetime.now(tz=timezone.utc):
|
2022-01-18 09:00:51 +00:00
|
|
|
logger.warning('Backtest result caching disabled due to use of open-ended timerange.')
|
|
|
|
elif backtest_cache_age == 'day':
|
|
|
|
min_backtest_date = datetime.now(tz=timezone.utc) - timedelta(days=1)
|
|
|
|
elif backtest_cache_age == 'week':
|
|
|
|
min_backtest_date = datetime.now(tz=timezone.utc) - timedelta(weeks=1)
|
|
|
|
elif backtest_cache_age == 'month':
|
|
|
|
min_backtest_date = datetime.now(tz=timezone.utc) - timedelta(weeks=4)
|
|
|
|
return min_backtest_date
|
|
|
|
|
2022-01-20 05:44:41 +00:00
|
|
|
def load_prior_backtest(self):
|
2022-01-18 09:00:51 +00:00
|
|
|
self.run_ids = {
|
2022-01-06 09:53:11 +00:00
|
|
|
strategy.get_strategy_name(): get_strategy_run_id(strategy)
|
|
|
|
for strategy in self.strategylist
|
|
|
|
}
|
|
|
|
|
|
|
|
# Load previous result that will be updated incrementally.
|
2022-01-16 17:01:05 +00:00
|
|
|
# This can be circumvented in certain instances in combination with downloading more data
|
2022-01-18 09:00:51 +00:00
|
|
|
min_backtest_date = self._get_min_cached_backtest_date()
|
|
|
|
if min_backtest_date is not None:
|
2022-01-06 09:53:11 +00:00
|
|
|
self.results = find_existing_backtest_stats(
|
2022-01-18 09:00:51 +00:00
|
|
|
self.config['user_data_dir'] / 'backtest_results', self.run_ids, min_backtest_date)
|
2022-01-06 09:53:11 +00:00
|
|
|
|
2018-02-09 07:35:38 +00:00
|
|
|
def start(self) -> None:
|
|
|
|
"""
|
2019-12-13 23:12:16 +00:00
|
|
|
Run backtesting end-to-end
|
2018-02-09 07:35:38 +00:00
|
|
|
:return: None
|
|
|
|
"""
|
2018-08-19 17:39:22 +00:00
|
|
|
data: Dict[str, Any] = {}
|
2019-12-13 23:12:16 +00:00
|
|
|
|
2019-10-23 18:13:43 +00:00
|
|
|
data, timerange = self.load_bt_data()
|
2021-08-14 13:34:43 +00:00
|
|
|
self.load_bt_data_detail()
|
2021-04-23 17:22:41 +00:00
|
|
|
logger.info("Dataload complete. Calculating indicators")
|
2019-06-14 17:37:54 +00:00
|
|
|
|
2022-01-20 05:44:41 +00:00
|
|
|
self.load_prior_backtest()
|
|
|
|
|
2018-07-28 05:41:38 +00:00
|
|
|
for strat in self.strategylist:
|
2022-01-06 09:53:11 +00:00
|
|
|
if self.results and strat.get_strategy_name() in self.results['strategy']:
|
|
|
|
# When previous result hash matches - reuse that result and skip backtesting.
|
|
|
|
logger.info(f'Reusing result of previous backtest for {strat.get_strategy_name()}')
|
|
|
|
continue
|
2020-09-18 05:44:11 +00:00
|
|
|
min_date, max_date = self.backtest_one_strategy(strat, data, timerange)
|
2020-12-31 18:46:54 +00:00
|
|
|
|
2022-01-06 09:53:11 +00:00
|
|
|
# Update old results with new ones.
|
|
|
|
if len(self.all_results) > 0:
|
|
|
|
results = generate_backtest_stats(
|
|
|
|
data, self.all_results, min_date=min_date, max_date=max_date)
|
|
|
|
if self.results:
|
|
|
|
self.results['metadata'].update(results['metadata'])
|
|
|
|
self.results['strategy'].update(results['strategy'])
|
|
|
|
self.results['strategy_comparison'].extend(results['strategy_comparison'])
|
|
|
|
else:
|
|
|
|
self.results = results
|
2022-06-15 04:53:52 +00:00
|
|
|
dt_appendix = datetime.now().strftime("%Y-%m-%d_%H-%M-%S")
|
2022-04-23 07:23:53 +00:00
|
|
|
if self.config.get('export', 'none') in ('trades', 'signals'):
|
2022-06-15 04:53:52 +00:00
|
|
|
store_backtest_stats(self.config['exportfilename'], self.results, dt_appendix)
|
2020-09-26 12:55:12 +00:00
|
|
|
|
2022-04-23 06:51:52 +00:00
|
|
|
if (self.config.get('export', 'none') == 'signals' and
|
|
|
|
self.dataprovider.runmode == RunMode.BACKTEST):
|
2023-03-19 13:56:41 +00:00
|
|
|
store_backtest_analysis_results(
|
|
|
|
self.config['exportfilename'], self.processed_dfs, self.rejected_df,
|
|
|
|
dt_appendix)
|
2022-12-05 15:34:31 +00:00
|
|
|
|
2022-01-06 09:53:11 +00:00
|
|
|
# Results may be mixed up now. Sort them so they follow --strategy-list order.
|
|
|
|
if 'strategy_list' in self.config and len(self.results) > 0:
|
|
|
|
self.results['strategy_comparison'] = sorted(
|
|
|
|
self.results['strategy_comparison'],
|
|
|
|
key=lambda c: self.config['strategy_list'].index(c['key']))
|
|
|
|
self.results['strategy'] = dict(
|
|
|
|
sorted(self.results['strategy'].items(),
|
|
|
|
key=lambda kv: self.config['strategy_list'].index(kv[0])))
|
|
|
|
|
|
|
|
if len(self.strategylist) > 0:
|
2021-03-01 07:57:57 +00:00
|
|
|
# Show backtest results
|
2020-12-31 18:46:54 +00:00
|
|
|
show_backtest_results(self.config, self.results)
|