-
-
Notifications
You must be signed in to change notification settings - Fork 568
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add tests for descriptorcollection and fix encountered issues
- Loading branch information
Showing
5 changed files
with
221 additions
and
14 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,192 @@ | ||
import pytest | ||
|
||
from miio import ( | ||
AccessFlags, | ||
ActionDescriptor, | ||
DescriptorCollection, | ||
Device, | ||
DeviceStatus, | ||
EnumDescriptor, | ||
PropertyDescriptor, | ||
RangeDescriptor, | ||
ValidSettingRange, | ||
) | ||
from miio.devicestatus import action, sensor, setting | ||
|
||
|
||
@pytest.fixture | ||
def dev(mocker): | ||
d = Device("127.0.0.1", "68ffffffffffffffffffffffffffffff") | ||
mocker.patch("miio.Device.send") | ||
mocker.patch("miio.Device.send_handshake") | ||
yield d | ||
|
||
|
||
def test_descriptors_from_device_object(mocker): | ||
"""Test descriptor collection from device class.""" | ||
|
||
class DummyDevice(Device): | ||
@action(id="test", name="test") | ||
def test_action(self): | ||
pass | ||
|
||
dev = DummyDevice("127.0.0.1", "68ffffffffffffffffffffffffffffff") | ||
mocker.patch("miio.Device.send") | ||
mocker.patch("miio.Device.send_handshake") | ||
|
||
coll = DescriptorCollection(device=dev) | ||
coll.descriptors_from_object(DummyDevice()) | ||
assert len(coll) == 1 | ||
assert isinstance(coll["test"], ActionDescriptor) | ||
|
||
|
||
def test_descriptors_from_status_object(dev): | ||
coll = DescriptorCollection(device=dev) | ||
|
||
class TestStatus(DeviceStatus): | ||
@sensor(id="test", name="test sensor") | ||
def test_sensor(self): | ||
pass | ||
|
||
@setting(id="test-setting", name="test setting", setter=lambda _: _) | ||
def test_setting(self): | ||
pass | ||
|
||
status = TestStatus() | ||
coll.descriptors_from_object(status) | ||
assert len(coll) == 2 | ||
assert isinstance(coll["test"], PropertyDescriptor) | ||
assert isinstance(coll["test-setting"], PropertyDescriptor) | ||
assert coll["test-setting"].access & AccessFlags.Write | ||
|
||
|
||
@pytest.mark.parametrize( | ||
"cls, params", | ||
[ | ||
pytest.param(ActionDescriptor, {"method": lambda _: _}, id="action"), | ||
pytest.param(PropertyDescriptor, {"status_attribute": "foo"}), | ||
], | ||
) | ||
def test_add_descriptor(dev: Device, cls, params): | ||
"""Test that adding a descriptor works.""" | ||
coll: DescriptorCollection = DescriptorCollection(device=dev) | ||
coll.add_descriptor(cls(id="id", name="test name", **params)) | ||
assert len(coll) == 1 | ||
assert coll["id"] is not None | ||
|
||
|
||
def test_handle_action_descriptor(mocker, dev): | ||
coll = DescriptorCollection(device=dev) | ||
invalid_desc = ActionDescriptor(id="action", name="test name") | ||
with pytest.raises(ValueError, match="Neither method or method_name was defined"): | ||
coll.add_descriptor(invalid_desc) | ||
|
||
mocker.patch.object(dev, "existing_method", create=True) | ||
|
||
# Test method name binding | ||
act_with_method_name = ActionDescriptor( | ||
id="with-method-name", name="with-method-name", method_name="existing_method" | ||
) | ||
coll.add_descriptor(act_with_method_name) | ||
assert act_with_method_name.method is not None | ||
|
||
# Test non-existing method | ||
act_with_method_name_missing = ActionDescriptor( | ||
id="with-method-name-missing", | ||
name="with-method-name-missing", | ||
method_name="nonexisting_method", | ||
) | ||
with pytest.raises(AttributeError): | ||
coll.add_descriptor(act_with_method_name_missing) | ||
|
||
|
||
def test_handle_writable_property_descriptor(mocker, dev): | ||
coll = DescriptorCollection(device=dev) | ||
data = { | ||
"name": "", | ||
"status_attribute": "", | ||
"access": AccessFlags.Write, | ||
} | ||
invalid = PropertyDescriptor(id="missing_setter", **data) | ||
with pytest.raises(ValueError, match="Neither setter or setter_name was defined"): | ||
coll.add_descriptor(invalid) | ||
|
||
mocker.patch.object(dev, "existing_method", create=True) | ||
|
||
# Test name binding | ||
setter_name_desc = PropertyDescriptor( | ||
**data, id="setter_name", setter_name="existing_method" | ||
) | ||
coll.add_descriptor(setter_name_desc) | ||
assert setter_name_desc.setter is not None | ||
|
||
with pytest.raises(AttributeError): | ||
coll.add_descriptor( | ||
PropertyDescriptor( | ||
**data, id="missing_setter", setter_name="non_existing_setter" | ||
) | ||
) | ||
|
||
|
||
def test_handle_enum_constraints(dev, mocker): | ||
coll = DescriptorCollection(device=dev) | ||
|
||
data = { | ||
"name": "enum", | ||
"status_attribute": "attr", | ||
} | ||
|
||
mocker.patch.object(dev, "choices_attr", create=True) | ||
|
||
# Check that error is raised if choices are missing | ||
invalid = EnumDescriptor(id="missing", **data) | ||
with pytest.raises( | ||
ValueError, match="Neither choices nor choices_attribute was defined" | ||
): | ||
coll.add_descriptor(invalid) | ||
|
||
# Check that binding works | ||
choices_attribute = EnumDescriptor( | ||
id="with_choices_attr", choices_attribute="choices_attr", **data | ||
) | ||
coll.add_descriptor(choices_attribute) | ||
assert len(coll) == 1 | ||
assert coll["with_choices_attr"].choices is not None | ||
|
||
|
||
def test_handle_range_constraints(dev, mocker): | ||
coll = DescriptorCollection(device=dev) | ||
|
||
data = { | ||
"name": "name", | ||
"status_attribute": "attr", | ||
"min_value": 0, | ||
"max_value": 100, | ||
"step": 1, | ||
} | ||
|
||
# Check regular descriptor | ||
desc = RangeDescriptor(id="regular", **data) | ||
coll.add_descriptor(desc) | ||
assert coll["regular"].max_value == 100 | ||
|
||
valid_range = mocker.patch.object(dev, "range", create=True) | ||
valid_range.return_value = ValidSettingRange(-1, 1000, 10) | ||
range_attr = RangeDescriptor(id="range_attribute", range_attribute="range", **data) | ||
coll.add_descriptor(range_attr) | ||
|
||
assert coll["range_attribute"].min_value == -1 | ||
assert coll["range_attribute"].max_value == 1000 | ||
assert coll["range_attribute"].step == 10 | ||
|
||
|
||
def test_duplicate_identifiers(dev): | ||
coll = DescriptorCollection(device=dev) | ||
for i in range(3): | ||
coll.add_descriptor( | ||
ActionDescriptor(id="action", name=f"action {i}", method=lambda _: _) | ||
) | ||
|
||
assert coll["action"] | ||
assert coll["action-2"] | ||
assert coll["action-3"] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters