2019-04-03 15:40:03 +00:00
|
|
|
"""Nuki.io lock platform."""
|
2020-05-20 12:44:57 +00:00
|
|
|
from abc import ABC, abstractmethod
|
2017-05-02 16:18:47 +00:00
|
|
|
|
2022-01-05 21:57:55 +00:00
|
|
|
from pynuki.constants import MODE_OPENER_CONTINUOUS
|
2017-02-02 14:15:27 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2021-12-21 09:01:43 +00:00
|
|
|
from homeassistant.components.lock import SUPPORT_OPEN, LockEntity
|
2022-01-03 14:45:15 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant
|
2020-12-11 17:02:23 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv, entity_platform
|
2022-01-03 14:45:15 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2019-09-17 09:44:43 +00:00
|
|
|
|
2021-04-06 19:20:57 +00:00
|
|
|
from . import NukiEntity
|
|
|
|
from .const import (
|
|
|
|
ATTR_BATTERY_CRITICAL,
|
2021-06-30 06:43:02 +00:00
|
|
|
ATTR_ENABLE,
|
2021-04-06 19:20:57 +00:00
|
|
|
ATTR_NUKI_ID,
|
|
|
|
ATTR_UNLATCH,
|
|
|
|
DATA_COORDINATOR,
|
|
|
|
DATA_LOCKS,
|
|
|
|
DATA_OPENERS,
|
|
|
|
DOMAIN as NUKI_DOMAIN,
|
|
|
|
ERROR_STATES,
|
|
|
|
)
|
2017-02-02 14:15:27 +00:00
|
|
|
|
2021-01-29 10:05:13 +00:00
|
|
|
|
2022-01-03 14:45:15 +00:00
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
|
|
|
) -> None:
|
2021-01-29 10:05:13 +00:00
|
|
|
"""Set up the Nuki lock platform."""
|
2021-04-06 19:20:57 +00:00
|
|
|
data = hass.data[NUKI_DOMAIN][entry.entry_id]
|
|
|
|
coordinator = data[DATA_COORDINATOR]
|
2017-08-07 12:58:31 +00:00
|
|
|
|
2022-01-19 12:29:24 +00:00
|
|
|
entities: list[NukiDeviceEntity] = [
|
|
|
|
NukiLockEntity(coordinator, lock) for lock in data[DATA_LOCKS]
|
|
|
|
]
|
2021-04-06 19:20:57 +00:00
|
|
|
entities.extend(
|
|
|
|
[NukiOpenerEntity(coordinator, opener) for opener in data[DATA_OPENERS]]
|
|
|
|
)
|
2020-12-11 17:02:23 +00:00
|
|
|
async_add_entities(entities)
|
2017-08-07 12:58:31 +00:00
|
|
|
|
2021-05-03 16:34:28 +00:00
|
|
|
platform = entity_platform.async_get_current_platform()
|
2020-12-11 17:02:23 +00:00
|
|
|
platform.async_register_entity_service(
|
|
|
|
"lock_n_go",
|
|
|
|
{
|
|
|
|
vol.Optional(ATTR_UNLATCH, default=False): cv.boolean,
|
|
|
|
},
|
|
|
|
"lock_n_go",
|
2021-07-07 15:25:52 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
platform.async_register_entity_service(
|
2021-06-30 06:43:02 +00:00
|
|
|
"set_continuous_mode",
|
|
|
|
{
|
|
|
|
vol.Required(ATTR_ENABLE): cv.boolean,
|
|
|
|
},
|
|
|
|
"set_continuous_mode",
|
2020-12-11 17:02:23 +00:00
|
|
|
)
|
2019-09-17 09:44:43 +00:00
|
|
|
|
2017-02-02 14:15:27 +00:00
|
|
|
|
2021-04-06 19:20:57 +00:00
|
|
|
class NukiDeviceEntity(NukiEntity, LockEntity, ABC):
|
2020-05-20 12:44:57 +00:00
|
|
|
"""Representation of a Nuki device."""
|
2017-02-02 14:15:27 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the lock."""
|
2020-05-20 12:44:57 +00:00
|
|
|
return self._nuki_device.name
|
2020-02-19 16:16:02 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def unique_id(self) -> str:
|
|
|
|
"""Return a unique ID."""
|
2020-05-20 12:44:57 +00:00
|
|
|
return self._nuki_device.nuki_id
|
2017-02-02 14:15:27 +00:00
|
|
|
|
|
|
|
@property
|
2020-05-20 12:44:57 +00:00
|
|
|
@abstractmethod
|
2017-02-02 14:15:27 +00:00
|
|
|
def is_locked(self):
|
|
|
|
"""Return true if lock is locked."""
|
|
|
|
|
2017-08-07 12:58:31 +00:00
|
|
|
@property
|
2021-03-11 19:11:25 +00:00
|
|
|
def extra_state_attributes(self):
|
2017-08-07 12:58:31 +00:00
|
|
|
"""Return the device specific state attributes."""
|
|
|
|
data = {
|
2020-05-20 12:44:57 +00:00
|
|
|
ATTR_BATTERY_CRITICAL: self._nuki_device.battery_critical,
|
|
|
|
ATTR_NUKI_ID: self._nuki_device.nuki_id,
|
2019-07-31 19:25:30 +00:00
|
|
|
}
|
2017-08-07 12:58:31 +00:00
|
|
|
return data
|
|
|
|
|
2019-07-16 15:06:47 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Flag supported features."""
|
|
|
|
return SUPPORT_OPEN
|
|
|
|
|
|
|
|
@property
|
|
|
|
def available(self) -> bool:
|
|
|
|
"""Return True if entity is available."""
|
2021-04-06 19:20:57 +00:00
|
|
|
return super().available and self._nuki_device.state not in ERROR_STATES
|
2019-08-12 03:48:56 +00:00
|
|
|
|
2020-05-20 12:44:57 +00:00
|
|
|
@abstractmethod
|
|
|
|
def lock(self, **kwargs):
|
|
|
|
"""Lock the device."""
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def unlock(self, **kwargs):
|
|
|
|
"""Unlock the device."""
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def open(self, **kwargs):
|
|
|
|
"""Open the door latch."""
|
|
|
|
|
|
|
|
|
|
|
|
class NukiLockEntity(NukiDeviceEntity):
|
|
|
|
"""Representation of a Nuki lock."""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_locked(self):
|
|
|
|
"""Return true if lock is locked."""
|
|
|
|
return self._nuki_device.is_locked
|
|
|
|
|
2017-02-02 14:15:27 +00:00
|
|
|
def lock(self, **kwargs):
|
|
|
|
"""Lock the device."""
|
2020-05-20 12:44:57 +00:00
|
|
|
self._nuki_device.lock()
|
2017-02-02 14:15:27 +00:00
|
|
|
|
|
|
|
def unlock(self, **kwargs):
|
|
|
|
"""Unlock the device."""
|
2020-05-20 12:44:57 +00:00
|
|
|
self._nuki_device.unlock()
|
2017-08-07 12:58:31 +00:00
|
|
|
|
2019-07-16 15:06:47 +00:00
|
|
|
def open(self, **kwargs):
|
|
|
|
"""Open the door latch."""
|
2020-05-20 12:44:57 +00:00
|
|
|
self._nuki_device.unlatch()
|
2019-07-16 15:06:47 +00:00
|
|
|
|
2020-12-11 17:02:23 +00:00
|
|
|
def lock_n_go(self, unlatch):
|
2017-08-07 12:58:31 +00:00
|
|
|
"""Lock and go.
|
|
|
|
|
|
|
|
This will first unlock the door, then wait for 20 seconds (or another
|
|
|
|
amount of time depending on the lock settings) and relock.
|
|
|
|
"""
|
2020-12-11 17:02:23 +00:00
|
|
|
self._nuki_device.lock_n_go(unlatch)
|
2020-05-20 12:44:57 +00:00
|
|
|
|
|
|
|
|
|
|
|
class NukiOpenerEntity(NukiDeviceEntity):
|
|
|
|
"""Representation of a Nuki opener."""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_locked(self):
|
2021-05-25 19:05:56 +00:00
|
|
|
"""Return true if either ring-to-open or continuous mode is enabled."""
|
|
|
|
return not (
|
|
|
|
self._nuki_device.is_rto_activated
|
|
|
|
or self._nuki_device.mode == MODE_OPENER_CONTINUOUS
|
|
|
|
)
|
2020-05-20 12:44:57 +00:00
|
|
|
|
|
|
|
def lock(self, **kwargs):
|
|
|
|
"""Disable ring-to-open."""
|
|
|
|
self._nuki_device.deactivate_rto()
|
|
|
|
|
|
|
|
def unlock(self, **kwargs):
|
|
|
|
"""Enable ring-to-open."""
|
|
|
|
self._nuki_device.activate_rto()
|
|
|
|
|
|
|
|
def open(self, **kwargs):
|
|
|
|
"""Buzz open the door."""
|
|
|
|
self._nuki_device.electric_strike_actuation()
|
2020-12-11 17:02:23 +00:00
|
|
|
|
|
|
|
def lock_n_go(self, unlatch):
|
|
|
|
"""Stub service."""
|
2021-06-30 06:43:02 +00:00
|
|
|
|
|
|
|
def set_continuous_mode(self, enable):
|
|
|
|
"""Continuous Mode.
|
|
|
|
|
|
|
|
This feature will cause the door to automatically open when anyone
|
|
|
|
rings the bell. This is similar to ring-to-open, except that it does
|
|
|
|
not automatically deactivate
|
|
|
|
"""
|
|
|
|
if enable:
|
|
|
|
self._nuki_device.activate_continuous_mode()
|
|
|
|
else:
|
|
|
|
self._nuki_device.deactivate_continuous_mode()
|