Source code for project_composer.compose

import sys
import inspect
from pathlib import Path

from .app_storage import AppStore
from .exceptions import ComposerError
from .importer import import_module
from .logger import LoggerBase
from .manifest import Manifest
from .utils.tree_printer import TreePrinter


[docs]class Composer(LoggerBase): """ Composer base implements everything about application module discovering and manifest loading. Arguments: manifest (string or pathlib.Path or dict or Manifest): The Manifest source to load. It can be either: * A Manifest object, it will just be returned as it without any validation. You are responsible of its correctness; * A string for the file path to load in JSON or TOML format; * A Path object to the file to load in JSON or TOML format; * A Dictionnary which respect the manifest structure; Source file format are guessed from their file extension such as JSON for ``.json`` or TOML for ``.toml``. processors (list): List of available composition processors classes. Attributes: _APPLICATION_MODULE_PYTHONPATH (string): A template string to build the full Python path of founded class. It expected two variables ``parent`` and ``name``, respectively the module path and the class name. """ _APPLICATION_MODULE_PYTHONPATH = "{parent}.{name}" def __init__(self, manifest, processors=[]): super().__init__() self.manifest = self.get_manifest(manifest) self.set_syspaths(self.manifest.syspaths or []) self.store = AppStore(default_app=self.manifest.default_store_app) self.apps = [] # Register and initialize all given processors self.processors = { proc.__name__: proc(self) for proc in processors }
[docs] def get_manifest(self, manifest): """ Return loaded manifest object. Arguments: manifest (string or pathlib.Path or dict or Manifest): The Manifest source to load. Returns: Manifest: The manifest object loaded from given source. """ return Manifest.load(manifest)
[docs] def set_syspaths(self, paths): """ Append each item path to ``sys.path``. This won't never append a same path twice. Arguments: paths (list): A list of path to append. """ for path in paths: if path not in sys.path: sys.path.append(path)
[docs] def get_application_base_module_path(self, name): """ Return the Python path to the application base module. Commonly this should be the ``__init__.py`` file from application directory. Arguments: name (string): Module name. Returns: string: Module name prefixed with repository path if it is not empty else returns just the module name. """ if self.manifest.repository: return self._APPLICATION_MODULE_PYTHONPATH.format( parent=self.manifest.repository, name=name, ) return name
[docs] def get_module_path(self, name): """ Return a Python path for a module name. Arguments: name (string): Module name. Returns: string: Module name prefixed with repository path if it is not empty else returns just the module name. """ return self.get_application_base_module_path(name)
[docs] def find_app_module(self, name): """ Find a module (by its pythonpath) from application. Arguments: name (string): Module pythonpath. Returns: object: Module object if found else None. """ try: module = import_module(name) except ModuleNotFoundError: msg = "{klass} is unable to find module: {path}".format( klass=self.__class__.__name__, path=name, ) self.log.debug(msg) return None else: return module
[docs] def _is_elligible_class(self, obj): """ Find if given object is an enabled class for composition. Criterias for eligibility are in order: * Object is a class; * Class is not named ``EnabledApplicationMarker``; * Class got attribute ``_ENABLED_COMPOSABLE_APPLICATION`` which value is not ``None``; Arguments: obj (object): Object to check for eligibility. Returns: boolean: ``True`` if object is eligibile to criterias else ``False``. """ if ( inspect.isclass(obj) and getattr(obj, "__name__", None) != "EnabledApplicationMarker" and getattr(obj, "_ENABLED_COMPOSABLE_APPLICATION", None) is not None ): return True return False
[docs] def _get_elligible_module_classes(self, path, module): """ Get all elligible classes from a module. Arguments: path (string): The Python path to a module used for reporting and logging messages. module (object): The module object where to find elligible classes. Returns: list: List of elligible classes objects. """ enabled = [] if not hasattr(module, "__dict__"): msg = "Module object from '{}' must have a '__dict__' attribute." raise ComposerError(msg.format(path)) for object_name in module.__dict__.keys(): if not object_name.startswith("_"): obj = getattr(module, object_name) if self._is_elligible_class(obj): msg = "{klass} found enabled Class at: {path}.{object_name}".format( klass=self.__class__.__name__, path=path, object_name=object_name, ) self.log.debug(msg) if obj not in enabled: enabled.append(obj) return enabled
[docs] def _scan_app_module(self, name): """ Load an application module to get its options. Arguments: path (string): The Python path to a module to check for. Returns: dict: Application payload (name, dependencies and push_end options). """ path = self.get_module_path(name) # Try to find application module and get its possible parameter variables module = self.find_app_module(path) if module: msg = "{klass} found application at: {path}".format( klass=self.__class__.__name__, path=path, ) self.log.debug(msg) payload = { "name": name, "filepath": module.__file__ } if hasattr(module, "DEPENDENCIES"): payload["dependencies"] = getattr(module, "DEPENDENCIES") if hasattr(module, "PUSH_END"): payload["push_end"] = getattr(module, "PUSH_END") return payload return None
[docs] def call_processor(self, name, method, **kwargs): """ Execute a processor method. Arguments: name (string): Processor name in registry. method (string): Processor method to execute. **kwargs: Keyword arguments to pass to method if any. Returns: object: Content type depend from what processor method returns. """ if name not in self.processors: msg = "Given processor name is not registered from composer: {}" raise ComposerError(msg.format(name)) if not hasattr(self.processors[name], method): msg = "Processor '{proc}' don't have any method named '{method}'" raise ComposerError(msg.format(proc=name, method=method)) return getattr(self.processors[name], method)(**kwargs)
[docs] def resolve_collection(self, lazy=True): """ Resolve collection with AppStore. Keyword Arguments: lazy (boolean): If True, there won't be any dependency order resolving and the application list will just be the collection with ``AppNode`` objects instead of name strings. If False, the resolving will be processed. Default is ``True``. Returns: list: List of ``AppNode`` objects. """ collection = [] for name in self.manifest.collection: payload = self._scan_app_module(name) # Ignore unfound application if payload: collection.append(payload) if lazy: self.apps = self.store.resolve( collection, no_ordering=True ) else: self.apps = self.store.resolve( collection, no_ordering=self.manifest.no_ordering ) return collection
[docs] def check(self, lazy=True, printer=None): """ Output some informations about given manifest, app resolving and processors. It is strongly recommended that every checking is directly outputted. It means you should not build a list of messages to output at the end, instead each job should directly output what it has checked. This is to ensure the debugging won't hide what have been done before a critical error. Remember this method should be almost safe since it is for debugging. Keyword Arguments: lazy (boolean): Wheither to use the lazy mode or not with composer resolver. printer (callable): A callable to use to output debugging informations. Default to builtin function ``utils.tree_printer.TreePrinter`` to benefit from the tree alike display. """ printer = printer or TreePrinter(printable=True) printer("👷 Checking composer") printer() printer("📄 Manifest") printer("T", "Name: {}".format(self.manifest.name)) printer("T", "Repository: {}".format(self.manifest.repository)) printer("X", "Collection:") if self.manifest.collection: last = len(self.manifest.collection) for i, item in enumerate(self.manifest.collection, start=1): printer( "OX" if (i == last) else "OT", item, ) # Scan repository printer() printer("🌐 Repository directory") repository_mod = self.find_app_module(self.manifest.repository) if repository_mod: repository_dirpath = Path(repository_mod.__file__).parents[0] printer("X", repository_dirpath, yes_or_no=repository_dirpath.exists()) last = len(self.manifest.collection) for i, item in enumerate(self.manifest.collection, start=1): app_dirpath = repository_dirpath / item printer( "OX" if (i == last) else "OT", app_dirpath, yes_or_no=app_dirpath.exists(), ) else: printer("X", "Unable to find repository directory") self.resolve_collection(lazy=lazy) printer() printer("🗃️ Resolved applications") if self.apps: last = len(self.apps) for i, app in enumerate(self.apps, start=1): printer( "X" if (i == last) else "T", app, ) # Push end option printer( "OT" if (i == last) else "IT", "Push end: {}".format(app.push_end), ) # Dependencies dep_msg = "No dependency" if app.dependency_names: dep_msg = "Dependencies: {}".format(", ".join(app.dependency_names)) printer( "OX" if (i == last) else "IX", dep_msg, ) else: printer("X", "Unable to find any applications") # Call for processors check for name, proc in self.processors.items(): proc.check(printer=printer)