stable/freqtrade/strategy/resolver.py

167 lines
6.0 KiB
Python
Raw Normal View History

# pragma pylint: disable=attribute-defined-outside-init
2018-01-28 05:26:57 +00:00
"""
This module load custom strategies
"""
import importlib.util
import inspect
2018-03-25 19:37:14 +00:00
import logging
2018-01-15 08:35:11 +00:00
import os
2018-02-06 14:31:50 +00:00
from collections import OrderedDict
from typing import Optional, Dict, Type
2018-03-17 21:44:47 +00:00
2018-01-15 08:35:11 +00:00
from pandas import DataFrame
2018-03-17 21:44:47 +00:00
2018-02-07 04:22:17 +00:00
from freqtrade.constants import Constants
2018-01-15 08:35:11 +00:00
from freqtrade.strategy.interface import IStrategy
2018-03-25 19:37:14 +00:00
logger = logging.getLogger(__name__)
2018-03-24 20:56:20 +00:00
2018-03-24 17:11:21 +00:00
class StrategyResolver(object):
2018-01-28 05:26:57 +00:00
"""
This class contains all the logic to load custom strategy class
"""
2018-03-24 17:14:05 +00:00
def __init__(self, config: Optional[Dict] = None) -> None:
2018-01-28 05:26:57 +00:00
"""
Load the custom class from config parameter
:param config:
:return:
"""
2018-03-24 17:14:05 +00:00
config = config or {}
2018-01-15 08:35:11 +00:00
# Verify the strategy is in the configuration, otherwise fallback to the default strategy
if 'strategy' in config:
strategy = config['strategy']
else:
2018-02-07 04:22:17 +00:00
strategy = Constants.DEFAULT_STRATEGY
2018-01-15 08:35:11 +00:00
# Try to load the strategy
2018-01-15 08:35:11 +00:00
self._load_strategy(strategy)
# Set attributes
# Check if we need to override configuration
if 'minimal_roi' in config:
self.custom_strategy.minimal_roi = config['minimal_roi']
2018-03-24 20:56:20 +00:00
logger.info("Override strategy \'minimal_roi\' with value in config file.")
2018-01-15 08:35:11 +00:00
if 'stoploss' in config:
self.custom_strategy.stoploss = config['stoploss']
2018-03-24 20:56:20 +00:00
logger.info(
2018-01-28 05:26:57 +00:00
"Override strategy \'stoploss\' with value in config file: %s.", config['stoploss']
)
if 'ticker_interval' in config:
self.custom_strategy.ticker_interval = config['ticker_interval']
2018-03-24 20:56:20 +00:00
logger.info(
2018-01-28 05:26:57 +00:00
"Override strategy \'ticker_interval\' with value in config file: %s.",
config['ticker_interval']
)
2018-01-15 08:35:11 +00:00
2018-01-28 05:26:57 +00:00
# Minimal ROI designed for the strategy
2018-02-06 14:31:50 +00:00
self.minimal_roi = OrderedDict(sorted(
{int(key): value for (key, value) in self.custom_strategy.minimal_roi.items()}.items(),
2018-03-17 23:01:22 +00:00
key=lambda t: t[0])) # sort after converting to number
2018-01-28 05:26:57 +00:00
# Optimal stoploss designed for the strategy
self.stoploss = float(self.custom_strategy.stoploss)
2018-01-28 05:26:57 +00:00
self.ticker_interval = int(self.custom_strategy.ticker_interval)
2018-01-15 08:35:11 +00:00
def _load_strategy(self, strategy_name: str) -> None:
"""
Search and loads the specified strategy.
2018-01-15 08:35:11 +00:00
:param strategy_name: name of the module to import
:return: None
"""
try:
current_path = os.path.dirname(os.path.realpath(__file__))
abs_paths = [
os.path.join(current_path, '..', '..', 'user_data', 'strategies'),
current_path,
]
for path in abs_paths:
self.custom_strategy = self._search_strategy(path, strategy_name)
if self.custom_strategy:
2018-03-24 20:56:20 +00:00
logger.info('Using resolved strategy %s from \'%s\'', strategy_name, path)
return None
raise ImportError('not found')
2018-01-15 08:35:11 +00:00
# Fallback to the default strategy
except (ImportError, TypeError) as error:
2018-03-24 20:56:20 +00:00
logger.error(
"Impossible to load Strategy '%s'. This class does not exist"
" or contains Python code errors",
strategy_name
)
2018-03-24 20:56:20 +00:00
logger.error(
"The error is:\n%s.",
error
)
2018-01-15 08:35:11 +00:00
@staticmethod
def _get_valid_strategies(module_path: str, strategy_name: str) -> Optional[Type[IStrategy]]:
2018-01-15 08:35:11 +00:00
"""
Returns a list of all possible strategies for the given module_path
:param module_path: absolute path to the module
:param strategy_name: Class name of the strategy
:return: Tuple with (name, class) or None
2018-01-15 08:35:11 +00:00
"""
# Generate spec based on absolute path
spec = importlib.util.spec_from_file_location('user_data.strategies', module_path)
module = importlib.util.module_from_spec(spec)
spec.loader.exec_module(module)
2018-01-15 08:35:11 +00:00
valid_strategies_gen = (
obj for name, obj in inspect.getmembers(module, inspect.isclass)
if strategy_name == name and IStrategy in obj.__bases__
)
return next(valid_strategies_gen, None)
2018-01-15 08:35:11 +00:00
2018-03-24 20:56:20 +00:00
@staticmethod
def _search_strategy(directory: str, strategy_name: str) -> Optional[IStrategy]:
2018-01-15 08:35:11 +00:00
"""
Search for the strategy_name in the given directory
:param directory: relative or absolute directory path
:return: name of the strategy class
2018-01-15 08:35:11 +00:00
"""
2018-03-24 20:56:20 +00:00
logger.debug('Searching for strategy %s in \'%s\'', strategy_name, directory)
for entry in os.listdir(directory):
# Only consider python files
if not entry.endswith('.py'):
2018-03-24 20:56:20 +00:00
logger.debug('Ignoring %s', entry)
continue
strategy = StrategyResolver._get_valid_strategies(
os.path.abspath(os.path.join(directory, entry)), strategy_name
)
if strategy:
return strategy()
return None
2018-01-15 08:35:11 +00:00
def populate_indicators(self, dataframe: DataFrame) -> DataFrame:
"""
Populate indicators that will be used in the Buy and Sell strategy
:param dataframe: Raw data from the exchange and parsed by parse_ticker_dataframe()
:return: a Dataframe with all mandatory indicators for the strategies
"""
return self.custom_strategy.populate_indicators(dataframe)
def populate_buy_trend(self, dataframe: DataFrame) -> DataFrame:
"""
Based on TA indicators, populates the buy signal for the given dataframe
:param dataframe: DataFrame
:return: DataFrame with buy column
:return:
"""
return self.custom_strategy.populate_buy_trend(dataframe)
def populate_sell_trend(self, dataframe: DataFrame) -> DataFrame:
"""
Based on TA indicators, populates the sell signal for the given dataframe
:param dataframe: DataFrame
:return: DataFrame with buy column
"""
return self.custom_strategy.populate_sell_trend(dataframe)