Source code for plaster.loaders

try:
    from importlib import metadata
except ImportError:  # pragma: no cover < py38
    import importlib_metadata as metadata

from .exceptions import LoaderNotFound, MultipleLoadersFound
from .interfaces import ILoaderInfo
from .uri import parse_uri


def get_sections(config_uri):
    """
    Load the list of named sections.

    .. code-block:: python

        sections = plaster.get_sections('development.ini')
        full_config = {
            section: plaster.get_settings('development.ini', section)
            for section in sections
        }

    :param config_uri: Anything that can be parsed by
        :func:`plaster.parse_uri`.

    :returns: A list of section names in the config file.

    """
    loader = get_loader(config_uri)

    return loader.get_sections()


[docs] def get_settings(config_uri, section=None, defaults=None): """ Load the settings from a named section. .. code-block:: python settings = plaster.get_settings(...) print(settings['foo']) :param config_uri: Anything that can be parsed by :func:`plaster.parse_uri`. :param section: The name of the section in the config file. If this is ``None`` then it is up to the loader to determine a sensible default usually derived from the fragment in the ``path#name`` syntax of the ``config_uri``. :param defaults: A ``dict`` of default values used to populate the settings and support variable interpolation. Any values in ``defaults`` may be overridden by the loader prior to returning the final configuration dictionary. :returns: A ``dict`` of settings. This should return a dictionary object even if no data is available. """ loader = get_loader(config_uri) return loader.get_settings(section, defaults)
[docs] def setup_logging(config_uri, defaults=None): """ Execute the logging configuration defined in the config file. This function should, at least, configure the Python standard logging module. However, it may also be used to configure any other logging subsystems that serve a similar purpose. :param config_uri: Anything that can be parsed by :func:`plaster.parse_uri`. :param defaults: A ``dict`` of default values used to populate the settings and support variable interpolation. Any values in ``defaults`` may be overridden by the loader prior to returning the final configuration dictionary. """ loader = get_loader(config_uri) return loader.setup_logging(defaults)
[docs] def get_loader(config_uri, protocols=None): """ Find a :class:`plaster.ILoader` object capable of handling ``config_uri``. :param config_uri: Anything that can be parsed by :func:`plaster.parse_uri`. :param protocols: Zero or more :term:`loader protocol` identifiers that the loader must implement to match the desired ``config_uri``. :returns: A :class:`plaster.ILoader` object. :raises plaster.LoaderNotFound: If no loader could be found. :raises plaster.MultipleLoadersFound: If multiple loaders match the requested criteria. If this happens, you can disambiguate the lookup by appending the package name to the scheme for the loader you wish to use. For example if ``ini`` is ambiguous then specify ``ini+myapp`` to use the ini loader from the ``myapp`` package. """ config_uri = parse_uri(config_uri) requested_scheme = config_uri.scheme matched_loaders = find_loaders(requested_scheme, protocols=protocols) if len(matched_loaders) < 1: raise LoaderNotFound(requested_scheme, protocols=protocols) if len(matched_loaders) > 1: raise MultipleLoadersFound( requested_scheme, matched_loaders, protocols=protocols ) loader_info = matched_loaders[0] loader = loader_info.load(config_uri) return loader
[docs] def find_loaders(scheme, protocols=None): """ Find all loaders that match the requested scheme and protocols. :param scheme: Any valid scheme. Examples would be something like ``ini`` or ``pastedeploy+ini``. :param protocols: Zero or more :term:`loader protocol` identifiers that the loader must implement. If ``None`` then only generic loaders will be returned. :returns: A list containing zero or more :class:`plaster.ILoaderInfo` objects. """ # build a list of all required entry points matching_groups = ["plaster.loader_factory"] if protocols: matching_groups += [f"plaster.{proto}_loader_factory" for proto in protocols] scheme = scheme.lower() # if a distribution is specified then it overrides the default search parts = scheme.split("+", 1) if len(parts) == 2: try: dist = metadata.distribution(parts[0]) except metadata.PackageNotFoundError: pass else: ep = _find_ep_in_dist(dist, parts[1], matching_groups) # if we got one or more loaders from a specific distribution # then they override everything else so we'll just return them if ep: return [EntryPointLoaderInfo(dist, ep, protocols)] return [ EntryPointLoaderInfo(dist, ep, protocols=protocols) for (dist, ep) in _iter_ep_in_dists(scheme, matching_groups) ]
def _iter_ep_in_dists(scheme, groups): # XXX this is a hack to deduplicate distributions because importlib.metadata # does not do this for us by default, at least up to Python 3.11. # Specifically, if ``lib`` is symlinked to ``lib64`` then a Distribution # object will be returned for each path, causing duplicate entry points # to be found. # # See https://github.com/Pylons/plaster/issues/25 dups = set() for dist in metadata.distributions(): name = dist.metadata["Name"] if name in dups: # pragma: no cover continue dups.add(name) ep = _find_ep_in_dist(dist, scheme, groups) if ep: yield (dist, ep) def _find_ep_in_dist(dist, scheme, groups): entry_points = [ entry_point for entry_point in dist.entry_points if entry_point.group in groups and (scheme is None or scheme == entry_point.name.lower()) ] # verify that the entry point from each group points to the same factory if len({ep.value for ep in entry_points}) == 1: return entry_points[0] class EntryPointLoaderInfo(ILoaderInfo): def __init__(self, dist, ep, protocols=None): self.entry_point = ep self.scheme = "{}+{}".format( dist.metadata["name"] if "name" in dist.metadata else "Unknown", ep.name ) self.protocols = protocols self._factory = None @property def factory(self): if self._factory is None: self._factory = self.entry_point.load() return self._factory def load(self, config_uri): config_uri = parse_uri(config_uri) return self.factory(config_uri)