Source code for clout._loaders.multi

import collections
import collections.abc
import typing as t

import attr
import desert
import marshmallow

from .. import exceptions


@attr.dataclass(frozen=True)
class Multi:
    loaders: t.List = attr.ib(factory=list)
    inherits: t.FrozenSet[str] = attr.ib(default=frozenset())
    metadata_key: str = "cli"
    data: dict = attr.ib(factory=dict)

    def set(self, **kw):
        return attr.evolve(self, **kw)

    def set_data_on_loaders(self,):
        loaders = []
        for loader in self.loaders:
            loaders.append(
                loader.set(
                    **{k: v for k, v in self.data.items() if k in loader.inherits}
                )
            )
        return self.set(loaders=loaders)

    def prep(self, cls):
        multi = self.set_data_on_loaders()
        return DeepChainMap(*[loader.prep(cls) or {} for loader in multi.loaders])

    def build(self, cls):
        schema = desert.schema_class(cls)()
        prepped = self.prep(cls)
        try:
            return schema.load(prepped)
        except marshmallow.exceptions.ValidationError as e:
            raise exceptions.ValidationError(*e.args) from e


[docs]class DeepChainMap(collections.ChainMap): """Combine multiple dicts into a deep mapping. Lookups that fail in the first dict will be checked in the next one. >>> import clout >>> maps = [{"a": {}}, {"a": {"b": {}}}, {"a": {"b": {"c": 1337}}}] >>> dcm = clout.DeepChainMap(*maps) >>> dcm["a"]["b"]["c"] 1337 """ def __getitem__(self, key): value = super().__getitem__(key) if isinstance(value, collections.abc.Mapping): return type(self)(*[m[key] for m in self.maps if key in m]) return value