to run this: freqtrade trade --strategy LitmusMetaStrategy --config user_data/strategies/config.LitmusMeta.json --freqaimodel LitmusMultiTargetClassifier --verbose
Timeframe
N/A
Direction
Long & Short
Stoploss
-3.0%
Trailing Stop
Yes
ROI
0m: 100.0%, 60m: 0.0%
Interface Version
N/A
Startup Candles
120
Indicators
17
freqtrade/freqtrade-strategies
Strategy 003 author@: Gerald Lonlas github@: https://github.com/freqtrade/freqtrade-strategies
import logging
import numpy as np
import pandas as pd
import pandas_ta as pta
import talib.abstract as ta
from feature_engine.creation import CyclicalFeatures
from freqtrade.strategy import IStrategy, BooleanParameter, IntParameter
from functools import reduce
from freqtrade.litmus.label_helpers import tripple_barrier
from freqtrade.litmus import indicator_helpers as ih
from pandas import DataFrame
from technical import qtpylib
logger = logging.getLogger(__name__)
# Temp
pd.set_option('display.max_rows', 500)
pd.set_option('display.max_columns', 500)
pd.set_option('display.width', 2000)
class LitmusMLDPStrategy(IStrategy):
"""
to run this:
freqtrade trade --strategy LitmusMetaStrategy
--config user_data/strategies/config.LitmusMeta.json
--freqaimodel LitmusMultiTargetClassifier --verbose
"""
plot_config = {
"main_plot": {
"mm_bb_lowerband": {"color": "grey"},
"mm_bb_upperband": {"color": "grey"},
},
"subplots": {
"do_predict": {
"do_predict": {"color": "brown"},
"DI_values": {"color": "grey"}
},
"Long": {
"a_win_long": {"color": "PaleGreen"},
"primary_enter_long_threshold": {"color": "Grey"}
},
"Short": {
"a_win_short": {"color": "Salmon"},
"primary_enter_short_threshold": {"color": "Grey"},
},
"GT": {
"primary_enter_long_tbm": {"color": "PaleGreen"},
"primary_enter_short_tbm": {"color": "Salmon"},
},
"Time": {
"total_time_&-primary_enter_long": {"color": "PaleGreen"},
"total_time_&-primary_enter_short": {"color": "Salmon"}
},
"Returns": {
"value_max_returns_&-primary_enter_long": {"color": "PaleGreen"},
"value_max_returns_&-primary_enter_short": {"color": "Salmon"}
},
"Feat": {
"num_features_selected_&-primary_enter_long": {"color": "PaleGreen"},
"num_features_selected_&-primary_enter_short": {"color": "Salmon"}
}
},
}
prot_cooldown_lookback = IntParameter(2, 48, default=2, space="protection", optimize=True)
prot_stoploss_enabled = BooleanParameter(default=False, space="protection", optimize=False)
prot_stoploss_duration = IntParameter(1, 60, default=17, space="protection", optimize=False)
# ROI table:
minimal_roi = {
"0": 1.0,
"60": 0
}
# Stoploss:
stoploss = -0.03
# Stop loss config
trailing_stop = True
# Other strategy flags
process_only_new_candles = True
use_exit_signal = True
can_short = True
startup_candle_count = 120
"""@property
def protections(self):
prot = []
prot.append({
"method": "CooldownPeriod",
"stop_duration_candles": self.prot_cooldown_lookback.value
})
if self.prot_stoploss_enabled.value:
prot.append({
"method": "StoplossGuard",
"lookback_period": 60,
"trade_limit": 2,
"stop_duration_candles": self.prot_stoploss_duration.value,
"required_profit": 0.0,
"only_per_pair": False,
"only_per_side": False
})
return prot"""
def feature_engineering_expand_all(self, dataframe, period, **kwargs):
"""
Will expand:
`indicator_periods_candles` *`include_timeframes` * `include_shifted_candles`
* `include_corr_pairs`
"""
dataframe["%-rsi"] = ta.RSI(dataframe, timeperiod=period)
dataframe["%-mfi"] = ta.MFI(dataframe, timeperiod=period)
dataframe["%-adx"] = ta.ADX(dataframe, timeperiod=period)
dataframe["%-sma"] = ta.SMA(dataframe, timeperiod=period)
dataframe["%-ema"] = ta.EMA(dataframe, timeperiod=period)
dataframe["%-cci"] = ta.CCI(dataframe, timeperiod=period)
dataframe["%-er"] = pta.er(dataframe['close'], length=period)
dataframe["%-rocr"] = ta.ROCR(dataframe, timeperiod=period)
dataframe["%-cmf"] = ih.chaikin_mf(dataframe, periods=period)
dataframe["%-tcp"] = ih.top_percent_change(dataframe, period)
dataframe["%-cti"] = pta.cti(dataframe['close'], length=period)
dataframe["%-chop"] = qtpylib.chopiness(dataframe, period)
dataframe["%-linear"] = ta.LINEARREG_ANGLE(dataframe['close'], timeperiod=period)
dataframe["%-obv"] = ta.OBV(dataframe)
bollinger = qtpylib.bollinger_bands(
qtpylib.typical_price(dataframe), window=period, stds=2.2
)
dataframe["bb_lowerband"] = bollinger["lower"]
dataframe["bb_middleband"] = bollinger["mid"]
dataframe["bb_upperband"] = bollinger["upper"]
dataframe["%-bb_width"] = (
dataframe["bb_upperband"]
- dataframe["bb_lowerband"]
) / dataframe["bb_middleband"]
dataframe["%-close-bb_lower"] = (
dataframe["close"] / dataframe["bb_lowerband"]
)
dataframe["%-roc"] = ta.ROC(dataframe, timeperiod=period)
dataframe["%-relative_volume"] = (
dataframe["volume"] / dataframe["volume"].rolling(period).mean()
)
# Absolute Price Oscillator
dataframe["%-apo"] = ta.APO(
dataframe["close"], fastperiod=int(period / 2), slowperiod=period, matype=0)
# PPO (Percentage Price Oscilator)
dataframe["%-ppo"] = ta.PPO(
dataframe["close"], fastperiod=int(period / 2), slowperiod=period, matype=0)
# MACD (macd, macdsignal, macdhist)
_, _, macdhist = ta.MACD(
dataframe["close"], fastperiod=int(period / 2),
slowperiod=period, signalperiod=int(3 * period / 4))
dataframe["%-macdhist"] = macdhist
# Average True Range
dataframe["%-atr"] = ta.ATR(
dataframe["high"], dataframe["low"], dataframe["close"], timeperiod=period)
return dataframe
def feature_engineering_expand_basic(self, dataframe, **kwargs) -> pd.DataFrame:
"""
Will expand:
`include_timeframes` * `include_shifted_candles` * `include_corr_pairs`
Will not expand: `indicator_periods_candles`
"""
t = 20
dataframe["%-raw_price"] = dataframe["close"]
dataframe["%-price_change"] = dataframe["close"].pct_change()
dataframe["%-volatility_price"] = \
dataframe["%-price_change"].rolling(t).std() * np.sqrt(t)
dataframe["%-raw_volume"] = dataframe["volume"]
dataframe["%-volume_change"] = np.log(dataframe['volume'] / dataframe['volume'].shift())
dataframe["%-volatility_volume"] = \
dataframe["%-volume_change"].rolling(t).std() * np.sqrt(t)
return dataframe
def feature_engineering_standard(self, dataframe, **kwargs) -> pd.DataFrame:
"""
*Only functional with FreqAI enabled strategies*
This optional function will be called once with the dataframe of the base timeframe.
This is the final function to be called, which means that the dataframe entering this
function will contain all the features and columns created by all other
freqai_feature_engineering_* functions.
This function is a good place to do custom exotic feature extractions (e.g. tsfresh).
This function is a good place for any feature that should not be auto-expanded upon
(e.g. day of the week).
All features must be prepended with `%` to be recognized by FreqAI internals.
More details about feature engineering available:
https://www.freqtrade.io/en/latest/freqai-feature-engineering
:param df: strategy dataframe which will receive the features
usage example: dataframe["%-day_of_week"] = (dataframe["date"].dt.dayofweek + 1) / 7
"""
# Time features
dataframe["%-day_of_week"] = dataframe["date"].dt.dayofweek
dataframe["%-hour_of_day"] = dataframe["date"].dt.hour
cyclical_transform = CyclicalFeatures(
variables=["%-day_of_week", "%-hour_of_day"], max_values=None, drop_original=True
)
dataframe = cyclical_transform.fit_transform(dataframe)
# Target features
target_params = self.freqai_info["target_parameters"]
# Indicators for primary
bollinger = qtpylib.bollinger_bands(
qtpylib.typical_price(dataframe), window=25, stds=1.9
)
dataframe["mm_bb_lowerband"] = bollinger["lower"]
dataframe["mm_bb_middleband"] = bollinger["mid"]
dataframe["mm_bb_upperband"] = bollinger["upper"]
dataframe["psar"] = ta.SAR(
dataframe["high"], dataframe["low"], acceleration=0.01, maximum=0.2)
# Primary: Enter Long
"""dataframe["primary_enter_long"] = qtpylib.crossed_above(
dataframe["psar"], dataframe["psar"].shift(1))"""
dataframe["primary_enter_long"] = dataframe["close"] < dataframe["mm_bb_lowerband"]
# Long: TBM Labeling
logger.info("Starting TBM: Long")
window = target_params["tmb_long_window"]
params = {
"upper_pct": target_params["tmb_long_upper"],
"lower_pct": target_params["tmb_long_lower"]
}
dataframe["primary_enter_long_tbm"] = (
dataframe["close"]
.shift(-window)
.rolling(window + 1)
.apply(tripple_barrier, kwargs=params)
)
# Long: Primary model performance a feature for meta model
logger.info("Starting model performance: Long")
dataframe["masked_shift_long_tbm"] = np.where(
dataframe["primary_enter_long"].shift(window),
dataframe["primary_enter_long_tbm"].shift(window),
np.nan)
dataframe["%-primary_long_perf_mean"] = dataframe["masked_shift_long_tbm"].rolling(
window=target_params["primary_perf_window"],
min_periods=0).apply(lambda x: x[~np.isnan(x)].mean()).fillna(0)
dataframe["%-primary_long_perf_std"] = dataframe["masked_shift_long_tbm"].rolling(
window=target_params["primary_perf_window"],
min_periods=0).apply(lambda x: x[~np.isnan(x)].std()).fillna(0)
dataframe["%-primary_long_perf_count"] = dataframe["masked_shift_long_tbm"].rolling(
window=target_params["primary_perf_window"],
min_periods=0).apply(lambda x: x[~np.isnan(x)].count()).fillna(0)
# Primary: Enter Short
"""dataframe["primary_enter_short"] = qtpylib.crossed_below(
dataframe["psar"], dataframe["psar"].shift(1))"""
dataframe["primary_enter_short"] = dataframe["close"] > dataframe["mm_bb_upperband"]
# Short: TBM Labeling
logger.info("Starting TBM: Short")
window = target_params["tmb_short_window"]
params = {
"upper_pct": target_params["tmb_short_upper"],
"lower_pct": target_params["tmb_short_lower"]
}
dataframe["primary_enter_short_tbm"] = (
dataframe["close"]
.shift(-window)
.rolling(window + 1)
.apply(tripple_barrier, kwargs=params)
)
# Short: Primary model performance a feature for meta model
logger.info("Starting model performance: Short")
dataframe["masked_shift_short_tbm"] = np.where(
dataframe["primary_enter_short"].shift(window),
dataframe["primary_enter_short_tbm"].shift(window),
np.nan)
dataframe["%-primary_short_perf_mean"] = dataframe["masked_shift_short_tbm"].rolling(
window=target_params["primary_perf_window"],
min_periods=0).apply(lambda x: x[~np.isnan(x)].mean()).fillna(0)
dataframe["%-primary_short_perf_std"] = dataframe["masked_shift_short_tbm"].rolling(
window=target_params["primary_perf_window"],
min_periods=0).apply(lambda x: x[~np.isnan(x)].std()).fillna(0)
dataframe["%-primary_short_perf_count"] = dataframe["masked_shift_short_tbm"].rolling(
window=target_params["primary_perf_window"],
min_periods=0).apply(lambda x: x[~np.isnan(x)].count()).fillna(0)
# Crude forward-looking return of current candle (Note: cannot be used as feature)
dataframe["!-trade_return"] = dataframe["close"].pct_change(window).shift(-window)
return dataframe
def set_freqai_targets(self, dataframe, **kwargs) -> pd.DataFrame:
"""
*Only functional with FreqAI enabled strategies*
Required function to set the targets for the model.
All targets must be prepended with `&` to be recognized by the FreqAI internals.
More details about feature engineering available:
https://www.freqtrade.io/en/latest/freqai-feature-engineering
:param df: strategy dataframe which will receive the targets
usage example: dataframe["&-target"] = dataframe["close"].shift(-1) / dataframe["close"]
"""
# Long Entry
tbm_map = {1: "a_win_long", 0: "loss_long", -1: "loss_long"}
dataframe["&-primary_enter_long"] = dataframe["primary_enter_long_tbm"].map(tbm_map)
dataframe["&-primary_enter_long"] = np.where(
dataframe["primary_enter_long"], dataframe["&-primary_enter_long"], "drop-row")
logger.info("Label counts for primary long: ")
print(dataframe.groupby("&-primary_enter_long").size())
# Short Entry
tbm_map = {1: "loss_short", 0: "loss_short", -1: "a_win_short"}
dataframe["&-primary_enter_short"] = dataframe["primary_enter_short_tbm"].map(tbm_map)
dataframe["&-primary_enter_short"] = np.where(
dataframe["primary_enter_short"], dataframe["&-primary_enter_short"], "drop-row")
logger.info("Label counts for primary short: ")
print(dataframe.groupby("&-primary_enter_short").size())
return dataframe
def populate_indicators(self, dataframe: DataFrame, metadata: dict) -> pd.DataFrame:
dataframe = self.freqai.start(dataframe, metadata, self)
# Add rolling smoothing function over entry thresholds
smoothing_window = self.freqai_info["trigger_parameters"].get("smoothing_window", 30)
dataframe["primary_enter_long_threshold"] = dataframe[
"threshold_max_returns_&-primary_enter_long"].rolling(smoothing_window).mean()
dataframe["primary_enter_short_threshold"] = dataframe[
"threshold_max_returns_&-primary_enter_short"].rolling(smoothing_window).mean()
return dataframe
def populate_entry_trend(self, df: DataFrame, metadata: dict) -> pd.DataFrame:
# Long Entry
conditions = [df["primary_enter_long"],
df["a_win_long"] >= df["primary_enter_long_threshold"]]
if conditions:
df.loc[
reduce(lambda x, y: x & y, conditions), ["enter_long", "enter_tag"]
] = (1, "primary_enter_long")
# Short Entry
conditions = [df["primary_enter_short"],
df["a_win_short"] >= df["primary_enter_short_threshold"]]
if conditions:
df.loc[
reduce(lambda x, y: x & y, conditions), ["enter_short", "enter_tag"]
] = (1, "primary_enter_short")
return df
def populate_exit_trend(self, df: DataFrame, metadata: dict) -> pd.DataFrame:
# Long Exit
conditions = [df["primary_enter_short"]]
if conditions:
df.loc[
reduce(lambda x, y: x & y, conditions), ["exit_long", "exit_tag"]
] = (1, "primary_exit_long")
# Short Exit
conditions = [df["primary_enter_long"]]
if conditions:
df.loc[
reduce(lambda x, y: x & y, conditions), ["exit_short", "exit_tag"]
] = (1, "primary_exit_short")
return df
def get_ticker_indicator(self):
return int(self.config["timeframe"][:-1])