refactor(pyargus): define a runtime checkable Signal protocol

This commit is contained in:
Anand Balakrishnan 2023-09-07 16:34:13 -07:00
parent 8027f86213
commit 3d6157e03a
No known key found for this signature in database
6 changed files with 70 additions and 17 deletions

View file

@ -1,4 +1,4 @@
from typing import ClassVar, Generic, Protocol, TypeVar, final
from typing import ClassVar, Protocol, final
from typing_extensions import Self
@ -134,9 +134,7 @@ class dtype: # noqa: N801
def __eq__(self, other: object) -> bool: ...
def __int__(self) -> int: ...
_SignalKind = TypeVar("_SignalKind", bool, int, float, covariant=True)
class Signal(Generic[_SignalKind], Protocol):
class Signal:
def is_empty(self) -> bool: ...
@property
def start_time(self) -> float | None: ...
@ -146,16 +144,16 @@ class Signal(Generic[_SignalKind], Protocol):
def kind(self) -> dtype: ...
@final
class BoolSignal(Signal[bool]):
class BoolSignal(Signal):
@classmethod
def constant(cls, value: bool) -> Self: ...
@classmethod
def from_samples(cls, samples: list[tuple[float, bool]]) -> Self: ...
def push(self, time: float, value: bool) -> None: ...
def at(self, time: float) -> _SignalKind | None: ...
def at(self, time: float) -> bool | None: ...
@final
class IntSignal(Signal[int]):
class IntSignal(Signal):
@classmethod
def constant(cls, value: int) -> Self: ...
@classmethod
@ -164,7 +162,7 @@ class IntSignal(Signal[int]):
def at(self, time: float) -> int | None: ...
@final
class UnsignedIntSignal(Signal[int]):
class UnsignedIntSignal(Signal):
@classmethod
def constant(cls, value: int) -> Self: ...
@classmethod
@ -173,7 +171,7 @@ class UnsignedIntSignal(Signal[int]):
def at(self, time: float) -> int | None: ...
@final
class FloatSignal(Signal[float]):
class FloatSignal(Signal):
@classmethod
def constant(cls, value: float) -> Self: ...
@classmethod

View file

@ -1,4 +1,43 @@
from argus._argus import BoolSignal, FloatSignal, IntSignal, Signal, UnsignedIntSignal
from typing import List, Optional, Protocol, Tuple, TypeVar, runtime_checkable
from typing_extensions import Self
from argus._argus import BoolSignal, FloatSignal, IntSignal, UnsignedIntSignal, dtype
T = TypeVar("T", bool, int, float)
@runtime_checkable
class Signal(Protocol[T]):
def is_empty(self) -> bool:
...
@property
def start_time(self) -> Optional[float]:
...
@property
def end_time(self) -> Optional[float]:
...
@property
def kind(self) -> dtype:
...
@classmethod
def constant(cls, value: T) -> Self:
...
@classmethod
def from_samples(cls, samples: List[Tuple[float, T]]) -> Self:
...
def push(self, time: float, value: T) -> None:
...
def at(self, time: float) -> Optional[T]:
...
__all__ = [
"Signal",

View file

@ -41,6 +41,7 @@ addopts = ["--import-mode=importlib"]
testpaths = ["tests"]
[tool.mypy]
packages = ["argus"]
# ignore_missing_imports = true
show_error_codes = true
plugins = ["numpy.typing.mypy_plugin"]

View file

@ -0,0 +1,5 @@
argus.signals.Protocol
argus.signals.TypeVar.__bound__
argus.signals.TypeVar.__constraints__
argus.signals.TypeVar.__contravariant__
argus.signals.TypeVar.__covariant__

View file

@ -95,6 +95,7 @@ def gen_dtype() -> SearchStrategy[Union[Type[AllowedDtype], dtype]]:
def test_correct_constant_signals(data: st.DataObject) -> None:
dtype_ = data.draw(gen_dtype())
signal = data.draw(constant_signal(dtype_))
assert isinstance(signal, argus.Signal)
assert not signal.is_empty()
assert signal.start_time is None
@ -108,6 +109,7 @@ def test_correctly_create_signals(data: st.DataObject) -> None:
note(f"Samples: {gen_samples}")
signal = argus.signal(dtype_, data=xs)
assert isinstance(signal, argus.Signal)
if len(xs) > 0:
expected_start_time = xs[0][0]
expected_end_time = xs[-1][0]
@ -165,18 +167,20 @@ def test_signal_create_should_fail(data: st.DataObject) -> None:
@given(st.data())
def test_push_to_empty_signal(data: st.DataObject) -> None:
dtype_ = data.draw(gen_dtype())
sig = data.draw(empty_signal(dtype_=dtype_))
assert sig.is_empty()
signal = data.draw(empty_signal(dtype_=dtype_))
assert isinstance(signal, argus.Signal)
assert signal.is_empty()
element = data.draw(gen_element_fn(dtype_))
with pytest.raises(RuntimeError, match="cannot push value to non-sampled signal"):
sig.push(0.0, element) # type: ignore[attr-defined]
signal.push(0.0, element) # type: ignore[attr-defined]
@given(st.data())
def test_push_to_constant_signal(data: st.DataObject) -> None:
dtype_ = data.draw(gen_dtype())
sig = data.draw(constant_signal(dtype_=dtype_))
assert not sig.is_empty()
signal = data.draw(constant_signal(dtype_=dtype_))
assert isinstance(signal, argus.Signal)
assert not signal.is_empty()
sample = data.draw(gen_samples(min_size=1, max_size=1, dtype_=dtype_))[0]
with pytest.raises(RuntimeError, match="cannot push value to non-sampled signal"):
sig.push(*sample) # type: ignore[attr-defined]
signal.push(*sample) # type: ignore[attr-defined]