Add lock support to deCONZ (#40807)

Co-authored-by: Paulus Schoutsen <paulus@home-assistant.io>
pull/40930/head
Robert Svensson 2020-10-01 09:50:06 +02:00 committed by GitHub
parent 95d228cace
commit 9116061262
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 168 additions and 4 deletions

View File

@ -23,6 +23,7 @@ SUPPORTED_PLATFORMS = [
"climate", "climate",
"cover", "cover",
"light", "light",
"lock",
"scene", "scene",
"sensor", "sensor",
"switch", "switch",
@ -38,10 +39,16 @@ ATTR_OFFSET = "offset"
ATTR_ON = "on" ATTR_ON = "on"
ATTR_VALVE = "valve" ATTR_VALVE = "valve"
# Covers
DAMPERS = ["Level controllable output"] DAMPERS = ["Level controllable output"]
WINDOW_COVERS = ["Window covering device", "Window covering controller"] WINDOW_COVERS = ["Window covering device", "Window covering controller"]
COVER_TYPES = DAMPERS + WINDOW_COVERS COVER_TYPES = DAMPERS + WINDOW_COVERS
# Locks
LOCKS = ["Door Lock"]
LOCK_TYPES = LOCKS
# Switches
POWER_PLUGS = ["On/Off light", "On/Off plug-in unit", "Smart plug"] POWER_PLUGS = ["On/Off light", "On/Off plug-in unit", "Smart plug"]
SIRENS = ["Warning device"] SIRENS = ["Warning device"]
SWITCH_TYPES = POWER_PLUGS + SIRENS SWITCH_TYPES = POWER_PLUGS + SIRENS

View File

@ -26,6 +26,7 @@ from .const import (
CONF_GROUP_ID_BASE, CONF_GROUP_ID_BASE,
COVER_TYPES, COVER_TYPES,
DOMAIN as DECONZ_DOMAIN, DOMAIN as DECONZ_DOMAIN,
LOCK_TYPES,
NEW_GROUP, NEW_GROUP,
NEW_LIGHT, NEW_LIGHT,
SWITCH_TYPES, SWITCH_TYPES,
@ -50,7 +51,7 @@ async def async_setup_entry(hass, config_entry, async_add_entities):
for light in lights: for light in lights:
if ( if (
light.type not in COVER_TYPES + SWITCH_TYPES light.type not in COVER_TYPES + LOCK_TYPES + SWITCH_TYPES
and light.uniqueid not in gateway.entities[DOMAIN] and light.uniqueid not in gateway.entities[DOMAIN]
): ):
entities.append(DeconzLight(light, gateway)) entities.append(DeconzLight(light, gateway))

View File

@ -0,0 +1,59 @@
"""Support for deCONZ locks."""
from homeassistant.components.lock import DOMAIN, LockEntity
from homeassistant.core import callback
from homeassistant.helpers.dispatcher import async_dispatcher_connect
from .const import LOCKS, NEW_LIGHT
from .deconz_device import DeconzDevice
from .gateway import get_gateway_from_config_entry
async def async_setup_entry(hass, config_entry, async_add_entities):
"""Set up locks for deCONZ component.
Locks are based on the same device class as lights in deCONZ.
"""
gateway = get_gateway_from_config_entry(hass, config_entry)
gateway.entities[DOMAIN] = set()
@callback
def async_add_lock(lights):
"""Add lock from deCONZ."""
entities = []
for light in lights:
if light.type in LOCKS and light.uniqueid not in gateway.entities[DOMAIN]:
entities.append(DeconzLock(light, gateway))
if entities:
async_add_entities(entities, True)
gateway.listeners.append(
async_dispatcher_connect(
hass, gateway.async_signal_new_device(NEW_LIGHT), async_add_lock
)
)
async_add_lock(gateway.api.lights.values())
class DeconzLock(DeconzDevice, LockEntity):
"""Representation of a deCONZ lock."""
TYPE = DOMAIN
@property
def is_locked(self):
"""Return true if lock is on."""
return self._device.state
async def async_lock(self, **kwargs):
"""Lock the lock."""
data = {"on": True}
await self._device.async_set_state(data)
async def async_unlock(self, **kwargs):
"""Unlock the lock."""
data = {"on": False}
await self._device.async_set_state(data)

View File

@ -91,9 +91,10 @@ async def test_gateway_setup(hass):
assert forward_entry_setup.mock_calls[1][1] == (entry, "climate") assert forward_entry_setup.mock_calls[1][1] == (entry, "climate")
assert forward_entry_setup.mock_calls[2][1] == (entry, "cover") assert forward_entry_setup.mock_calls[2][1] == (entry, "cover")
assert forward_entry_setup.mock_calls[3][1] == (entry, "light") assert forward_entry_setup.mock_calls[3][1] == (entry, "light")
assert forward_entry_setup.mock_calls[4][1] == (entry, "scene") assert forward_entry_setup.mock_calls[4][1] == (entry, "lock")
assert forward_entry_setup.mock_calls[5][1] == (entry, "sensor") assert forward_entry_setup.mock_calls[5][1] == (entry, "scene")
assert forward_entry_setup.mock_calls[6][1] == (entry, "switch") assert forward_entry_setup.mock_calls[6][1] == (entry, "sensor")
assert forward_entry_setup.mock_calls[7][1] == (entry, "switch")
async def test_gateway_retry(hass): async def test_gateway_retry(hass):

View File

@ -0,0 +1,96 @@
"""deCONZ lock platform tests."""
from copy import deepcopy
from homeassistant.components import deconz
import homeassistant.components.lock as lock
from homeassistant.const import STATE_LOCKED, STATE_UNLOCKED
from homeassistant.setup import async_setup_component
from .test_gateway import DECONZ_WEB_REQUEST, setup_deconz_integration
from tests.async_mock import patch
LOCKS = {
"1": {
"etag": "5c2ec06cde4bd654aef3a555fcd8ad12",
"hascolor": False,
"lastannounced": None,
"lastseen": "2020-08-22T15:29:03Z",
"manufacturername": "Danalock",
"modelid": "V3-BTZB",
"name": "Door lock",
"state": {"alert": "none", "on": False, "reachable": True},
"swversion": "19042019",
"type": "Door Lock",
"uniqueid": "00:00:00:00:00:00:00:00-00",
}
}
async def test_platform_manually_configured(hass):
"""Test that we do not discover anything or try to set up a gateway."""
assert (
await async_setup_component(
hass, lock.DOMAIN, {"lock": {"platform": deconz.DOMAIN}}
)
is True
)
assert deconz.DOMAIN not in hass.data
async def test_no_locks(hass):
"""Test that no lock entities are created."""
gateway = await setup_deconz_integration(hass)
assert len(gateway.deconz_ids) == 0
assert len(hass.states.async_all()) == 0
async def test_locks(hass):
"""Test that all supported lock entities are created."""
data = deepcopy(DECONZ_WEB_REQUEST)
data["lights"] = deepcopy(LOCKS)
gateway = await setup_deconz_integration(hass, get_state_response=data)
assert "lock.door_lock" in gateway.deconz_ids
assert len(hass.states.async_all()) == 1
door_lock = hass.states.get("lock.door_lock")
assert door_lock.state == STATE_UNLOCKED
state_changed_event = {
"t": "event",
"e": "changed",
"r": "lights",
"id": "1",
"state": {"on": True},
}
gateway.api.event_handler(state_changed_event)
await hass.async_block_till_done()
door_lock = hass.states.get("lock.door_lock")
assert door_lock.state == STATE_LOCKED
door_lock_device = gateway.api.lights["1"]
with patch.object(door_lock_device, "_request", return_value=True) as set_callback:
await hass.services.async_call(
lock.DOMAIN,
lock.SERVICE_LOCK,
{"entity_id": "lock.door_lock"},
blocking=True,
)
await hass.async_block_till_done()
set_callback.assert_called_with("put", "/lights/1/state", json={"on": True})
with patch.object(door_lock_device, "_request", return_value=True) as set_callback:
await hass.services.async_call(
lock.DOMAIN,
lock.SERVICE_UNLOCK,
{"entity_id": "lock.door_lock"},
blocking=True,
)
await hass.async_block_till_done()
set_callback.assert_called_with("put", "/lights/1/state", json={"on": False})
await gateway.async_reset()
assert len(hass.states.async_all()) == 0