2022-01-25 17:31:17 +00:00
|
|
|
"""Support for locks on Ubiquiti's UniFi Protect NVR."""
|
2024-03-08 15:35:23 +00:00
|
|
|
|
2022-01-25 17:31:17 +00:00
|
|
|
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
|
|
|
|
2024-06-09 23:25:39 +00:00
|
|
|
from uiprotect.data import (
|
2022-06-27 21:03:25 +00:00
|
|
|
Doorlock,
|
|
|
|
LockStatusType,
|
2022-08-26 11:46:11 +00:00
|
|
|
ModelType,
|
2022-06-27 21:03:25 +00:00
|
|
|
ProtectAdoptableDeviceModel,
|
|
|
|
)
|
2022-01-25 17:31:17 +00:00
|
|
|
|
|
|
|
from homeassistant.components.lock import LockEntity, LockEntityDescription
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
|
2024-07-05 07:31:31 +00:00
|
|
|
from .data import ProtectDeviceType, UFPConfigEntry
|
2022-01-25 17:31:17 +00:00
|
|
|
from .entity import ProtectDeviceEntity
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
2024-06-12 15:49:18 +00:00
|
|
|
entry: UFPConfigEntry,
|
2022-01-25 17:31:17 +00:00
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
|
|
|
"""Set up locks on a UniFi Protect NVR."""
|
2024-06-12 15:49:18 +00:00
|
|
|
data = entry.runtime_data
|
2022-01-25 17:31:17 +00:00
|
|
|
|
2024-02-21 05:47:31 +00:00
|
|
|
@callback
|
|
|
|
def _add_new_device(device: ProtectAdoptableDeviceModel) -> None:
|
2022-06-27 21:03:25 +00:00
|
|
|
if isinstance(device, Doorlock):
|
|
|
|
async_add_entities([ProtectLock(data, device)])
|
|
|
|
|
2024-06-13 16:44:29 +00:00
|
|
|
data.async_subscribe_adopt(_add_new_device)
|
2022-06-27 21:03:25 +00:00
|
|
|
|
2024-06-12 20:23:18 +00:00
|
|
|
async_add_entities(
|
2024-06-16 14:00:14 +00:00
|
|
|
ProtectLock(
|
|
|
|
data, cast(Doorlock, device), LockEntityDescription(key="lock", name="Lock")
|
|
|
|
)
|
2024-06-12 20:23:18 +00:00
|
|
|
for device in data.get_by_types({ModelType.DOORLOCK})
|
|
|
|
)
|
2022-01-25 17:31:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ProtectLock(ProtectDeviceEntity, LockEntity):
|
|
|
|
"""A Ubiquiti UniFi Protect Speaker."""
|
|
|
|
|
|
|
|
device: Doorlock
|
|
|
|
entity_description: LockEntityDescription
|
2024-06-14 19:29:18 +00:00
|
|
|
_state_attrs = (
|
|
|
|
"_attr_available",
|
|
|
|
"_attr_is_locked",
|
|
|
|
"_attr_is_locking",
|
|
|
|
"_attr_is_unlocking",
|
|
|
|
"_attr_is_jammed",
|
|
|
|
)
|
2022-01-25 17:31:17 +00:00
|
|
|
|
|
|
|
@callback
|
2024-07-05 07:31:31 +00:00
|
|
|
def _async_update_device_from_protect(self, device: ProtectDeviceType) -> None:
|
2022-06-21 03:52:41 +00:00
|
|
|
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
|
2024-01-05 12:27:10 +00:00
|
|
|
if lock_status is LockStatusType.CLOSED:
|
2022-01-25 17:31:17 +00:00
|
|
|
self._attr_is_locked = True
|
2024-01-05 12:27:10 +00:00
|
|
|
elif lock_status is LockStatusType.CLOSING:
|
2022-01-25 17:31:17 +00:00
|
|
|
self._attr_is_locking = True
|
2024-01-05 12:27:10 +00:00
|
|
|
elif lock_status is 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()
|