freqtrade_origin/tests/optimize/test_recursive_analysis.py

189 lines
6.7 KiB
Python
Raw Normal View History

2023-09-21 05:00:17 +00:00
# pragma pylint: disable=missing-docstring, W0212, line-too-long, C0103, unused-argument
from copy import deepcopy
from pathlib import Path
from unittest.mock import MagicMock, PropertyMock
import pytest
from freqtrade.commands.optimize_commands import start_recursive_analysis
from freqtrade.data.history import get_timerange
from freqtrade.exceptions import OperationalException
2023-10-01 06:00:09 +00:00
from freqtrade.optimize.analysis.recursive import RecursiveAnalysis
from freqtrade.optimize.analysis.recursive_helpers import RecursiveAnalysisSubFunctions
2024-04-27 17:52:48 +00:00
from tests.conftest import EXMS, get_args, log_has_re, patch_exchange
2023-09-21 05:00:17 +00:00
@pytest.fixture
def recursive_conf(default_conf_usdt, tmp_path):
2024-05-12 13:59:04 +00:00
default_conf_usdt["user_data_dir"] = tmp_path
default_conf_usdt["timerange"] = "20220101-20220501"
default_conf_usdt["strategy_path"] = str(Path(__file__).parent.parent / "strategy/strats")
default_conf_usdt["strategy"] = "strategy_test_v3_recursive_issue"
default_conf_usdt["pairs"] = ["UNITTEST/USDT"]
default_conf_usdt["startup_candle"] = [100]
2023-09-21 05:00:17 +00:00
return default_conf_usdt
2023-09-21 08:51:37 +00:00
def test_start_recursive_analysis(mocker):
single_mock = MagicMock()
text_table_mock = MagicMock()
mocker.patch.multiple(
2024-05-12 13:59:04 +00:00
"freqtrade.optimize.analysis.recursive_helpers.RecursiveAnalysisSubFunctions",
2023-09-21 08:51:37 +00:00
initialize_single_recursive_analysis=single_mock,
text_table_recursive_analysis_instances=text_table_mock,
)
args = [
"recursive-analysis",
"--strategy",
"strategy_test_v3_recursive_issue",
"--strategy-path",
str(Path(__file__).parent.parent / "strategy/strats"),
"--pairs",
"UNITTEST/BTC",
"--timerange",
2024-05-12 13:59:04 +00:00
"20220101-20220201",
2023-09-21 08:51:37 +00:00
]
pargs = get_args(args)
2024-05-12 13:59:04 +00:00
pargs["config"] = None
2023-09-21 08:51:37 +00:00
start_recursive_analysis(pargs)
assert single_mock.call_count == 1
assert text_table_mock.call_count == 1
single_mock.reset_mock()
# Missing timerange
args = [
"recursive-analysis",
"--strategy",
"strategy_test_v3_with_recursive_bias",
"--strategy-path",
str(Path(__file__).parent.parent / "strategy/strats"),
"--pairs",
2024-05-12 13:59:04 +00:00
"UNITTEST/BTC",
2023-09-21 08:51:37 +00:00
]
pargs = get_args(args)
2024-05-12 13:59:04 +00:00
pargs["config"] = None
with pytest.raises(OperationalException, match=r"Please set a timerange\..*"):
2023-09-21 08:51:37 +00:00
start_recursive_analysis(pargs)
def test_recursive_helper_no_strategy_defined(recursive_conf):
conf = deepcopy(recursive_conf)
2024-05-12 13:59:04 +00:00
conf["pairs"] = ["UNITTEST/USDT"]
del conf["strategy"]
with pytest.raises(OperationalException, match=r"No Strategy specified"):
2023-09-21 08:51:37 +00:00
RecursiveAnalysisSubFunctions.start(conf)
2023-09-21 05:21:54 +00:00
2023-09-21 08:47:51 +00:00
def test_recursive_helper_start(recursive_conf, mocker) -> None:
single_mock = MagicMock()
text_table_mock = MagicMock()
mocker.patch.multiple(
2024-05-12 13:59:04 +00:00
"freqtrade.optimize.analysis.recursive_helpers.RecursiveAnalysisSubFunctions",
2023-09-21 08:47:51 +00:00
initialize_single_recursive_analysis=single_mock,
text_table_recursive_analysis_instances=text_table_mock,
)
RecursiveAnalysisSubFunctions.start(recursive_conf)
assert single_mock.call_count == 1
assert text_table_mock.call_count == 1
single_mock.reset_mock()
text_table_mock.reset_mock()
def test_recursive_helper_text_table_recursive_analysis_instances(recursive_conf):
dict_diff = dict()
2024-05-12 13:59:04 +00:00
dict_diff["rsi"] = {}
dict_diff["rsi"][100] = "0.078%"
2023-09-21 08:47:51 +00:00
strategy_obj = {
2024-05-12 13:59:04 +00:00
"name": "strategy_test_v3_recursive_issue",
"location": Path(recursive_conf["strategy_path"], f"{recursive_conf['strategy']}.py"),
2023-09-21 08:47:51 +00:00
}
instance = RecursiveAnalysis(recursive_conf, strategy_obj)
instance.dict_recursive = dict_diff
2024-05-12 13:59:04 +00:00
_table, _headers, data = RecursiveAnalysisSubFunctions.text_table_recursive_analysis_instances(
[instance]
)
2023-09-21 08:47:51 +00:00
# check row contents for a try that has too few signals
2024-05-12 13:59:04 +00:00
assert data[0][0] == "rsi"
assert data[0][1] == "0.078%"
2023-09-21 08:47:51 +00:00
assert len(data[0]) == 2
# now check when there is no issue
dict_diff = dict()
instance = RecursiveAnalysis(recursive_conf, strategy_obj)
instance.dict_recursive = dict_diff
2024-05-12 13:59:04 +00:00
_table, _headers, data = RecursiveAnalysisSubFunctions.text_table_recursive_analysis_instances(
[instance]
)
2023-09-21 08:47:51 +00:00
assert len(data) == 0
def test_initialize_single_recursive_analysis(recursive_conf, mocker, caplog):
2024-05-12 13:59:04 +00:00
mocker.patch("freqtrade.data.history.get_timerange", get_timerange)
2023-09-21 08:47:51 +00:00
patch_exchange(mocker)
2024-05-12 13:59:04 +00:00
mocker.patch(
"freqtrade.plugins.pairlistmanager.PairListManager.whitelist",
PropertyMock(return_value=["UNITTEST/BTC"]),
)
recursive_conf["pairs"] = ["UNITTEST/BTC"]
2023-09-21 05:21:54 +00:00
2024-05-12 13:59:04 +00:00
recursive_conf["timeframe"] = "5m"
recursive_conf["timerange"] = "20180119-20180122"
start_mock = mocker.patch("freqtrade.optimize.analysis.recursive.RecursiveAnalysis.start")
2023-09-21 08:47:51 +00:00
strategy_obj = {
2024-05-12 13:59:04 +00:00
"name": "strategy_test_v3_recursive_issue",
"location": Path(recursive_conf["strategy_path"], f"{recursive_conf['strategy']}.py"),
2023-09-21 08:47:51 +00:00
}
2023-09-21 05:21:54 +00:00
2023-09-21 08:47:51 +00:00
instance = RecursiveAnalysisSubFunctions.initialize_single_recursive_analysis(
2024-05-12 13:59:04 +00:00
recursive_conf, strategy_obj
)
2023-09-21 08:47:51 +00:00
assert log_has_re(r"Recursive test of .* started\.", caplog)
assert start_mock.call_count == 1
2023-09-21 05:21:54 +00:00
2024-05-12 13:59:04 +00:00
assert instance.strategy_obj["name"] == "strategy_test_v3_recursive_issue"
2023-09-21 05:00:17 +00:00
2024-05-12 13:59:04 +00:00
@pytest.mark.parametrize("scenario", ["no_bias", "bias1", "bias2"])
2023-09-29 04:58:16 +00:00
def test_recursive_biased_strategy(recursive_conf, mocker, caplog, scenario) -> None:
2023-09-21 05:00:17 +00:00
patch_exchange(mocker)
2024-05-12 13:59:04 +00:00
mocker.patch(f"{EXMS}.get_fee", return_value=0.0)
mocker.patch("freqtrade.data.history.get_timerange", get_timerange)
mocker.patch(
"freqtrade.plugins.pairlistmanager.PairListManager.whitelist",
PropertyMock(return_value=["UNITTEST/BTC"]),
)
recursive_conf["pairs"] = ["UNITTEST/BTC"]
2023-09-21 05:00:17 +00:00
2024-05-12 13:59:04 +00:00
recursive_conf["timeframe"] = "5m"
recursive_conf["timerange"] = "20180119-20180122"
recursive_conf["startup_candle"] = [100]
2023-09-21 05:00:17 +00:00
# Patch scenario Parameter to allow for easy selection
2024-05-12 13:59:04 +00:00
mocker.patch(
"freqtrade.strategy.hyper.HyperStrategyMixin.load_params_from_file",
return_value={"params": {"buy": {"scenario": scenario}}},
)
strategy_obj = {"name": "strategy_test_v3_recursive_issue"}
2023-09-21 05:21:54 +00:00
instance = RecursiveAnalysis(recursive_conf, strategy_obj)
2023-09-21 05:00:17 +00:00
instance.start()
# Assert init correct
assert log_has_re(f"Strategy Parameter: scenario = {scenario}", caplog)
if scenario == "bias2":
assert log_has_re("=> found lookahead in indicator rsi", caplog)
2024-05-12 13:59:04 +00:00
diff_pct = abs(float(instance.dict_recursive["rsi"][100].replace("%", "")))
2023-09-29 05:09:48 +00:00
# check non-biased strategy
if scenario == "no_bias":
assert diff_pct < 0.01
# check biased strategy
elif scenario in ("bias1", "bias2"):
assert diff_pct >= 0.01