lock.py 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. """
  2. Setup for different kinds of Tuya lock devices
  3. """
  4. from homeassistant.components.lock import LockEntity
  5. from .device import TuyaLocalDevice
  6. from .helpers.config import async_tuya_setup_platform
  7. from .helpers.device_config import TuyaEntityConfig
  8. from .helpers.mixin import TuyaLocalEntity
  9. async def async_setup_entry(hass, config_entry, async_add_entities):
  10. config = {**config_entry.data, **config_entry.options}
  11. await async_tuya_setup_platform(
  12. hass,
  13. async_add_entities,
  14. config,
  15. "lock",
  16. TuyaLocalLock,
  17. )
  18. class TuyaLocalLock(TuyaLocalEntity, LockEntity):
  19. """Representation of a Tuya Wi-Fi connected lock."""
  20. def __init__(self, device: TuyaLocalDevice, config: TuyaEntityConfig):
  21. """
  22. Initialise the lock.
  23. Args:
  24. device (TuyaLocalDevice): The device API instance.
  25. config (TuyaEntityConfig): The configuration for this entity.
  26. """
  27. super().__init__()
  28. dps_map = self._init_begin(device, config)
  29. self._lock_dp = dps_map.pop("lock", None)
  30. self._unlock_fp_dp = dps_map.pop("unlock_fingerprint", None)
  31. self._unlock_pw_dp = dps_map.pop("unlock_password", None)
  32. self._unlock_tmppw_dp = dps_map.pop("unlock_temp_pwd", None)
  33. self._unlock_dynpw_dp = dps_map.pop("unlock_dynamic_pwd", None)
  34. self._unlock_offlinepw_dp = dps_map.pop("unlock_offline_pwd", None)
  35. self._unlock_card_dp = dps_map.pop("unlock_card", None)
  36. self._unlock_app_dp = dps_map.pop("unlock_app", None)
  37. self._unlock_key_dp = dps_map.pop("unlock_key", None)
  38. self._unlock_ble_dp = dps_map.pop("unlock_ble", None)
  39. self._unlock_voice_dp = dps_map.pop("unlock_voice", None)
  40. self._unlock_face_dp = dps_map.pop("unlock_face", None)
  41. self._unlock_multi_dp = dps_map.pop("unlock_multi", None)
  42. self._req_unlock_dp = dps_map.pop("request_unlock", None)
  43. self._approve_unlock_dp = dps_map.pop("approve_unlock", None)
  44. self._req_intercom_dp = dps_map.pop("request_intercom", None)
  45. self._approve_intercom_dp = dps_map.pop("approve_intercom", None)
  46. self._jam_dp = dps_map.pop("jammed", None)
  47. self._init_end(dps_map)
  48. @property
  49. def is_locked(self):
  50. """Return the a boolean representing whether the lock is locked."""
  51. lock = None
  52. if self._lock_dp:
  53. lock = self._lock_dp.get_value(self._device)
  54. else:
  55. for d in (
  56. self._unlock_card_dp,
  57. self._unlock_dynpw_dp,
  58. self._unlock_fp_dp,
  59. self._unlock_offlinepw_dp,
  60. self._unlock_pw_dp,
  61. self._unlock_tmppw_dp,
  62. self._unlock_app_dp,
  63. self._unlock_key_dp,
  64. self._unlock_ble_dp,
  65. self._unlock_voice_dp,
  66. self._unlock_face_dp,
  67. self._unlock_multi_dp,
  68. ):
  69. if d:
  70. if d.get_value(self._device):
  71. lock = False
  72. elif lock is None:
  73. lock = True
  74. return lock
  75. @property
  76. def is_jammed(self):
  77. if self._jam_dp:
  78. return self._jam_dp.get_value(self._device)
  79. def unlocker_id(self, dp, type):
  80. if dp:
  81. unlock = dp.get_value(self._device)
  82. if unlock:
  83. if unlock is True:
  84. return f"{type}"
  85. else:
  86. return f"{type} #{unlock}"
  87. @property
  88. def changed_by(self):
  89. for dp, desc in {
  90. self._unlock_app_dp: "App",
  91. self._unlock_ble_dp: "Bluetooth",
  92. self._unlock_card_dp: "Card",
  93. self._unlock_dynpw_dp: "Dynamic Password",
  94. self._unlock_fp_dp: "Finger",
  95. self._unlock_key_dp: "Key",
  96. self._unlock_offlinepw_dp: "Offline Password",
  97. self._unlock_pw_dp: "Password",
  98. self._unlock_tmppw_dp: "Temporary Password",
  99. self._unlock_voice_dp: "Voice",
  100. self._unlock_face_dp: "Face",
  101. self._unlock_multi_dp: "Multifactor",
  102. }.items():
  103. by = self.unlocker_id(dp, desc)
  104. if by:
  105. # clear non-persistent dps immediately on reporting, instead
  106. # of waiting for the next poll, to make the lock more responsive
  107. # to multiple attempts
  108. if not dp.persist:
  109. self._device._cached_state.pop(dp.id, None)
  110. return by
  111. async def async_lock(self, **kwargs):
  112. """Lock the lock."""
  113. if self._lock_dp:
  114. await self._lock_dp.async_set_value(self._device, True)
  115. else:
  116. raise NotImplementedError()
  117. async def async_unlock(self, **kwargs):
  118. """Unlock the lock."""
  119. if self._lock_dp:
  120. await self._lock_dp.async_set_value(self._device, False)
  121. elif self._approve_unlock_dp:
  122. if self._req_unlock_dp and not self._req_unlock_dp.get_value(self._device):
  123. raise TimeoutError()
  124. await self._approve_unlock_dp.async_set_value(self._device, True)
  125. elif self._approve_intercom_dp:
  126. if self._req_intercom_dp and not self._req_intercom_dp.get_value(
  127. self._device
  128. ):
  129. raise TimeoutError()
  130. await self._approve_intercom_dp.async_set_value(self._device, True)
  131. else:
  132. raise NotImplementedError()