Timeframe
5m
Direction
Long Only
Stoploss
-99.0%
Trailing Stop
Yes
ROI
0m: 1.8%
Interface Version
3
Startup Candles
N/A
Indicators
5
freqtrade/freqtrade-strategies
Strategy 003 author@: Gerald Lonlas github@: https://github.com/freqtrade/freqtrade-strategies
# --- Do not remove these libs --------------------------------------------------------------------
import freqtrade.vendor.qtpylib.indicators as qtpylib
import talib.abstract as ta
import logging
import pandas as pd
import numpy as np
from pandas import DataFrame, Series, DatetimeIndex, merge
from freqtrade.exchange import timeframe_to_minutes
from freqtrade.strategy import IStrategy, IntParameter, DecimalParameter, CategoricalParameter, merge_informative_pair
from freqtrade.persistence import Trade
from datetime import datetime, timedelta
from functools import reduce
###########################################################################################################
## CombinedBinHAndClucV6 by iterativ ##
## hyperoptable additions and refactoring by nightshift2k ##
## ##
## source & discussion -> freqtrade discord: ##
## https://discord.gg/ZdsMQ3tE ##
## ##
## Freqtrade https://github.com/freqtrade/freqtrade ##
## The authors of the original CombinedBinHAndCluc https://github.com/freqtrade/freqtrade-strategies ##
## V6 by iterativ. ##
## ##
###########################################################################################################
## GENERAL RECOMMENDATIONS ##
## ##
## For optimal performance, suggested to use between 4 and 6 open trades, with unlimited stake. ##
## A pairlist with 20 to 40 pairs. Volume pairlist works well. ##
## Prefer stable coin (USDT, BUSDT etc) pairs, instead of BTC or ETH pairs. ##
## Highly recommended to blacklist leveraged tokens (*BULL, *BEAR, *UP, *DOWN etc). ##
## Ensure that you don't override any variables in you config.json. Especially ##
## the timeframe (must be 5m) & exit_profit_only (must be true). ##
## ##
###########################################################################################################
## DONATIONS ##
## ##
## Absolutely not required. However, will be accepted as a token of appreciation. ##
## ##
## BTC: bc1qvflsvddkmxh7eqhc4jyu5z5k6xcw3ay8jl49sk ##
## ETH: 0x83D3cFb8001BDC5d2211cBeBB8cB3461E5f7Ec91 ##
## ##
###########################################################################################################
# -------------------------------------------------------------------------------------------------
# --- logger for parameter merging output, only remove if you remove it further down too! ---------
logger = logging.getLogger(__name__)
# -------------------------------------------------------------------------------------------------
class CombinedBinHAndClucV6H(IStrategy):
INTERFACE_VERSION = 3
minimal_roi = {'0': 0.0181}
max_open_trades = 5
# Using custom stoploss
stoploss = -0.99
use_custom_stoploss = True
# Trailing stoploss
trailing_stop = True
trailing_only_offset_is_reached = True
trailing_stop_positive = 0.01
trailing_stop_positive_offset = 0.025
timeframe = '5m'
informative_timeframe = '1h'
# derived from EMA 200
startup_candle_count: int = 200
# Run "populate_indicators()" only for new candle.
process_only_new_candles = False
use_exit_signal = True
exit_profit_only = True
exit_profit_offset = 0.001
ignore_roi_if_entry_signal = True
# hyperspace default entry params
entry_params = {'entry_bin_bbdelta_close': 0.031, 'entry_bin_closedelta_close': 0.018, 'entry_bin_tail_bbdelta': 0.233, 'entry_bin_guard': True, 'entry_cluc_close_bblowerband': 0.993, 'entry_cluc_volume': 21, 'entry_cluc_guard': True, 'entry_long_rsi_diff': 43.276, 'entry_bin_enable': True, 'entry_cluc_enable': True, 'entry_long_enable': True, 'entry_minimum_conditions': 1}
# hyperspace default exit params
# to disable holding with high RSI set to 100
exit_params = {'exit_roi_override_rsi_threshold': 50, 'cstp_loss_threshold': 0, 'cstp_bail_time': 5}
# if you want to see which entry conditions were met
# or if there is an trade exit override due to high RSI set to True
# logger will output the entry and trade exit conditions
cust_log_verbose = False
###########################################################################################################
## IMPORTANT NOTICE REGARDING HYPEROPT: ##
## Use the optimize dict below to enable optimization (set to True) of one or more ##
## hyperspace parameters. ##
## ##
## Please be aware: Enabling ALL at once will very likely produce very low results! ##
## So please enable max. 1-2 parameters at once, you'll need to play around ##
## what makes more or less sense. At the end it always depends on a lot more ##
## parameters (stake, max trades, hyperopt-loss, etc.) ##
###########################################################################################################
# use this switches to enable the optimization of hyperspace params
cust_optimize = {'entry_bin_bbdelta_close': False, 'entry_bin_closedelta_close': False, 'entry_bin_tail_bbdelta': False, 'entry_bin_guard': False, 'entry_cluc_close_bblowerband': False, 'entry_cluc_volume': False, 'entry_cluc_guard': False, 'entry_long_rsi_diff': False, 'entry_bin_enable': False, 'entry_cluc_enable': False, 'entry_long_enable': False, 'entry_minimum_conditions': False, 'exit_roi_override_rsi_threshold': False, 'cstp_bail_time': False, 'cstp_loss_threshold': False}
# bin entry parameters
entry_bin_bbdelta_close = DecimalParameter(0.0, 0.05, default=0.031, space='entry', optimize=cust_optimize['entry_bin_bbdelta_close'], load=True)
entry_bin_closedelta_close = DecimalParameter(0.0, 0.03, default=0.018, decimals=4, space='entry', optimize=cust_optimize['entry_bin_closedelta_close'], load=True)
entry_bin_tail_bbdelta = DecimalParameter(0.0, 1.0, default=0.233, decimals=3, space='entry', optimize=cust_optimize['entry_bin_tail_bbdelta'], load=True)
entry_bin_guard = CategoricalParameter([True, False], default=True, space='entry', optimize=cust_optimize['entry_bin_guard'], load=True)
# cluc entry parameters
entry_cluc_close_bblowerband = DecimalParameter(0.0, 1.5, default=0.993, decimals=3, space='entry', optimize=cust_optimize['entry_cluc_close_bblowerband'], load=True)
entry_cluc_volume = IntParameter(10, 40, default=21, space='entry', optimize=cust_optimize['entry_cluc_volume'], load=True)
entry_cluc_guard = CategoricalParameter([True, False], default=True, space='entry', optimize=cust_optimize['entry_cluc_guard'], load=True)
# log entry parameters
entry_long_rsi_diff = DecimalParameter(40, 45, default=43.276, decimals=3, space='entry', optimize=cust_optimize['entry_long_rsi_diff'], load=True)
# enable bin, cluc or long
entry_bin_enable = CategoricalParameter([True, False], default=True, space='entry', optimize=cust_optimize['entry_bin_enable'], load=True)
entry_cluc_enable = CategoricalParameter([True, False], default=True, space='entry', optimize=cust_optimize['entry_cluc_enable'], load=True)
entry_long_enable = CategoricalParameter([True, False], default=True, space='entry', optimize=cust_optimize['entry_long_enable'], load=True)
# minimum conditions to match in entry
entry_minimum_conditions = IntParameter(1, 2, default=1, space='entry', optimize=cust_optimize['entry_minimum_conditions'], load=True)
# if RSI above threshold, override ROI
exit_roi_override_rsi_threshold = IntParameter(40, 70, default=50, space='exit', optimize=cust_optimize['exit_roi_override_rsi_threshold'], load=True)
# custom stoploss, if trade open > x hours and loss > threshold then exit
cstp_bail_time = IntParameter(1, 36, default=5, space='exit', optimize=cust_optimize['cstp_bail_time'])
cstp_loss_threshold = DecimalParameter(-0.25, 0, default=0, decimals=2, space='exit', optimize=cust_optimize['cstp_loss_threshold'])
'\n Informative Pairs\n '
def informative_pairs(self):
pairs = self.dp.current_whitelist()
informative_pairs = [(pair, self.informative_timeframe) for pair in pairs]
return informative_pairs
'\n Informative Timeframe Indicators\n '
def get_informative_indicators(self, dataframe: DataFrame, metadata: dict) -> DataFrame:
dataframe = self.dp.get_pair_dataframe(pair=metadata['pair'], timeframe=self.informative_timeframe)
# EMA
dataframe['ema_50'] = ta.EMA(dataframe, timeperiod=50)
dataframe['ema_200'] = ta.EMA(dataframe, timeperiod=200)
# RSI
dataframe['rsi'] = ta.RSI(dataframe, timeperiod=14)
# SSL Channels
ssldown, sslup = SSLChannels_ATR(dataframe, 20)
dataframe['ssl-up'] = sslup
dataframe['ssl-down'] = ssldown
dataframe['ssl-dir'] = np.where(sslup > ssldown, 'up', 'down')
return dataframe
'\n Main Timeframe Indicators\n '
def get_main_indicators(self, dataframe: DataFrame, metadata: dict) -> DataFrame:
# strategy BinHV45
bb_40 = qtpylib.bollinger_bands(dataframe['close'], window=40, stds=2)
dataframe['lower'] = bb_40['lower']
dataframe['mid'] = bb_40['mid']
dataframe['bbdelta'] = (bb_40['mid'] - dataframe['lower']).abs()
dataframe['closedelta'] = (dataframe['close'] - dataframe['close'].shift()).abs()
dataframe['tail'] = (dataframe['close'] - dataframe['low']).abs()
# strategy ClucMay72018
bollinger = qtpylib.bollinger_bands(qtpylib.typical_price(dataframe), window=20, stds=2)
dataframe['bb_lowerband'] = bollinger['lower']
dataframe['bb_middleband'] = bollinger['mid']
dataframe['bb_upperband'] = bollinger['upper']
dataframe['volume_mean_slow'] = dataframe['volume'].rolling(window=30).mean()
# EMA
dataframe['ema_50'] = ta.EMA(dataframe, timeperiod=50)
dataframe['ema_200'] = ta.EMA(dataframe, timeperiod=200)
# SMA
dataframe['sma_5'] = ta.EMA(dataframe, timeperiod=5)
# RSI
dataframe['rsi'] = ta.RSI(dataframe, timeperiod=14)
# drop NAN in hyperopt to fix "'<' not supported between instances of 'str' and 'int' error
if self.config['runmode'].value == 'hyperopt':
dataframe = dataframe.dropna()
return dataframe
'\n Populate Informative and Main Timeframe Indicators\n '
def populate_indicators(self, dataframe: DataFrame, metadata: dict) -> DataFrame:
if self.config['runmode'].value in ('backtest', 'hyperopt'):
assert timeframe_to_minutes(self.timeframe) <= 5, 'Backtest this strategy in a timeframe of 5m or less.'
assert self.dp, 'DataProvider is required for multiple timeframes.'
informative = self.get_informative_indicators(dataframe, metadata)
dataframe = merge_informative_pair(dataframe, informative, self.timeframe, self.informative_timeframe, ffill=True)
dataframe = self.get_main_indicators(dataframe, metadata)
return dataframe
'\n Buy Signal\n '
def populate_entry_trend(self, dataframe: DataFrame, metadata: dict) -> DataFrame:
conditions = []
# reset additional dataframe rows
dataframe.loc[:, 'entry_cond_bin'] = False
dataframe.loc[:, 'entry_cond_cluc'] = False
dataframe.loc[:, 'entry_cond_long'] = False
dataframe.loc[:, 'conditions_count'] = 0
# strategy BinHV45 + guards by @iterativ
# @iterativ adition (guard)
# strategy BinHV45
dataframe.loc[((dataframe['close'] > dataframe['ema_200_1h']) & (dataframe['ema_50'] > dataframe['ema_200']) & (dataframe['ema_50_1h'] > dataframe['ema_200_1h']) & (self.entry_bin_guard.value == True) | (self.entry_bin_guard.value == False)) & dataframe['lower'].shift().gt(0) & dataframe['bbdelta'].gt(dataframe['close'] * self.entry_bin_bbdelta_close.value) & dataframe['closedelta'].gt(dataframe['close'] * self.entry_bin_closedelta_close.value) & dataframe['tail'].lt(dataframe['bbdelta'] * self.entry_bin_tail_bbdelta.value) & dataframe['close'].lt(dataframe['lower'].shift()) & dataframe['close'].le(dataframe['close'].shift()) & (self.entry_bin_enable.value == True), 'entry_cond_bin'] = 1
# strategy ClucMay72018 + guards by @iterativ
# @iterativ addition (guard)
# strategy ClucMay72018
dataframe.loc[((dataframe['close'] > dataframe['ema_200']) & (dataframe['close'] > dataframe['ema_200_1h']) & (self.entry_cluc_guard.value == True) | (self.entry_cluc_guard.value == False)) & (dataframe['close'] < dataframe['ema_50']) & (dataframe['close'] < self.entry_cluc_close_bblowerband.value * dataframe['bb_lowerband']) & (dataframe['volume'] < dataframe['volume_mean_slow'].shift(1) * self.entry_cluc_volume.value) & (self.entry_cluc_enable.value == True), 'entry_cond_cluc'] = 1
# long strategy by @iterativ
dataframe.loc[(dataframe['close'] < dataframe['sma_5']) & (dataframe['ssl-dir_1h'] == 'up') & (dataframe['ema_50'] > dataframe['ema_200']) & (dataframe['ema_50_1h'] > dataframe['ema_200_1h']) & (dataframe['rsi'] < dataframe['rsi_1h'] - self.entry_long_rsi_diff.value) & (self.entry_long_enable.value == True), 'entry_cond_long'] = 1
# count the amount of conditions met
dataframe.loc[:, 'conditions_count'] = dataframe['entry_cond_bin'].astype(int) + dataframe['entry_cond_cluc'].astype(int) + dataframe['entry_cond_long'].astype(int)
# append the minimum amount of conditions to be met
conditions.append(dataframe['conditions_count'] >= self.entry_minimum_conditions.value)
conditions.append(dataframe['volume'].gt(0))
if conditions:
dataframe.loc[reduce(lambda x, y: x & y, conditions), 'enter_long'] = 1
# verbose logging enable only for verbose information or troubleshooting
if self.cust_log_verbose == True:
for index, row in dataframe.iterrows():
if row['enter_long'] == 1:
entry_cond_details = f"count={int(row['conditions_count'])}/bin={int(row['entry_cond_bin'])}/cluc={int(row['entry_cond_cluc'])}/long={int(row['entry_cond_long'])}"
logger.info(f"{metadata['pair']} - candle: {row['date']} - entry condition - details: {entry_cond_details}")
return dataframe
'\n Sell Signal\n '
def populate_exit_trend(self, dataframe: DataFrame, metadata: dict) -> DataFrame:
conditions = [] # Make sure Volume is not 0
conditions.append((dataframe['close'] > dataframe['bb_upperband']) & (dataframe['close'].shift(1) > dataframe['bb_upperband'].shift(1)) & (dataframe['volume'] > 0))
if conditions:
dataframe.loc[reduce(lambda x, y: x & y, conditions), 'exit_long'] = 1
return dataframe
'\n Custom Stop Loss\n '
def custom_stoploss(self, pair: str, trade: 'Trade', current_time: datetime, current_rate: float, current_profit: float, **kwargs) -> float:
if (current_profit < self.cstp_loss_threshold.value) & (current_time - timedelta(hours=int(self.cstp_bail_time.value)) > trade.open_date_utc):
return 0.01
return self.stoploss
'\n Trade Exit Confirmation\n '
def confirm_trade_exit(self, pair: str, trade: Trade, order_type: str, amount: float, rate: float, time_in_force: str, exit_reason: str, **kwargs) -> bool:
dataframe, _ = self.dp.get_analyzed_dataframe(pair, self.timeframe)
last_candle = dataframe.iloc[-1].squeeze()
if self.cust_log_verbose == True:
logger.info(f"{pair} - candle: {last_candle['date']} - exit trade {exit_reason} with profit {trade.calc_profit_ratio(rate)}")
# failsafe for user triggered forced exits > always have highest prio!
if exit_reason == 'force_exit':
return True
# Prevent ROI trigger, if there is more potential, in order to maximize profit
if last_candle is not None and exit_reason == 'roi':
rsi = 0
if 'rsi' in last_candle.index:
rsi = last_candle['rsi']
if rsi >= self.exit_roi_override_rsi_threshold.value:
if self.cust_log_verbose == True:
logger.info(f"{pair} - candle: {last_candle['date']} - not exiting trade with current profit {trade.calc_profit_ratio(rate)}, rsi = {rsi} which is > than {self.exit_roi_override_rsi_threshold.value}")
return False
else:
return True
return True
# --- custom indicators ---------------------------------------------------------------------------
def SSLChannels_ATR(dataframe, length=7):
"""
SSL Channels with ATR: https://www.tradingview.com/script/SKHqWzql-SSL-ATR-channel/
Credit to @JimmyNixx for python
"""
df = dataframe.copy()
df['ATR'] = ta.ATR(df, timeperiod=14)
df['smaHigh'] = df['high'].rolling(length).mean() + df['ATR']
df['smaLow'] = df['low'].rolling(length).mean() - df['ATR']
df['hlv'] = np.where(df['close'] > df['smaHigh'], 1, np.where(df['close'] < df['smaLow'], -1, np.NAN))
df['hlv'] = df['hlv'].ffill()
df['sslDown'] = np.where(df['hlv'] < 0, df['smaHigh'], df['smaLow'])
df['sslUp'] = np.where(df['hlv'] < 0, df['smaLow'], df['smaHigh'])
return (df['sslDown'], df['sslUp'])