2022-01-04 12:40:29 +00:00
|
|
|
"""Test ZHA select entities."""
|
|
|
|
|
2022-06-17 16:41:10 +00:00
|
|
|
from unittest.mock import call, patch
|
2022-04-28 18:57:26 +00:00
|
|
|
|
2022-01-04 12:40:29 +00:00
|
|
|
import pytest
|
|
|
|
from zigpy.const import SIG_EP_PROFILE
|
|
|
|
import zigpy.profiles.zha as zha
|
|
|
|
import zigpy.zcl.clusters.general as general
|
|
|
|
import zigpy.zcl.clusters.security as security
|
|
|
|
|
|
|
|
from homeassistant.const import ENTITY_CATEGORY_CONFIG, STATE_UNKNOWN, Platform
|
|
|
|
from homeassistant.helpers import entity_registry as er, restore_state
|
|
|
|
from homeassistant.util import dt as dt_util
|
|
|
|
|
|
|
|
from .common import find_entity_id
|
|
|
|
from .conftest import SIG_EP_INPUT, SIG_EP_OUTPUT, SIG_EP_TYPE
|
|
|
|
|
|
|
|
|
2022-06-17 16:41:10 +00:00
|
|
|
@pytest.fixture(autouse=True)
|
|
|
|
def select_select_only():
|
|
|
|
"""Only setup the select and required base platforms to speed up tests."""
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.zha.PLATFORMS",
|
|
|
|
(
|
|
|
|
Platform.BUTTON,
|
|
|
|
Platform.DEVICE_TRACKER,
|
|
|
|
Platform.SIREN,
|
|
|
|
Platform.LIGHT,
|
|
|
|
Platform.NUMBER,
|
|
|
|
Platform.SELECT,
|
|
|
|
Platform.SENSOR,
|
|
|
|
),
|
|
|
|
):
|
|
|
|
yield
|
|
|
|
|
|
|
|
|
2022-01-04 12:40:29 +00:00
|
|
|
@pytest.fixture
|
|
|
|
async def siren(hass, zigpy_device_mock, zha_device_joined_restored):
|
|
|
|
"""Siren fixture."""
|
|
|
|
|
|
|
|
zigpy_device = zigpy_device_mock(
|
|
|
|
{
|
|
|
|
1: {
|
|
|
|
SIG_EP_INPUT: [general.Basic.cluster_id, security.IasWd.cluster_id],
|
|
|
|
SIG_EP_OUTPUT: [],
|
|
|
|
SIG_EP_TYPE: zha.DeviceType.IAS_WARNING_DEVICE,
|
|
|
|
SIG_EP_PROFILE: zha.PROFILE_ID,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
zha_device = await zha_device_joined_restored(zigpy_device)
|
|
|
|
return zha_device, zigpy_device.endpoints[1].ias_wd
|
|
|
|
|
|
|
|
|
2022-04-24 16:50:06 +00:00
|
|
|
@pytest.fixture
|
|
|
|
async def light(hass, zigpy_device_mock):
|
|
|
|
"""Siren fixture."""
|
|
|
|
|
|
|
|
zigpy_device = zigpy_device_mock(
|
|
|
|
{
|
|
|
|
1: {
|
|
|
|
SIG_EP_PROFILE: zha.PROFILE_ID,
|
|
|
|
SIG_EP_TYPE: zha.DeviceType.ON_OFF_LIGHT,
|
|
|
|
SIG_EP_INPUT: [
|
|
|
|
general.Basic.cluster_id,
|
|
|
|
general.Identify.cluster_id,
|
|
|
|
general.OnOff.cluster_id,
|
|
|
|
],
|
|
|
|
SIG_EP_OUTPUT: [general.Ota.cluster_id],
|
|
|
|
}
|
|
|
|
},
|
2022-04-28 18:57:26 +00:00
|
|
|
node_descriptor=b"\x02@\x84_\x11\x7fd\x00\x00,d\x00\x00",
|
2022-04-24 16:50:06 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
return zigpy_device
|
|
|
|
|
|
|
|
|
2022-01-04 12:40:29 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def core_rs(hass_storage):
|
|
|
|
"""Core.restore_state fixture."""
|
|
|
|
|
|
|
|
def _storage(entity_id, state):
|
|
|
|
now = dt_util.utcnow().isoformat()
|
|
|
|
|
|
|
|
hass_storage[restore_state.STORAGE_KEY] = {
|
|
|
|
"version": restore_state.STORAGE_VERSION,
|
|
|
|
"key": restore_state.STORAGE_KEY,
|
|
|
|
"data": [
|
|
|
|
{
|
|
|
|
"state": {
|
|
|
|
"entity_id": entity_id,
|
|
|
|
"state": str(state),
|
|
|
|
"last_changed": now,
|
|
|
|
"last_updated": now,
|
|
|
|
"context": {
|
|
|
|
"id": "3c2243ff5f30447eb12e7348cfd5b8ff",
|
|
|
|
"user_id": None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
"last_seen": now,
|
|
|
|
}
|
|
|
|
],
|
|
|
|
}
|
|
|
|
return
|
|
|
|
|
|
|
|
return _storage
|
|
|
|
|
|
|
|
|
|
|
|
async def test_select(hass, siren):
|
|
|
|
"""Test zha select platform."""
|
|
|
|
|
|
|
|
entity_registry = er.async_get(hass)
|
|
|
|
zha_device, cluster = siren
|
|
|
|
assert cluster is not None
|
|
|
|
select_name = security.IasWd.Warning.WarningMode.__name__
|
|
|
|
entity_id = await find_entity_id(
|
|
|
|
Platform.SELECT,
|
|
|
|
zha_device,
|
|
|
|
hass,
|
|
|
|
qualifier=select_name.lower(),
|
|
|
|
)
|
|
|
|
assert entity_id is not None
|
|
|
|
|
|
|
|
state = hass.states.get(entity_id)
|
|
|
|
assert state
|
|
|
|
assert state.state == STATE_UNKNOWN
|
|
|
|
assert state.attributes["options"] == [
|
|
|
|
"Stop",
|
|
|
|
"Burglar",
|
|
|
|
"Fire",
|
|
|
|
"Emergency",
|
|
|
|
"Police Panic",
|
|
|
|
"Fire Panic",
|
|
|
|
"Emergency Panic",
|
|
|
|
]
|
|
|
|
|
|
|
|
entity_entry = entity_registry.async_get(entity_id)
|
|
|
|
assert entity_entry
|
|
|
|
assert entity_entry.entity_category == ENTITY_CATEGORY_CONFIG
|
|
|
|
|
|
|
|
# Test select option with string value
|
|
|
|
await hass.services.async_call(
|
|
|
|
"select",
|
|
|
|
"select_option",
|
|
|
|
{
|
|
|
|
"entity_id": entity_id,
|
|
|
|
"option": security.IasWd.Warning.WarningMode.Burglar.name,
|
|
|
|
},
|
|
|
|
blocking=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
state = hass.states.get(entity_id)
|
|
|
|
assert state
|
|
|
|
assert state.state == security.IasWd.Warning.WarningMode.Burglar.name
|
|
|
|
|
|
|
|
|
|
|
|
async def test_select_restore_state(
|
|
|
|
hass,
|
|
|
|
zigpy_device_mock,
|
|
|
|
core_rs,
|
|
|
|
zha_device_restored,
|
|
|
|
):
|
|
|
|
"""Test zha select entity restore state."""
|
|
|
|
|
|
|
|
entity_id = "select.fakemanufacturer_fakemodel_e769900a_ias_wd_warningmode"
|
|
|
|
core_rs(entity_id, state="Burglar")
|
|
|
|
|
|
|
|
zigpy_device = zigpy_device_mock(
|
|
|
|
{
|
|
|
|
1: {
|
|
|
|
SIG_EP_INPUT: [general.Basic.cluster_id, security.IasWd.cluster_id],
|
|
|
|
SIG_EP_OUTPUT: [],
|
|
|
|
SIG_EP_TYPE: zha.DeviceType.IAS_WARNING_DEVICE,
|
|
|
|
SIG_EP_PROFILE: zha.PROFILE_ID,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
zha_device = await zha_device_restored(zigpy_device)
|
|
|
|
cluster = zigpy_device.endpoints[1].ias_wd
|
|
|
|
assert cluster is not None
|
|
|
|
select_name = security.IasWd.Warning.WarningMode.__name__
|
|
|
|
entity_id = await find_entity_id(
|
|
|
|
Platform.SELECT,
|
|
|
|
zha_device,
|
|
|
|
hass,
|
|
|
|
qualifier=select_name.lower(),
|
|
|
|
)
|
|
|
|
|
|
|
|
assert entity_id is not None
|
|
|
|
state = hass.states.get(entity_id)
|
|
|
|
assert state
|
|
|
|
assert state.state == security.IasWd.Warning.WarningMode.Burglar.name
|
2022-04-24 16:50:06 +00:00
|
|
|
|
|
|
|
|
2022-04-28 18:57:26 +00:00
|
|
|
async def test_on_off_select_new_join(hass, light, zha_device_joined):
|
|
|
|
"""Test zha on off select - new join."""
|
2022-04-24 16:50:06 +00:00
|
|
|
|
|
|
|
entity_registry = er.async_get(hass)
|
|
|
|
on_off_cluster = light.endpoints[1].on_off
|
|
|
|
on_off_cluster.PLUGGED_ATTR_READS = {
|
|
|
|
"start_up_on_off": general.OnOff.StartUpOnOff.On
|
|
|
|
}
|
2022-04-28 18:57:26 +00:00
|
|
|
zha_device = await zha_device_joined(light)
|
2022-04-24 16:50:06 +00:00
|
|
|
select_name = general.OnOff.StartUpOnOff.__name__
|
|
|
|
entity_id = await find_entity_id(
|
|
|
|
Platform.SELECT,
|
|
|
|
zha_device,
|
|
|
|
hass,
|
|
|
|
qualifier=select_name.lower(),
|
|
|
|
)
|
|
|
|
assert entity_id is not None
|
|
|
|
|
2022-04-28 18:57:26 +00:00
|
|
|
assert on_off_cluster.read_attributes.call_count == 2
|
|
|
|
assert (
|
|
|
|
call(["start_up_on_off"], allow_cache=True, only_cache=False, manufacturer=None)
|
|
|
|
in on_off_cluster.read_attributes.call_args_list
|
|
|
|
)
|
|
|
|
assert (
|
|
|
|
call(["on_off"], allow_cache=False, only_cache=False, manufacturer=None)
|
|
|
|
in on_off_cluster.read_attributes.call_args_list
|
|
|
|
)
|
|
|
|
|
2022-04-24 16:50:06 +00:00
|
|
|
state = hass.states.get(entity_id)
|
|
|
|
assert state
|
2022-04-28 18:57:26 +00:00
|
|
|
assert state.state == general.OnOff.StartUpOnOff.On.name
|
2022-04-28 00:52:32 +00:00
|
|
|
|
2022-04-24 16:50:06 +00:00
|
|
|
assert state.attributes["options"] == ["Off", "On", "Toggle", "PreviousValue"]
|
|
|
|
|
|
|
|
entity_entry = entity_registry.async_get(entity_id)
|
|
|
|
assert entity_entry
|
|
|
|
assert entity_entry.entity_category == ENTITY_CATEGORY_CONFIG
|
|
|
|
|
|
|
|
# Test select option with string value
|
|
|
|
await hass.services.async_call(
|
|
|
|
"select",
|
|
|
|
"select_option",
|
|
|
|
{
|
|
|
|
"entity_id": entity_id,
|
|
|
|
"option": general.OnOff.StartUpOnOff.Off.name,
|
|
|
|
},
|
|
|
|
blocking=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
assert on_off_cluster.write_attributes.call_count == 1
|
|
|
|
assert on_off_cluster.write_attributes.call_args[0][0] == {
|
|
|
|
"start_up_on_off": general.OnOff.StartUpOnOff.Off
|
|
|
|
}
|
|
|
|
|
|
|
|
state = hass.states.get(entity_id)
|
|
|
|
assert state
|
|
|
|
assert state.state == general.OnOff.StartUpOnOff.Off.name
|
|
|
|
|
|
|
|
|
2022-04-28 18:57:26 +00:00
|
|
|
async def test_on_off_select_restored(hass, light, zha_device_restored):
|
|
|
|
"""Test zha on off select - restored."""
|
|
|
|
|
|
|
|
entity_registry = er.async_get(hass)
|
|
|
|
on_off_cluster = light.endpoints[1].on_off
|
|
|
|
on_off_cluster.PLUGGED_ATTR_READS = {
|
|
|
|
"start_up_on_off": general.OnOff.StartUpOnOff.On
|
|
|
|
}
|
|
|
|
zha_device = await zha_device_restored(light)
|
|
|
|
|
|
|
|
assert zha_device.is_mains_powered
|
|
|
|
|
|
|
|
assert on_off_cluster.read_attributes.call_count == 4
|
|
|
|
# first 2 calls hit cache only
|
|
|
|
assert (
|
|
|
|
call(["start_up_on_off"], allow_cache=True, only_cache=True, manufacturer=None)
|
|
|
|
in on_off_cluster.read_attributes.call_args_list
|
|
|
|
)
|
|
|
|
assert (
|
|
|
|
call(["on_off"], allow_cache=True, only_cache=True, manufacturer=None)
|
|
|
|
in on_off_cluster.read_attributes.call_args_list
|
|
|
|
)
|
|
|
|
|
|
|
|
# 2nd set of calls can actually read from the device
|
|
|
|
assert (
|
|
|
|
call(["start_up_on_off"], allow_cache=True, only_cache=False, manufacturer=None)
|
|
|
|
in on_off_cluster.read_attributes.call_args_list
|
|
|
|
)
|
|
|
|
assert (
|
|
|
|
call(["on_off"], allow_cache=False, only_cache=False, manufacturer=None)
|
|
|
|
in on_off_cluster.read_attributes.call_args_list
|
|
|
|
)
|
|
|
|
|
|
|
|
select_name = general.OnOff.StartUpOnOff.__name__
|
|
|
|
entity_id = await find_entity_id(
|
|
|
|
Platform.SELECT,
|
|
|
|
zha_device,
|
|
|
|
hass,
|
|
|
|
qualifier=select_name.lower(),
|
|
|
|
)
|
|
|
|
assert entity_id is not None
|
|
|
|
|
|
|
|
state = hass.states.get(entity_id)
|
|
|
|
assert state
|
|
|
|
assert state.state == general.OnOff.StartUpOnOff.On.name
|
|
|
|
assert state.attributes["options"] == ["Off", "On", "Toggle", "PreviousValue"]
|
|
|
|
|
|
|
|
entity_entry = entity_registry.async_get(entity_id)
|
|
|
|
assert entity_entry
|
|
|
|
assert entity_entry.entity_category == ENTITY_CATEGORY_CONFIG
|
|
|
|
|
|
|
|
|
2022-04-24 16:50:06 +00:00
|
|
|
async def test_on_off_select_unsupported(hass, light, zha_device_joined_restored):
|
|
|
|
"""Test zha on off select unsupported."""
|
|
|
|
|
|
|
|
on_off_cluster = light.endpoints[1].on_off
|
|
|
|
on_off_cluster.add_unsupported_attribute("start_up_on_off")
|
|
|
|
zha_device = await zha_device_joined_restored(light)
|
|
|
|
select_name = general.OnOff.StartUpOnOff.__name__
|
|
|
|
entity_id = await find_entity_id(
|
|
|
|
Platform.SELECT,
|
|
|
|
zha_device,
|
|
|
|
hass,
|
|
|
|
qualifier=select_name.lower(),
|
|
|
|
)
|
|
|
|
assert entity_id is None
|