2018-11-07 18:46:04 +00:00
|
|
|
# pragma pylint: disable=missing-docstring, invalid-name, pointless-string-statement
|
2020-09-28 17:39:41 +00:00
|
|
|
# isort: skip_file
|
2018-11-07 18:46:04 +00:00
|
|
|
|
2019-11-21 05:49:16 +00:00
|
|
|
# --- Do not remove these libs ---
|
2018-11-07 18:46:04 +00:00
|
|
|
from functools import reduce
|
2019-07-15 19:35:42 +00:00
|
|
|
from typing import Any, Callable, Dict, List
|
2018-11-07 18:46:04 +00:00
|
|
|
|
2019-11-02 10:11:13 +00:00
|
|
|
import numpy as np # noqa
|
2019-11-21 05:49:16 +00:00
|
|
|
import pandas as pd # noqa
|
2019-07-15 19:35:42 +00:00
|
|
|
from pandas import DataFrame
|
2019-11-02 10:11:13 +00:00
|
|
|
from skopt.space import Categorical, Dimension, Integer, Real # noqa
|
2018-11-07 18:46:04 +00:00
|
|
|
|
2018-11-20 16:40:45 +00:00
|
|
|
from freqtrade.optimize.hyperopt_interface import IHyperOpt
|
2018-11-07 18:46:04 +00:00
|
|
|
|
2019-11-21 05:49:16 +00:00
|
|
|
# --------------------------------
|
|
|
|
# Add your lib to import here
|
|
|
|
import talib.abstract as ta # noqa
|
|
|
|
import freqtrade.vendor.qtpylib.indicators as qtpylib
|
|
|
|
|
2018-11-07 18:46:04 +00:00
|
|
|
|
2019-11-13 08:38:06 +00:00
|
|
|
class SampleHyperOpt(IHyperOpt):
|
2018-11-07 18:46:04 +00:00
|
|
|
"""
|
2019-08-11 23:19:50 +00:00
|
|
|
This is a sample Hyperopt to inspire you.
|
2019-08-05 14:54:53 +00:00
|
|
|
|
2020-03-09 14:04:28 +00:00
|
|
|
More information in the documentation: https://www.freqtrade.io/en/latest/hyperopt/
|
2019-08-05 14:54:53 +00:00
|
|
|
|
2019-08-08 19:45:37 +00:00
|
|
|
You should:
|
|
|
|
- Rename the class name to some unique name.
|
2019-08-05 14:54:53 +00:00
|
|
|
- Add any methods you want to build your hyperopt.
|
|
|
|
- Add any lib you need to build your hyperopt.
|
|
|
|
|
2020-03-09 14:04:28 +00:00
|
|
|
An easier way to get a new hyperopt file is by using
|
|
|
|
`freqtrade new-hyperopt --hyperopt MyCoolHyperopt`.
|
|
|
|
|
2019-08-05 14:54:53 +00:00
|
|
|
You must keep:
|
|
|
|
- The prototypes for the methods: populate_indicators, indicator_space, buy_strategy_generator.
|
|
|
|
|
2020-03-09 14:04:28 +00:00
|
|
|
The methods roi_space, generate_roi_table and stoploss_space are not required
|
|
|
|
and are provided by default.
|
|
|
|
However, you may override them if you need 'roi' and 'stoploss' spaces that
|
|
|
|
differ from the defaults offered by Freqtrade.
|
|
|
|
Sample implementation of these methods will be copied to `user_data/hyperopts` when
|
|
|
|
creating the user-data directory using `freqtrade create-userdir --userdir user_data`,
|
|
|
|
or is available online under the following URL:
|
|
|
|
https://github.com/freqtrade/freqtrade/blob/develop/freqtrade/templates/sample_hyperopt_advanced.py.
|
2018-11-07 18:46:04 +00:00
|
|
|
"""
|
|
|
|
|
2021-03-14 18:49:46 +00:00
|
|
|
@staticmethod
|
2021-08-21 23:06:04 +00:00
|
|
|
def indicator_space() -> List[Dimension]:
|
2021-03-14 18:49:46 +00:00
|
|
|
"""
|
|
|
|
Define your Hyperopt space for searching buy strategy parameters.
|
|
|
|
"""
|
|
|
|
return [
|
|
|
|
Integer(10, 25, name='mfi-value'),
|
|
|
|
Integer(15, 45, name='fastd-value'),
|
|
|
|
Integer(20, 50, name='adx-value'),
|
|
|
|
Integer(20, 40, name='rsi-value'),
|
|
|
|
Categorical([True, False], name='mfi-enabled'),
|
|
|
|
Categorical([True, False], name='fastd-enabled'),
|
|
|
|
Categorical([True, False], name='adx-enabled'),
|
|
|
|
Categorical([True, False], name='rsi-enabled'),
|
2021-08-18 12:03:44 +00:00
|
|
|
Categorical(['boll', 'macd_cross_signal', 'sar_reversal'], name='trigger')
|
2021-03-14 18:49:46 +00:00
|
|
|
]
|
|
|
|
|
2018-11-20 18:41:07 +00:00
|
|
|
@staticmethod
|
|
|
|
def buy_strategy_generator(params: Dict[str, Any]) -> Callable:
|
2018-11-07 18:46:04 +00:00
|
|
|
"""
|
2019-08-11 23:19:50 +00:00
|
|
|
Define the buy strategy parameters to be used by Hyperopt.
|
2018-11-07 18:46:04 +00:00
|
|
|
"""
|
|
|
|
def populate_buy_trend(dataframe: DataFrame, metadata: dict) -> DataFrame:
|
|
|
|
"""
|
2019-08-11 23:19:50 +00:00
|
|
|
Buy strategy Hyperopt will build and use.
|
2018-11-07 18:46:04 +00:00
|
|
|
"""
|
2021-08-18 10:19:17 +00:00
|
|
|
long_conditions = []
|
2019-08-11 23:19:50 +00:00
|
|
|
|
2018-11-07 18:46:04 +00:00
|
|
|
# GUARDS AND TRENDS
|
|
|
|
if 'mfi-enabled' in params and params['mfi-enabled']:
|
2021-08-18 10:19:17 +00:00
|
|
|
long_conditions.append(dataframe['mfi'] < params['mfi-value'])
|
2018-11-07 18:46:04 +00:00
|
|
|
if 'fastd-enabled' in params and params['fastd-enabled']:
|
2021-08-18 10:19:17 +00:00
|
|
|
long_conditions.append(dataframe['fastd'] < params['fastd-value'])
|
2018-11-07 18:46:04 +00:00
|
|
|
if 'adx-enabled' in params and params['adx-enabled']:
|
2021-08-18 10:19:17 +00:00
|
|
|
long_conditions.append(dataframe['adx'] > params['adx-value'])
|
2018-11-07 18:46:04 +00:00
|
|
|
if 'rsi-enabled' in params and params['rsi-enabled']:
|
2021-08-18 10:19:17 +00:00
|
|
|
long_conditions.append(dataframe['rsi'] < params['rsi-value'])
|
2018-11-07 18:46:04 +00:00
|
|
|
|
|
|
|
# TRIGGERS
|
2019-01-06 09:30:58 +00:00
|
|
|
if 'trigger' in params:
|
2021-08-18 10:19:17 +00:00
|
|
|
if params['trigger'] == 'boll':
|
|
|
|
long_conditions.append(dataframe['close'] < dataframe['bb_lowerband'])
|
2019-01-06 09:30:58 +00:00
|
|
|
if params['trigger'] == 'macd_cross_signal':
|
2021-08-18 10:19:17 +00:00
|
|
|
long_conditions.append(qtpylib.crossed_above(
|
|
|
|
dataframe['macd'],
|
|
|
|
dataframe['macdsignal']
|
|
|
|
))
|
2019-01-06 09:30:58 +00:00
|
|
|
if params['trigger'] == 'sar_reversal':
|
2021-08-18 10:19:17 +00:00
|
|
|
long_conditions.append(qtpylib.crossed_above(
|
|
|
|
dataframe['close'],
|
|
|
|
dataframe['sar']
|
|
|
|
))
|
2018-11-07 18:46:04 +00:00
|
|
|
|
2020-03-10 15:05:33 +00:00
|
|
|
# Check that volume is not 0
|
2021-08-18 10:19:17 +00:00
|
|
|
long_conditions.append(dataframe['volume'] > 0)
|
2020-03-10 15:05:33 +00:00
|
|
|
|
2021-08-18 10:19:17 +00:00
|
|
|
if long_conditions:
|
2019-05-24 20:08:56 +00:00
|
|
|
dataframe.loc[
|
2021-08-18 10:19:17 +00:00
|
|
|
reduce(lambda x, y: x & y, long_conditions),
|
2019-05-24 20:08:56 +00:00
|
|
|
'buy'] = 1
|
2018-11-07 18:46:04 +00:00
|
|
|
|
|
|
|
return dataframe
|
|
|
|
|
|
|
|
return populate_buy_trend
|
|
|
|
|
|
|
|
@staticmethod
|
2021-03-14 18:49:46 +00:00
|
|
|
def sell_indicator_space() -> List[Dimension]:
|
2018-11-07 18:46:04 +00:00
|
|
|
"""
|
2021-03-14 18:49:46 +00:00
|
|
|
Define your Hyperopt space for searching sell strategy parameters.
|
2018-11-07 18:46:04 +00:00
|
|
|
"""
|
|
|
|
return [
|
2021-03-14 18:49:46 +00:00
|
|
|
Integer(75, 100, name='sell-mfi-value'),
|
|
|
|
Integer(50, 100, name='sell-fastd-value'),
|
|
|
|
Integer(50, 100, name='sell-adx-value'),
|
|
|
|
Integer(60, 100, name='sell-rsi-value'),
|
|
|
|
Categorical([True, False], name='sell-mfi-enabled'),
|
|
|
|
Categorical([True, False], name='sell-fastd-enabled'),
|
|
|
|
Categorical([True, False], name='sell-adx-enabled'),
|
|
|
|
Categorical([True, False], name='sell-rsi-enabled'),
|
2021-08-18 10:19:17 +00:00
|
|
|
Categorical(['sell-boll',
|
2021-03-14 18:49:46 +00:00
|
|
|
'sell-macd_cross_signal',
|
2021-08-18 10:19:17 +00:00
|
|
|
'sell-sar_reversal'],
|
|
|
|
name='sell-trigger'
|
2021-08-18 12:03:44 +00:00
|
|
|
)
|
2018-11-07 18:46:04 +00:00
|
|
|
]
|
|
|
|
|
2019-01-06 09:16:30 +00:00
|
|
|
@staticmethod
|
|
|
|
def sell_strategy_generator(params: Dict[str, Any]) -> Callable:
|
|
|
|
"""
|
2019-08-11 23:19:50 +00:00
|
|
|
Define the sell strategy parameters to be used by Hyperopt.
|
2019-01-06 09:16:30 +00:00
|
|
|
"""
|
|
|
|
def populate_sell_trend(dataframe: DataFrame, metadata: dict) -> DataFrame:
|
|
|
|
"""
|
2019-08-11 23:19:50 +00:00
|
|
|
Sell strategy Hyperopt will build and use.
|
2019-01-06 09:16:30 +00:00
|
|
|
"""
|
2021-08-18 10:19:17 +00:00
|
|
|
exit_long_conditions = []
|
2019-08-11 23:19:50 +00:00
|
|
|
|
2019-01-06 09:16:30 +00:00
|
|
|
# GUARDS AND TRENDS
|
|
|
|
if 'sell-mfi-enabled' in params and params['sell-mfi-enabled']:
|
2021-08-18 10:19:17 +00:00
|
|
|
exit_long_conditions.append(dataframe['mfi'] > params['sell-mfi-value'])
|
2019-01-06 09:16:30 +00:00
|
|
|
if 'sell-fastd-enabled' in params and params['sell-fastd-enabled']:
|
2021-08-18 10:19:17 +00:00
|
|
|
exit_long_conditions.append(dataframe['fastd'] > params['sell-fastd-value'])
|
2019-01-06 09:16:30 +00:00
|
|
|
if 'sell-adx-enabled' in params and params['sell-adx-enabled']:
|
2021-08-18 10:19:17 +00:00
|
|
|
exit_long_conditions.append(dataframe['adx'] < params['sell-adx-value'])
|
2019-01-06 09:16:30 +00:00
|
|
|
if 'sell-rsi-enabled' in params and params['sell-rsi-enabled']:
|
2021-08-18 10:19:17 +00:00
|
|
|
exit_long_conditions.append(dataframe['rsi'] > params['sell-rsi-value'])
|
2019-01-06 09:16:30 +00:00
|
|
|
|
|
|
|
# TRIGGERS
|
|
|
|
if 'sell-trigger' in params:
|
2021-08-18 10:19:17 +00:00
|
|
|
if params['sell-trigger'] == 'sell-boll':
|
|
|
|
exit_long_conditions.append(dataframe['close'] > dataframe['bb_upperband'])
|
2019-01-06 09:16:30 +00:00
|
|
|
if params['sell-trigger'] == 'sell-macd_cross_signal':
|
2021-08-18 10:19:17 +00:00
|
|
|
exit_long_conditions.append(qtpylib.crossed_above(
|
|
|
|
dataframe['macdsignal'],
|
|
|
|
dataframe['macd']
|
|
|
|
))
|
2019-01-06 09:16:30 +00:00
|
|
|
if params['sell-trigger'] == 'sell-sar_reversal':
|
2021-08-18 10:19:17 +00:00
|
|
|
exit_long_conditions.append(qtpylib.crossed_above(
|
|
|
|
dataframe['sar'],
|
|
|
|
dataframe['close']
|
|
|
|
))
|
2019-01-06 09:16:30 +00:00
|
|
|
|
2020-03-10 15:05:33 +00:00
|
|
|
# Check that volume is not 0
|
2021-08-18 10:19:17 +00:00
|
|
|
exit_long_conditions.append(dataframe['volume'] > 0)
|
2020-03-10 15:05:33 +00:00
|
|
|
|
2021-08-18 10:19:17 +00:00
|
|
|
if exit_long_conditions:
|
2019-05-24 20:08:56 +00:00
|
|
|
dataframe.loc[
|
2021-08-18 10:19:17 +00:00
|
|
|
reduce(lambda x, y: x & y, exit_long_conditions),
|
2019-05-24 20:08:56 +00:00
|
|
|
'sell'] = 1
|
2019-01-06 09:16:30 +00:00
|
|
|
|
2021-08-08 09:38:34 +00:00
|
|
|
return dataframe
|
|
|
|
|
2021-08-18 10:19:17 +00:00
|
|
|
return populate_sell_trend
|