mirror of
https://github.com/python-kasa/python-kasa.git
synced 2025-01-08 22:07:06 +00:00
Implement action feature (#849)
Adds `FeatureType.Action` making it possible to expose features like "reboot", "test alarm", "pair" etc. The `attribute_getter` is no longer mandatory, but it will raise an exception if not defined for other types than actions. Trying to read returns a static string `<Action>`. This overloads the `set_value` to call the given callable on any value. This also fixes the `play` and `stop` coroutines of the alarm module to await the call.
This commit is contained in:
parent
b860c32d5f
commit
6e5cae1f47
@ -17,7 +17,7 @@ class FeatureType(Enum):
|
|||||||
Sensor = auto()
|
Sensor = auto()
|
||||||
BinarySensor = auto()
|
BinarySensor = auto()
|
||||||
Switch = auto()
|
Switch = auto()
|
||||||
Button = auto()
|
Action = auto()
|
||||||
Number = auto()
|
Number = auto()
|
||||||
|
|
||||||
|
|
||||||
@ -46,7 +46,7 @@ class Feature:
|
|||||||
#: User-friendly short description
|
#: User-friendly short description
|
||||||
name: str
|
name: str
|
||||||
#: Name of the property that allows accessing the value
|
#: Name of the property that allows accessing the value
|
||||||
attribute_getter: str | Callable
|
attribute_getter: str | Callable | None = None
|
||||||
#: Name of the method that allows changing the value
|
#: Name of the method that allows changing the value
|
||||||
attribute_setter: str | None = None
|
attribute_setter: str | None = None
|
||||||
#: Container storing the data, this overrides 'device' for getters
|
#: Container storing the data, this overrides 'device' for getters
|
||||||
@ -95,6 +95,11 @@ class Feature:
|
|||||||
@property
|
@property
|
||||||
def value(self):
|
def value(self):
|
||||||
"""Return the current value."""
|
"""Return the current value."""
|
||||||
|
if self.type == FeatureType.Action:
|
||||||
|
return "<Action>"
|
||||||
|
if self.attribute_getter is None:
|
||||||
|
raise ValueError("Not an action and no attribute_getter set")
|
||||||
|
|
||||||
container = self.container if self.container is not None else self.device
|
container = self.container if self.container is not None else self.device
|
||||||
if isinstance(self.attribute_getter, Callable):
|
if isinstance(self.attribute_getter, Callable):
|
||||||
return self.attribute_getter(container)
|
return self.attribute_getter(container)
|
||||||
@ -112,6 +117,9 @@ class Feature:
|
|||||||
)
|
)
|
||||||
|
|
||||||
container = self.container if self.container is not None else self.device
|
container = self.container if self.container is not None else self.device
|
||||||
|
if self.type == FeatureType.Action:
|
||||||
|
return await getattr(container, self.attribute_setter)()
|
||||||
|
|
||||||
return await getattr(container, self.attribute_setter)(value)
|
return await getattr(container, self.attribute_setter)(value)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
@ -54,6 +54,24 @@ class AlarmModule(SmartModule):
|
|||||||
device, "Alarm volume", container=self, attribute_getter="alarm_volume"
|
device, "Alarm volume", container=self, attribute_getter="alarm_volume"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
self._add_feature(
|
||||||
|
Feature(
|
||||||
|
device,
|
||||||
|
"Test alarm",
|
||||||
|
container=self,
|
||||||
|
attribute_setter="play",
|
||||||
|
type=FeatureType.Action,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
self._add_feature(
|
||||||
|
Feature(
|
||||||
|
device,
|
||||||
|
"Stop alarm",
|
||||||
|
container=self,
|
||||||
|
attribute_setter="stop",
|
||||||
|
type=FeatureType.Action,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def alarm_sound(self):
|
def alarm_sound(self):
|
||||||
@ -83,8 +101,8 @@ class AlarmModule(SmartModule):
|
|||||||
|
|
||||||
async def play(self):
|
async def play(self):
|
||||||
"""Play alarm."""
|
"""Play alarm."""
|
||||||
return self.call("play_alarm")
|
return await self.call("play_alarm")
|
||||||
|
|
||||||
async def stop(self):
|
async def stop(self):
|
||||||
"""Stop alarm."""
|
"""Stop alarm."""
|
||||||
return self.call("stop_alarm")
|
return await self.call("stop_alarm")
|
||||||
|
@ -3,11 +3,13 @@ import pytest
|
|||||||
from kasa import Feature, FeatureType
|
from kasa import Feature, FeatureType
|
||||||
|
|
||||||
|
|
||||||
|
class DummyDevice:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def dummy_feature() -> Feature:
|
def dummy_feature() -> Feature:
|
||||||
# create_autospec for device slows tests way too much, so we use a dummy here
|
# create_autospec for device slows tests way too much, so we use a dummy here
|
||||||
class DummyDevice:
|
|
||||||
pass
|
|
||||||
|
|
||||||
feat = Feature(
|
feat = Feature(
|
||||||
device=DummyDevice(), # type: ignore[arg-type]
|
device=DummyDevice(), # type: ignore[arg-type]
|
||||||
@ -79,3 +81,19 @@ async def test_feature_setter_read_only(dummy_feature):
|
|||||||
dummy_feature.attribute_setter = None
|
dummy_feature.attribute_setter = None
|
||||||
with pytest.raises(ValueError):
|
with pytest.raises(ValueError):
|
||||||
await dummy_feature.set_value("value for read only feature")
|
await dummy_feature.set_value("value for read only feature")
|
||||||
|
|
||||||
|
|
||||||
|
async def test_feature_action(mocker):
|
||||||
|
"""Test that setting value on button calls the setter."""
|
||||||
|
feat = Feature(
|
||||||
|
device=DummyDevice(), # type: ignore[arg-type]
|
||||||
|
name="dummy_feature",
|
||||||
|
attribute_setter="call_action",
|
||||||
|
container=None,
|
||||||
|
icon="mdi:dummy",
|
||||||
|
type=FeatureType.Action,
|
||||||
|
)
|
||||||
|
mock_call_action = mocker.patch.object(feat.device, "call_action", create=True)
|
||||||
|
assert feat.value == "<Action>"
|
||||||
|
await feat.set_value(1234)
|
||||||
|
mock_call_action.assert_called()
|
||||||
|
Loading…
Reference in New Issue
Block a user