2018-10-11 10:53:54 +00:00
|
|
|
"""The tests for the Template lock platform."""
|
2020-10-16 11:25:57 +00:00
|
|
|
import pytest
|
|
|
|
|
2018-10-11 10:53:54 +00:00
|
|
|
from homeassistant import setup
|
|
|
|
from homeassistant.components import lock
|
2022-05-13 16:33:25 +00:00
|
|
|
from homeassistant.const import ATTR_ENTITY_ID, STATE_OFF, STATE_ON, STATE_UNAVAILABLE
|
2022-05-12 14:29:48 +00:00
|
|
|
|
|
|
|
OPTIMISTIC_LOCK_CONFIG = {
|
|
|
|
"platform": "template",
|
|
|
|
"lock": {
|
2022-05-13 16:33:25 +00:00
|
|
|
"service": "test.automation",
|
|
|
|
"data_template": {
|
|
|
|
"action": "lock",
|
|
|
|
"caller": "{{ this.entity_id }}",
|
|
|
|
},
|
2022-05-12 14:29:48 +00:00
|
|
|
},
|
|
|
|
"unlock": {
|
2022-05-13 16:33:25 +00:00
|
|
|
"service": "test.automation",
|
|
|
|
"data_template": {
|
|
|
|
"action": "unlock",
|
|
|
|
"caller": "{{ this.entity_id }}",
|
|
|
|
},
|
2022-05-12 14:29:48 +00:00
|
|
|
},
|
|
|
|
}
|
2018-10-11 10:53:54 +00:00
|
|
|
|
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(1, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
|
|
|
{
|
|
|
|
lock.DOMAIN: {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2021-09-11 18:48:25 +00:00
|
|
|
"name": "Test template lock",
|
|
|
|
"value_template": "{{ states.switch.test_state.state }}",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
],
|
|
|
|
)
|
|
|
|
async def test_template_state(hass, start_ha):
|
|
|
|
"""Test template."""
|
2020-10-16 11:25:57 +00:00
|
|
|
hass.states.async_set("switch.test_state", STATE_ON)
|
|
|
|
await hass.async_block_till_done()
|
2018-10-11 10:53:54 +00:00
|
|
|
|
2020-10-16 11:25:57 +00:00
|
|
|
state = hass.states.get("lock.test_template_lock")
|
|
|
|
assert state.state == lock.STATE_LOCKED
|
2018-10-11 10:53:54 +00:00
|
|
|
|
2020-10-16 11:25:57 +00:00
|
|
|
hass.states.async_set("switch.test_state", STATE_OFF)
|
|
|
|
await hass.async_block_till_done()
|
2018-10-11 10:53:54 +00:00
|
|
|
|
2020-10-16 11:25:57 +00:00
|
|
|
state = hass.states.get("lock.test_template_lock")
|
|
|
|
assert state.state == lock.STATE_UNLOCKED
|
2018-10-11 10:53:54 +00:00
|
|
|
|
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(1, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
|
|
|
{
|
|
|
|
lock.DOMAIN: {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2021-09-11 18:48:25 +00:00
|
|
|
"value_template": "{{ 1 == 1 }}",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
],
|
|
|
|
)
|
|
|
|
async def test_template_state_boolean_on(hass, start_ha):
|
|
|
|
"""Test the setting of the state with boolean on."""
|
2020-10-16 11:25:57 +00:00
|
|
|
state = hass.states.get("lock.template_lock")
|
|
|
|
assert state.state == lock.STATE_LOCKED
|
2018-10-11 10:53:54 +00:00
|
|
|
|
2020-10-16 11:25:57 +00:00
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(1, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
|
|
|
{
|
|
|
|
lock.DOMAIN: {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2021-09-11 18:48:25 +00:00
|
|
|
"value_template": "{{ 1 == 2 }}",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
],
|
|
|
|
)
|
|
|
|
async def test_template_state_boolean_off(hass, start_ha):
|
|
|
|
"""Test the setting of the state with off."""
|
2020-10-16 11:25:57 +00:00
|
|
|
state = hass.states.get("lock.template_lock")
|
|
|
|
assert state.state == lock.STATE_UNLOCKED
|
2018-10-11 10:53:54 +00:00
|
|
|
|
2020-10-16 11:25:57 +00:00
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(0, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
|
|
|
{
|
|
|
|
lock.DOMAIN: {
|
|
|
|
"platform": "template",
|
|
|
|
"value_template": "{% if rubbish %}",
|
2020-10-16 11:25:57 +00:00
|
|
|
"lock": {
|
2021-09-11 18:48:25 +00:00
|
|
|
"service": "switch.turn_on",
|
|
|
|
"entity_id": "switch.test_state",
|
|
|
|
},
|
|
|
|
"unlock": {
|
|
|
|
"service": "switch.turn_off",
|
|
|
|
"entity_id": "switch.test_state",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"switch": {
|
|
|
|
"platform": "lock",
|
|
|
|
"name": "{{%}",
|
|
|
|
"value_template": "{{ rubbish }",
|
2019-07-31 19:25:30 +00:00
|
|
|
"lock": {
|
2021-09-11 18:48:25 +00:00
|
|
|
"service": "switch.turn_on",
|
|
|
|
"entity_id": "switch.test_state",
|
|
|
|
},
|
|
|
|
"unlock": {
|
|
|
|
"service": "switch.turn_off",
|
|
|
|
"entity_id": "switch.test_state",
|
|
|
|
},
|
2019-07-31 19:25:30 +00:00
|
|
|
},
|
2021-09-11 18:48:25 +00:00
|
|
|
},
|
|
|
|
{lock.DOMAIN: {"platform": "template", "value_template": "Invalid"}},
|
|
|
|
{
|
|
|
|
lock.DOMAIN: {
|
|
|
|
"platform": "template",
|
|
|
|
"not_value_template": "{{ states.switch.test_state.state }}",
|
|
|
|
"lock": {
|
|
|
|
"service": "switch.turn_on",
|
|
|
|
"entity_id": "switch.test_state",
|
|
|
|
},
|
|
|
|
"unlock": {
|
|
|
|
"service": "switch.turn_off",
|
|
|
|
"entity_id": "switch.test_state",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
],
|
|
|
|
)
|
|
|
|
async def test_template_syntax_error(hass, start_ha):
|
|
|
|
"""Test templating syntax error."""
|
2021-10-07 10:58:00 +00:00
|
|
|
assert hass.states.async_all("lock") == []
|
2018-10-11 10:53:54 +00:00
|
|
|
|
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(1, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
|
|
|
{
|
|
|
|
lock.DOMAIN: {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2021-09-11 18:48:25 +00:00
|
|
|
"value_template": "{{ 1 + 1 }}",
|
|
|
|
}
|
|
|
|
},
|
|
|
|
],
|
|
|
|
)
|
|
|
|
async def test_template_static(hass, start_ha):
|
|
|
|
"""Test that we allow static templates."""
|
2020-10-16 11:25:57 +00:00
|
|
|
state = hass.states.get("lock.template_lock")
|
|
|
|
assert state.state == lock.STATE_UNLOCKED
|
|
|
|
|
|
|
|
hass.states.async_set("lock.template_lock", lock.STATE_LOCKED)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
state = hass.states.get("lock.template_lock")
|
|
|
|
assert state.state == lock.STATE_LOCKED
|
|
|
|
|
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(1, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
2020-10-16 11:25:57 +00:00
|
|
|
{
|
2021-09-11 18:48:25 +00:00
|
|
|
lock.DOMAIN: {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2020-10-16 11:25:57 +00:00
|
|
|
"value_template": "{{ states.switch.test_state.state }}",
|
|
|
|
}
|
|
|
|
},
|
2021-09-11 18:48:25 +00:00
|
|
|
],
|
|
|
|
)
|
2022-05-13 16:33:25 +00:00
|
|
|
async def test_lock_action(hass, start_ha, calls):
|
2021-09-11 18:48:25 +00:00
|
|
|
"""Test lock action."""
|
2022-05-12 14:29:48 +00:00
|
|
|
await setup.async_setup_component(hass, "switch", {})
|
2020-10-16 11:25:57 +00:00
|
|
|
hass.states.async_set("switch.test_state", STATE_OFF)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
state = hass.states.get("lock.template_lock")
|
|
|
|
assert state.state == lock.STATE_UNLOCKED
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
lock.DOMAIN, lock.SERVICE_LOCK, {ATTR_ENTITY_ID: "lock.template_lock"}
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
2022-05-13 16:33:25 +00:00
|
|
|
assert len(calls) == 1
|
|
|
|
assert calls[0].data["action"] == "lock"
|
|
|
|
assert calls[0].data["caller"] == "lock.template_lock"
|
2020-10-16 11:25:57 +00:00
|
|
|
|
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(1, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
2020-10-16 11:25:57 +00:00
|
|
|
{
|
2021-09-11 18:48:25 +00:00
|
|
|
lock.DOMAIN: {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2020-10-16 11:25:57 +00:00
|
|
|
"value_template": "{{ states.switch.test_state.state }}",
|
|
|
|
}
|
|
|
|
},
|
2021-09-11 18:48:25 +00:00
|
|
|
],
|
|
|
|
)
|
2022-05-13 16:33:25 +00:00
|
|
|
async def test_unlock_action(hass, start_ha, calls):
|
2021-09-11 18:48:25 +00:00
|
|
|
"""Test unlock action."""
|
2022-05-12 14:29:48 +00:00
|
|
|
await setup.async_setup_component(hass, "switch", {})
|
2020-10-16 11:25:57 +00:00
|
|
|
hass.states.async_set("switch.test_state", STATE_ON)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
state = hass.states.get("lock.template_lock")
|
|
|
|
assert state.state == lock.STATE_LOCKED
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
lock.DOMAIN, lock.SERVICE_UNLOCK, {ATTR_ENTITY_ID: "lock.template_lock"}
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
2022-05-13 16:33:25 +00:00
|
|
|
assert len(calls) == 1
|
|
|
|
assert calls[0].data["action"] == "unlock"
|
|
|
|
assert calls[0].data["caller"] == "lock.template_lock"
|
2019-10-01 12:15:15 +00:00
|
|
|
|
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(1, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
2021-07-21 04:50:21 +00:00
|
|
|
{
|
2021-09-11 18:48:25 +00:00
|
|
|
lock.DOMAIN: {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2021-07-21 04:50:21 +00:00
|
|
|
"value_template": "{{ states.input_select.test_state.state }}",
|
|
|
|
}
|
|
|
|
},
|
2021-09-11 18:48:25 +00:00
|
|
|
],
|
|
|
|
)
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"test_state", [lock.STATE_UNLOCKING, lock.STATE_LOCKING, lock.STATE_JAMMED]
|
|
|
|
)
|
|
|
|
async def test_lock_state(hass, test_state, start_ha):
|
2022-05-12 14:29:48 +00:00
|
|
|
"""Test value template."""
|
2021-09-11 18:48:25 +00:00
|
|
|
hass.states.async_set("input_select.test_state", test_state)
|
2021-07-21 04:50:21 +00:00
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
state = hass.states.get("lock.template_lock")
|
2021-09-11 18:48:25 +00:00
|
|
|
assert state.state == test_state
|
2021-07-21 04:50:21 +00:00
|
|
|
|
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(1, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
2021-07-21 04:50:21 +00:00
|
|
|
{
|
2021-09-11 18:48:25 +00:00
|
|
|
lock.DOMAIN: {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2019-11-26 00:30:49 +00:00
|
|
|
"value_template": "{{ states('switch.test_state') }}",
|
2019-10-01 12:15:15 +00:00
|
|
|
"availability_template": "{{ is_state('availability_state.state', 'on') }}",
|
|
|
|
}
|
|
|
|
},
|
2021-09-11 18:48:25 +00:00
|
|
|
],
|
|
|
|
)
|
|
|
|
async def test_available_template_with_entities(hass, start_ha):
|
|
|
|
"""Test availability templates with values from other entities."""
|
2019-10-01 12:15:15 +00:00
|
|
|
# When template returns true..
|
|
|
|
hass.states.async_set("availability_state.state", STATE_ON)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# Device State should not be unavailable
|
|
|
|
assert hass.states.get("lock.template_lock").state != STATE_UNAVAILABLE
|
|
|
|
|
|
|
|
# When Availability template returns false
|
|
|
|
hass.states.async_set("availability_state.state", STATE_OFF)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# device state should be unavailable
|
|
|
|
assert hass.states.get("lock.template_lock").state == STATE_UNAVAILABLE
|
|
|
|
|
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(1, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
2019-10-01 12:15:15 +00:00
|
|
|
{
|
2021-09-11 18:48:25 +00:00
|
|
|
lock.DOMAIN: {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2019-10-01 12:15:15 +00:00
|
|
|
"value_template": "{{ 1 + 1 }}",
|
|
|
|
"availability_template": "{{ x - 12 }}",
|
|
|
|
}
|
|
|
|
},
|
2021-09-11 18:48:25 +00:00
|
|
|
],
|
|
|
|
)
|
|
|
|
async def test_invalid_availability_template_keeps_component_available(
|
|
|
|
hass, start_ha, caplog_setup_text
|
|
|
|
):
|
|
|
|
"""Test that an invalid availability keeps the device available."""
|
2019-10-01 12:15:15 +00:00
|
|
|
assert hass.states.get("lock.template_lock").state != STATE_UNAVAILABLE
|
2021-09-11 18:48:25 +00:00
|
|
|
assert ("UndefinedError: 'x' is undefined") in caplog_setup_text
|
2020-08-01 22:45:55 +00:00
|
|
|
|
|
|
|
|
2023-02-15 13:09:50 +00:00
|
|
|
@pytest.mark.parametrize(("count", "domain"), [(1, lock.DOMAIN)])
|
2021-09-11 18:48:25 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"config",
|
|
|
|
[
|
2020-08-01 22:45:55 +00:00
|
|
|
{
|
2021-09-11 18:48:25 +00:00
|
|
|
lock.DOMAIN: {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2020-08-01 22:45:55 +00:00
|
|
|
"name": "test_template_lock_01",
|
|
|
|
"unique_id": "not-so-unique-anymore",
|
|
|
|
"value_template": "{{ true }}",
|
2021-09-11 18:48:25 +00:00
|
|
|
}
|
2020-08-01 22:45:55 +00:00
|
|
|
},
|
2021-09-11 18:48:25 +00:00
|
|
|
],
|
|
|
|
)
|
|
|
|
async def test_unique_id(hass, start_ha):
|
|
|
|
"""Test unique_id option only creates one lock per id."""
|
2020-08-01 22:45:55 +00:00
|
|
|
await setup.async_setup_component(
|
|
|
|
hass,
|
2020-10-16 11:25:57 +00:00
|
|
|
lock.DOMAIN,
|
2020-08-01 22:45:55 +00:00
|
|
|
{
|
|
|
|
"lock": {
|
2022-05-12 14:29:48 +00:00
|
|
|
**OPTIMISTIC_LOCK_CONFIG,
|
2020-08-01 22:45:55 +00:00
|
|
|
"name": "test_template_lock_02",
|
|
|
|
"unique_id": "not-so-unique-anymore",
|
|
|
|
"value_template": "{{ false }}",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
await hass.async_start()
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
2021-10-07 10:58:00 +00:00
|
|
|
assert len(hass.states.async_all("lock")) == 1
|