Source code for kerno.start

"""Eko is the class used to start a Kerno application.

It works much like initial configuration of a Pyramid app.
"""

from configparser import NoSectionError
from types import ModuleType
from typing import Callable, Iterable, List, Union

from bag.settings import read_ini_files, resolve

from kerno.kerno import Kerno
from kerno.typing import DictStr
from kerno.utility_registry import ConfigurationError, UtilityRegistryBuilder

ResourceType = Union[str, ModuleType, Callable]


[docs]class Eko: r"""At startup builds the Kerno instance for the app. In the Esperanto language, Eko is a noun that means "a start". And "eki" is a verb that means "to start". The Eko class can build an application core from multiple Python modules, much like the Configurator in Pyramid. If the same module is included twice, Eko will raise ``kerno.start.ConfigurationError``, helping you ensure your startup code stays clean. Usage example:: from kerno.start import Eko eko = Eko.from_ini('main_config.ini', 'production.ini') eko.settings # lets you access the merged settings from INI files eko.kerno # is the Kerno instance that will be the core of the app eko.utilities.register('mailer', 'some.package:mailer_function') # ...and later you can retrieve *mailer_function* by doing: # kerno.utilities['mailer'] # This finds the function some.extension.package:eki() and runs it, # passing it the eko configurator object: eko.include('some.extension.package') # For instance, here is how you can build a repository from 2 classes: eko.include('kerno.repository') # adds add_repository_mixin() to eko eko.add_repository_mixin( 'kerno.repository.sqlalchemy.BaseSQLAlchemyRepository') eko.add_repository_mixin('my.package:MyRepoMixinClass') # After this, the first time you access ``kerno.Repository``, # the Repository class is assembled from the mixin classes, # and stored for usage as kerno.Repository. If you need to debug the order in which modules got included, you can:: print(*eko._included_modules, sep='\n') """
[docs] @classmethod def from_ini(cls, *config_files, encoding: str = "utf-8"): """Return an instance after reading some INI file(s).""" return cls(settings=read_ini_files(*config_files, encoding=encoding))
def __init__(self, settings: DictStr = {}): # noqa if settings and not hasattr(settings, "__getitem__"): raise TypeError( "The *settings* argument must be dict-like. " "Received: {}".format(type(settings)) ) self._included_modules: List[ModuleType] = [] self.kerno = Kerno(settings) self.utilities = UtilityRegistryBuilder(kerno=self.kerno) try: main_config_section = settings["kerno"] except (NoSectionError, KeyError): return for extension in main_config_section.get("includes", []): self.include(extension)
[docs] def include(self, resource: ResourceType, throw: bool = True) -> None: """Execute a configuration callable for imperative extension. If the argument ``throw`` is True (which is the default) and ``resource`` does not have an ``eki()`` function, raise ConfigurationError. Otherwise, ignore the current module and return without an error. """ # TODO log.debug('Including {}'.format(module)) obj = resolve(resource) if isinstance(resource, str) else resource if obj in self._included_modules: raise ConfigurationError(f"{obj} has already been included!") else: self._included_modules.append(obj) if isinstance(obj, ModuleType): try: fn = getattr(obj, "eki") except AttributeError: if throw: raise ConfigurationError( "The module {} has no function called 'eki'.".format( obj.__name__ ) ) else: return else: fn = obj fn(self)
[docs] def include_many( self, resources: Iterable[ResourceType], throw: bool = True ) -> None: """Initialize multiple app modules.""" for resource in resources: self.include(resource, throw=throw)