2020-01-12 03:42:14 +00:00
|
|
|
"""Test zha cover."""
|
|
|
|
from unittest.mock import MagicMock, call, patch
|
|
|
|
|
2020-02-10 02:45:35 +00:00
|
|
|
import asynctest
|
|
|
|
import pytest
|
2020-01-12 03:42:14 +00:00
|
|
|
import zigpy.types
|
|
|
|
import zigpy.zcl.clusters.closures as closures
|
|
|
|
import zigpy.zcl.foundation as zcl_f
|
|
|
|
|
|
|
|
from homeassistant.components.cover import DOMAIN
|
|
|
|
from homeassistant.const import STATE_CLOSED, STATE_OPEN, STATE_UNAVAILABLE
|
|
|
|
|
|
|
|
from .common import (
|
|
|
|
async_enable_traffic,
|
2020-02-12 21:12:14 +00:00
|
|
|
async_test_rejoin,
|
2020-01-12 03:42:14 +00:00
|
|
|
find_entity_id,
|
2020-03-13 23:17:50 +00:00
|
|
|
send_attributes_report,
|
2020-01-12 03:42:14 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
from tests.common import mock_coro
|
|
|
|
|
|
|
|
|
2020-02-10 02:45:35 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def zigpy_cover_device(zigpy_device_mock):
|
|
|
|
"""Zigpy cover device."""
|
|
|
|
|
|
|
|
endpoints = {
|
|
|
|
1: {
|
|
|
|
"device_type": 1026,
|
|
|
|
"in_clusters": [closures.WindowCovering.cluster_id],
|
|
|
|
"out_clusters": [],
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return zigpy_device_mock(endpoints)
|
2020-01-12 03:42:14 +00:00
|
|
|
|
2020-02-10 02:45:35 +00:00
|
|
|
|
|
|
|
@asynctest.patch(
|
|
|
|
"homeassistant.components.zha.core.channels.closures.WindowCovering.async_initialize"
|
|
|
|
)
|
2020-02-12 21:12:14 +00:00
|
|
|
async def test_cover(m1, hass, zha_device_joined_restored, zigpy_cover_device):
|
2020-02-10 02:45:35 +00:00
|
|
|
"""Test zha cover platform."""
|
2020-01-12 03:42:14 +00:00
|
|
|
|
|
|
|
async def get_chan_attr(*args, **kwargs):
|
|
|
|
return 100
|
|
|
|
|
|
|
|
with patch(
|
2020-02-21 23:06:57 +00:00
|
|
|
"homeassistant.components.zha.core.channels.base.ZigbeeChannel.get_attribute_value",
|
2020-01-12 03:42:14 +00:00
|
|
|
new=MagicMock(side_effect=get_chan_attr),
|
|
|
|
) as get_attr_mock:
|
|
|
|
# load up cover domain
|
2020-02-10 02:45:35 +00:00
|
|
|
zha_device = await zha_device_joined_restored(zigpy_cover_device)
|
2020-01-12 03:42:14 +00:00
|
|
|
assert get_attr_mock.call_count == 2
|
|
|
|
assert get_attr_mock.call_args[0][0] == "current_position_lift_percentage"
|
|
|
|
|
2020-02-10 02:45:35 +00:00
|
|
|
cluster = zigpy_cover_device.endpoints.get(1).window_covering
|
2020-01-12 03:42:14 +00:00
|
|
|
entity_id = await find_entity_id(DOMAIN, zha_device, hass)
|
|
|
|
assert entity_id is not None
|
|
|
|
|
|
|
|
# test that the cover was created and that it is unavailable
|
|
|
|
assert hass.states.get(entity_id).state == STATE_UNAVAILABLE
|
|
|
|
|
|
|
|
# allow traffic to flow through the gateway and device
|
2020-02-12 21:12:14 +00:00
|
|
|
await async_enable_traffic(hass, [zha_device])
|
2020-01-12 03:42:14 +00:00
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# test that the state has changed from unavailable to off
|
2020-03-13 23:17:50 +00:00
|
|
|
await send_attributes_report(hass, cluster, {0: 0, 8: 100, 1: 1})
|
2020-01-12 03:42:14 +00:00
|
|
|
assert hass.states.get(entity_id).state == STATE_CLOSED
|
|
|
|
|
|
|
|
# test to see if it opens
|
2020-03-13 23:17:50 +00:00
|
|
|
await send_attributes_report(hass, cluster, {0: 1, 8: 0, 1: 100})
|
2020-01-12 03:42:14 +00:00
|
|
|
assert hass.states.get(entity_id).state == STATE_OPEN
|
|
|
|
|
|
|
|
# close from UI
|
|
|
|
with patch(
|
|
|
|
"zigpy.zcl.Cluster.request", return_value=mock_coro([0x1, zcl_f.Status.SUCCESS])
|
|
|
|
):
|
|
|
|
await hass.services.async_call(
|
|
|
|
DOMAIN, "close_cover", {"entity_id": entity_id}, blocking=True
|
|
|
|
)
|
|
|
|
assert cluster.request.call_count == 1
|
|
|
|
assert cluster.request.call_args == call(
|
2020-03-09 16:39:41 +00:00
|
|
|
False, 0x1, (), expect_reply=True, manufacturer=None, tsn=None
|
2020-01-12 03:42:14 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
# open from UI
|
|
|
|
with patch(
|
|
|
|
"zigpy.zcl.Cluster.request", return_value=mock_coro([0x0, zcl_f.Status.SUCCESS])
|
|
|
|
):
|
|
|
|
await hass.services.async_call(
|
|
|
|
DOMAIN, "open_cover", {"entity_id": entity_id}, blocking=True
|
|
|
|
)
|
|
|
|
assert cluster.request.call_count == 1
|
|
|
|
assert cluster.request.call_args == call(
|
2020-03-09 16:39:41 +00:00
|
|
|
False, 0x0, (), expect_reply=True, manufacturer=None, tsn=None
|
2020-01-12 03:42:14 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
# set position UI
|
|
|
|
with patch(
|
|
|
|
"zigpy.zcl.Cluster.request", return_value=mock_coro([0x5, zcl_f.Status.SUCCESS])
|
|
|
|
):
|
|
|
|
await hass.services.async_call(
|
|
|
|
DOMAIN,
|
|
|
|
"set_cover_position",
|
|
|
|
{"entity_id": entity_id, "position": 47},
|
|
|
|
blocking=True,
|
|
|
|
)
|
|
|
|
assert cluster.request.call_count == 1
|
|
|
|
assert cluster.request.call_args == call(
|
2020-03-09 16:39:41 +00:00
|
|
|
False,
|
|
|
|
0x5,
|
|
|
|
(zigpy.types.uint8_t,),
|
|
|
|
53,
|
|
|
|
expect_reply=True,
|
|
|
|
manufacturer=None,
|
|
|
|
tsn=None,
|
2020-01-12 03:42:14 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
# stop from UI
|
|
|
|
with patch(
|
|
|
|
"zigpy.zcl.Cluster.request", return_value=mock_coro([0x2, zcl_f.Status.SUCCESS])
|
|
|
|
):
|
|
|
|
await hass.services.async_call(
|
|
|
|
DOMAIN, "stop_cover", {"entity_id": entity_id}, blocking=True
|
|
|
|
)
|
|
|
|
assert cluster.request.call_count == 1
|
|
|
|
assert cluster.request.call_args == call(
|
2020-03-09 16:39:41 +00:00
|
|
|
False, 0x2, (), expect_reply=True, manufacturer=None, tsn=None
|
2020-01-12 03:42:14 +00:00
|
|
|
)
|
|
|
|
|
2020-02-10 02:45:35 +00:00
|
|
|
# test rejoin
|
2020-02-12 21:12:14 +00:00
|
|
|
await async_test_rejoin(hass, zigpy_cover_device, [cluster], (1,))
|
2020-02-10 02:45:35 +00:00
|
|
|
assert hass.states.get(entity_id).state == STATE_OPEN
|