stable/freqtrade/resolvers/iresolver.py

67 lines
2.4 KiB
Python
Raw Normal View History

# pragma pylint: disable=attribute-defined-outside-init
"""
This module load custom objects
"""
import importlib.util
import inspect
import logging
2018-11-24 19:39:16 +00:00
from pathlib import Path
2019-07-12 20:45:49 +00:00
from typing import Any, Optional, Tuple, Type, Union
logger = logging.getLogger(__name__)
class IResolver(object):
"""
This class contains all the logic to load custom classes
"""
@staticmethod
def _get_valid_object(object_type, module_path: Path,
object_name: str) -> Optional[Type[Any]]:
"""
Returns the first object with matching object_type and object_name in the path given.
:param object_type: object_type (class)
:param module_path: absolute path to the module
:param object_name: Class name of the object
:return: class or None
"""
# Generate spec based on absolute path
2018-11-24 19:39:16 +00:00
spec = importlib.util.spec_from_file_location('unknown', str(module_path))
module = importlib.util.module_from_spec(spec)
try:
spec.loader.exec_module(module) # type: ignore # importlib does not use typehints
2019-03-15 18:50:38 +00:00
except (ModuleNotFoundError, SyntaxError) as err:
# Catch errors in case a specific module is not installed
2019-03-15 18:50:38 +00:00
logger.warning(f"Could not import {module_path} due to '{err}'")
valid_objects_gen = (
obj for name, obj in inspect.getmembers(module, inspect.isclass)
if object_name == name and object_type in obj.__bases__
)
return next(valid_objects_gen, None)
@staticmethod
2018-11-24 19:39:16 +00:00
def _search_object(directory: Path, object_type, object_name: str,
2019-07-12 20:45:49 +00:00
kwargs: dict = {}) -> Union[Tuple[Any, Path], Tuple[None, None]]:
"""
Search for the objectname in the given directory
:param directory: relative or absolute directory path
:return: object instance
"""
2019-02-19 21:27:20 +00:00
logger.debug("Searching for %s %s in '%s'", object_type.__name__, object_name, directory)
2018-11-24 19:39:16 +00:00
for entry in directory.iterdir():
# Only consider python files
2018-11-24 19:39:16 +00:00
if not str(entry).endswith('.py'):
logger.debug('Ignoring %s', entry)
continue
2019-07-12 20:45:49 +00:00
module_path = Path.resolve(directory.joinpath(entry))
obj = IResolver._get_valid_object(
2019-07-12 20:45:49 +00:00
object_type, module_path, object_name
)
if obj:
2019-07-12 20:45:49 +00:00
return (obj(**kwargs), module_path)
return (None, None)