Coverage for src/metador_core/container/protocols.py: 100%
14 statements
« prev ^ index » next coverage.py v7.3.1, created at 2023-09-11 11:01 +0000
« prev ^ index » next coverage.py v7.3.1, created at 2023-09-11 11:01 +0000
1"""Protocol roughly formalizing the overlap of h5py.File and IH5Record API.
3We build the MetadorContainer interface assuming only these methods.
4"""
5from __future__ import annotations
7from typing import (
8 Any,
9 Callable,
10 ItemsView,
11 Iterable,
12 KeysView,
13 Literal,
14 MutableMapping,
15 Optional,
16 Protocol,
17 TypeVar,
18 Union,
19 ValuesView,
20 runtime_checkable,
21)
23from typing_extensions import get_args
26@runtime_checkable
27class H5NodeLike(Protocol): # pragma: no cover
28 """HDF5 Files, Groups and Datasets are all Nodes."""
30 @property
31 def name(self) -> str:
32 """Absolute path of the node."""
34 @property
35 def attrs(self) -> MutableMapping:
36 """Attached HDF5 attributes."""
38 @property
39 def parent(self) -> H5GroupLike:
40 """Parent group."""
42 @property
43 def file(self) -> H5FileLike:
44 """Original file-like object this node belongs to."""
47@runtime_checkable
48class H5DatasetLike(H5NodeLike, Protocol): # pragma: no cover
49 """Datasets provide numpy-style indexing into data.
51 Metador containers use it for storing bytes,
52 and for getting bytes out again using [()].
53 """
55 def __getitem__(self, key: Any) -> Any:
56 ...
58 def __setitem__(self, key: Any, value) -> None:
59 ...
61 # needed to distinguish from other types:
62 @property
63 def ndim(self) -> int:
64 """Numpy-style dimensionality."""
67CallbackResult = TypeVar("CallbackResult")
68VisitCallback = Callable[[str], Optional[CallbackResult]]
69VisititemsCallback = Callable[[str, H5NodeLike], Optional[CallbackResult]]
72@runtime_checkable
73class H5GroupLike(H5NodeLike, Protocol): # pragma: no cover
74 # MutableMapping-like
76 def __getitem__(self, name: str) -> Union[H5GroupLike, H5DatasetLike]:
77 ...
79 def __setitem__(self, name: str, obj) -> None:
80 ...
82 def __delitem__(self, name: str) -> None:
83 ...
85 def __iter__(self) -> Iterable[str]:
86 ...
88 def __len__(self) -> int:
89 ...
91 # Container-like
93 def __contains__(self, name: str) -> bool:
94 ...
96 # dict-like extras
98 def keys(self) -> KeysView[str]:
99 ...
101 def values(self) -> ValuesView[Union[H5GroupLike, H5DatasetLike]]:
102 ...
104 def items(self) -> ItemsView[str, Union[H5GroupLike, H5DatasetLike]]:
105 ...
107 def get(self, name: str, *args, **kwargs) -> Union[H5GroupLike, H5DatasetLike]:
108 ...
110 # h5py specific
112 def visititems(self, func: VisititemsCallback) -> Optional[CallbackResult]:
113 # returns value returned from callback
114 ...
116 def visit(self, func: VisitCallback) -> Optional[CallbackResult]:
117 # returns value returned from callback
118 ...
120 def create_dataset(self, path, *args, **kwargs) -> H5DatasetLike:
121 # returns original passed-in data
122 ...
124 def require_dataset(self, path, *args, **kwargs) -> H5DatasetLike:
125 ...
127 def create_group(self, path: str) -> H5GroupLike:
128 # returns new group
129 ...
131 def require_group(self, path: str) -> H5GroupLike:
132 ...
134 def move(self, source: str, dest: str) -> None:
135 ...
137 def copy(self, source, dest, **kwargs) -> None:
138 ...
141OpenMode = Literal["r", "r+", "a", "w", "w-", "x"]
142"""User open modes that can be passed during initialization."""
143_OPEN_MODES = list(get_args(OpenMode))
146@runtime_checkable
147class H5FileLike(H5GroupLike, Protocol): # pragma: no cover
148 """A HDF5 File acts like the root group and has some extra features."""
150 @property
151 def mode(self) -> Literal["r", "r+"]:
152 """Return 'r' if container is immutable, otherwise 'r+'."""
154 def close(self) -> None:
155 ...
157 # context manager (`with` notation)
159 def __enter__(self) -> H5FileLike:
160 ...
162 def __exit__(self, ex_type, ex_value, ex_traceback) -> None:
163 ...
166__all__ = ["H5FileLike", "H5GroupLike", "H5DatasetLike", "H5NodeLike", "OpenMode"]