core/homeassistant/components/verisure/lock.py

124 lines
4.0 KiB
Python
Raw Normal View History

"""Support for Verisure locks."""
from __future__ import annotations
import asyncio
from typing import Any, Callable
from homeassistant.components.lock import LockEntity
from homeassistant.const import ATTR_CODE, STATE_LOCKED, STATE_UNLOCKED
from homeassistant.core import HomeAssistant
from homeassistant.helpers.update_coordinator import CoordinatorEntity
from .const import CONF_CODE_DIGITS, CONF_DEFAULT_LOCK_CODE, CONF_LOCKS, DOMAIN, LOGGER
from .coordinator import VerisureDataUpdateCoordinator
def setup_platform(
hass: HomeAssistant,
config: dict[str, Any],
add_entities: Callable[[list[VerisureDoorlock]], None],
discovery_info: dict[str, Any] | None = None,
) -> None:
"""Set up the Verisure lock platform."""
coordinator = hass.data[DOMAIN]
locks = []
if int(coordinator.config.get(CONF_LOCKS, 1)):
2019-07-31 19:25:30 +00:00
locks.extend(
[
VerisureDoorlock(coordinator, serial_number)
for serial_number in coordinator.data["locks"]
2019-07-31 19:25:30 +00:00
]
)
add_entities(locks)
class VerisureDoorlock(CoordinatorEntity, LockEntity):
2016-03-07 21:13:18 +00:00
"""Representation of a Verisure doorlock."""
coordinator: VerisureDataUpdateCoordinator
def __init__(
self, coordinator: VerisureDataUpdateCoordinator, serial_number: str
) -> None:
"""Initialize the Verisure lock."""
super().__init__(coordinator)
self.serial_number = serial_number
self._state = None
self._digits = coordinator.config.get(CONF_CODE_DIGITS)
self._default_lock_code = coordinator.config.get(CONF_DEFAULT_LOCK_CODE)
@property
def name(self) -> str:
2016-03-07 21:13:18 +00:00
"""Return the name of the lock."""
return self.coordinator.data["locks"][self.serial_number]["area"]
@property
def available(self) -> bool:
"""Return True if entity is available."""
2019-07-31 19:25:30 +00:00
return (
super().available and self.serial_number in self.coordinator.data["locks"]
2019-07-31 19:25:30 +00:00
)
@property
def changed_by(self) -> str | None:
"""Last change triggered by."""
return self.coordinator.data["locks"][self.serial_number].get("userString")
@property
def code_format(self) -> str:
2016-03-07 21:13:18 +00:00
"""Return the required six digit code."""
2019-07-31 19:25:30 +00:00
return "^\\d{%s}$" % self._digits
@property
def is_locked(self) -> bool:
2016-02-28 11:03:47 +00:00
"""Return true if lock is locked."""
return (
self.coordinator.data["locks"][self.serial_number]["lockedState"]
== "LOCKED"
)
async def async_unlock(self, **kwargs) -> None:
2016-02-28 11:03:47 +00:00
"""Send unlock command."""
code = kwargs.get(ATTR_CODE, self._default_lock_code)
if code is None:
2020-12-30 08:55:18 +00:00
LOGGER.error("Code required but none provided")
return
await self.async_set_lock_state(code, STATE_UNLOCKED)
async def async_lock(self, **kwargs) -> None:
2016-02-28 11:03:47 +00:00
"""Send lock command."""
code = kwargs.get(ATTR_CODE, self._default_lock_code)
if code is None:
2020-12-30 08:55:18 +00:00
LOGGER.error("Code required but none provided")
return
await self.async_set_lock_state(code, STATE_LOCKED)
async def async_set_lock_state(self, code: str, state: str) -> None:
"""Send set lock state command."""
target_state = "lock" if state == STATE_LOCKED else "unlock"
lock_state = await self.hass.async_add_executor_job(
self.coordinator.verisure.set_lock_state,
code,
self.serial_number,
target_state,
)
2020-12-30 08:55:18 +00:00
LOGGER.debug("Verisure doorlock %s", state)
transaction = {}
attempts = 0
2019-07-31 19:25:30 +00:00
while "result" not in transaction:
transaction = await self.hass.async_add_executor_job(
self.coordinator.verisure.get_lock_state_transaction,
lock_state["doorLockStateChangeTransactionId"],
)
attempts += 1
if attempts == 30:
break
if attempts > 1:
await asyncio.sleep(0.5)
2019-07-31 19:25:30 +00:00
if transaction["result"] == "OK":
self._state = state