2018-03-15 01:48:21 +00:00
|
|
|
"""Test HomeKit util module."""
|
2021-09-04 03:15:28 +00:00
|
|
|
from unittest.mock import MagicMock, Mock, patch
|
2021-02-24 00:22:23 +00:00
|
|
|
|
2018-04-12 13:01:41 +00:00
|
|
|
import pytest
|
2018-05-10 23:21:59 +00:00
|
|
|
import voluptuous as vol
|
2018-03-15 01:48:21 +00:00
|
|
|
|
2018-05-25 09:37:20 +00:00
|
|
|
from homeassistant.components.homekit.const import (
|
2021-02-02 13:30:38 +00:00
|
|
|
BRIDGE_NAME,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_FEATURE,
|
|
|
|
CONF_FEATURE_LIST,
|
|
|
|
CONF_LINKED_BATTERY_SENSOR,
|
|
|
|
CONF_LOW_BATTERY_THRESHOLD,
|
2020-05-01 04:05:06 +00:00
|
|
|
DEFAULT_CONFIG_FLOW_PORT,
|
|
|
|
DOMAIN,
|
2019-07-31 19:25:30 +00:00
|
|
|
FEATURE_ON_OFF,
|
|
|
|
FEATURE_PLAY_PAUSE,
|
2020-04-21 22:38:43 +00:00
|
|
|
HOMEKIT_PAIRING_QR,
|
|
|
|
HOMEKIT_PAIRING_QR_SECRET,
|
2019-07-31 19:25:30 +00:00
|
|
|
TYPE_FAUCET,
|
|
|
|
TYPE_OUTLET,
|
|
|
|
TYPE_SHOWER,
|
|
|
|
TYPE_SPRINKLER,
|
|
|
|
TYPE_SWITCH,
|
|
|
|
TYPE_VALVE,
|
|
|
|
)
|
2018-03-15 01:48:21 +00:00
|
|
|
from homeassistant.components.homekit.util import (
|
2021-02-24 00:22:23 +00:00
|
|
|
accessory_friendly_name,
|
2021-11-23 17:40:20 +00:00
|
|
|
async_dismiss_setup_message,
|
2021-02-02 13:30:38 +00:00
|
|
|
async_find_next_available_port,
|
2021-09-04 03:15:28 +00:00
|
|
|
async_port_is_available,
|
2021-11-23 17:40:20 +00:00
|
|
|
async_show_setup_message,
|
2020-05-02 21:15:44 +00:00
|
|
|
cleanup_name_for_homekit,
|
2019-07-31 19:25:30 +00:00
|
|
|
convert_to_float,
|
|
|
|
density_to_air_quality,
|
2022-01-04 15:19:12 +00:00
|
|
|
format_version,
|
2021-03-31 16:22:30 +00:00
|
|
|
state_needs_accessory_mode,
|
2019-07-31 19:25:30 +00:00
|
|
|
temperature_to_homekit,
|
|
|
|
temperature_to_states,
|
|
|
|
validate_entity_config as vec,
|
|
|
|
validate_media_player_features,
|
|
|
|
)
|
2021-11-23 17:40:20 +00:00
|
|
|
from homeassistant.components.persistent_notification import async_create, async_dismiss
|
2018-03-15 01:48:21 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_CODE,
|
|
|
|
ATTR_SUPPORTED_FEATURES,
|
|
|
|
CONF_NAME,
|
2021-02-02 13:30:38 +00:00
|
|
|
CONF_PORT,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_TYPE,
|
|
|
|
STATE_UNKNOWN,
|
|
|
|
TEMP_CELSIUS,
|
|
|
|
TEMP_FAHRENHEIT,
|
|
|
|
)
|
2018-06-17 11:37:44 +00:00
|
|
|
from homeassistant.core import State
|
2018-03-15 01:48:21 +00:00
|
|
|
|
2020-05-01 04:05:06 +00:00
|
|
|
from .util import async_init_integration
|
|
|
|
|
2021-10-07 10:58:00 +00:00
|
|
|
from tests.common import MockConfigEntry
|
2018-03-15 01:48:21 +00:00
|
|
|
|
|
|
|
|
2021-09-04 03:15:28 +00:00
|
|
|
def _mock_socket(failure_attempts: int = 0) -> MagicMock:
|
|
|
|
"""Mock a socket that fails to bind failure_attempts amount of times."""
|
|
|
|
mock_socket = MagicMock()
|
|
|
|
attempts = 0
|
|
|
|
|
|
|
|
def _simulate_bind(*_):
|
|
|
|
nonlocal attempts
|
|
|
|
attempts += 1
|
|
|
|
if attempts <= failure_attempts:
|
|
|
|
raise OSError
|
|
|
|
return
|
|
|
|
|
|
|
|
mock_socket.bind = Mock(side_effect=_simulate_bind)
|
|
|
|
return mock_socket
|
|
|
|
|
|
|
|
|
2018-04-12 13:01:41 +00:00
|
|
|
def test_validate_entity_config():
|
|
|
|
"""Test validate entities."""
|
2019-07-31 19:25:30 +00:00
|
|
|
configs = [
|
|
|
|
None,
|
|
|
|
[],
|
|
|
|
"string",
|
|
|
|
12345,
|
|
|
|
{"invalid_entity_id": {}},
|
|
|
|
{"demo.test": 1},
|
|
|
|
{"binary_sensor.demo": {CONF_LINKED_BATTERY_SENSOR: None}},
|
|
|
|
{"binary_sensor.demo": {CONF_LINKED_BATTERY_SENSOR: "switch.demo"}},
|
|
|
|
{"binary_sensor.demo": {CONF_LOW_BATTERY_THRESHOLD: "switch.demo"}},
|
|
|
|
{"binary_sensor.demo": {CONF_LOW_BATTERY_THRESHOLD: -10}},
|
|
|
|
{"demo.test": "test"},
|
|
|
|
{"demo.test": [1, 2]},
|
|
|
|
{"demo.test": None},
|
|
|
|
{"demo.test": {CONF_NAME: None}},
|
|
|
|
{"media_player.test": {CONF_FEATURE_LIST: [{CONF_FEATURE: "invalid_feature"}]}},
|
|
|
|
{
|
|
|
|
"media_player.test": {
|
|
|
|
CONF_FEATURE_LIST: [
|
2018-05-28 14:26:33 +00:00
|
|
|
{CONF_FEATURE: FEATURE_ON_OFF},
|
2019-07-31 19:25:30 +00:00
|
|
|
{CONF_FEATURE: FEATURE_ON_OFF},
|
|
|
|
]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{"switch.test": {CONF_TYPE: "invalid_type"}},
|
|
|
|
]
|
2018-04-12 13:01:41 +00:00
|
|
|
|
|
|
|
for conf in configs:
|
|
|
|
with pytest.raises(vol.Invalid):
|
|
|
|
vec(conf)
|
|
|
|
|
|
|
|
assert vec({}) == {}
|
2019-07-31 19:25:30 +00:00
|
|
|
assert vec({"demo.test": {CONF_NAME: "Name"}}) == {
|
|
|
|
"demo.test": {CONF_NAME: "Name", CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert vec(
|
|
|
|
{"binary_sensor.demo": {CONF_LINKED_BATTERY_SENSOR: "sensor.demo_battery"}}
|
|
|
|
) == {
|
|
|
|
"binary_sensor.demo": {
|
|
|
|
CONF_LINKED_BATTERY_SENSOR: "sensor.demo_battery",
|
|
|
|
CONF_LOW_BATTERY_THRESHOLD: 20,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert vec({"binary_sensor.demo": {CONF_LOW_BATTERY_THRESHOLD: 50}}) == {
|
|
|
|
"binary_sensor.demo": {CONF_LOW_BATTERY_THRESHOLD: 50}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert vec({"alarm_control_panel.demo": {}}) == {
|
|
|
|
"alarm_control_panel.demo": {ATTR_CODE: None, CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
assert vec({"alarm_control_panel.demo": {ATTR_CODE: "1234"}}) == {
|
|
|
|
"alarm_control_panel.demo": {ATTR_CODE: "1234", CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert vec({"lock.demo": {}}) == {
|
|
|
|
"lock.demo": {ATTR_CODE: None, CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
assert vec({"lock.demo": {ATTR_CODE: "1234"}}) == {
|
|
|
|
"lock.demo": {ATTR_CODE: "1234", CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert vec({"media_player.demo": {}}) == {
|
|
|
|
"media_player.demo": {CONF_FEATURE_LIST: {}, CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
config = {
|
|
|
|
CONF_FEATURE_LIST: [
|
|
|
|
{CONF_FEATURE: FEATURE_ON_OFF},
|
|
|
|
{CONF_FEATURE: FEATURE_PLAY_PAUSE},
|
|
|
|
]
|
|
|
|
}
|
|
|
|
assert vec({"media_player.demo": config}) == {
|
|
|
|
"media_player.demo": {
|
|
|
|
CONF_FEATURE_LIST: {FEATURE_ON_OFF: {}, FEATURE_PLAY_PAUSE: {}},
|
|
|
|
CONF_LOW_BATTERY_THRESHOLD: 20,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert vec({"switch.demo": {CONF_TYPE: TYPE_FAUCET}}) == {
|
|
|
|
"switch.demo": {CONF_TYPE: TYPE_FAUCET, CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
assert vec({"switch.demo": {CONF_TYPE: TYPE_OUTLET}}) == {
|
|
|
|
"switch.demo": {CONF_TYPE: TYPE_OUTLET, CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
assert vec({"switch.demo": {CONF_TYPE: TYPE_SHOWER}}) == {
|
|
|
|
"switch.demo": {CONF_TYPE: TYPE_SHOWER, CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
assert vec({"switch.demo": {CONF_TYPE: TYPE_SPRINKLER}}) == {
|
|
|
|
"switch.demo": {CONF_TYPE: TYPE_SPRINKLER, CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
assert vec({"switch.demo": {CONF_TYPE: TYPE_SWITCH}}) == {
|
|
|
|
"switch.demo": {CONF_TYPE: TYPE_SWITCH, CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
|
|
|
assert vec({"switch.demo": {CONF_TYPE: TYPE_VALVE}}) == {
|
|
|
|
"switch.demo": {CONF_TYPE: TYPE_VALVE, CONF_LOW_BATTERY_THRESHOLD: 20}
|
|
|
|
}
|
2018-05-25 09:37:20 +00:00
|
|
|
|
|
|
|
|
2018-05-28 14:26:33 +00:00
|
|
|
def test_validate_media_player_features():
|
2018-05-25 09:37:20 +00:00
|
|
|
"""Test validate modes for media players."""
|
|
|
|
config = {}
|
|
|
|
attrs = {ATTR_SUPPORTED_FEATURES: 20873}
|
2019-07-31 19:25:30 +00:00
|
|
|
entity_state = State("media_player.demo", "on", attrs)
|
2018-05-28 14:26:33 +00:00
|
|
|
assert validate_media_player_features(entity_state, config) is True
|
|
|
|
|
|
|
|
config = {FEATURE_ON_OFF: None}
|
|
|
|
assert validate_media_player_features(entity_state, config) is True
|
2018-05-25 09:37:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
entity_state = State("media_player.demo", "on")
|
2018-05-28 14:26:33 +00:00
|
|
|
assert validate_media_player_features(entity_state, config) is False
|
2018-05-25 09:37:20 +00:00
|
|
|
|
2018-04-12 13:01:41 +00:00
|
|
|
|
|
|
|
def test_convert_to_float():
|
|
|
|
"""Test convert_to_float method."""
|
|
|
|
assert convert_to_float(12) == 12
|
|
|
|
assert convert_to_float(12.4) == 12.4
|
|
|
|
assert convert_to_float(STATE_UNKNOWN) is None
|
|
|
|
assert convert_to_float(None) is None
|
|
|
|
|
|
|
|
|
2020-05-02 21:15:44 +00:00
|
|
|
def test_cleanup_name_for_homekit():
|
|
|
|
"""Ensure name sanitize works as expected."""
|
|
|
|
|
|
|
|
assert cleanup_name_for_homekit("abc") == "abc"
|
|
|
|
assert cleanup_name_for_homekit("a b c") == "a b c"
|
|
|
|
assert cleanup_name_for_homekit("ab_c") == "ab c"
|
|
|
|
assert (
|
|
|
|
cleanup_name_for_homekit('ab!@#$%^&*()-=":.,><?//\\ frog')
|
|
|
|
== "ab--#---&----- -.,------ frog"
|
|
|
|
)
|
|
|
|
assert cleanup_name_for_homekit("の日本_語文字セット") == "の日本 語文字セット"
|
|
|
|
|
|
|
|
|
2018-04-12 13:01:41 +00:00
|
|
|
def test_temperature_to_homekit():
|
|
|
|
"""Test temperature conversion from HA to HomeKit."""
|
|
|
|
assert temperature_to_homekit(20.46, TEMP_CELSIUS) == 20.5
|
2019-10-04 00:44:07 +00:00
|
|
|
assert temperature_to_homekit(92.1, TEMP_FAHRENHEIT) == 33.4
|
2018-04-12 13:01:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_temperature_to_states():
|
|
|
|
"""Test temperature conversion from HomeKit to HA."""
|
|
|
|
assert temperature_to_states(20, TEMP_CELSIUS) == 20.0
|
2018-11-04 21:04:51 +00:00
|
|
|
assert temperature_to_states(20.2, TEMP_FAHRENHEIT) == 68.5
|
2018-04-12 13:01:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_density_to_air_quality():
|
|
|
|
"""Test map PM2.5 density to HomeKit AirQuality level."""
|
|
|
|
assert density_to_air_quality(0) == 1
|
|
|
|
assert density_to_air_quality(35) == 1
|
|
|
|
assert density_to_air_quality(35.1) == 2
|
|
|
|
assert density_to_air_quality(75) == 2
|
|
|
|
assert density_to_air_quality(115) == 3
|
|
|
|
assert density_to_air_quality(150) == 4
|
|
|
|
assert density_to_air_quality(300) == 5
|
|
|
|
|
|
|
|
|
2021-11-23 17:40:20 +00:00
|
|
|
async def test_async_show_setup_msg(hass, hk_driver, mock_get_source_ip):
|
2018-05-10 23:21:59 +00:00
|
|
|
"""Test show setup message as persistence notification."""
|
2019-07-31 19:25:30 +00:00
|
|
|
pincode = b"123-45-678"
|
2018-05-10 23:21:59 +00:00
|
|
|
|
2020-05-01 04:05:06 +00:00
|
|
|
entry = await async_init_integration(hass)
|
|
|
|
assert entry
|
|
|
|
|
2021-10-07 10:58:00 +00:00
|
|
|
with patch(
|
2021-11-23 17:40:20 +00:00
|
|
|
"homeassistant.components.persistent_notification.async_create",
|
|
|
|
side_effect=async_create,
|
2021-10-07 10:58:00 +00:00
|
|
|
) as mock_create:
|
2021-11-23 17:40:20 +00:00
|
|
|
async_show_setup_message(
|
|
|
|
hass, entry.entry_id, "bridge_name", pincode, "X-HM://0"
|
2021-10-07 10:58:00 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
2020-05-01 04:05:06 +00:00
|
|
|
assert hass.data[DOMAIN][entry.entry_id][HOMEKIT_PAIRING_QR_SECRET]
|
|
|
|
assert hass.data[DOMAIN][entry.entry_id][HOMEKIT_PAIRING_QR]
|
2018-05-10 23:21:59 +00:00
|
|
|
|
2021-10-07 10:58:00 +00:00
|
|
|
assert len(mock_create.mock_calls) == 1
|
|
|
|
assert mock_create.mock_calls[0][1][3] == entry.entry_id
|
|
|
|
assert pincode.decode() in mock_create.mock_calls[0][1][1]
|
2018-05-10 23:21:59 +00:00
|
|
|
|
|
|
|
|
2021-11-23 17:40:20 +00:00
|
|
|
async def test_async_dismiss_setup_msg(hass):
|
2018-05-10 23:21:59 +00:00
|
|
|
"""Test dismiss setup message."""
|
2021-10-07 10:58:00 +00:00
|
|
|
with patch(
|
2021-11-23 17:40:20 +00:00
|
|
|
"homeassistant.components.persistent_notification.async_dismiss",
|
|
|
|
side_effect=async_dismiss,
|
2021-10-07 10:58:00 +00:00
|
|
|
) as mock_dismiss:
|
2021-11-23 17:40:20 +00:00
|
|
|
async_dismiss_setup_message(hass, "entry_id")
|
2021-10-07 10:58:00 +00:00
|
|
|
await hass.async_block_till_done()
|
2018-05-10 23:21:59 +00:00
|
|
|
|
2021-10-07 10:58:00 +00:00
|
|
|
assert len(mock_dismiss.mock_calls) == 1
|
|
|
|
assert mock_dismiss.mock_calls[0][1][1] == "entry_id"
|
2019-02-05 15:11:19 +00:00
|
|
|
|
|
|
|
|
2020-05-01 04:05:06 +00:00
|
|
|
async def test_port_is_available(hass):
|
|
|
|
"""Test we can get an available port and it is actually available."""
|
2021-09-04 03:15:28 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.components.homekit.util.socket.socket",
|
|
|
|
return_value=_mock_socket(0),
|
|
|
|
):
|
|
|
|
next_port = async_find_next_available_port(hass, DEFAULT_CONFIG_FLOW_PORT)
|
2021-02-02 13:30:38 +00:00
|
|
|
assert next_port
|
2021-09-04 03:15:28 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.components.homekit.util.socket.socket",
|
|
|
|
return_value=_mock_socket(0),
|
|
|
|
):
|
|
|
|
assert async_port_is_available(next_port)
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.homekit.util.socket.socket",
|
|
|
|
return_value=_mock_socket(5),
|
|
|
|
):
|
|
|
|
next_port = async_find_next_available_port(hass, DEFAULT_CONFIG_FLOW_PORT)
|
|
|
|
assert next_port == DEFAULT_CONFIG_FLOW_PORT + 5
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.homekit.util.socket.socket",
|
|
|
|
return_value=_mock_socket(0),
|
|
|
|
):
|
|
|
|
assert async_port_is_available(next_port)
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.homekit.util.socket.socket",
|
|
|
|
return_value=_mock_socket(1),
|
|
|
|
):
|
|
|
|
assert not async_port_is_available(next_port)
|
2021-02-02 13:30:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_port_is_available_skips_existing_entries(hass):
|
|
|
|
"""Test we can get an available port and it is actually available."""
|
|
|
|
entry = MockConfigEntry(
|
|
|
|
domain=DOMAIN,
|
|
|
|
data={CONF_NAME: BRIDGE_NAME, CONF_PORT: DEFAULT_CONFIG_FLOW_PORT},
|
|
|
|
options={},
|
2020-05-01 04:05:06 +00:00
|
|
|
)
|
2021-02-02 13:30:38 +00:00
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
2021-09-04 03:15:28 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.components.homekit.util.socket.socket",
|
|
|
|
return_value=_mock_socket(),
|
|
|
|
):
|
|
|
|
next_port = async_find_next_available_port(hass, DEFAULT_CONFIG_FLOW_PORT)
|
|
|
|
|
|
|
|
assert next_port == DEFAULT_CONFIG_FLOW_PORT + 1
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.homekit.util.socket.socket",
|
|
|
|
return_value=_mock_socket(),
|
|
|
|
):
|
|
|
|
assert async_port_is_available(next_port)
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.homekit.util.socket.socket",
|
|
|
|
return_value=_mock_socket(4),
|
|
|
|
):
|
|
|
|
next_port = async_find_next_available_port(hass, DEFAULT_CONFIG_FLOW_PORT)
|
|
|
|
|
|
|
|
assert next_port == DEFAULT_CONFIG_FLOW_PORT + 5
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.homekit.util.socket.socket",
|
|
|
|
return_value=_mock_socket(),
|
|
|
|
):
|
|
|
|
assert async_port_is_available(next_port)
|
|
|
|
|
|
|
|
with pytest.raises(OSError), patch(
|
|
|
|
"homeassistant.components.homekit.util.socket.socket",
|
|
|
|
return_value=_mock_socket(10),
|
|
|
|
):
|
|
|
|
async_find_next_available_port(hass, 65530)
|
2020-05-17 20:51:51 +00:00
|
|
|
|
|
|
|
|
2022-01-04 15:19:12 +00:00
|
|
|
async def test_format_version():
|
|
|
|
"""Test format_version method."""
|
|
|
|
assert format_version("soho+3.6.8+soho-release-rt120+10") == "3.6.8"
|
|
|
|
assert format_version("undefined-undefined-1.6.8") == "1.6.8"
|
|
|
|
assert format_version("56.0-76060") == "56.0.76060"
|
|
|
|
assert format_version(3.6) == "3.6"
|
|
|
|
assert format_version("AK001-ZJ100") == "001.100"
|
|
|
|
assert format_version("HF-LPB100-") == "100"
|
|
|
|
assert format_version("AK001-ZJ2149") == "001.2149"
|
|
|
|
assert format_version("0.1") == "0.1"
|
|
|
|
assert format_version("unknown") is None
|
2021-02-24 00:22:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_accessory_friendly_name():
|
|
|
|
"""Test we provide a helpful friendly name."""
|
|
|
|
|
|
|
|
accessory = Mock()
|
|
|
|
accessory.display_name = "same"
|
2021-03-12 06:05:03 +00:00
|
|
|
assert accessory_friendly_name("Same", accessory) == "Same"
|
2021-02-24 00:22:23 +00:00
|
|
|
assert accessory_friendly_name("hass title", accessory) == "hass title (same)"
|
2021-03-12 06:05:03 +00:00
|
|
|
accessory.display_name = "Hass title 123"
|
|
|
|
assert accessory_friendly_name("hass title", accessory) == "Hass title 123"
|
2021-03-31 16:22:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_lock_state_needs_accessory_mode(hass):
|
|
|
|
"""Test that locks are setup as accessories."""
|
|
|
|
hass.states.async_set("lock.mine", "locked")
|
|
|
|
assert state_needs_accessory_mode(hass.states.get("lock.mine")) is True
|