2017-12-26 08:08:10 +00:00
|
|
|
# pragma pylint: disable=missing-docstring,W0212,C0103
|
2019-11-06 18:33:15 +00:00
|
|
|
import locale
|
2020-04-28 19:56:19 +00:00
|
|
|
import logging
|
2020-09-28 17:43:15 +00:00
|
|
|
from copy import deepcopy
|
2019-04-26 17:51:24 +00:00
|
|
|
from datetime import datetime
|
2019-09-08 07:54:15 +00:00
|
|
|
from pathlib import Path
|
2020-02-10 09:35:48 +00:00
|
|
|
from typing import Dict, List
|
2019-07-21 13:56:44 +00:00
|
|
|
from unittest.mock import MagicMock, PropertyMock
|
2018-03-17 21:44:47 +00:00
|
|
|
|
2018-02-08 19:49:43 +00:00
|
|
|
import pandas as pd
|
2018-05-02 20:49:55 +00:00
|
|
|
import pytest
|
2019-07-15 19:54:41 +00:00
|
|
|
from arrow import Arrow
|
|
|
|
from filelock import Timeout
|
2018-03-15 22:37:34 +00:00
|
|
|
|
2020-03-10 09:42:31 +00:00
|
|
|
from freqtrade import constants
|
2020-09-28 17:43:15 +00:00
|
|
|
from freqtrade.commands.optimize_commands import setup_optimize_configuration, start_hyperopt
|
2019-12-27 09:25:38 +00:00
|
|
|
from freqtrade.data.history import load_data
|
2020-03-10 09:42:31 +00:00
|
|
|
from freqtrade.exceptions import DependencyException, OperationalException
|
2019-07-16 04:50:25 +00:00
|
|
|
from freqtrade.optimize.default_hyperopt_loss import DefaultHyperOptLoss
|
2019-07-21 14:07:06 +00:00
|
|
|
from freqtrade.optimize.hyperopt import Hyperopt
|
2020-09-28 17:43:15 +00:00
|
|
|
from freqtrade.resolvers.hyperopt_resolver import HyperOptLossResolver, HyperOptResolver
|
2019-04-22 18:24:45 +00:00
|
|
|
from freqtrade.state import RunMode
|
2019-07-15 19:54:41 +00:00
|
|
|
from freqtrade.strategy.interface import SellType
|
2019-09-08 07:54:15 +00:00
|
|
|
from tests.conftest import (get_args, log_has, log_has_re, patch_exchange,
|
|
|
|
patched_configuration_load_config_file)
|
2018-01-07 11:54:00 +00:00
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
from .hyperopts.default_hyperopt import DefaultHyperOpt
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(scope='function')
|
|
|
|
def hyperopt_conf(default_conf):
|
|
|
|
hyperconf = deepcopy(default_conf)
|
|
|
|
hyperconf.update({
|
|
|
|
'hyperopt': 'DefaultHyperOpt',
|
|
|
|
'hyperopt_path': str(Path(__file__).parent / 'hyperopts'),
|
|
|
|
'epochs': 1,
|
|
|
|
'timerange': None,
|
|
|
|
'spaces': ['default'],
|
|
|
|
'hyperopt_jobs': 1,
|
|
|
|
})
|
|
|
|
return hyperconf
|
|
|
|
|
2018-04-08 09:11:14 +00:00
|
|
|
|
|
|
|
@pytest.fixture(scope='function')
|
2020-08-06 06:51:01 +00:00
|
|
|
def hyperopt(hyperopt_conf, mocker):
|
|
|
|
|
2018-07-30 10:26:54 +00:00
|
|
|
patch_exchange(mocker)
|
2020-08-06 06:51:01 +00:00
|
|
|
return Hyperopt(hyperopt_conf)
|
2017-12-26 08:08:10 +00:00
|
|
|
|
|
|
|
|
2019-07-15 19:54:41 +00:00
|
|
|
@pytest.fixture(scope='function')
|
|
|
|
def hyperopt_results():
|
|
|
|
return pd.DataFrame(
|
|
|
|
{
|
|
|
|
'pair': ['ETH/BTC', 'ETH/BTC', 'ETH/BTC'],
|
2019-09-16 12:05:39 +00:00
|
|
|
'profit_percent': [-0.1, 0.2, 0.3],
|
|
|
|
'profit_abs': [-0.2, 0.4, 0.6],
|
2019-07-15 19:54:41 +00:00
|
|
|
'trade_duration': [10, 30, 10],
|
2020-02-06 05:49:08 +00:00
|
|
|
'sell_reason': [SellType.STOP_LOSS, SellType.ROI, SellType.ROI],
|
2020-06-26 07:21:28 +00:00
|
|
|
'close_date':
|
2020-02-06 05:49:08 +00:00
|
|
|
[
|
|
|
|
datetime(2019, 1, 1, 9, 26, 3, 478039),
|
|
|
|
datetime(2019, 2, 1, 9, 26, 3, 478039),
|
|
|
|
datetime(2019, 3, 1, 9, 26, 3, 478039)
|
|
|
|
]
|
2019-07-15 19:54:41 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2018-03-02 13:46:32 +00:00
|
|
|
# Functions for recurrent object patching
|
2020-04-28 19:56:19 +00:00
|
|
|
def create_results(mocker, hyperopt, testdatadir) -> List[Dict]:
|
2018-01-09 09:37:27 +00:00
|
|
|
"""
|
2020-04-28 19:56:19 +00:00
|
|
|
When creating results, mock the hyperopt so that *by default*
|
2018-01-09 09:37:27 +00:00
|
|
|
- we don't create any pickle'd files in the filesystem
|
|
|
|
- we might have a pickle'd file so make sure that we return
|
|
|
|
false when looking for it
|
|
|
|
"""
|
2020-04-28 19:56:19 +00:00
|
|
|
hyperopt.results_file = testdatadir / 'optimize/ut_results.pickle'
|
2018-03-02 13:46:32 +00:00
|
|
|
|
2019-07-21 13:56:44 +00:00
|
|
|
mocker.patch.object(Path, "is_file", MagicMock(return_value=False))
|
|
|
|
stat_mock = MagicMock()
|
|
|
|
stat_mock.st_size = PropertyMock(return_value=1)
|
|
|
|
mocker.patch.object(Path, "stat", MagicMock(return_value=False))
|
|
|
|
|
|
|
|
mocker.patch.object(Path, "unlink", MagicMock(return_value=True))
|
2018-07-03 19:51:48 +00:00
|
|
|
mocker.patch('freqtrade.optimize.hyperopt.dump', return_value=None)
|
2018-03-02 13:46:32 +00:00
|
|
|
|
2018-06-24 12:27:53 +00:00
|
|
|
return [{'loss': 1, 'result': 'foo', 'params': {}}]
|
2017-12-26 08:08:10 +00:00
|
|
|
|
|
|
|
|
2019-04-22 18:24:45 +00:00
|
|
|
def test_setup_hyperopt_configuration_without_arguments(mocker, default_conf, caplog) -> None:
|
2019-07-14 22:44:25 +00:00
|
|
|
patched_configuration_load_config_file(mocker, default_conf)
|
2019-04-22 18:24:45 +00:00
|
|
|
|
|
|
|
args = [
|
2019-09-16 04:44:07 +00:00
|
|
|
'hyperopt',
|
2019-04-22 18:24:45 +00:00
|
|
|
'--config', 'config.json',
|
2019-10-20 17:54:38 +00:00
|
|
|
'--hyperopt', 'DefaultHyperOpt',
|
2019-04-22 18:24:45 +00:00
|
|
|
]
|
|
|
|
|
2020-01-26 12:33:13 +00:00
|
|
|
config = setup_optimize_configuration(get_args(args), RunMode.HYPEROPT)
|
2019-04-22 18:24:45 +00:00
|
|
|
assert 'max_open_trades' in config
|
|
|
|
assert 'stake_currency' in config
|
|
|
|
assert 'stake_amount' in config
|
|
|
|
assert 'exchange' in config
|
|
|
|
assert 'pair_whitelist' in config['exchange']
|
|
|
|
assert 'datadir' in config
|
2019-08-11 18:16:52 +00:00
|
|
|
assert log_has('Using data directory: {} ...'.format(config['datadir']), caplog)
|
2020-06-01 18:33:26 +00:00
|
|
|
assert 'timeframe' in config
|
2019-08-11 18:16:52 +00:00
|
|
|
assert not log_has_re('Parameter -i/--ticker-interval detected .*', caplog)
|
2019-04-22 18:24:45 +00:00
|
|
|
|
|
|
|
assert 'position_stacking' not in config
|
2019-08-11 18:16:52 +00:00
|
|
|
assert not log_has('Parameter --enable-position-stacking detected ...', caplog)
|
2019-04-22 18:24:45 +00:00
|
|
|
|
|
|
|
assert 'timerange' not in config
|
|
|
|
assert 'runmode' in config
|
|
|
|
assert config['runmode'] == RunMode.HYPEROPT
|
|
|
|
|
|
|
|
|
|
|
|
def test_setup_hyperopt_configuration_with_arguments(mocker, default_conf, caplog) -> None:
|
2019-07-14 22:44:25 +00:00
|
|
|
patched_configuration_load_config_file(mocker, default_conf)
|
2019-07-11 22:41:09 +00:00
|
|
|
mocker.patch(
|
2019-07-12 00:26:27 +00:00
|
|
|
'freqtrade.configuration.configuration.create_datadir',
|
|
|
|
lambda c, x: x
|
2019-07-11 22:41:09 +00:00
|
|
|
)
|
2019-04-22 18:24:45 +00:00
|
|
|
|
|
|
|
args = [
|
2019-09-16 04:44:07 +00:00
|
|
|
'hyperopt',
|
2019-04-22 18:24:45 +00:00
|
|
|
'--config', 'config.json',
|
2019-10-20 17:54:38 +00:00
|
|
|
'--hyperopt', 'DefaultHyperOpt',
|
2019-04-22 18:24:45 +00:00
|
|
|
'--datadir', '/foo/bar',
|
2020-06-01 18:47:27 +00:00
|
|
|
'--timeframe', '1m',
|
2019-04-22 18:24:45 +00:00
|
|
|
'--timerange', ':100',
|
|
|
|
'--enable-position-stacking',
|
|
|
|
'--disable-max-market-positions',
|
|
|
|
'--epochs', '1000',
|
2019-11-07 22:55:14 +00:00
|
|
|
'--spaces', 'default',
|
2019-04-22 18:24:45 +00:00
|
|
|
'--print-all'
|
|
|
|
]
|
|
|
|
|
2020-01-26 12:33:13 +00:00
|
|
|
config = setup_optimize_configuration(get_args(args), RunMode.HYPEROPT)
|
2019-04-22 18:24:45 +00:00
|
|
|
assert 'max_open_trades' in config
|
|
|
|
assert 'stake_currency' in config
|
|
|
|
assert 'stake_amount' in config
|
|
|
|
assert 'exchange' in config
|
|
|
|
assert 'pair_whitelist' in config['exchange']
|
|
|
|
assert 'datadir' in config
|
|
|
|
assert config['runmode'] == RunMode.HYPEROPT
|
|
|
|
|
2019-08-11 18:16:52 +00:00
|
|
|
assert log_has('Using data directory: {} ...'.format(config['datadir']), caplog)
|
2020-06-01 18:33:26 +00:00
|
|
|
assert 'timeframe' in config
|
2020-06-01 18:47:27 +00:00
|
|
|
assert log_has('Parameter -i/--timeframe detected ... Using timeframe: 1m ...',
|
2019-08-11 18:16:52 +00:00
|
|
|
caplog)
|
2019-04-22 18:24:45 +00:00
|
|
|
|
|
|
|
assert 'position_stacking' in config
|
2019-08-11 18:16:52 +00:00
|
|
|
assert log_has('Parameter --enable-position-stacking detected ...', caplog)
|
2019-04-22 18:24:45 +00:00
|
|
|
|
|
|
|
assert 'use_max_market_positions' in config
|
2019-08-11 18:16:52 +00:00
|
|
|
assert log_has('Parameter --disable-max-market-positions detected ...', caplog)
|
|
|
|
assert log_has('max_open_trades set to unlimited ...', caplog)
|
2019-04-22 18:24:45 +00:00
|
|
|
|
|
|
|
assert 'timerange' in config
|
2019-08-11 18:16:52 +00:00
|
|
|
assert log_has('Parameter --timerange detected: {} ...'.format(config['timerange']), caplog)
|
2019-04-22 18:24:45 +00:00
|
|
|
|
|
|
|
assert 'epochs' in config
|
2019-04-24 19:12:23 +00:00
|
|
|
assert log_has('Parameter --epochs detected ... Will run Hyperopt with for 1000 epochs ...',
|
2019-08-11 18:16:52 +00:00
|
|
|
caplog)
|
2019-04-22 18:24:45 +00:00
|
|
|
|
|
|
|
assert 'spaces' in config
|
2019-08-11 18:16:52 +00:00
|
|
|
assert log_has('Parameter -s/--spaces detected: {}'.format(config['spaces']), caplog)
|
2019-04-22 18:24:45 +00:00
|
|
|
assert 'print_all' in config
|
2019-08-11 18:16:52 +00:00
|
|
|
assert log_has('Parameter --print-all detected ...', caplog)
|
2019-04-22 18:24:45 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_setup_hyperopt_configuration_unlimited_stake_amount(mocker, default_conf) -> None:
|
2020-03-10 09:42:31 +00:00
|
|
|
default_conf['stake_amount'] = constants.UNLIMITED_STAKE_AMOUNT
|
|
|
|
|
|
|
|
patched_configuration_load_config_file(mocker, default_conf)
|
|
|
|
|
|
|
|
args = [
|
|
|
|
'hyperopt',
|
|
|
|
'--config', 'config.json',
|
|
|
|
'--hyperopt', 'DefaultHyperOpt',
|
|
|
|
]
|
|
|
|
|
|
|
|
with pytest.raises(DependencyException, match=r'.`stake_amount`.*'):
|
|
|
|
setup_optimize_configuration(get_args(args), RunMode.HYPEROPT)
|
|
|
|
|
|
|
|
|
2019-01-06 18:38:32 +00:00
|
|
|
def test_hyperoptresolver(mocker, default_conf, caplog) -> None:
|
2019-07-11 21:39:42 +00:00
|
|
|
patched_configuration_load_config_file(mocker, default_conf)
|
2019-01-06 18:38:32 +00:00
|
|
|
|
2019-10-18 20:29:19 +00:00
|
|
|
hyperopt = DefaultHyperOpt
|
2019-11-02 13:17:15 +00:00
|
|
|
delattr(hyperopt, 'populate_indicators')
|
2019-10-18 20:29:19 +00:00
|
|
|
delattr(hyperopt, 'populate_buy_trend')
|
|
|
|
delattr(hyperopt, 'populate_sell_trend')
|
2019-01-06 18:38:32 +00:00
|
|
|
mocker.patch(
|
2019-12-24 12:54:46 +00:00
|
|
|
'freqtrade.resolvers.hyperopt_resolver.HyperOptResolver.load_object',
|
2019-10-18 20:29:19 +00:00
|
|
|
MagicMock(return_value=hyperopt(default_conf))
|
2019-01-06 18:38:32 +00:00
|
|
|
)
|
2019-10-20 17:54:38 +00:00
|
|
|
default_conf.update({'hyperopt': 'DefaultHyperOpt'})
|
2019-12-23 09:06:19 +00:00
|
|
|
x = HyperOptResolver.load_hyperopt(default_conf)
|
2019-11-02 13:17:15 +00:00
|
|
|
assert not hasattr(x, 'populate_indicators')
|
2019-01-06 18:38:32 +00:00
|
|
|
assert not hasattr(x, 'populate_buy_trend')
|
|
|
|
assert not hasattr(x, 'populate_sell_trend')
|
2019-11-02 13:17:15 +00:00
|
|
|
assert log_has("Hyperopt class does not provide populate_indicators() method. "
|
|
|
|
"Using populate_indicators from the strategy.", caplog)
|
2019-09-18 19:57:37 +00:00
|
|
|
assert log_has("Hyperopt class does not provide populate_sell_trend() method. "
|
|
|
|
"Using populate_sell_trend from the strategy.", caplog)
|
|
|
|
assert log_has("Hyperopt class does not provide populate_buy_trend() method. "
|
|
|
|
"Using populate_buy_trend from the strategy.", caplog)
|
2020-06-02 08:02:55 +00:00
|
|
|
assert hasattr(x, "ticker_interval") # DEPRECATED
|
|
|
|
assert hasattr(x, "timeframe")
|
2019-01-06 18:38:32 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_hyperoptresolver_wrongname(default_conf) -> None:
|
2019-07-28 13:19:17 +00:00
|
|
|
default_conf.update({'hyperopt': "NonExistingHyperoptClass"})
|
|
|
|
|
|
|
|
with pytest.raises(OperationalException, match=r'Impossible to load Hyperopt.*'):
|
2019-12-23 09:06:19 +00:00
|
|
|
HyperOptResolver.load_hyperopt(default_conf)
|
2019-10-10 01:37:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_hyperoptresolver_noname(default_conf):
|
|
|
|
default_conf['hyperopt'] = ''
|
|
|
|
with pytest.raises(OperationalException,
|
2019-10-14 17:42:28 +00:00
|
|
|
match="No Hyperopt set. Please use `--hyperopt` to specify "
|
2019-10-10 01:37:32 +00:00
|
|
|
"the Hyperopt class to use."):
|
2019-12-23 09:06:19 +00:00
|
|
|
HyperOptResolver.load_hyperopt(default_conf)
|
2019-07-28 13:19:17 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_hyperoptlossresolver(mocker, default_conf) -> None:
|
2019-07-16 04:50:25 +00:00
|
|
|
|
|
|
|
hl = DefaultHyperOptLoss
|
|
|
|
mocker.patch(
|
2019-12-24 12:54:46 +00:00
|
|
|
'freqtrade.resolvers.hyperopt_resolver.HyperOptLossResolver.load_object',
|
2019-07-16 04:50:25 +00:00
|
|
|
MagicMock(return_value=hl)
|
|
|
|
)
|
2019-12-23 09:09:08 +00:00
|
|
|
x = HyperOptLossResolver.load_hyperoptloss(default_conf)
|
2019-07-28 13:19:17 +00:00
|
|
|
assert hasattr(x, "hyperopt_loss_function")
|
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_hyperoptlossresolver_wrongname(default_conf) -> None:
|
2019-07-28 13:19:17 +00:00
|
|
|
default_conf.update({'hyperopt_loss': "NonExistingLossClass"})
|
|
|
|
|
|
|
|
with pytest.raises(OperationalException, match=r'Impossible to load HyperoptLoss.*'):
|
2019-12-23 09:09:08 +00:00
|
|
|
HyperOptLossResolver.load_hyperoptloss(default_conf)
|
2019-07-16 04:50:25 +00:00
|
|
|
|
|
|
|
|
2020-08-06 07:00:28 +00:00
|
|
|
def test_start_not_installed(mocker, default_conf, import_fails) -> None:
|
2019-09-24 12:39:28 +00:00
|
|
|
start_mock = MagicMock()
|
|
|
|
patched_configuration_load_config_file(mocker, default_conf)
|
|
|
|
|
|
|
|
mocker.patch('freqtrade.optimize.hyperopt.Hyperopt.start', start_mock)
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
|
|
|
args = [
|
|
|
|
'hyperopt',
|
2019-09-29 14:25:25 +00:00
|
|
|
'--config', 'config.json',
|
2019-10-20 17:54:38 +00:00
|
|
|
'--hyperopt', 'DefaultHyperOpt',
|
2020-08-06 07:00:28 +00:00
|
|
|
'--hyperopt-path',
|
|
|
|
str(Path(__file__).parent / "hyperopts"),
|
2019-09-24 12:39:28 +00:00
|
|
|
'--epochs', '5'
|
|
|
|
]
|
2020-02-10 09:35:48 +00:00
|
|
|
pargs = get_args(args)
|
2019-09-24 12:39:28 +00:00
|
|
|
|
|
|
|
with pytest.raises(OperationalException, match=r"Please ensure that the hyperopt dependencies"):
|
2020-02-10 09:35:48 +00:00
|
|
|
start_hyperopt(pargs)
|
2019-09-24 12:39:28 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_start(mocker, hyperopt_conf, caplog) -> None:
|
2018-03-06 06:02:03 +00:00
|
|
|
start_mock = MagicMock()
|
2020-08-06 06:51:01 +00:00
|
|
|
patched_configuration_load_config_file(mocker, hyperopt_conf)
|
2018-03-06 06:02:03 +00:00
|
|
|
mocker.patch('freqtrade.optimize.hyperopt.Hyperopt.start', start_mock)
|
2018-06-17 20:32:56 +00:00
|
|
|
patch_exchange(mocker)
|
|
|
|
|
2018-03-06 06:02:03 +00:00
|
|
|
args = [
|
|
|
|
'hyperopt',
|
2019-09-16 04:44:07 +00:00
|
|
|
'--config', 'config.json',
|
2019-10-20 17:54:38 +00:00
|
|
|
'--hyperopt', 'DefaultHyperOpt',
|
2018-03-06 06:02:03 +00:00
|
|
|
'--epochs', '5'
|
|
|
|
]
|
2020-02-10 09:35:48 +00:00
|
|
|
pargs = get_args(args)
|
|
|
|
start_hyperopt(pargs)
|
2018-03-06 06:02:03 +00:00
|
|
|
|
2019-08-11 18:16:52 +00:00
|
|
|
assert log_has('Starting freqtrade in Hyperopt mode', caplog)
|
2018-03-06 06:02:03 +00:00
|
|
|
assert start_mock.call_count == 1
|
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_start_no_data(mocker, hyperopt_conf) -> None:
|
|
|
|
patched_configuration_load_config_file(mocker, hyperopt_conf)
|
2019-12-04 05:57:44 +00:00
|
|
|
mocker.patch('freqtrade.data.history.load_pair_history', MagicMock(return_value=pd.DataFrame))
|
2019-05-14 06:23:09 +00:00
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-05-14 06:23:09 +00:00
|
|
|
MagicMock(return_value=(datetime(2017, 12, 10), datetime(2017, 12, 13)))
|
|
|
|
)
|
|
|
|
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
|
|
|
args = [
|
|
|
|
'hyperopt',
|
2019-09-16 04:44:07 +00:00
|
|
|
'--config', 'config.json',
|
2019-10-20 17:54:38 +00:00
|
|
|
'--hyperopt', 'DefaultHyperOpt',
|
2019-05-14 06:23:09 +00:00
|
|
|
'--epochs', '5'
|
|
|
|
]
|
2020-02-10 09:35:48 +00:00
|
|
|
pargs = get_args(args)
|
2019-10-23 18:27:51 +00:00
|
|
|
with pytest.raises(OperationalException, match='No data found. Terminating.'):
|
2020-02-10 09:35:48 +00:00
|
|
|
start_hyperopt(pargs)
|
2019-05-14 06:23:09 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_start_filelock(mocker, hyperopt_conf, caplog) -> None:
|
|
|
|
start_mock = MagicMock(side_effect=Timeout(Hyperopt.get_lock_filename(hyperopt_conf)))
|
|
|
|
patched_configuration_load_config_file(mocker, hyperopt_conf)
|
2019-05-25 18:25:59 +00:00
|
|
|
mocker.patch('freqtrade.optimize.hyperopt.Hyperopt.start', start_mock)
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
|
|
|
args = [
|
|
|
|
'hyperopt',
|
2019-09-16 04:44:07 +00:00
|
|
|
'--config', 'config.json',
|
2019-10-20 17:54:38 +00:00
|
|
|
'--hyperopt', 'DefaultHyperOpt',
|
2019-05-25 18:25:59 +00:00
|
|
|
'--epochs', '5'
|
|
|
|
]
|
2020-02-10 09:35:48 +00:00
|
|
|
pargs = get_args(args)
|
|
|
|
start_hyperopt(pargs)
|
2019-08-11 18:16:52 +00:00
|
|
|
assert log_has("Another running instance of freqtrade Hyperopt detected.", caplog)
|
2019-05-25 18:25:59 +00:00
|
|
|
|
|
|
|
|
2019-07-16 04:45:13 +00:00
|
|
|
def test_loss_calculation_prefer_correct_trade_count(default_conf, hyperopt_results) -> None:
|
2019-12-23 09:09:08 +00:00
|
|
|
hl = HyperOptLossResolver.load_hyperoptloss(default_conf)
|
2020-02-10 15:32:41 +00:00
|
|
|
correct = hl.hyperopt_loss_function(hyperopt_results, 600,
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
over = hl.hyperopt_loss_function(hyperopt_results, 600 + 100,
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
under = hl.hyperopt_loss_function(hyperopt_results, 600 - 100,
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
2018-03-02 13:46:32 +00:00
|
|
|
assert over > correct
|
|
|
|
assert under > correct
|
2017-12-26 08:08:10 +00:00
|
|
|
|
|
|
|
|
2019-07-16 04:45:13 +00:00
|
|
|
def test_loss_calculation_prefer_shorter_trades(default_conf, hyperopt_results) -> None:
|
2019-07-15 19:54:41 +00:00
|
|
|
resultsb = hyperopt_results.copy()
|
2019-07-27 20:24:06 +00:00
|
|
|
resultsb.loc[1, 'trade_duration'] = 20
|
2019-07-15 19:54:41 +00:00
|
|
|
|
2019-12-23 09:09:08 +00:00
|
|
|
hl = HyperOptLossResolver.load_hyperoptloss(default_conf)
|
2020-02-10 15:32:41 +00:00
|
|
|
longer = hl.hyperopt_loss_function(hyperopt_results, 100,
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
shorter = hl.hyperopt_loss_function(resultsb, 100,
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
2018-03-02 13:46:32 +00:00
|
|
|
assert shorter < longer
|
2017-12-26 08:08:10 +00:00
|
|
|
|
|
|
|
|
2019-07-16 04:45:13 +00:00
|
|
|
def test_loss_calculation_has_limited_profit(default_conf, hyperopt_results) -> None:
|
2019-07-15 19:54:41 +00:00
|
|
|
results_over = hyperopt_results.copy()
|
|
|
|
results_over['profit_percent'] = hyperopt_results['profit_percent'] * 2
|
|
|
|
results_under = hyperopt_results.copy()
|
|
|
|
results_under['profit_percent'] = hyperopt_results['profit_percent'] / 2
|
|
|
|
|
2019-12-23 09:09:08 +00:00
|
|
|
hl = HyperOptLossResolver.load_hyperoptloss(default_conf)
|
2020-02-10 15:32:41 +00:00
|
|
|
correct = hl.hyperopt_loss_function(hyperopt_results, 600,
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
over = hl.hyperopt_loss_function(results_over, 600,
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
under = hl.hyperopt_loss_function(results_under, 600,
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
2019-07-15 20:52:33 +00:00
|
|
|
assert over < correct
|
|
|
|
assert under > correct
|
2019-07-15 20:59:28 +00:00
|
|
|
|
2019-07-15 20:52:33 +00:00
|
|
|
|
2019-07-16 04:45:13 +00:00
|
|
|
def test_sharpe_loss_prefers_higher_profits(default_conf, hyperopt_results) -> None:
|
2019-07-15 20:52:33 +00:00
|
|
|
results_over = hyperopt_results.copy()
|
|
|
|
results_over['profit_percent'] = hyperopt_results['profit_percent'] * 2
|
|
|
|
results_under = hyperopt_results.copy()
|
|
|
|
results_under['profit_percent'] = hyperopt_results['profit_percent'] / 2
|
|
|
|
|
2019-07-16 04:45:13 +00:00
|
|
|
default_conf.update({'hyperopt_loss': 'SharpeHyperOptLoss'})
|
2019-12-23 09:09:08 +00:00
|
|
|
hl = HyperOptLossResolver.load_hyperoptloss(default_conf)
|
2019-07-25 17:54:12 +00:00
|
|
|
correct = hl.hyperopt_loss_function(hyperopt_results, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
over = hl.hyperopt_loss_function(results_over, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
under = hl.hyperopt_loss_function(results_under, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
assert over < correct
|
|
|
|
assert under > correct
|
|
|
|
|
|
|
|
|
2020-02-06 05:49:08 +00:00
|
|
|
def test_sharpe_loss_daily_prefers_higher_profits(default_conf, hyperopt_results) -> None:
|
|
|
|
results_over = hyperopt_results.copy()
|
|
|
|
results_over['profit_percent'] = hyperopt_results['profit_percent'] * 2
|
|
|
|
results_under = hyperopt_results.copy()
|
|
|
|
results_under['profit_percent'] = hyperopt_results['profit_percent'] / 2
|
|
|
|
|
|
|
|
default_conf.update({'hyperopt_loss': 'SharpeHyperOptLossDaily'})
|
|
|
|
hl = HyperOptLossResolver.load_hyperoptloss(default_conf)
|
|
|
|
correct = hl.hyperopt_loss_function(hyperopt_results, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
2019-07-25 17:54:12 +00:00
|
|
|
over = hl.hyperopt_loss_function(results_over, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
under = hl.hyperopt_loss_function(results_under, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
assert over < correct
|
|
|
|
assert under > correct
|
|
|
|
|
|
|
|
|
2020-02-07 00:50:51 +00:00
|
|
|
def test_sortino_loss_prefers_higher_profits(default_conf, hyperopt_results) -> None:
|
|
|
|
results_over = hyperopt_results.copy()
|
|
|
|
results_over['profit_percent'] = hyperopt_results['profit_percent'] * 2
|
|
|
|
results_under = hyperopt_results.copy()
|
|
|
|
results_under['profit_percent'] = hyperopt_results['profit_percent'] / 2
|
|
|
|
|
|
|
|
default_conf.update({'hyperopt_loss': 'SortinoHyperOptLoss'})
|
|
|
|
hl = HyperOptLossResolver.load_hyperoptloss(default_conf)
|
|
|
|
correct = hl.hyperopt_loss_function(hyperopt_results, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
over = hl.hyperopt_loss_function(results_over, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
under = hl.hyperopt_loss_function(results_under, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
assert over < correct
|
|
|
|
assert under > correct
|
|
|
|
|
|
|
|
|
|
|
|
def test_sortino_loss_daily_prefers_higher_profits(default_conf, hyperopt_results) -> None:
|
|
|
|
results_over = hyperopt_results.copy()
|
2020-02-07 01:22:05 +00:00
|
|
|
results_over['profit_percent'] = hyperopt_results['profit_percent'] * 2
|
2020-02-07 00:50:51 +00:00
|
|
|
results_under = hyperopt_results.copy()
|
2020-02-07 01:22:05 +00:00
|
|
|
results_under['profit_percent'] = hyperopt_results['profit_percent'] / 2
|
2020-02-07 00:50:51 +00:00
|
|
|
|
|
|
|
default_conf.update({'hyperopt_loss': 'SortinoHyperOptLossDaily'})
|
|
|
|
hl = HyperOptLossResolver.load_hyperoptloss(default_conf)
|
|
|
|
correct = hl.hyperopt_loss_function(hyperopt_results, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
2019-07-25 17:54:12 +00:00
|
|
|
over = hl.hyperopt_loss_function(results_over, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
under = hl.hyperopt_loss_function(results_under, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
assert over < correct
|
|
|
|
assert under > correct
|
|
|
|
|
|
|
|
|
|
|
|
def test_onlyprofit_loss_prefers_higher_profits(default_conf, hyperopt_results) -> None:
|
|
|
|
results_over = hyperopt_results.copy()
|
|
|
|
results_over['profit_percent'] = hyperopt_results['profit_percent'] * 2
|
|
|
|
results_under = hyperopt_results.copy()
|
|
|
|
results_under['profit_percent'] = hyperopt_results['profit_percent'] / 2
|
|
|
|
|
|
|
|
default_conf.update({'hyperopt_loss': 'OnlyProfitHyperOptLoss'})
|
2019-12-23 09:09:08 +00:00
|
|
|
hl = HyperOptLossResolver.load_hyperoptloss(default_conf)
|
2019-07-16 04:45:13 +00:00
|
|
|
correct = hl.hyperopt_loss_function(hyperopt_results, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
over = hl.hyperopt_loss_function(results_over, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
|
|
|
under = hl.hyperopt_loss_function(results_under, len(hyperopt_results),
|
|
|
|
datetime(2019, 1, 1), datetime(2019, 5, 1))
|
2019-07-15 19:54:41 +00:00
|
|
|
assert over < correct
|
2018-03-02 13:46:32 +00:00
|
|
|
assert under > correct
|
2017-12-26 08:08:10 +00:00
|
|
|
|
|
|
|
|
2018-07-30 10:26:54 +00:00
|
|
|
def test_log_results_if_loss_improves(hyperopt, capsys) -> None:
|
2018-03-02 13:46:32 +00:00
|
|
|
hyperopt.current_best_loss = 2
|
2019-07-30 08:47:46 +00:00
|
|
|
hyperopt.total_epochs = 2
|
2020-03-01 02:11:00 +00:00
|
|
|
|
2019-11-27 19:52:43 +00:00
|
|
|
hyperopt.print_results(
|
2018-03-02 13:46:32 +00:00
|
|
|
{
|
|
|
|
'loss': 1,
|
2020-03-01 02:11:00 +00:00
|
|
|
'results_metrics':
|
|
|
|
{
|
|
|
|
'trade_count': 1,
|
|
|
|
'avg_profit': 0.1,
|
|
|
|
'total_profit': 0.001,
|
|
|
|
'profit': 1.0,
|
|
|
|
'duration': 20.0
|
|
|
|
},
|
|
|
|
'total_profit': 0,
|
2019-11-23 09:00:43 +00:00
|
|
|
'current_epoch': 2, # This starts from 1 (in a human-friendly manner)
|
2020-03-01 02:11:00 +00:00
|
|
|
'is_initial_point': False,
|
|
|
|
'is_best': True
|
2018-03-02 13:46:32 +00:00
|
|
|
}
|
|
|
|
)
|
2018-03-25 20:57:40 +00:00
|
|
|
out, err = capsys.readouterr()
|
2020-03-03 00:14:56 +00:00
|
|
|
assert all(x in out
|
2020-03-04 19:51:09 +00:00
|
|
|
for x in ["Best", "2/2", " 1", "0.10%", "0.00100000 BTC (1.00%)", "20.0 m"])
|
2017-12-26 08:08:10 +00:00
|
|
|
|
|
|
|
|
2018-07-30 10:26:54 +00:00
|
|
|
def test_no_log_if_loss_does_not_improve(hyperopt, caplog) -> None:
|
2018-03-02 13:46:32 +00:00
|
|
|
hyperopt.current_best_loss = 2
|
2019-11-27 19:52:43 +00:00
|
|
|
hyperopt.print_results(
|
2018-03-02 13:46:32 +00:00
|
|
|
{
|
2019-11-27 19:52:43 +00:00
|
|
|
'is_best': False,
|
2018-03-02 13:46:32 +00:00
|
|
|
'loss': 3,
|
2019-11-23 09:00:43 +00:00
|
|
|
'current_epoch': 1,
|
2018-03-02 13:46:32 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
assert caplog.record_tuples == []
|
2018-01-07 01:12:32 +00:00
|
|
|
|
|
|
|
|
2020-04-28 19:56:19 +00:00
|
|
|
def test_save_results_saves_epochs(mocker, hyperopt, testdatadir, caplog) -> None:
|
|
|
|
epochs = create_results(mocker, hyperopt, testdatadir)
|
2018-07-03 19:51:48 +00:00
|
|
|
mock_dump = mocker.patch('freqtrade.optimize.hyperopt.dump', return_value=None)
|
2020-04-28 19:56:19 +00:00
|
|
|
results_file = testdatadir / 'optimize' / 'ut_results.pickle'
|
2019-11-23 09:30:49 +00:00
|
|
|
|
2020-04-28 19:56:19 +00:00
|
|
|
caplog.set_level(logging.DEBUG)
|
|
|
|
|
|
|
|
hyperopt.epochs = epochs
|
|
|
|
hyperopt._save_results()
|
|
|
|
assert log_has(f"1 epoch saved to '{results_file}'.", caplog)
|
2018-03-02 13:46:32 +00:00
|
|
|
mock_dump.assert_called_once()
|
2018-01-09 10:19:44 +00:00
|
|
|
|
2020-04-28 19:56:19 +00:00
|
|
|
hyperopt.epochs = epochs + epochs
|
|
|
|
hyperopt._save_results()
|
|
|
|
assert log_has(f"2 epochs saved to '{results_file}'.", caplog)
|
2019-11-23 09:30:49 +00:00
|
|
|
|
2018-01-09 10:19:44 +00:00
|
|
|
|
2020-04-28 19:56:19 +00:00
|
|
|
def test_read_results_returns_epochs(mocker, hyperopt, testdatadir, caplog) -> None:
|
|
|
|
epochs = create_results(mocker, hyperopt, testdatadir)
|
|
|
|
mock_load = mocker.patch('freqtrade.optimize.hyperopt.load', return_value=epochs)
|
|
|
|
results_file = testdatadir / 'optimize' / 'ut_results.pickle'
|
|
|
|
hyperopt_epochs = hyperopt._read_results(results_file)
|
|
|
|
assert log_has(f"Reading epochs from '{results_file}'", caplog)
|
|
|
|
assert hyperopt_epochs == epochs
|
2018-01-09 10:19:44 +00:00
|
|
|
mock_load.assert_called_once()
|
2018-01-25 08:45:53 +00:00
|
|
|
|
|
|
|
|
2018-07-30 10:26:54 +00:00
|
|
|
def test_roi_table_generation(hyperopt) -> None:
|
2018-01-25 08:45:53 +00:00
|
|
|
params = {
|
|
|
|
'roi_t1': 5,
|
|
|
|
'roi_t2': 10,
|
|
|
|
'roi_t3': 15,
|
|
|
|
'roi_p1': 1,
|
|
|
|
'roi_p2': 2,
|
|
|
|
'roi_p3': 3,
|
|
|
|
}
|
2018-03-02 13:46:32 +00:00
|
|
|
|
2018-03-22 08:27:13 +00:00
|
|
|
assert hyperopt.custom_hyperopt.generate_roi_table(params) == {0: 6, 15: 3, 25: 1, 30: 0}
|
2018-03-02 13:46:32 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_start_calls_optimizer(mocker, hyperopt_conf, capsys) -> None:
|
2018-07-03 19:09:59 +00:00
|
|
|
dumper = mocker.patch('freqtrade.optimize.hyperopt.dump', MagicMock())
|
2019-10-27 09:56:38 +00:00
|
|
|
mocker.patch('freqtrade.optimize.backtesting.Backtesting.load_bt_data',
|
|
|
|
MagicMock(return_value=(MagicMock(), None)))
|
2019-05-14 06:23:09 +00:00
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-05-14 06:23:09 +00:00
|
|
|
MagicMock(return_value=(datetime(2017, 12, 10), datetime(2017, 12, 13)))
|
|
|
|
)
|
|
|
|
|
2018-06-24 12:27:53 +00:00
|
|
|
parallel = mocker.patch(
|
|
|
|
'freqtrade.optimize.hyperopt.Hyperopt.run_optimizer_parallel',
|
2020-03-01 02:24:04 +00:00
|
|
|
MagicMock(return_value=[{
|
|
|
|
'loss': 1, 'results_explanation': 'foo result',
|
|
|
|
'params': {'buy': {}, 'sell': {}, 'roi': {}, 'stoploss': 0.0},
|
|
|
|
'results_metrics':
|
|
|
|
{
|
|
|
|
'trade_count': 1,
|
|
|
|
'avg_profit': 0.1,
|
|
|
|
'total_profit': 0.001,
|
|
|
|
'profit': 1.0,
|
|
|
|
'duration': 20.0
|
|
|
|
},
|
|
|
|
}])
|
2018-06-24 12:27:53 +00:00
|
|
|
)
|
2018-06-17 20:32:56 +00:00
|
|
|
patch_exchange(mocker)
|
2020-03-08 10:35:31 +00:00
|
|
|
# Co-test loading timeframe from strategy
|
2020-08-06 06:51:01 +00:00
|
|
|
del hyperopt_conf['timeframe']
|
2018-03-02 13:46:32 +00:00
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt = Hyperopt(hyperopt_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
hyperopt.backtesting.strategy.ohlcvdata_to_dataframe = MagicMock()
|
2019-08-01 20:57:50 +00:00
|
|
|
hyperopt.custom_hyperopt.generate_roi_table = MagicMock(return_value={})
|
2018-03-02 13:46:32 +00:00
|
|
|
|
|
|
|
hyperopt.start()
|
2019-07-30 08:47:46 +00:00
|
|
|
|
2018-06-24 12:27:53 +00:00
|
|
|
parallel.assert_called_once()
|
2019-07-30 08:47:46 +00:00
|
|
|
|
|
|
|
out, err = capsys.readouterr()
|
2019-08-03 08:19:36 +00:00
|
|
|
assert 'Best result:\n\n* 1/1: foo result Objective: 1.00000\n' in out
|
2018-07-03 19:09:59 +00:00
|
|
|
assert dumper.called
|
2020-03-08 10:35:31 +00:00
|
|
|
# Should be called twice, once for historical candle data, once to save evaluations
|
2019-06-08 17:32:31 +00:00
|
|
|
assert dumper.call_count == 2
|
2019-09-18 19:57:37 +00:00
|
|
|
assert hasattr(hyperopt.backtesting.strategy, "advise_sell")
|
|
|
|
assert hasattr(hyperopt.backtesting.strategy, "advise_buy")
|
2019-07-15 18:28:55 +00:00
|
|
|
assert hasattr(hyperopt, "max_open_trades")
|
2020-08-06 06:51:01 +00:00
|
|
|
assert hyperopt.max_open_trades == hyperopt_conf['max_open_trades']
|
2019-07-16 03:50:27 +00:00
|
|
|
assert hasattr(hyperopt, "position_stacking")
|
2018-03-02 13:46:32 +00:00
|
|
|
|
|
|
|
|
2018-07-30 10:26:54 +00:00
|
|
|
def test_format_results(hyperopt):
|
2018-06-02 21:07:31 +00:00
|
|
|
# Test with BTC as stake_currency
|
2018-03-04 00:42:37 +00:00
|
|
|
trades = [
|
2018-02-03 16:15:40 +00:00
|
|
|
('ETH/BTC', 2, 2, 123),
|
|
|
|
('LTC/BTC', 1, 1, 123),
|
|
|
|
('XPR/BTC', -1, -2, -246)
|
2018-03-04 00:42:37 +00:00
|
|
|
]
|
2018-06-10 11:56:23 +00:00
|
|
|
labels = ['currency', 'profit_percent', 'profit_abs', 'trade_duration']
|
2018-02-08 19:49:43 +00:00
|
|
|
df = pd.DataFrame.from_records(trades, columns=labels)
|
2019-11-27 19:52:43 +00:00
|
|
|
results_metrics = hyperopt._calculate_results_metrics(df)
|
2019-12-01 15:01:59 +00:00
|
|
|
results_explanation = hyperopt._format_results_explanation_string(results_metrics)
|
|
|
|
total_profit = results_metrics['total_profit']
|
|
|
|
|
|
|
|
results = {
|
|
|
|
'loss': 0.0,
|
|
|
|
'params_dict': None,
|
|
|
|
'params_details': None,
|
|
|
|
'results_metrics': results_metrics,
|
|
|
|
'results_explanation': results_explanation,
|
|
|
|
'total_profit': total_profit,
|
|
|
|
'current_epoch': 1,
|
|
|
|
'is_initial_point': True,
|
|
|
|
}
|
2018-06-02 21:07:31 +00:00
|
|
|
|
2019-12-01 15:01:59 +00:00
|
|
|
result = hyperopt._format_explanation_string(results, 1)
|
2018-06-02 21:07:31 +00:00
|
|
|
assert result.find(' 66.67%')
|
|
|
|
assert result.find('Total profit 1.00000000 BTC')
|
|
|
|
assert result.find('2.0000Σ %')
|
|
|
|
|
|
|
|
# Test with EUR as stake_currency
|
|
|
|
trades = [
|
|
|
|
('ETH/EUR', 2, 2, 123),
|
|
|
|
('LTC/EUR', 1, 1, 123),
|
|
|
|
('XPR/EUR', -1, -2, -246)
|
|
|
|
]
|
|
|
|
df = pd.DataFrame.from_records(trades, columns=labels)
|
2019-11-27 19:52:43 +00:00
|
|
|
results_metrics = hyperopt._calculate_results_metrics(df)
|
2019-12-01 15:01:59 +00:00
|
|
|
results['total_profit'] = results_metrics['total_profit']
|
|
|
|
result = hyperopt._format_explanation_string(results, 1)
|
2018-06-02 21:07:31 +00:00
|
|
|
assert result.find('Total profit 1.00000000 EUR')
|
2018-02-08 19:49:43 +00:00
|
|
|
|
|
|
|
|
2019-11-07 22:55:14 +00:00
|
|
|
@pytest.mark.parametrize("spaces, expected_results", [
|
|
|
|
(['buy'],
|
|
|
|
{'buy': True, 'sell': False, 'roi': False, 'stoploss': False, 'trailing': False}),
|
|
|
|
(['sell'],
|
|
|
|
{'buy': False, 'sell': True, 'roi': False, 'stoploss': False, 'trailing': False}),
|
|
|
|
(['roi'],
|
|
|
|
{'buy': False, 'sell': False, 'roi': True, 'stoploss': False, 'trailing': False}),
|
|
|
|
(['stoploss'],
|
|
|
|
{'buy': False, 'sell': False, 'roi': False, 'stoploss': True, 'trailing': False}),
|
|
|
|
(['trailing'],
|
|
|
|
{'buy': False, 'sell': False, 'roi': False, 'stoploss': False, 'trailing': True}),
|
|
|
|
(['buy', 'sell', 'roi', 'stoploss'],
|
|
|
|
{'buy': True, 'sell': True, 'roi': True, 'stoploss': True, 'trailing': False}),
|
|
|
|
(['buy', 'sell', 'roi', 'stoploss', 'trailing'],
|
|
|
|
{'buy': True, 'sell': True, 'roi': True, 'stoploss': True, 'trailing': True}),
|
|
|
|
(['buy', 'roi'],
|
|
|
|
{'buy': True, 'sell': False, 'roi': True, 'stoploss': False, 'trailing': False}),
|
|
|
|
(['all'],
|
|
|
|
{'buy': True, 'sell': True, 'roi': True, 'stoploss': True, 'trailing': True}),
|
|
|
|
(['default'],
|
|
|
|
{'buy': True, 'sell': True, 'roi': True, 'stoploss': True, 'trailing': False}),
|
|
|
|
(['default', 'trailing'],
|
|
|
|
{'buy': True, 'sell': True, 'roi': True, 'stoploss': True, 'trailing': True}),
|
|
|
|
(['all', 'buy'],
|
|
|
|
{'buy': True, 'sell': True, 'roi': True, 'stoploss': True, 'trailing': True}),
|
|
|
|
(['default', 'buy'],
|
|
|
|
{'buy': True, 'sell': True, 'roi': True, 'stoploss': True, 'trailing': False}),
|
|
|
|
])
|
|
|
|
def test_has_space(hyperopt, spaces, expected_results):
|
|
|
|
for s in ['buy', 'sell', 'roi', 'stoploss', 'trailing']:
|
|
|
|
hyperopt.config.update({'spaces': spaces})
|
|
|
|
assert hyperopt.has_space(s) == expected_results[s]
|
2018-03-05 08:35:42 +00:00
|
|
|
|
|
|
|
|
2019-09-07 18:56:03 +00:00
|
|
|
def test_populate_indicators(hyperopt, testdatadir) -> None:
|
2020-03-08 10:35:31 +00:00
|
|
|
data = load_data(testdatadir, '1m', ['UNITTEST/BTC'], fill_up_missing=True)
|
|
|
|
dataframes = hyperopt.backtesting.strategy.ohlcvdata_to_dataframe(data)
|
2018-03-22 08:27:13 +00:00
|
|
|
dataframe = hyperopt.custom_hyperopt.populate_indicators(dataframes['UNITTEST/BTC'],
|
|
|
|
{'pair': 'UNITTEST/BTC'})
|
2018-03-05 08:35:42 +00:00
|
|
|
|
|
|
|
# Check if some indicators are generated. We will not test all of them
|
|
|
|
assert 'adx' in dataframe
|
2018-06-24 12:27:53 +00:00
|
|
|
assert 'mfi' in dataframe
|
|
|
|
assert 'rsi' in dataframe
|
2018-03-05 08:35:42 +00:00
|
|
|
|
|
|
|
|
2019-09-07 18:56:03 +00:00
|
|
|
def test_buy_strategy_generator(hyperopt, testdatadir) -> None:
|
2020-03-08 10:35:31 +00:00
|
|
|
data = load_data(testdatadir, '1m', ['UNITTEST/BTC'], fill_up_missing=True)
|
|
|
|
dataframes = hyperopt.backtesting.strategy.ohlcvdata_to_dataframe(data)
|
2018-03-22 08:27:13 +00:00
|
|
|
dataframe = hyperopt.custom_hyperopt.populate_indicators(dataframes['UNITTEST/BTC'],
|
|
|
|
{'pair': 'UNITTEST/BTC'})
|
2018-03-05 08:35:42 +00:00
|
|
|
|
2018-03-22 08:27:13 +00:00
|
|
|
populate_buy_trend = hyperopt.custom_hyperopt.buy_strategy_generator(
|
2018-03-05 08:35:42 +00:00
|
|
|
{
|
2018-06-24 12:27:53 +00:00
|
|
|
'adx-value': 20,
|
|
|
|
'fastd-value': 20,
|
|
|
|
'mfi-value': 20,
|
|
|
|
'rsi-value': 20,
|
|
|
|
'adx-enabled': True,
|
|
|
|
'fastd-enabled': True,
|
|
|
|
'mfi-enabled': True,
|
|
|
|
'rsi-enabled': True,
|
|
|
|
'trigger': 'bb_lower'
|
2018-03-05 08:35:42 +00:00
|
|
|
}
|
|
|
|
)
|
2018-07-29 19:07:21 +00:00
|
|
|
result = populate_buy_trend(dataframe, {'pair': 'UNITTEST/BTC'})
|
2018-03-05 08:35:42 +00:00
|
|
|
# Check if some indicators are generated. We will not test all of them
|
2018-03-06 06:02:03 +00:00
|
|
|
assert 'buy' in result
|
|
|
|
assert 1 in result['buy']
|
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_sell_strategy_generator(hyperopt, testdatadir) -> None:
|
|
|
|
data = load_data(testdatadir, '1m', ['UNITTEST/BTC'], fill_up_missing=True)
|
|
|
|
dataframes = hyperopt.backtesting.strategy.ohlcvdata_to_dataframe(data)
|
|
|
|
dataframe = hyperopt.custom_hyperopt.populate_indicators(dataframes['UNITTEST/BTC'],
|
|
|
|
{'pair': 'UNITTEST/BTC'})
|
|
|
|
|
|
|
|
populate_sell_trend = hyperopt.custom_hyperopt.sell_strategy_generator(
|
|
|
|
{
|
|
|
|
'sell-adx-value': 20,
|
|
|
|
'sell-fastd-value': 75,
|
|
|
|
'sell-mfi-value': 80,
|
|
|
|
'sell-rsi-value': 20,
|
|
|
|
'sell-adx-enabled': True,
|
|
|
|
'sell-fastd-enabled': True,
|
|
|
|
'sell-mfi-enabled': True,
|
|
|
|
'sell-rsi-enabled': True,
|
|
|
|
'sell-trigger': 'sell-bb_upper'
|
|
|
|
}
|
|
|
|
)
|
|
|
|
result = populate_sell_trend(dataframe, {'pair': 'UNITTEST/BTC'})
|
|
|
|
# Check if some indicators are generated. We will not test all of them
|
|
|
|
print(result)
|
|
|
|
assert 'sell' in result
|
|
|
|
assert 1 in result['sell']
|
|
|
|
|
|
|
|
|
|
|
|
def test_generate_optimizer(mocker, hyperopt_conf) -> None:
|
|
|
|
hyperopt_conf.update({'spaces': 'all',
|
|
|
|
'hyperopt_min_trades': 1,
|
|
|
|
})
|
2018-03-06 06:02:03 +00:00
|
|
|
|
|
|
|
trades = [
|
2019-10-30 08:20:56 +00:00
|
|
|
('TRX/BTC', 0.023117, 0.000233, 100)
|
2018-03-06 06:02:03 +00:00
|
|
|
]
|
2018-06-10 11:56:23 +00:00
|
|
|
labels = ['currency', 'profit_percent', 'profit_abs', 'trade_duration']
|
2018-03-06 06:02:03 +00:00
|
|
|
backtest_result = pd.DataFrame.from_records(trades, columns=labels)
|
|
|
|
|
|
|
|
mocker.patch(
|
2019-08-23 21:10:55 +00:00
|
|
|
'freqtrade.optimize.hyperopt.Backtesting.backtest',
|
2018-03-06 06:02:03 +00:00
|
|
|
MagicMock(return_value=backtest_result)
|
|
|
|
)
|
2018-10-16 18:32:33 +00:00
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-07-15 19:54:41 +00:00
|
|
|
MagicMock(return_value=(Arrow(2017, 12, 10), Arrow(2017, 12, 13)))
|
2018-10-16 18:32:33 +00:00
|
|
|
)
|
2018-06-17 20:32:56 +00:00
|
|
|
patch_exchange(mocker)
|
2018-07-03 19:09:59 +00:00
|
|
|
mocker.patch('freqtrade.optimize.hyperopt.load', MagicMock())
|
2018-03-06 06:02:03 +00:00
|
|
|
|
|
|
|
optimizer_param = {
|
2018-06-24 12:27:53 +00:00
|
|
|
'adx-value': 0,
|
|
|
|
'fastd-value': 35,
|
|
|
|
'mfi-value': 0,
|
|
|
|
'rsi-value': 0,
|
|
|
|
'adx-enabled': False,
|
|
|
|
'fastd-enabled': True,
|
|
|
|
'mfi-enabled': False,
|
|
|
|
'rsi-enabled': False,
|
|
|
|
'trigger': 'macd_cross_signal',
|
2019-01-06 09:35:18 +00:00
|
|
|
'sell-adx-value': 0,
|
|
|
|
'sell-fastd-value': 75,
|
|
|
|
'sell-mfi-value': 0,
|
|
|
|
'sell-rsi-value': 0,
|
|
|
|
'sell-adx-enabled': False,
|
|
|
|
'sell-fastd-enabled': True,
|
|
|
|
'sell-mfi-enabled': False,
|
|
|
|
'sell-rsi-enabled': False,
|
|
|
|
'sell-trigger': 'macd_cross_signal',
|
2018-03-06 06:02:03 +00:00
|
|
|
'roi_t1': 60.0,
|
|
|
|
'roi_t2': 30.0,
|
|
|
|
'roi_t3': 20.0,
|
2018-06-24 12:27:53 +00:00
|
|
|
'roi_p1': 0.01,
|
|
|
|
'roi_p2': 0.01,
|
|
|
|
'roi_p3': 0.1,
|
2018-03-06 06:02:03 +00:00
|
|
|
'stoploss': -0.4,
|
2019-11-13 20:32:37 +00:00
|
|
|
'trailing_stop': True,
|
|
|
|
'trailing_stop_positive': 0.02,
|
2019-12-10 11:06:17 +00:00
|
|
|
'trailing_stop_positive_offset_p1': 0.05,
|
2019-11-13 20:32:37 +00:00
|
|
|
'trailing_only_offset_is_reached': False,
|
2018-03-06 06:02:03 +00:00
|
|
|
}
|
|
|
|
response_expected = {
|
|
|
|
'loss': 1.9840569076926293,
|
2020-08-14 05:31:14 +00:00
|
|
|
'results_explanation': (' 1 trades. 1/0/0 Wins/Draws/Losses. '
|
2020-03-02 01:50:27 +00:00
|
|
|
'Avg profit 2.31%. Median profit 2.31%. Total profit '
|
|
|
|
'0.00023300 BTC ( 2.31\N{GREEK CAPITAL LETTER SIGMA}%). '
|
|
|
|
'Avg duration 100.0 min.'
|
2019-11-06 18:33:15 +00:00
|
|
|
).encode(locale.getpreferredencoding(), 'replace').decode('utf-8'),
|
2019-11-27 19:52:43 +00:00
|
|
|
'params_details': {'buy': {'adx-enabled': False,
|
|
|
|
'adx-value': 0,
|
|
|
|
'fastd-enabled': True,
|
|
|
|
'fastd-value': 35,
|
|
|
|
'mfi-enabled': False,
|
|
|
|
'mfi-value': 0,
|
|
|
|
'rsi-enabled': False,
|
|
|
|
'rsi-value': 0,
|
|
|
|
'trigger': 'macd_cross_signal'},
|
|
|
|
'roi': {0: 0.12000000000000001,
|
|
|
|
20.0: 0.02,
|
|
|
|
50.0: 0.01,
|
|
|
|
110.0: 0},
|
|
|
|
'sell': {'sell-adx-enabled': False,
|
|
|
|
'sell-adx-value': 0,
|
|
|
|
'sell-fastd-enabled': True,
|
|
|
|
'sell-fastd-value': 75,
|
|
|
|
'sell-mfi-enabled': False,
|
|
|
|
'sell-mfi-value': 0,
|
|
|
|
'sell-rsi-enabled': False,
|
|
|
|
'sell-rsi-value': 0,
|
|
|
|
'sell-trigger': 'macd_cross_signal'},
|
2019-12-04 22:08:38 +00:00
|
|
|
'stoploss': {'stoploss': -0.4},
|
|
|
|
'trailing': {'trailing_only_offset_is_reached': False,
|
|
|
|
'trailing_stop': True,
|
|
|
|
'trailing_stop_positive': 0.02,
|
2019-12-10 11:06:17 +00:00
|
|
|
'trailing_stop_positive_offset': 0.07}},
|
2019-11-27 19:52:43 +00:00
|
|
|
'params_dict': optimizer_param,
|
|
|
|
'results_metrics': {'avg_profit': 2.3117,
|
2020-03-02 01:50:27 +00:00
|
|
|
'draws': 0,
|
2019-11-27 19:52:43 +00:00
|
|
|
'duration': 100.0,
|
2020-03-02 01:50:27 +00:00
|
|
|
'losses': 0,
|
2020-09-19 15:32:22 +00:00
|
|
|
'winsdrawslosses': ' 1 0 0',
|
2020-03-02 01:50:27 +00:00
|
|
|
'median_profit': 2.3117,
|
2019-11-27 19:52:43 +00:00
|
|
|
'profit': 2.3117,
|
|
|
|
'total_profit': 0.000233,
|
2020-03-02 01:50:27 +00:00
|
|
|
'trade_count': 1,
|
|
|
|
'wins': 1},
|
2019-08-03 16:13:18 +00:00
|
|
|
'total_profit': 0.00023300
|
2018-03-06 06:02:03 +00:00
|
|
|
}
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt = Hyperopt(hyperopt_conf)
|
2019-09-16 18:53:19 +00:00
|
|
|
hyperopt.dimensions = hyperopt.hyperopt_space()
|
2018-06-24 12:27:53 +00:00
|
|
|
generate_optimizer_value = hyperopt.generate_optimizer(list(optimizer_param.values()))
|
2018-03-06 06:02:03 +00:00
|
|
|
assert generate_optimizer_value == response_expected
|
2019-07-15 18:27:34 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_clean_hyperopt(mocker, hyperopt_conf, caplog):
|
2019-07-15 18:27:34 +00:00
|
|
|
patch_exchange(mocker)
|
2020-08-06 06:51:01 +00:00
|
|
|
|
2019-07-15 18:27:34 +00:00
|
|
|
mocker.patch("freqtrade.optimize.hyperopt.Path.is_file", MagicMock(return_value=True))
|
|
|
|
unlinkmock = mocker.patch("freqtrade.optimize.hyperopt.Path.unlink", MagicMock())
|
2020-08-06 06:51:01 +00:00
|
|
|
h = Hyperopt(hyperopt_conf)
|
2019-07-15 18:27:34 +00:00
|
|
|
|
|
|
|
assert unlinkmock.call_count == 2
|
2020-03-08 10:35:31 +00:00
|
|
|
assert log_has(f"Removing `{h.data_pickle_file}`.", caplog)
|
2019-07-16 03:50:27 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_continue_hyperopt(mocker, hyperopt_conf, caplog):
|
2019-07-16 03:50:27 +00:00
|
|
|
patch_exchange(mocker)
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt_conf.update({'hyperopt_continue': True})
|
2019-07-16 03:50:27 +00:00
|
|
|
mocker.patch("freqtrade.optimize.hyperopt.Path.is_file", MagicMock(return_value=True))
|
|
|
|
unlinkmock = mocker.patch("freqtrade.optimize.hyperopt.Path.unlink", MagicMock())
|
2020-08-06 06:51:01 +00:00
|
|
|
Hyperopt(hyperopt_conf)
|
2019-07-16 03:50:27 +00:00
|
|
|
|
|
|
|
assert unlinkmock.call_count == 0
|
2020-05-18 09:40:25 +00:00
|
|
|
assert log_has("Continuing on previous hyperopt results.", caplog)
|
2019-08-15 21:49:49 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_print_json_spaces_all(mocker, hyperopt_conf, capsys) -> None:
|
2019-08-16 01:20:12 +00:00
|
|
|
dumper = mocker.patch('freqtrade.optimize.hyperopt.dump', MagicMock())
|
2019-10-27 09:56:38 +00:00
|
|
|
mocker.patch('freqtrade.optimize.backtesting.Backtesting.load_bt_data',
|
|
|
|
MagicMock(return_value=(MagicMock(), None)))
|
2019-08-15 21:49:49 +00:00
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-08-15 21:49:49 +00:00
|
|
|
MagicMock(return_value=(datetime(2017, 12, 10), datetime(2017, 12, 13)))
|
|
|
|
)
|
|
|
|
|
|
|
|
parallel = mocker.patch(
|
|
|
|
'freqtrade.optimize.hyperopt.Hyperopt.run_optimizer_parallel',
|
2020-03-01 02:11:00 +00:00
|
|
|
MagicMock(return_value=[{
|
|
|
|
'loss': 1, 'results_explanation': 'foo result', 'params': {},
|
|
|
|
'params_details': {
|
|
|
|
'buy': {'mfi-value': None},
|
|
|
|
'sell': {'sell-mfi-value': None},
|
|
|
|
'roi': {}, 'stoploss': {'stoploss': None},
|
|
|
|
'trailing': {'trailing_stop': None}
|
|
|
|
},
|
|
|
|
'results_metrics':
|
|
|
|
{
|
|
|
|
'trade_count': 1,
|
|
|
|
'avg_profit': 0.1,
|
|
|
|
'total_profit': 0.001,
|
|
|
|
'profit': 1.0,
|
|
|
|
'duration': 20.0
|
|
|
|
}
|
|
|
|
}])
|
2019-08-15 21:49:49 +00:00
|
|
|
)
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt_conf.update({'spaces': 'all',
|
|
|
|
'hyperopt_jobs': 1,
|
|
|
|
'print_json': True,
|
|
|
|
})
|
2019-08-15 21:49:49 +00:00
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt = Hyperopt(hyperopt_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
hyperopt.backtesting.strategy.ohlcvdata_to_dataframe = MagicMock()
|
2019-08-15 21:49:49 +00:00
|
|
|
hyperopt.custom_hyperopt.generate_roi_table = MagicMock(return_value={})
|
|
|
|
|
|
|
|
hyperopt.start()
|
|
|
|
|
|
|
|
parallel.assert_called_once()
|
|
|
|
|
|
|
|
out, err = capsys.readouterr()
|
2020-03-01 02:11:00 +00:00
|
|
|
result_str = (
|
|
|
|
'{"params":{"mfi-value":null,"sell-mfi-value":null},"minimal_roi"'
|
|
|
|
':{},"stoploss":null,"trailing_stop":null}'
|
|
|
|
)
|
|
|
|
assert result_str in out # noqa: E501
|
2019-08-16 01:20:12 +00:00
|
|
|
assert dumper.called
|
2020-03-08 10:35:31 +00:00
|
|
|
# Should be called twice, once for historical candle data, once to save evaluations
|
2019-08-16 01:20:12 +00:00
|
|
|
assert dumper.call_count == 2
|
2019-08-15 21:49:49 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_print_json_spaces_default(mocker, hyperopt_conf, capsys) -> None:
|
2019-11-13 20:09:05 +00:00
|
|
|
dumper = mocker.patch('freqtrade.optimize.hyperopt.dump', MagicMock())
|
|
|
|
mocker.patch('freqtrade.optimize.backtesting.Backtesting.load_bt_data',
|
|
|
|
MagicMock(return_value=(MagicMock(), None)))
|
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-11-13 20:09:05 +00:00
|
|
|
MagicMock(return_value=(datetime(2017, 12, 10), datetime(2017, 12, 13)))
|
|
|
|
)
|
|
|
|
|
|
|
|
parallel = mocker.patch(
|
|
|
|
'freqtrade.optimize.hyperopt.Hyperopt.run_optimizer_parallel',
|
2020-03-01 02:11:00 +00:00
|
|
|
MagicMock(return_value=[{
|
|
|
|
'loss': 1, 'results_explanation': 'foo result', 'params': {},
|
|
|
|
'params_details': {
|
|
|
|
'buy': {'mfi-value': None},
|
|
|
|
'sell': {'sell-mfi-value': None},
|
|
|
|
'roi': {}, 'stoploss': {'stoploss': None}
|
|
|
|
},
|
|
|
|
'results_metrics':
|
|
|
|
{
|
|
|
|
'trade_count': 1,
|
|
|
|
'avg_profit': 0.1,
|
|
|
|
'total_profit': 0.001,
|
|
|
|
'profit': 1.0,
|
|
|
|
'duration': 20.0
|
|
|
|
}
|
|
|
|
}])
|
2019-11-13 20:09:05 +00:00
|
|
|
)
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt_conf.update({'print_json': True})
|
2019-11-13 20:09:05 +00:00
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt = Hyperopt(hyperopt_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
hyperopt.backtesting.strategy.ohlcvdata_to_dataframe = MagicMock()
|
2019-11-13 20:09:05 +00:00
|
|
|
hyperopt.custom_hyperopt.generate_roi_table = MagicMock(return_value={})
|
|
|
|
|
|
|
|
hyperopt.start()
|
|
|
|
|
|
|
|
parallel.assert_called_once()
|
|
|
|
|
|
|
|
out, err = capsys.readouterr()
|
2019-12-01 15:34:25 +00:00
|
|
|
assert '{"params":{"mfi-value":null,"sell-mfi-value":null},"minimal_roi":{},"stoploss":null}' in out # noqa: E501
|
2019-11-13 20:09:05 +00:00
|
|
|
assert dumper.called
|
2020-03-08 10:35:31 +00:00
|
|
|
# Should be called twice, once for historical candle data, once to save evaluations
|
2019-11-13 20:09:05 +00:00
|
|
|
assert dumper.call_count == 2
|
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_print_json_spaces_roi_stoploss(mocker, hyperopt_conf, capsys) -> None:
|
2019-08-16 01:20:12 +00:00
|
|
|
dumper = mocker.patch('freqtrade.optimize.hyperopt.dump', MagicMock())
|
2019-10-27 09:56:38 +00:00
|
|
|
mocker.patch('freqtrade.optimize.backtesting.Backtesting.load_bt_data',
|
|
|
|
MagicMock(return_value=(MagicMock(), None)))
|
2019-08-15 21:49:49 +00:00
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-08-15 21:49:49 +00:00
|
|
|
MagicMock(return_value=(datetime(2017, 12, 10), datetime(2017, 12, 13)))
|
|
|
|
)
|
|
|
|
|
|
|
|
parallel = mocker.patch(
|
|
|
|
'freqtrade.optimize.hyperopt.Hyperopt.run_optimizer_parallel',
|
2020-03-01 02:11:00 +00:00
|
|
|
MagicMock(return_value=[{
|
|
|
|
'loss': 1, 'results_explanation': 'foo result', 'params': {},
|
|
|
|
'params_details': {'roi': {}, 'stoploss': {'stoploss': None}},
|
|
|
|
'results_metrics':
|
|
|
|
{
|
|
|
|
'trade_count': 1,
|
|
|
|
'avg_profit': 0.1,
|
|
|
|
'total_profit': 0.001,
|
|
|
|
'profit': 1.0,
|
|
|
|
'duration': 20.0
|
|
|
|
}
|
|
|
|
}])
|
2019-08-15 21:49:49 +00:00
|
|
|
)
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt_conf.update({'spaces': 'roi stoploss',
|
|
|
|
'hyperopt_jobs': 1,
|
|
|
|
'print_json': True,
|
|
|
|
})
|
2019-08-15 21:49:49 +00:00
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt = Hyperopt(hyperopt_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
hyperopt.backtesting.strategy.ohlcvdata_to_dataframe = MagicMock()
|
2019-08-15 21:49:49 +00:00
|
|
|
hyperopt.custom_hyperopt.generate_roi_table = MagicMock(return_value={})
|
|
|
|
|
|
|
|
hyperopt.start()
|
|
|
|
|
|
|
|
parallel.assert_called_once()
|
|
|
|
|
|
|
|
out, err = capsys.readouterr()
|
|
|
|
assert '{"minimal_roi":{},"stoploss":null}' in out
|
2019-08-16 01:20:12 +00:00
|
|
|
assert dumper.called
|
2020-03-08 10:35:31 +00:00
|
|
|
# Should be called twice, once for historical candle data, once to save evaluations
|
2019-08-16 01:20:12 +00:00
|
|
|
assert dumper.call_count == 2
|
2019-09-04 21:38:15 +00:00
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_simplified_interface_roi_stoploss(mocker, hyperopt_conf, capsys) -> None:
|
2019-09-04 21:38:15 +00:00
|
|
|
dumper = mocker.patch('freqtrade.optimize.hyperopt.dump', MagicMock())
|
2019-10-27 09:56:38 +00:00
|
|
|
mocker.patch('freqtrade.optimize.backtesting.Backtesting.load_bt_data',
|
|
|
|
MagicMock(return_value=(MagicMock(), None)))
|
2019-09-04 21:38:15 +00:00
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-09-04 21:38:15 +00:00
|
|
|
MagicMock(return_value=(datetime(2017, 12, 10), datetime(2017, 12, 13)))
|
|
|
|
)
|
|
|
|
|
|
|
|
parallel = mocker.patch(
|
|
|
|
'freqtrade.optimize.hyperopt.Hyperopt.run_optimizer_parallel',
|
2019-09-06 13:09:03 +00:00
|
|
|
MagicMock(return_value=[{
|
2020-03-01 02:11:00 +00:00
|
|
|
'loss': 1, 'results_explanation': 'foo result', 'params': {'stoploss': 0.0},
|
|
|
|
'results_metrics':
|
|
|
|
{
|
|
|
|
'trade_count': 1,
|
|
|
|
'avg_profit': 0.1,
|
|
|
|
'total_profit': 0.001,
|
|
|
|
'profit': 1.0,
|
|
|
|
'duration': 20.0
|
|
|
|
}
|
|
|
|
}])
|
2019-09-04 21:38:15 +00:00
|
|
|
)
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt_conf.update({'spaces': 'roi stoploss'})
|
2019-09-04 21:38:15 +00:00
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt = Hyperopt(hyperopt_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
hyperopt.backtesting.strategy.ohlcvdata_to_dataframe = MagicMock()
|
2019-09-04 21:38:15 +00:00
|
|
|
hyperopt.custom_hyperopt.generate_roi_table = MagicMock(return_value={})
|
|
|
|
|
|
|
|
del hyperopt.custom_hyperopt.__class__.buy_strategy_generator
|
|
|
|
del hyperopt.custom_hyperopt.__class__.sell_strategy_generator
|
|
|
|
del hyperopt.custom_hyperopt.__class__.indicator_space
|
|
|
|
del hyperopt.custom_hyperopt.__class__.sell_indicator_space
|
|
|
|
|
|
|
|
hyperopt.start()
|
|
|
|
|
|
|
|
parallel.assert_called_once()
|
|
|
|
|
|
|
|
out, err = capsys.readouterr()
|
|
|
|
assert 'Best result:\n\n* 1/1: foo result Objective: 1.00000\n' in out
|
|
|
|
assert dumper.called
|
2020-03-08 10:35:31 +00:00
|
|
|
# Should be called twice, once for historical candle data, once to save evaluations
|
2019-09-04 21:38:15 +00:00
|
|
|
assert dumper.call_count == 2
|
2019-09-18 19:57:37 +00:00
|
|
|
assert hasattr(hyperopt.backtesting.strategy, "advise_sell")
|
|
|
|
assert hasattr(hyperopt.backtesting.strategy, "advise_buy")
|
2019-09-04 21:38:15 +00:00
|
|
|
assert hasattr(hyperopt, "max_open_trades")
|
2020-08-06 06:51:01 +00:00
|
|
|
assert hyperopt.max_open_trades == hyperopt_conf['max_open_trades']
|
2019-09-04 21:38:15 +00:00
|
|
|
assert hasattr(hyperopt, "position_stacking")
|
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_simplified_interface_all_failed(mocker, hyperopt_conf) -> None:
|
2019-09-04 21:38:15 +00:00
|
|
|
mocker.patch('freqtrade.optimize.hyperopt.dump', MagicMock())
|
2019-10-27 09:56:38 +00:00
|
|
|
mocker.patch('freqtrade.optimize.backtesting.Backtesting.load_bt_data',
|
|
|
|
MagicMock(return_value=(MagicMock(), None)))
|
2019-09-04 21:38:15 +00:00
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-09-04 21:38:15 +00:00
|
|
|
MagicMock(return_value=(datetime(2017, 12, 10), datetime(2017, 12, 13)))
|
|
|
|
)
|
|
|
|
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt_conf.update({'spaces': 'all', })
|
2019-09-04 21:38:15 +00:00
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt = Hyperopt(hyperopt_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
hyperopt.backtesting.strategy.ohlcvdata_to_dataframe = MagicMock()
|
2019-09-04 21:38:15 +00:00
|
|
|
hyperopt.custom_hyperopt.generate_roi_table = MagicMock(return_value={})
|
|
|
|
|
|
|
|
del hyperopt.custom_hyperopt.__class__.buy_strategy_generator
|
|
|
|
del hyperopt.custom_hyperopt.__class__.sell_strategy_generator
|
|
|
|
del hyperopt.custom_hyperopt.__class__.indicator_space
|
|
|
|
del hyperopt.custom_hyperopt.__class__.sell_indicator_space
|
|
|
|
|
|
|
|
with pytest.raises(OperationalException, match=r"The 'buy' space is included into *"):
|
|
|
|
hyperopt.start()
|
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_simplified_interface_buy(mocker, hyperopt_conf, capsys) -> None:
|
2019-09-04 21:38:15 +00:00
|
|
|
dumper = mocker.patch('freqtrade.optimize.hyperopt.dump', MagicMock())
|
2019-10-27 09:56:38 +00:00
|
|
|
mocker.patch('freqtrade.optimize.backtesting.Backtesting.load_bt_data',
|
|
|
|
MagicMock(return_value=(MagicMock(), None)))
|
2019-09-04 21:38:15 +00:00
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-09-04 21:38:15 +00:00
|
|
|
MagicMock(return_value=(datetime(2017, 12, 10), datetime(2017, 12, 13)))
|
|
|
|
)
|
|
|
|
|
|
|
|
parallel = mocker.patch(
|
|
|
|
'freqtrade.optimize.hyperopt.Hyperopt.run_optimizer_parallel',
|
2020-03-01 02:11:00 +00:00
|
|
|
MagicMock(return_value=[{
|
|
|
|
'loss': 1, 'results_explanation': 'foo result', 'params': {},
|
|
|
|
'results_metrics':
|
|
|
|
{
|
|
|
|
'trade_count': 1,
|
|
|
|
'avg_profit': 0.1,
|
|
|
|
'total_profit': 0.001,
|
|
|
|
'profit': 1.0,
|
|
|
|
'duration': 20.0
|
|
|
|
}
|
|
|
|
}])
|
2019-09-04 21:38:15 +00:00
|
|
|
)
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt_conf.update({'spaces': 'buy'})
|
2019-09-04 21:38:15 +00:00
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt = Hyperopt(hyperopt_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
hyperopt.backtesting.strategy.ohlcvdata_to_dataframe = MagicMock()
|
2019-09-04 21:38:15 +00:00
|
|
|
hyperopt.custom_hyperopt.generate_roi_table = MagicMock(return_value={})
|
|
|
|
|
|
|
|
# TODO: sell_strategy_generator() is actually not called because
|
|
|
|
# run_optimizer_parallel() is mocked
|
|
|
|
del hyperopt.custom_hyperopt.__class__.sell_strategy_generator
|
|
|
|
del hyperopt.custom_hyperopt.__class__.sell_indicator_space
|
|
|
|
|
|
|
|
hyperopt.start()
|
|
|
|
|
|
|
|
parallel.assert_called_once()
|
|
|
|
|
|
|
|
out, err = capsys.readouterr()
|
|
|
|
assert 'Best result:\n\n* 1/1: foo result Objective: 1.00000\n' in out
|
|
|
|
assert dumper.called
|
2020-03-08 10:35:31 +00:00
|
|
|
# Should be called twice, once for historical candle data, once to save evaluations
|
2019-09-04 21:38:15 +00:00
|
|
|
assert dumper.call_count == 2
|
2019-09-18 19:57:37 +00:00
|
|
|
assert hasattr(hyperopt.backtesting.strategy, "advise_sell")
|
|
|
|
assert hasattr(hyperopt.backtesting.strategy, "advise_buy")
|
2019-09-04 21:38:15 +00:00
|
|
|
assert hasattr(hyperopt, "max_open_trades")
|
2020-08-06 06:51:01 +00:00
|
|
|
assert hyperopt.max_open_trades == hyperopt_conf['max_open_trades']
|
2019-09-04 21:38:15 +00:00
|
|
|
assert hasattr(hyperopt, "position_stacking")
|
|
|
|
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_simplified_interface_sell(mocker, hyperopt_conf, capsys) -> None:
|
2019-09-04 21:38:15 +00:00
|
|
|
dumper = mocker.patch('freqtrade.optimize.hyperopt.dump', MagicMock())
|
2019-10-27 09:56:38 +00:00
|
|
|
mocker.patch('freqtrade.optimize.backtesting.Backtesting.load_bt_data',
|
|
|
|
MagicMock(return_value=(MagicMock(), None)))
|
2019-09-04 21:38:15 +00:00
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-09-04 21:38:15 +00:00
|
|
|
MagicMock(return_value=(datetime(2017, 12, 10), datetime(2017, 12, 13)))
|
|
|
|
)
|
|
|
|
|
|
|
|
parallel = mocker.patch(
|
|
|
|
'freqtrade.optimize.hyperopt.Hyperopt.run_optimizer_parallel',
|
2020-03-01 02:11:00 +00:00
|
|
|
MagicMock(return_value=[{
|
|
|
|
'loss': 1, 'results_explanation': 'foo result', 'params': {},
|
|
|
|
'results_metrics':
|
|
|
|
{
|
|
|
|
'trade_count': 1,
|
|
|
|
'avg_profit': 0.1,
|
|
|
|
'total_profit': 0.001,
|
|
|
|
'profit': 1.0,
|
|
|
|
'duration': 20.0
|
|
|
|
}
|
|
|
|
}])
|
2019-09-04 21:38:15 +00:00
|
|
|
)
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt_conf.update({'spaces': 'sell', })
|
2019-09-04 21:38:15 +00:00
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt = Hyperopt(hyperopt_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
hyperopt.backtesting.strategy.ohlcvdata_to_dataframe = MagicMock()
|
2019-09-04 21:38:15 +00:00
|
|
|
hyperopt.custom_hyperopt.generate_roi_table = MagicMock(return_value={})
|
|
|
|
|
|
|
|
# TODO: buy_strategy_generator() is actually not called because
|
|
|
|
# run_optimizer_parallel() is mocked
|
|
|
|
del hyperopt.custom_hyperopt.__class__.buy_strategy_generator
|
|
|
|
del hyperopt.custom_hyperopt.__class__.indicator_space
|
|
|
|
|
|
|
|
hyperopt.start()
|
|
|
|
|
|
|
|
parallel.assert_called_once()
|
|
|
|
|
|
|
|
out, err = capsys.readouterr()
|
|
|
|
assert 'Best result:\n\n* 1/1: foo result Objective: 1.00000\n' in out
|
|
|
|
assert dumper.called
|
2020-03-08 10:35:31 +00:00
|
|
|
# Should be called twice, once for historical candle data, once to save evaluations
|
2019-09-04 21:38:15 +00:00
|
|
|
assert dumper.call_count == 2
|
2019-09-18 19:57:37 +00:00
|
|
|
assert hasattr(hyperopt.backtesting.strategy, "advise_sell")
|
|
|
|
assert hasattr(hyperopt.backtesting.strategy, "advise_buy")
|
2019-09-04 21:38:15 +00:00
|
|
|
assert hasattr(hyperopt, "max_open_trades")
|
2020-08-06 06:51:01 +00:00
|
|
|
assert hyperopt.max_open_trades == hyperopt_conf['max_open_trades']
|
2019-09-04 21:38:15 +00:00
|
|
|
assert hasattr(hyperopt, "position_stacking")
|
|
|
|
|
|
|
|
|
2019-09-06 12:08:44 +00:00
|
|
|
@pytest.mark.parametrize("method,space", [
|
|
|
|
('buy_strategy_generator', 'buy'),
|
|
|
|
('indicator_space', 'buy'),
|
|
|
|
('sell_strategy_generator', 'sell'),
|
|
|
|
('sell_indicator_space', 'sell'),
|
|
|
|
])
|
2020-08-06 06:51:01 +00:00
|
|
|
def test_simplified_interface_failed(mocker, hyperopt_conf, method, space) -> None:
|
2019-09-04 21:38:15 +00:00
|
|
|
mocker.patch('freqtrade.optimize.hyperopt.dump', MagicMock())
|
2019-10-27 09:56:38 +00:00
|
|
|
mocker.patch('freqtrade.optimize.backtesting.Backtesting.load_bt_data',
|
|
|
|
MagicMock(return_value=(MagicMock(), None)))
|
2019-09-04 21:38:15 +00:00
|
|
|
mocker.patch(
|
2019-12-17 22:06:03 +00:00
|
|
|
'freqtrade.optimize.hyperopt.get_timerange',
|
2019-09-04 21:38:15 +00:00
|
|
|
MagicMock(return_value=(datetime(2017, 12, 10), datetime(2017, 12, 13)))
|
|
|
|
)
|
|
|
|
|
|
|
|
patch_exchange(mocker)
|
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt_conf.update({'spaces': space})
|
2019-09-04 21:38:15 +00:00
|
|
|
|
2020-08-06 06:51:01 +00:00
|
|
|
hyperopt = Hyperopt(hyperopt_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
hyperopt.backtesting.strategy.ohlcvdata_to_dataframe = MagicMock()
|
2019-09-04 21:38:15 +00:00
|
|
|
hyperopt.custom_hyperopt.generate_roi_table = MagicMock(return_value={})
|
|
|
|
|
2019-09-06 12:08:44 +00:00
|
|
|
delattr(hyperopt.custom_hyperopt.__class__, method)
|
2019-09-04 21:38:15 +00:00
|
|
|
|
2019-09-06 12:08:44 +00:00
|
|
|
with pytest.raises(OperationalException, match=f"The '{space}' space is included into *"):
|
2019-09-04 21:38:15 +00:00
|
|
|
hyperopt.start()
|