freqtrade_origin/freqtrade/resolvers/iresolver.py

291 lines
10 KiB
Python
Raw Normal View History

# pragma pylint: disable=attribute-defined-outside-init
"""
This module load custom objects
"""
2024-05-12 14:21:12 +00:00
import importlib.util
import inspect
import logging
import sys
from collections.abc import Iterator
2018-11-24 19:39:16 +00:00
from pathlib import Path
from typing import Any, Optional, Union
2022-09-18 11:31:52 +00:00
from freqtrade.constants import Config
from freqtrade.exceptions import OperationalException
2019-12-24 12:54:46 +00:00
2020-09-28 17:39:41 +00:00
logger = logging.getLogger(__name__)
class PathModifier:
def __init__(self, path: Path):
self.path = path
def __enter__(self):
"""Inject path to allow importing with relative imports."""
sys.path.insert(0, str(self.path))
return self
def __exit__(self, exc_type, exc_val, exc_tb):
"""Undo insertion of local path."""
str_path = str(self.path)
if str_path in sys.path:
sys.path.remove(str_path)
2019-09-12 01:39:52 +00:00
class IResolver:
"""
This class contains all the logic to load custom classes
"""
2024-05-12 14:21:12 +00:00
2019-12-24 12:34:37 +00:00
# Childclasses need to override this
object_type: type[Any]
2019-12-24 12:54:46 +00:00
object_type_str: str
user_subdir: Optional[str] = None
initial_search_path: Optional[Path] = None
# Optional config setting containing a path (strategy_path, freqaimodel_path)
extra_path: Optional[str] = None
2019-12-24 12:54:46 +00:00
@classmethod
2024-05-12 14:21:12 +00:00
def build_search_paths(
cls,
config: Config,
user_subdir: Optional[str] = None,
extra_dirs: Optional[list[str]] = None,
) -> list[Path]:
abs_paths: list[Path] = []
if cls.initial_search_path:
abs_paths.append(cls.initial_search_path)
2019-10-30 14:55:35 +00:00
if user_subdir:
2024-05-12 14:21:12 +00:00
abs_paths.insert(0, config["user_data_dir"].joinpath(user_subdir))
2022-04-23 07:10:15 +00:00
# Add extra directory to the top of the search paths
if extra_dirs:
2024-07-05 06:49:27 +00:00
for directory in extra_dirs:
abs_paths.insert(0, Path(directory).resolve())
if cls.extra_path and (extra := config.get(cls.extra_path)):
abs_paths.insert(0, Path(extra).resolve())
return abs_paths
2019-12-24 12:34:37 +00:00
@classmethod
2024-05-12 14:21:12 +00:00
def _get_valid_object(
cls, module_path: Path, object_name: Optional[str], enum_failed: bool = False
) -> Iterator[Any]:
"""
Generator returning objects with matching object_type and object_name in the path given.
:param module_path: absolute path to the module
:param object_name: Class name of the object
2020-02-14 18:15:36 +00:00
:param enum_failed: If True, will return None for modules which fail.
Otherwise, failing modules are skipped.
2020-09-17 05:38:56 +00:00
:return: generator containing tuple of matching objects
Tuple format: [Object, source]
"""
# Generate spec based on absolute path
# Pass object_name as first argument to have logging print a reasonable name.
with PathModifier(module_path.parent):
2022-03-20 12:14:52 +00:00
module_name = module_path.stem or ""
spec = importlib.util.spec_from_file_location(module_name, str(module_path))
if not spec:
2020-02-15 03:18:00 +00:00
return iter([None])
module = importlib.util.module_from_spec(spec)
try:
spec.loader.exec_module(module) # type: ignore # importlib does not use typehints
2024-05-12 14:21:12 +00:00
except (
AttributeError,
ModuleNotFoundError,
SyntaxError,
ImportError,
NameError,
) as err:
# Catch errors in case a specific module is not installed
logger.warning(f"Could not import {module_path} due to '{err}'")
if enum_failed:
return iter([None])
def is_valid_class(obj):
try:
return (
inspect.isclass(obj)
and issubclass(obj, cls.object_type)
and obj is not cls.object_type
and obj.__module__ == module_name
)
except TypeError:
return False
valid_objects_gen = (
2024-05-12 14:21:12 +00:00
(obj, inspect.getsource(module))
for name, obj in inspect.getmembers(module, is_valid_class)
if (object_name is None or object_name == name)
)
# The __module__ check ensures we only use strategies that are defined in this folder.
return valid_objects_gen
2019-12-24 12:34:37 +00:00
@classmethod
2024-05-12 14:21:12 +00:00
def _search_object(
cls, directory: Path, *, object_name: str, add_source: bool = False
) -> Union[tuple[Any, Path], tuple[None, None]]:
"""
Search for the objectname in the given directory
:param directory: relative or absolute directory path
2019-12-24 12:34:37 +00:00
:param object_name: ClassName of the object to load
:return: object class
"""
2019-12-24 14:28:35 +00:00
logger.debug(f"Searching for {cls.object_type.__name__} {object_name} in '{directory}'")
2018-11-24 19:39:16 +00:00
for entry in directory.iterdir():
# Only consider python files
2024-05-12 14:21:12 +00:00
if entry.suffix != ".py":
logger.debug("Ignoring %s", entry)
continue
if entry.is_symlink() and not entry.is_file():
2024-05-12 14:21:12 +00:00
logger.debug("Ignoring broken symlink %s", entry)
continue
module_path = entry.resolve()
2020-02-15 17:43:11 +00:00
obj = next(cls._get_valid_object(module_path, object_name), None)
if obj:
2020-09-17 05:38:56 +00:00
obj[0].__file__ = str(entry)
if add_source:
obj[0].__source__ = obj[1]
2020-09-17 04:56:51 +00:00
return (obj[0], module_path)
2019-07-12 20:45:49 +00:00
return (None, None)
2019-12-24 12:34:37 +00:00
@classmethod
2024-05-12 14:21:12 +00:00
def _load_object(
cls, paths: list[Path], *, object_name: str, add_source: bool = False, kwargs: dict
2024-05-12 14:21:12 +00:00
) -> Optional[Any]:
"""
Try to load object from path list.
"""
for _path in paths:
try:
2024-05-12 14:21:12 +00:00
(module, module_path) = cls._search_object(
directory=_path, object_name=object_name, add_source=add_source
)
if module:
logger.info(
2019-12-24 12:34:37 +00:00
f"Using resolved {cls.object_type.__name__.lower()[1:]} {object_name} "
2024-05-12 14:21:12 +00:00
f"from '{module_path}'..."
)
return module(**kwargs)
except FileNotFoundError:
logger.warning('Path "%s" does not exist.', _path.resolve())
return None
2019-12-24 12:54:46 +00:00
@classmethod
2024-05-12 14:21:12 +00:00
def load_object(
cls, object_name: str, config: Config, *, kwargs: dict, extra_dir: Optional[str] = None
) -> Any:
2019-12-24 12:54:46 +00:00
"""
2024-04-18 20:51:25 +00:00
Search and loads the specified object as configured in the child class.
2021-06-25 17:13:31 +00:00
:param object_name: name of the module to import
2019-12-24 12:54:46 +00:00
:param config: configuration dictionary
:param extra_dir: additional directory to search for the given pairlist
:raises: OperationalException if the class is invalid or does not exist.
:return: Object instance or None
"""
extra_dirs: list[str] = []
if extra_dir:
extra_dirs.append(extra_dir)
2024-05-12 14:21:12 +00:00
abs_paths = cls.build_search_paths(
config, user_subdir=cls.user_subdir, extra_dirs=extra_dirs
)
2019-12-24 12:54:46 +00:00
2024-05-12 14:21:12 +00:00
found_object = cls._load_object(paths=abs_paths, object_name=object_name, kwargs=kwargs)
2020-09-17 04:56:51 +00:00
if found_object:
return found_object
2019-12-24 12:54:46 +00:00
raise OperationalException(
f"Impossible to load {cls.object_type_str} '{object_name}'. This class does not exist "
"or contains Python code errors."
)
2019-12-24 14:28:35 +00:00
@classmethod
2024-05-12 14:21:12 +00:00
def search_all_objects(
cls, config: Config, enum_failed: bool, recursive: bool = False
) -> list[dict[str, Any]]:
2019-12-24 14:28:35 +00:00
"""
2022-10-14 14:32:30 +00:00
Searches for valid objects
:param config: Config object
:param enum_failed: If True, will return None for modules which fail.
Otherwise, failing modules are skipped.
:param recursive: Recursively walk directory tree searching for strategies
:return: List of dicts containing 'name', 'class' and 'location' entries
"""
result = []
abs_paths = cls.build_search_paths(config, user_subdir=cls.user_subdir)
for path in abs_paths:
result.extend(cls._search_all_objects(path, enum_failed, recursive))
return result
@classmethod
def _build_rel_location(cls, directory: Path, entry: Path) -> str:
builtin = cls.initial_search_path == directory
2024-05-12 14:21:12 +00:00
return (
f"<builtin>/{entry.relative_to(directory)}"
if builtin
else str(entry.relative_to(directory))
)
2022-10-14 14:32:30 +00:00
@classmethod
2022-10-14 14:41:25 +00:00
def _search_all_objects(
2024-05-12 14:21:12 +00:00
cls,
directory: Path,
enum_failed: bool,
recursive: bool = False,
basedir: Optional[Path] = None,
) -> list[dict[str, Any]]:
2022-10-14 14:32:30 +00:00
"""
2019-12-24 14:28:35 +00:00
Searches a directory for valid objects
:param directory: Path to search
2020-02-14 18:15:36 +00:00
:param enum_failed: If True, will return None for modules which fail.
Otherwise, failing modules are skipped.
:param recursive: Recursively walk directory tree searching for strategies
2021-06-25 13:45:49 +00:00
:return: List of dicts containing 'name', 'class' and 'location' entries
2019-12-24 14:28:35 +00:00
"""
logger.debug(f"Searching for {cls.object_type.__name__} '{directory}'")
objects: list[dict[str, Any]] = []
if not directory.is_dir():
logger.info(f"'{directory}' is not a directory, skipping.")
return objects
2019-12-24 14:28:35 +00:00
for entry in directory.iterdir():
if (
2024-05-12 14:21:12 +00:00
recursive
and entry.is_dir()
and not entry.name.startswith("__")
and not entry.name.startswith(".")
):
2024-05-12 14:21:12 +00:00
objects.extend(
cls._search_all_objects(entry, enum_failed, recursive, basedir or directory)
)
2019-12-24 14:28:35 +00:00
# Only consider python files
2024-05-12 14:21:12 +00:00
if entry.suffix != ".py":
logger.debug("Ignoring %s", entry)
2019-12-24 14:28:35 +00:00
continue
module_path = entry.resolve()
logger.debug(f"Path {module_path}")
2024-05-12 14:21:12 +00:00
for obj in cls._get_valid_object(
module_path, object_name=None, enum_failed=enum_failed
):
2019-12-24 14:28:35 +00:00
objects.append(
2024-05-12 14:21:12 +00:00
{
"name": obj[0].__name__ if obj is not None else "",
"class": obj[0] if obj is not None else None,
"location": entry,
"location_rel": cls._build_rel_location(basedir or directory, entry),
}
)
2019-12-24 14:28:35 +00:00
return objects