2019-02-14 15:01:46 +00:00
|
|
|
"""Support for HomeKit Controller locks."""
|
2020-02-24 09:55:33 +00:00
|
|
|
from aiohomekit.model.characteristics import CharacteristicsTypes
|
2020-11-16 23:11:39 +00:00
|
|
|
from aiohomekit.model.services import ServicesTypes
|
2019-08-14 16:14:15 +00:00
|
|
|
|
2021-07-21 04:55:04 +00:00
|
|
|
from homeassistant.components.lock import STATE_JAMMED, LockEntity
|
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_BATTERY_LEVEL,
|
|
|
|
STATE_LOCKED,
|
|
|
|
STATE_UNKNOWN,
|
|
|
|
STATE_UNLOCKED,
|
|
|
|
)
|
2020-01-29 21:59:45 +00:00
|
|
|
from homeassistant.core import callback
|
2019-01-12 02:48:28 +00:00
|
|
|
|
2019-03-26 06:49:51 +00:00
|
|
|
from . import KNOWN_DEVICES, HomeKitEntity
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2021-07-21 04:55:04 +00:00
|
|
|
CURRENT_STATE_MAP = {
|
|
|
|
0: STATE_UNLOCKED,
|
|
|
|
1: STATE_LOCKED,
|
|
|
|
2: STATE_JAMMED,
|
|
|
|
3: STATE_UNKNOWN,
|
|
|
|
}
|
2019-01-12 02:48:28 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
TARGET_STATE_MAP = {STATE_UNLOCKED: 0, STATE_LOCKED: 1}
|
2019-01-12 02:48:28 +00:00
|
|
|
|
2021-07-21 04:55:04 +00:00
|
|
|
REVERSED_TARGET_STATE_MAP = {v: k for k, v in TARGET_STATE_MAP.items()}
|
|
|
|
|
2019-01-12 02:48:28 +00:00
|
|
|
|
2019-05-13 06:56:05 +00:00
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
|
|
|
"""Set up Homekit lock."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hkid = config_entry.data["AccessoryPairingID"]
|
2019-05-13 06:56:05 +00:00
|
|
|
conn = hass.data[KNOWN_DEVICES][hkid]
|
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
@callback
|
2020-11-16 23:11:39 +00:00
|
|
|
def async_add_service(service):
|
|
|
|
if service.short_type != ServicesTypes.LOCK_MECHANISM:
|
2019-05-13 06:56:05 +00:00
|
|
|
return False
|
2020-11-16 23:11:39 +00:00
|
|
|
info = {"aid": service.accessory.aid, "iid": service.iid}
|
2019-05-13 06:56:05 +00:00
|
|
|
async_add_entities([HomeKitLock(conn, info)], True)
|
|
|
|
return True
|
|
|
|
|
|
|
|
conn.add_listener(async_add_service)
|
2019-01-12 02:48:28 +00:00
|
|
|
|
|
|
|
|
2020-04-25 16:02:41 +00:00
|
|
|
class HomeKitLock(HomeKitEntity, LockEntity):
|
2019-01-12 02:48:28 +00:00
|
|
|
"""Representation of a HomeKit Controller Lock."""
|
|
|
|
|
2019-01-28 16:21:20 +00:00
|
|
|
def get_characteristic_types(self):
|
|
|
|
"""Define the homekit characteristics the entity cares about."""
|
|
|
|
return [
|
|
|
|
CharacteristicsTypes.LOCK_MECHANISM_CURRENT_STATE,
|
|
|
|
CharacteristicsTypes.LOCK_MECHANISM_TARGET_STATE,
|
|
|
|
CharacteristicsTypes.BATTERY_LEVEL,
|
|
|
|
]
|
|
|
|
|
2019-01-12 02:48:28 +00:00
|
|
|
@property
|
|
|
|
def is_locked(self):
|
|
|
|
"""Return true if device is locked."""
|
2020-03-11 11:40:47 +00:00
|
|
|
value = self.service.value(CharacteristicsTypes.LOCK_MECHANISM_CURRENT_STATE)
|
2021-07-21 04:55:04 +00:00
|
|
|
if CURRENT_STATE_MAP[value] == STATE_UNKNOWN:
|
|
|
|
return None
|
2020-03-11 11:40:47 +00:00
|
|
|
return CURRENT_STATE_MAP[value] == STATE_LOCKED
|
2019-01-12 02:48:28 +00:00
|
|
|
|
2021-07-21 04:55:04 +00:00
|
|
|
@property
|
|
|
|
def is_locking(self):
|
|
|
|
"""Return true if device is locking."""
|
|
|
|
current_value = self.service.value(
|
|
|
|
CharacteristicsTypes.LOCK_MECHANISM_CURRENT_STATE
|
|
|
|
)
|
|
|
|
target_value = self.service.value(
|
|
|
|
CharacteristicsTypes.LOCK_MECHANISM_TARGET_STATE
|
|
|
|
)
|
|
|
|
return (
|
|
|
|
CURRENT_STATE_MAP[current_value] == STATE_UNLOCKED
|
|
|
|
and REVERSED_TARGET_STATE_MAP.get(target_value) == STATE_LOCKED
|
|
|
|
)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_unlocking(self):
|
|
|
|
"""Return true if device is unlocking."""
|
|
|
|
current_value = self.service.value(
|
|
|
|
CharacteristicsTypes.LOCK_MECHANISM_CURRENT_STATE
|
|
|
|
)
|
|
|
|
target_value = self.service.value(
|
|
|
|
CharacteristicsTypes.LOCK_MECHANISM_TARGET_STATE
|
|
|
|
)
|
|
|
|
return (
|
|
|
|
CURRENT_STATE_MAP[current_value] == STATE_LOCKED
|
|
|
|
and REVERSED_TARGET_STATE_MAP.get(target_value) == STATE_UNLOCKED
|
|
|
|
)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_jammed(self):
|
|
|
|
"""Return true if device is jammed."""
|
|
|
|
value = self.service.value(CharacteristicsTypes.LOCK_MECHANISM_CURRENT_STATE)
|
|
|
|
return CURRENT_STATE_MAP[value] == STATE_JAMMED
|
|
|
|
|
2019-03-11 18:59:41 +00:00
|
|
|
async def async_lock(self, **kwargs):
|
2019-01-12 02:48:28 +00:00
|
|
|
"""Lock the device."""
|
2019-03-11 18:59:41 +00:00
|
|
|
await self._set_lock_state(STATE_LOCKED)
|
2019-01-12 02:48:28 +00:00
|
|
|
|
2019-03-11 18:59:41 +00:00
|
|
|
async def async_unlock(self, **kwargs):
|
2019-01-12 02:48:28 +00:00
|
|
|
"""Unlock the device."""
|
2019-03-11 18:59:41 +00:00
|
|
|
await self._set_lock_state(STATE_UNLOCKED)
|
2019-01-12 02:48:28 +00:00
|
|
|
|
2019-03-11 18:59:41 +00:00
|
|
|
async def _set_lock_state(self, state):
|
2019-01-12 02:48:28 +00:00
|
|
|
"""Send state command."""
|
2020-03-11 16:27:20 +00:00
|
|
|
await self.async_put_characteristics(
|
|
|
|
{CharacteristicsTypes.LOCK_MECHANISM_TARGET_STATE: TARGET_STATE_MAP[state]}
|
|
|
|
)
|
2019-01-12 02:48:28 +00:00
|
|
|
|
|
|
|
@property
|
2021-03-11 15:57:47 +00:00
|
|
|
def extra_state_attributes(self):
|
2019-01-12 02:48:28 +00:00
|
|
|
"""Return the optional state attributes."""
|
2020-03-11 11:40:47 +00:00
|
|
|
attributes = {}
|
|
|
|
|
|
|
|
battery_level = self.service.value(CharacteristicsTypes.BATTERY_LEVEL)
|
|
|
|
if battery_level:
|
|
|
|
attributes[ATTR_BATTERY_LEVEL] = battery_level
|
2019-01-12 02:48:28 +00:00
|
|
|
|
2020-03-11 11:40:47 +00:00
|
|
|
return attributes
|