mirror of
https://github.com/tiqi-group/pydase.git
synced 2025-06-05 13:10:41 +02:00
Merge branch 'main' into feature/ignore_coroutine
This commit is contained in:
commit
84abd63d56
@ -1,6 +1,6 @@
|
|||||||
[tool.poetry]
|
[tool.poetry]
|
||||||
name = "pydase"
|
name = "pydase"
|
||||||
version = "0.8.2"
|
version = "0.8.3"
|
||||||
description = "A flexible and robust Python library for creating, managing, and interacting with data services, with built-in support for web and RPC servers, and customizable features for diverse use cases."
|
description = "A flexible and robust Python library for creating, managing, and interacting with data services, with built-in support for web and RPC servers, and customizable features for diverse use cases."
|
||||||
authors = ["Mose Mueller <mosmuell@ethz.ch>"]
|
authors = ["Mose Mueller <mosmuell@ethz.ch>"]
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
|
import weakref
|
||||||
from abc import ABC, abstractmethod
|
from abc import ABC, abstractmethod
|
||||||
from typing import TYPE_CHECKING, Any, ClassVar, SupportsIndex
|
from typing import TYPE_CHECKING, Any, ClassVar, SupportsIndex
|
||||||
|
|
||||||
@ -15,8 +16,8 @@ logger = logging.getLogger(__name__)
|
|||||||
|
|
||||||
|
|
||||||
class ObservableObject(ABC):
|
class ObservableObject(ABC):
|
||||||
_list_mapping: ClassVar[dict[int, _ObservableList]] = {}
|
_list_mapping: ClassVar[dict[int, weakref.ReferenceType[_ObservableList]]] = {}
|
||||||
_dict_mapping: ClassVar[dict[int, _ObservableDict]] = {}
|
_dict_mapping: ClassVar[dict[int, weakref.ReferenceType[_ObservableDict]]] = {}
|
||||||
|
|
||||||
def __init__(self) -> None:
|
def __init__(self) -> None:
|
||||||
if not hasattr(self, "_observers"):
|
if not hasattr(self, "_observers"):
|
||||||
@ -34,6 +35,10 @@ class ObservableObject(ABC):
|
|||||||
if attribute in self._observers:
|
if attribute in self._observers:
|
||||||
self._observers[attribute].remove(observer)
|
self._observers[attribute].remove(observer)
|
||||||
|
|
||||||
|
# remove attribute key from observers dict if list of observers is empty
|
||||||
|
if not self._observers[attribute]:
|
||||||
|
del self._observers[attribute]
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def _remove_observer_if_observable(self, name: str) -> None:
|
def _remove_observer_if_observable(self, name: str) -> None:
|
||||||
"""Removes the current object as an observer from an observable attribute.
|
"""Removes the current object as an observer from an observable attribute.
|
||||||
@ -91,19 +96,23 @@ class ObservableObject(ABC):
|
|||||||
if isinstance(value, list):
|
if isinstance(value, list):
|
||||||
if id(value) in self._list_mapping:
|
if id(value) in self._list_mapping:
|
||||||
# If the list `value` was already referenced somewhere else
|
# If the list `value` was already referenced somewhere else
|
||||||
new_value = self._list_mapping[id(value)]
|
new_value = self._list_mapping[id(value)]()
|
||||||
else:
|
else:
|
||||||
# convert the builtin list into a ObservableList
|
# convert the builtin list into a ObservableList
|
||||||
new_value = _ObservableList(original_list=value)
|
new_value = _ObservableList(original_list=value)
|
||||||
self._list_mapping[id(value)] = new_value
|
|
||||||
|
# Use weakref to allow the GC to collect unused objects
|
||||||
|
self._list_mapping[id(value)] = weakref.ref(new_value)
|
||||||
elif isinstance(value, dict):
|
elif isinstance(value, dict):
|
||||||
if id(value) in self._dict_mapping:
|
if id(value) in self._dict_mapping:
|
||||||
# If the dict `value` was already referenced somewhere else
|
# If the dict `value` was already referenced somewhere else
|
||||||
new_value = self._dict_mapping[id(value)]
|
new_value = self._dict_mapping[id(value)]()
|
||||||
else:
|
else:
|
||||||
# convert the builtin list into a ObservableList
|
# convert the builtin dict into a ObservableDict
|
||||||
new_value = _ObservableDict(original_dict=value)
|
new_value = _ObservableDict(original_dict=value)
|
||||||
self._dict_mapping[id(value)] = new_value
|
|
||||||
|
# Use weakref to allow the GC to collect unused objects
|
||||||
|
self._dict_mapping[id(value)] = weakref.ref(new_value)
|
||||||
if isinstance(new_value, ObservableObject):
|
if isinstance(new_value, ObservableObject):
|
||||||
new_value.add_observer(self, attr_name_or_key)
|
new_value.add_observer(self, attr_name_or_key)
|
||||||
return new_value
|
return new_value
|
||||||
@ -142,6 +151,9 @@ class _ObservableList(ObservableObject, list[Any]):
|
|||||||
for i, item in enumerate(self._original_list):
|
for i, item in enumerate(self._original_list):
|
||||||
super().__setitem__(i, self._initialise_new_objects(f"[{i}]", item))
|
super().__setitem__(i, self._initialise_new_objects(f"[{i}]", item))
|
||||||
|
|
||||||
|
def __del__(self) -> None:
|
||||||
|
self._list_mapping.pop(id(self._original_list))
|
||||||
|
|
||||||
def __setitem__(self, key: int, value: Any) -> None: # type: ignore[override]
|
def __setitem__(self, key: int, value: Any) -> None: # type: ignore[override]
|
||||||
if hasattr(self, "_observers"):
|
if hasattr(self, "_observers"):
|
||||||
self._remove_observer_if_observable(f"[{key}]")
|
self._remove_observer_if_observable(f"[{key}]")
|
||||||
@ -240,6 +252,9 @@ class _ObservableDict(ObservableObject, dict[str, Any]):
|
|||||||
for key, value in self._original_dict.items():
|
for key, value in self._original_dict.items():
|
||||||
self.__setitem__(key, self._initialise_new_objects(f'["{key}"]', value))
|
self.__setitem__(key, self._initialise_new_objects(f'["{key}"]', value))
|
||||||
|
|
||||||
|
def __del__(self) -> None:
|
||||||
|
self._dict_mapping.pop(id(self._original_dict))
|
||||||
|
|
||||||
def __setitem__(self, key: str, value: Any) -> None:
|
def __setitem__(self, key: str, value: Any) -> None:
|
||||||
if not isinstance(key, str):
|
if not isinstance(key, str):
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
|
@ -36,7 +36,8 @@ def test_unexpected_type_change_warning(caplog: LogCaptureFixture) -> None:
|
|||||||
|
|
||||||
|
|
||||||
def test_basic_inheritance_warning(caplog: LogCaptureFixture) -> None:
|
def test_basic_inheritance_warning(caplog: LogCaptureFixture) -> None:
|
||||||
class SubService(DataService): ...
|
class SubService(DataService):
|
||||||
|
...
|
||||||
|
|
||||||
class SomeEnum(Enum):
|
class SomeEnum(Enum):
|
||||||
HI = 0
|
HI = 0
|
||||||
@ -56,9 +57,11 @@ def test_basic_inheritance_warning(caplog: LogCaptureFixture) -> None:
|
|||||||
def name(self) -> str:
|
def name(self) -> str:
|
||||||
return self._name
|
return self._name
|
||||||
|
|
||||||
def some_method(self) -> None: ...
|
def some_method(self) -> None:
|
||||||
|
...
|
||||||
|
|
||||||
async def some_task(self) -> None: ...
|
async def some_task(self) -> None:
|
||||||
|
...
|
||||||
|
|
||||||
ServiceClass()
|
ServiceClass()
|
||||||
|
|
||||||
|
@ -138,7 +138,6 @@ def test_removed_observer_on_class_dict_attr(caplog: pytest.LogCaptureFixture) -
|
|||||||
caplog.clear()
|
caplog.clear()
|
||||||
|
|
||||||
assert nested_instance._observers == {
|
assert nested_instance._observers == {
|
||||||
'["nested"]': [],
|
|
||||||
"nested_attr": [instance],
|
"nested_attr": [instance],
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -172,7 +171,6 @@ def test_removed_observer_on_instance_dict_attr(
|
|||||||
caplog.clear()
|
caplog.clear()
|
||||||
|
|
||||||
assert nested_instance._observers == {
|
assert nested_instance._observers == {
|
||||||
'["nested"]': [],
|
|
||||||
"nested_attr": [instance],
|
"nested_attr": [instance],
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -211,6 +209,6 @@ def test_pop(caplog: pytest.LogCaptureFixture) -> None:
|
|||||||
instance = MyObservable()
|
instance = MyObservable()
|
||||||
MyObserver(instance)
|
MyObserver(instance)
|
||||||
assert instance.dict_attr.pop("nested") == nested_instance
|
assert instance.dict_attr.pop("nested") == nested_instance
|
||||||
assert nested_instance._observers == {'["nested"]': []}
|
assert nested_instance._observers == {}
|
||||||
|
|
||||||
assert f"'dict_attr' changed to '{instance.dict_attr}'" in caplog.text
|
assert f"'dict_attr' changed to '{instance.dict_attr}'" in caplog.text
|
||||||
|
@ -81,11 +81,21 @@ def test_removed_observer_on_class_list_attr(caplog: pytest.LogCaptureFixture) -
|
|||||||
|
|
||||||
instance = MyObservable()
|
instance = MyObservable()
|
||||||
MyObserver(instance)
|
MyObserver(instance)
|
||||||
|
|
||||||
|
assert nested_instance._observers == {
|
||||||
|
"[0]": [instance.changed_list_attr],
|
||||||
|
"nested_attr": [instance],
|
||||||
|
}
|
||||||
|
|
||||||
instance.changed_list_attr[0] = "Ciao"
|
instance.changed_list_attr[0] = "Ciao"
|
||||||
|
|
||||||
assert "'changed_list_attr[0]' changed to 'Ciao'" in caplog.text
|
assert "'changed_list_attr[0]' changed to 'Ciao'" in caplog.text
|
||||||
caplog.clear()
|
caplog.clear()
|
||||||
|
|
||||||
|
assert nested_instance._observers == {
|
||||||
|
"nested_attr": [instance],
|
||||||
|
}
|
||||||
|
|
||||||
instance.nested_attr.name = "Hi"
|
instance.nested_attr.name = "Hi"
|
||||||
|
|
||||||
assert "'nested_attr.name' changed to 'Hi'" in caplog.text
|
assert "'nested_attr.name' changed to 'Hi'" in caplog.text
|
||||||
@ -115,6 +125,10 @@ def test_removed_observer_on_instance_list_attr(
|
|||||||
assert "'changed_list_attr[0]' changed to 'Ciao'" in caplog.text
|
assert "'changed_list_attr[0]' changed to 'Ciao'" in caplog.text
|
||||||
caplog.clear()
|
caplog.clear()
|
||||||
|
|
||||||
|
assert nested_instance._observers == {
|
||||||
|
"nested_attr": [instance],
|
||||||
|
}
|
||||||
|
|
||||||
instance.nested_attr.name = "Hi"
|
instance.nested_attr.name = "Hi"
|
||||||
|
|
||||||
assert "'nested_attr.name' changed to 'Hi'" in caplog.text
|
assert "'nested_attr.name' changed to 'Hi'" in caplog.text
|
||||||
@ -311,3 +325,51 @@ def test_list_remove(caplog: pytest.LogCaptureFixture) -> None:
|
|||||||
# checks if observer key was updated correctly (was index 1)
|
# checks if observer key was updated correctly (was index 1)
|
||||||
other_observable_instance_2.greeting = "Ciao"
|
other_observable_instance_2.greeting = "Ciao"
|
||||||
assert "'my_list[0].greeting' changed to 'Ciao'" in caplog.text
|
assert "'my_list[0].greeting' changed to 'Ciao'" in caplog.text
|
||||||
|
|
||||||
|
|
||||||
|
def test_list_garbage_collection() -> None:
|
||||||
|
"""Makes sure that the GC collects lists that are not referenced anymore."""
|
||||||
|
|
||||||
|
import gc
|
||||||
|
import json
|
||||||
|
|
||||||
|
list_json = """
|
||||||
|
[1]
|
||||||
|
"""
|
||||||
|
|
||||||
|
class MyObservable(Observable):
|
||||||
|
def __init__(self) -> None:
|
||||||
|
super().__init__()
|
||||||
|
self.list_attr = json.loads(list_json)
|
||||||
|
|
||||||
|
observable = MyObservable()
|
||||||
|
list_mapping_length = len(observable._list_mapping)
|
||||||
|
observable.list_attr = json.loads(list_json)
|
||||||
|
|
||||||
|
gc.collect()
|
||||||
|
assert len(observable._list_mapping) <= list_mapping_length
|
||||||
|
|
||||||
|
|
||||||
|
def test_dict_garbage_collection() -> None:
|
||||||
|
"""Makes sure that the GC collects dicts that are not referenced anymore."""
|
||||||
|
|
||||||
|
import gc
|
||||||
|
import json
|
||||||
|
|
||||||
|
dict_json = """
|
||||||
|
{
|
||||||
|
"foo": "bar"
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
|
||||||
|
class MyObservable(Observable):
|
||||||
|
def __init__(self) -> None:
|
||||||
|
super().__init__()
|
||||||
|
self.dict_attr = json.loads(dict_json)
|
||||||
|
|
||||||
|
observable = MyObservable()
|
||||||
|
dict_mapping_length = len(observable._dict_mapping)
|
||||||
|
observable.dict_attr = json.loads(dict_json)
|
||||||
|
|
||||||
|
gc.collect()
|
||||||
|
assert len(observable._dict_mapping) <= dict_mapping_length
|
||||||
|
@ -16,6 +16,7 @@ def test_inherited_property_dependency_resolution() -> None:
|
|||||||
_name = "DerivedObservable"
|
_name = "DerivedObservable"
|
||||||
|
|
||||||
class MyObserver(PropertyObserver):
|
class MyObserver(PropertyObserver):
|
||||||
def on_change(self, full_access_path: str, value: Any) -> None: ...
|
def on_change(self, full_access_path: str, value: Any) -> None:
|
||||||
|
...
|
||||||
|
|
||||||
assert MyObserver(DerivedObservable()).property_deps_dict == {"_name": ["name"]}
|
assert MyObserver(DerivedObservable()).property_deps_dict == {"_name": ["name"]}
|
||||||
|
@ -476,7 +476,8 @@ def test_derived_data_service_serialization() -> None:
|
|||||||
def name(self, value: str) -> None:
|
def name(self, value: str) -> None:
|
||||||
self._name = value
|
self._name = value
|
||||||
|
|
||||||
class DerivedService(BaseService): ...
|
class DerivedService(BaseService):
|
||||||
|
...
|
||||||
|
|
||||||
base_service_serialization = dump(BaseService())
|
base_service_serialization = dump(BaseService())
|
||||||
derived_service_serialization = dump(DerivedService())
|
derived_service_serialization = dump(DerivedService())
|
||||||
|
Loading…
x
Reference in New Issue
Block a user