Coverage for src/metador_core/container/provider.py: 97%
32 statements
« prev ^ index » next coverage.py v7.3.2, created at 2023-11-02 09:33 +0000
« prev ^ index » next coverage.py v7.3.2, created at 2023-11-02 09:33 +0000
1"""Abstract Metador container provider interface."""
2from typing import Any, Dict, Generic, Optional, Protocol, Tuple, Type, TypeVar, Union
4from .wrappers import MetadorContainer, MetadorDriver
6T = TypeVar("T")
9class ContainerProxy(Protocol[T]):
10 """Abstract interface for Metador container providers.
12 This interface acts like a proxy to access containers by some identifier.
14 The identifier type parameter T is in the simplest case the Metador
15 container UUID. In more complex cases, it could be a different unique
16 identifier with a non-trivial relationship to Metador container UUIDs
17 (many-to-many). Therefore, T is implementation-specific.
19 There are many ways to store and organize containers, this interface serves
20 as the implementation target for generic service components such as
21 container-centric Flask blueprints, so they can be easier reused in
22 different backends and services.
24 Note that only containment and retrieval are possible - on purpose.
25 Knowing and iterating over all containers in a system is not always possible.
26 """
28 def __contains__(self, key: T) -> bool:
29 """Return whether a resource key is known to the proxy."""
30 # return self.get(key) is not None
32 def get(self, key: T) -> Optional[MetadorContainer]:
33 """Get a container instance, if resource key is known to the proxy.
35 Implement this method in subclasses to support the minimal interface.
36 """
38 def __getitem__(self, key: T) -> T:
39 """Get a container instance, if resource key is known to the proxy.
41 Default implementation is in terms of `get`.
42 """
43 if ret := self.get(key):
44 return ret
45 raise KeyError(key)
48ContainerArgs = Tuple[Type[MetadorDriver], Any]
49"""Pair of (driver class, suitable driver arguments).
51Must be such that `MetadorContainer(driver(source))` yields a working container.
52"""
55class SimpleContainerProvider(Generic[T], ContainerProxy[T]):
56 """Dict-backed container proxy.
58 It is a minimal reasonable implementation for the interface that can be
59 used in small apps and does not depend on the container driver,
60 thus can support all container interface implementations.
61 """
63 _known: Dict[T, ContainerArgs]
64 """Mapping from container identifier to MetadorContainer constructor args."""
66 def __init__(self):
67 self._known = {}
69 def __contains__(self, key: T) -> bool:
70 return key in self._known
72 def get(self, key: T) -> Optional[MetadorContainer]:
73 """Get an open container file to access data and metadata, if it exists."""
74 if key not in self._known:
75 return None
76 driver, source = self._known[key]
77 return MetadorContainer(driver(source))
79 # ----
81 def __delitem__(self, key: T):
82 del self._known[key]
84 def __setitem__(self, key: T, value: Union[ContainerArgs, MetadorContainer]):
85 # NOTE: can't do instance check here, because MetadorContainer is a wrapper itself
86 # so we check if a container is passed by presence of the .metador attribute
87 if container_toc := getattr(value, "metador", None):
88 self._known[key] = (container_toc.driver, container_toc.source)
89 else:
90 self._known[key] = value
92 def keys(self):
93 return self._known.keys()