2022-01-25 17:31:17 +00:00
|
|
|
"""Support for locks on Ubiquiti's UniFi Protect NVR."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
import logging
|
2022-08-26 11:46:11 +00:00
|
|
|
from typing import Any, cast
|
2022-01-25 17:31:17 +00:00
|
|
|
|
2022-06-27 21:03:25 +00:00
|
|
|
from pyunifiprotect.data import (
|
|
|
|
Doorlock,
|
|
|
|
LockStatusType,
|
2022-08-26 11:46:11 +00:00
|
|
|
ModelType,
|
2022-06-27 21:03:25 +00:00
|
|
|
ProtectAdoptableDeviceModel,
|
|
|
|
ProtectModelWithId,
|
|
|
|
)
|
2022-01-25 17:31:17 +00:00
|
|
|
|
|
|
|
from homeassistant.components.lock import LockEntity, LockEntityDescription
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
2022-06-27 21:03:25 +00:00
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
2022-01-25 17:31:17 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
|
2022-06-27 21:03:25 +00:00
|
|
|
from .const import DISPATCH_ADOPT, DOMAIN
|
2022-01-25 17:31:17 +00:00
|
|
|
from .data import ProtectData
|
|
|
|
from .entity import ProtectDeviceEntity
|
2022-06-27 21:03:25 +00:00
|
|
|
from .utils import async_dispatch_id as _ufpd
|
2022-01-25 17:31:17 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
entry: ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
|
|
|
"""Set up locks on a UniFi Protect NVR."""
|
|
|
|
data: ProtectData = hass.data[DOMAIN][entry.entry_id]
|
|
|
|
|
2022-06-27 21:03:25 +00:00
|
|
|
async def _add_new_device(device: ProtectAdoptableDeviceModel) -> None:
|
|
|
|
if isinstance(device, Doorlock):
|
|
|
|
async_add_entities([ProtectLock(data, device)])
|
|
|
|
|
2022-06-29 03:00:26 +00:00
|
|
|
entry.async_on_unload(
|
|
|
|
async_dispatcher_connect(hass, _ufpd(entry, DISPATCH_ADOPT), _add_new_device)
|
|
|
|
)
|
2022-06-27 21:03:25 +00:00
|
|
|
|
2022-06-22 20:57:21 +00:00
|
|
|
entities = []
|
2022-08-26 11:46:11 +00:00
|
|
|
for device in data.get_by_types({ModelType.DOORLOCK}):
|
|
|
|
device = cast(Doorlock, device)
|
2022-06-22 20:57:21 +00:00
|
|
|
entities.append(ProtectLock(data, device))
|
|
|
|
|
|
|
|
async_add_entities(entities)
|
2022-01-25 17:31:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ProtectLock(ProtectDeviceEntity, LockEntity):
|
|
|
|
"""A Ubiquiti UniFi Protect Speaker."""
|
|
|
|
|
|
|
|
device: Doorlock
|
|
|
|
entity_description: LockEntityDescription
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
data: ProtectData,
|
|
|
|
doorlock: Doorlock,
|
|
|
|
) -> None:
|
|
|
|
"""Initialize an UniFi lock."""
|
|
|
|
super().__init__(
|
|
|
|
data,
|
|
|
|
doorlock,
|
|
|
|
LockEntityDescription(key="lock"),
|
|
|
|
)
|
|
|
|
|
2022-06-22 20:57:21 +00:00
|
|
|
self._attr_name = f"{self.device.display_name} Lock"
|
2022-01-25 17:31:17 +00:00
|
|
|
|
|
|
|
@callback
|
2022-06-21 03:52:41 +00:00
|
|
|
def _async_update_device_from_protect(self, device: ProtectModelWithId) -> None:
|
|
|
|
super()._async_update_device_from_protect(device)
|
2023-07-16 16:24:27 +00:00
|
|
|
lock_status = self.device.lock_status
|
2022-01-25 17:31:17 +00:00
|
|
|
|
|
|
|
self._attr_is_locked = False
|
|
|
|
self._attr_is_locking = False
|
|
|
|
self._attr_is_unlocking = False
|
|
|
|
self._attr_is_jammed = False
|
2023-07-16 16:24:27 +00:00
|
|
|
if lock_status == LockStatusType.CLOSED:
|
2022-01-25 17:31:17 +00:00
|
|
|
self._attr_is_locked = True
|
2023-07-16 16:24:27 +00:00
|
|
|
elif lock_status == LockStatusType.CLOSING:
|
2022-01-25 17:31:17 +00:00
|
|
|
self._attr_is_locking = True
|
2023-07-16 16:24:27 +00:00
|
|
|
elif lock_status == LockStatusType.OPENING:
|
2022-01-25 17:31:17 +00:00
|
|
|
self._attr_is_unlocking = True
|
2023-07-16 16:24:27 +00:00
|
|
|
elif lock_status in (
|
2022-01-25 17:31:17 +00:00
|
|
|
LockStatusType.FAILED_WHILE_CLOSING,
|
|
|
|
LockStatusType.FAILED_WHILE_OPENING,
|
|
|
|
LockStatusType.JAMMED_WHILE_CLOSING,
|
|
|
|
LockStatusType.JAMMED_WHILE_OPENING,
|
|
|
|
):
|
|
|
|
self._attr_is_jammed = True
|
|
|
|
# lock is not fully initialized yet
|
2023-07-16 16:24:27 +00:00
|
|
|
elif lock_status != LockStatusType.OPEN:
|
2022-01-25 17:31:17 +00:00
|
|
|
self._attr_available = False
|
|
|
|
|
|
|
|
async def async_unlock(self, **kwargs: Any) -> None:
|
|
|
|
"""Unlock the lock."""
|
2022-06-22 20:57:21 +00:00
|
|
|
_LOGGER.debug("Unlocking %s", self.device.display_name)
|
2022-01-25 17:31:17 +00:00
|
|
|
return await self.device.open_lock()
|
|
|
|
|
|
|
|
async def async_lock(self, **kwargs: Any) -> None:
|
|
|
|
"""Lock the lock."""
|
2022-06-22 20:57:21 +00:00
|
|
|
_LOGGER.debug("Locking %s", self.device.display_name)
|
2022-01-25 17:31:17 +00:00
|
|
|
return await self.device.close_lock()
|