405 lines
13 KiB
Python
405 lines
13 KiB
Python
"""Test ZHA Gateway."""
|
|
|
|
import asyncio
|
|
from unittest.mock import MagicMock, PropertyMock, patch
|
|
|
|
import pytest
|
|
from zigpy.application import ControllerApplication
|
|
from zigpy.profiles import zha
|
|
import zigpy.types
|
|
from zigpy.zcl.clusters import general, lighting
|
|
import zigpy.zdo.types
|
|
|
|
from homeassistant.components.zha.core.gateway import ZHAGateway
|
|
from homeassistant.components.zha.core.group import GroupMember
|
|
from homeassistant.components.zha.core.helpers import get_zha_gateway
|
|
from homeassistant.const import Platform
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from .common import async_find_group_entity_id
|
|
from .conftest import SIG_EP_INPUT, SIG_EP_OUTPUT, SIG_EP_PROFILE, SIG_EP_TYPE
|
|
|
|
from tests.common import MockConfigEntry
|
|
|
|
IEEE_GROUPABLE_DEVICE = "01:2d:6f:00:0a:90:69:e8"
|
|
IEEE_GROUPABLE_DEVICE2 = "02:2d:6f:00:0a:90:69:e8"
|
|
|
|
|
|
@pytest.fixture
|
|
def zigpy_dev_basic(zigpy_device_mock):
|
|
"""Zigpy device with just a basic cluster."""
|
|
return zigpy_device_mock(
|
|
{
|
|
1: {
|
|
SIG_EP_INPUT: [general.Basic.cluster_id],
|
|
SIG_EP_OUTPUT: [],
|
|
SIG_EP_TYPE: zha.DeviceType.ON_OFF_SWITCH,
|
|
SIG_EP_PROFILE: zha.PROFILE_ID,
|
|
}
|
|
}
|
|
)
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def required_platform_only():
|
|
"""Only set up the required and required base platforms to speed up tests."""
|
|
with patch(
|
|
"homeassistant.components.zha.PLATFORMS",
|
|
(
|
|
Platform.SENSOR,
|
|
Platform.LIGHT,
|
|
Platform.DEVICE_TRACKER,
|
|
Platform.NUMBER,
|
|
Platform.SELECT,
|
|
),
|
|
):
|
|
yield
|
|
|
|
|
|
@pytest.fixture
|
|
async def zha_dev_basic(hass, zha_device_restored, zigpy_dev_basic):
|
|
"""ZHA device with just a basic cluster."""
|
|
|
|
return await zha_device_restored(zigpy_dev_basic)
|
|
|
|
|
|
@pytest.fixture
|
|
async def coordinator(hass, zigpy_device_mock, zha_device_joined):
|
|
"""Test ZHA light platform."""
|
|
|
|
zigpy_device = zigpy_device_mock(
|
|
{
|
|
1: {
|
|
SIG_EP_INPUT: [],
|
|
SIG_EP_OUTPUT: [],
|
|
SIG_EP_TYPE: zha.DeviceType.COLOR_DIMMABLE_LIGHT,
|
|
SIG_EP_PROFILE: zha.PROFILE_ID,
|
|
}
|
|
},
|
|
ieee="00:15:8d:00:02:32:4f:32",
|
|
nwk=0x0000,
|
|
node_descriptor=b"\xf8\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff",
|
|
)
|
|
zha_device = await zha_device_joined(zigpy_device)
|
|
zha_device.available = True
|
|
return zha_device
|
|
|
|
|
|
@pytest.fixture
|
|
async def device_light_1(hass, zigpy_device_mock, zha_device_joined):
|
|
"""Test ZHA light platform."""
|
|
|
|
zigpy_device = zigpy_device_mock(
|
|
{
|
|
1: {
|
|
SIG_EP_INPUT: [
|
|
general.OnOff.cluster_id,
|
|
general.LevelControl.cluster_id,
|
|
lighting.Color.cluster_id,
|
|
general.Groups.cluster_id,
|
|
],
|
|
SIG_EP_OUTPUT: [],
|
|
SIG_EP_TYPE: zha.DeviceType.COLOR_DIMMABLE_LIGHT,
|
|
SIG_EP_PROFILE: zha.PROFILE_ID,
|
|
}
|
|
},
|
|
ieee=IEEE_GROUPABLE_DEVICE,
|
|
)
|
|
zha_device = await zha_device_joined(zigpy_device)
|
|
zha_device.available = True
|
|
return zha_device
|
|
|
|
|
|
@pytest.fixture
|
|
async def device_light_2(hass, zigpy_device_mock, zha_device_joined):
|
|
"""Test ZHA light platform."""
|
|
|
|
zigpy_device = zigpy_device_mock(
|
|
{
|
|
1: {
|
|
SIG_EP_INPUT: [
|
|
general.OnOff.cluster_id,
|
|
general.LevelControl.cluster_id,
|
|
lighting.Color.cluster_id,
|
|
general.Groups.cluster_id,
|
|
],
|
|
SIG_EP_OUTPUT: [],
|
|
SIG_EP_TYPE: zha.DeviceType.COLOR_DIMMABLE_LIGHT,
|
|
SIG_EP_PROFILE: zha.PROFILE_ID,
|
|
}
|
|
},
|
|
ieee=IEEE_GROUPABLE_DEVICE2,
|
|
)
|
|
zha_device = await zha_device_joined(zigpy_device)
|
|
zha_device.available = True
|
|
return zha_device
|
|
|
|
|
|
async def test_device_left(hass: HomeAssistant, zigpy_dev_basic, zha_dev_basic) -> None:
|
|
"""Device leaving the network should become unavailable."""
|
|
|
|
assert zha_dev_basic.available is True
|
|
|
|
get_zha_gateway(hass).device_left(zigpy_dev_basic)
|
|
await hass.async_block_till_done()
|
|
assert zha_dev_basic.available is False
|
|
|
|
|
|
async def test_gateway_group_methods(
|
|
hass: HomeAssistant, device_light_1, device_light_2, coordinator
|
|
) -> None:
|
|
"""Test creating a group with 2 members."""
|
|
zha_gateway = get_zha_gateway(hass)
|
|
assert zha_gateway is not None
|
|
zha_gateway.coordinator_zha_device = coordinator
|
|
coordinator._zha_gateway = zha_gateway
|
|
device_light_1._zha_gateway = zha_gateway
|
|
device_light_2._zha_gateway = zha_gateway
|
|
member_ieee_addresses = [device_light_1.ieee, device_light_2.ieee]
|
|
members = [GroupMember(device_light_1.ieee, 1), GroupMember(device_light_2.ieee, 1)]
|
|
|
|
# test creating a group with 2 members
|
|
zha_group = await zha_gateway.async_create_zigpy_group("Test Group", members)
|
|
await hass.async_block_till_done()
|
|
|
|
assert zha_group is not None
|
|
assert len(zha_group.members) == 2
|
|
for member in zha_group.members:
|
|
assert member.device.ieee in member_ieee_addresses
|
|
|
|
entity_id = async_find_group_entity_id(hass, Platform.LIGHT, zha_group)
|
|
assert hass.states.get(entity_id) is not None
|
|
|
|
# test get group by name
|
|
assert zha_group == zha_gateway.async_get_group_by_name(zha_group.name)
|
|
|
|
# test removing a group
|
|
await zha_gateway.async_remove_zigpy_group(zha_group.group_id)
|
|
await hass.async_block_till_done()
|
|
|
|
# we shouldn't have the group anymore
|
|
assert zha_gateway.async_get_group_by_name(zha_group.name) is None
|
|
|
|
# the group entity should be cleaned up
|
|
assert entity_id not in hass.states.async_entity_ids(Platform.LIGHT)
|
|
|
|
# test creating a group with 1 member
|
|
zha_group = await zha_gateway.async_create_zigpy_group(
|
|
"Test Group", [GroupMember(device_light_1.ieee, 1)]
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert zha_group is not None
|
|
assert len(zha_group.members) == 1
|
|
for member in zha_group.members:
|
|
assert member.device.ieee in [device_light_1.ieee]
|
|
|
|
# the group entity should not have been cleaned up
|
|
assert entity_id not in hass.states.async_entity_ids(Platform.LIGHT)
|
|
|
|
with patch("zigpy.zcl.Cluster.request", side_effect=TimeoutError):
|
|
await zha_group.members[0].async_remove_from_group()
|
|
assert len(zha_group.members) == 1
|
|
for member in zha_group.members:
|
|
assert member.device.ieee in [device_light_1.ieee]
|
|
|
|
|
|
async def test_gateway_create_group_with_id(
|
|
hass: HomeAssistant, device_light_1, coordinator
|
|
) -> None:
|
|
"""Test creating a group with a specific ID."""
|
|
zha_gateway = get_zha_gateway(hass)
|
|
assert zha_gateway is not None
|
|
zha_gateway.coordinator_zha_device = coordinator
|
|
coordinator._zha_gateway = zha_gateway
|
|
device_light_1._zha_gateway = zha_gateway
|
|
|
|
zha_group = await zha_gateway.async_create_zigpy_group(
|
|
"Test Group", [GroupMember(device_light_1.ieee, 1)], group_id=0x1234
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(zha_group.members) == 1
|
|
assert zha_group.members[0].device is device_light_1
|
|
assert zha_group.group_id == 0x1234
|
|
|
|
|
|
@patch(
|
|
"homeassistant.components.zha.core.gateway.ZHAGateway.async_load_devices",
|
|
MagicMock(),
|
|
)
|
|
@patch(
|
|
"homeassistant.components.zha.core.gateway.ZHAGateway.async_load_groups",
|
|
MagicMock(),
|
|
)
|
|
@pytest.mark.parametrize(
|
|
("device_path", "thread_state", "config_override"),
|
|
[
|
|
("/dev/ttyUSB0", True, {}),
|
|
("socket://192.168.1.123:9999", False, {}),
|
|
("socket://192.168.1.123:9999", True, {"use_thread": True}),
|
|
],
|
|
)
|
|
async def test_gateway_initialize_bellows_thread(
|
|
device_path: str,
|
|
thread_state: bool,
|
|
config_override: dict,
|
|
hass: HomeAssistant,
|
|
zigpy_app_controller: ControllerApplication,
|
|
config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test ZHA disabling the UART thread when connecting to a TCP coordinator."""
|
|
data = dict(config_entry.data)
|
|
data["device"]["path"] = device_path
|
|
config_entry.add_to_hass(hass)
|
|
hass.config_entries.async_update_entry(config_entry, data=data)
|
|
|
|
zha_gateway = ZHAGateway(hass, {"zigpy_config": config_override}, config_entry)
|
|
|
|
with patch(
|
|
"bellows.zigbee.application.ControllerApplication.new",
|
|
return_value=zigpy_app_controller,
|
|
) as mock_new:
|
|
await zha_gateway.async_initialize()
|
|
|
|
assert mock_new.mock_calls[-1].kwargs["config"]["use_thread"] is thread_state
|
|
|
|
await zha_gateway.shutdown()
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("device_path", "config_override", "expected_channel"),
|
|
[
|
|
("/dev/ttyUSB0", {}, None),
|
|
("socket://192.168.1.123:9999", {}, None),
|
|
("socket://192.168.1.123:9999", {"network": {"channel": 20}}, 20),
|
|
("socket://core-silabs-multiprotocol:9999", {}, 15),
|
|
("socket://core-silabs-multiprotocol:9999", {"network": {"channel": 20}}, 20),
|
|
],
|
|
)
|
|
async def test_gateway_force_multi_pan_channel(
|
|
device_path: str,
|
|
config_override: dict,
|
|
expected_channel: int | None,
|
|
hass: HomeAssistant,
|
|
config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test ZHA disabling the UART thread when connecting to a TCP coordinator."""
|
|
data = dict(config_entry.data)
|
|
data["device"]["path"] = device_path
|
|
config_entry.add_to_hass(hass)
|
|
hass.config_entries.async_update_entry(config_entry, data=data)
|
|
|
|
zha_gateway = ZHAGateway(hass, {"zigpy_config": config_override}, config_entry)
|
|
|
|
_, config = zha_gateway.get_application_controller_data()
|
|
assert config["network"]["channel"] == expected_channel
|
|
|
|
|
|
async def test_single_reload_on_multiple_connection_loss(
|
|
hass: HomeAssistant,
|
|
zigpy_app_controller: ControllerApplication,
|
|
config_entry: MockConfigEntry,
|
|
):
|
|
"""Test that we only reload once when we lose the connection multiple times."""
|
|
config_entry.add_to_hass(hass)
|
|
|
|
zha_gateway = ZHAGateway(hass, {}, config_entry)
|
|
|
|
with patch(
|
|
"bellows.zigbee.application.ControllerApplication.new",
|
|
return_value=zigpy_app_controller,
|
|
):
|
|
await zha_gateway.async_initialize()
|
|
|
|
with patch.object(
|
|
hass.config_entries, "async_reload", wraps=hass.config_entries.async_reload
|
|
) as mock_reload:
|
|
zha_gateway.connection_lost(RuntimeError())
|
|
zha_gateway.connection_lost(RuntimeError())
|
|
zha_gateway.connection_lost(RuntimeError())
|
|
zha_gateway.connection_lost(RuntimeError())
|
|
zha_gateway.connection_lost(RuntimeError())
|
|
|
|
assert len(mock_reload.mock_calls) == 1
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
@pytest.mark.parametrize("radio_concurrency", [1, 2, 8])
|
|
async def test_startup_concurrency_limit(
|
|
radio_concurrency: int,
|
|
hass: HomeAssistant,
|
|
zigpy_app_controller: ControllerApplication,
|
|
config_entry: MockConfigEntry,
|
|
zigpy_device_mock,
|
|
):
|
|
"""Test ZHA gateway limits concurrency on startup."""
|
|
config_entry.add_to_hass(hass)
|
|
zha_gateway = ZHAGateway(hass, {}, config_entry)
|
|
|
|
with patch(
|
|
"bellows.zigbee.application.ControllerApplication.new",
|
|
return_value=zigpy_app_controller,
|
|
):
|
|
await zha_gateway.async_initialize()
|
|
|
|
for i in range(50):
|
|
zigpy_dev = zigpy_device_mock(
|
|
{
|
|
1: {
|
|
SIG_EP_INPUT: [
|
|
general.OnOff.cluster_id,
|
|
general.LevelControl.cluster_id,
|
|
lighting.Color.cluster_id,
|
|
general.Groups.cluster_id,
|
|
],
|
|
SIG_EP_OUTPUT: [],
|
|
SIG_EP_TYPE: zha.DeviceType.COLOR_DIMMABLE_LIGHT,
|
|
SIG_EP_PROFILE: zha.PROFILE_ID,
|
|
}
|
|
},
|
|
ieee=f"11:22:33:44:{i:08x}",
|
|
nwk=0x1234 + i,
|
|
)
|
|
zigpy_dev.node_desc.mac_capability_flags |= (
|
|
zigpy.zdo.types.NodeDescriptor.MACCapabilityFlags.MainsPowered
|
|
)
|
|
|
|
zha_gateway._async_get_or_create_device(zigpy_dev)
|
|
|
|
# Keep track of request concurrency during initialization
|
|
current_concurrency = 0
|
|
concurrencies = []
|
|
|
|
async def mock_send_packet(*args, **kwargs):
|
|
nonlocal current_concurrency
|
|
|
|
current_concurrency += 1
|
|
concurrencies.append(current_concurrency)
|
|
|
|
await asyncio.sleep(0.001)
|
|
|
|
current_concurrency -= 1
|
|
concurrencies.append(current_concurrency)
|
|
|
|
type(zha_gateway).radio_concurrency = PropertyMock(return_value=radio_concurrency)
|
|
assert zha_gateway.radio_concurrency == radio_concurrency
|
|
|
|
with patch(
|
|
"homeassistant.components.zha.core.device.ZHADevice.async_initialize",
|
|
side_effect=mock_send_packet,
|
|
):
|
|
await zha_gateway.async_fetch_updated_state_mains()
|
|
|
|
await zha_gateway.shutdown()
|
|
|
|
# Make sure concurrency was always limited
|
|
assert current_concurrency == 0
|
|
assert min(concurrencies) == 0
|
|
|
|
if radio_concurrency > 1:
|
|
assert 1 <= max(concurrencies) < zha_gateway.radio_concurrency
|
|
else:
|
|
assert 1 == max(concurrencies) == zha_gateway.radio_concurrency
|