lock.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  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_multi_dp = dps_map.pop("unlock_multi", None)
  41. self._req_unlock_dp = dps_map.pop("request_unlock", None)
  42. self._approve_unlock_dp = dps_map.pop("approve_unlock", None)
  43. self._req_intercom_dp = dps_map.pop("request_intercom", None)
  44. self._approve_intercom_dp = dps_map.pop("approve_intercom", None)
  45. self._jam_dp = dps_map.pop("jammed", None)
  46. self._init_end(dps_map)
  47. @property
  48. def is_locked(self):
  49. """Return the a boolean representing whether the lock is locked."""
  50. lock = None
  51. if self._lock_dp:
  52. lock = self._lock_dp.get_value(self._device)
  53. else:
  54. for d in (
  55. self._unlock_card_dp,
  56. self._unlock_dynpw_dp,
  57. self._unlock_fp_dp,
  58. self._unlock_offlinepw_dp,
  59. self._unlock_pw_dp,
  60. self._unlock_tmppw_dp,
  61. self._unlock_app_dp,
  62. self._unlock_key_dp,
  63. self._unlock_ble_dp,
  64. self._unlock_voice_dp,
  65. self._unlock_multi_dp,
  66. ):
  67. if d:
  68. if d.get_value(self._device):
  69. lock = False
  70. elif lock is None:
  71. lock = True
  72. return lock
  73. @property
  74. def is_jammed(self):
  75. if self._jam_dp:
  76. return self._jam_dp.get_value(self._device)
  77. def unlocker_id(self, dp, type):
  78. if dp:
  79. unlock = dp.get_value(self._device)
  80. if unlock:
  81. if unlock is True:
  82. return f"{type}"
  83. else:
  84. return f"{type} #{unlock}"
  85. @property
  86. def changed_by(self):
  87. for dp, desc in {
  88. self._unlock_app_dp: "App",
  89. self._unlock_ble_dp: "Bluetooth",
  90. self._unlock_card_dp: "Card",
  91. self._unlock_dynpw_dp: "Dynamic Password",
  92. self._unlock_fp_dp: "Finger",
  93. self._unlock_key_dp: "Key",
  94. self._unlock_offlinepw_dp: "Offline Password",
  95. self._unlock_pw_dp: "Password",
  96. self._unlock_tmppw_dp: "Temporary Password",
  97. self._unlock_voice_dp: "Voice",
  98. self._unlock_multi_dp: "Multifactor",
  99. }.items():
  100. by = self.unlocker_id(dp, desc)
  101. if by:
  102. # clear non-persistent dps immediately on reporting, instead
  103. # of waiting for the next poll, to make the lock more responsive
  104. # to multiple attempts
  105. if not dp.persist:
  106. self._device._cached_state.pop(dp.id, None)
  107. return by
  108. async def async_lock(self, **kwargs):
  109. """Lock the lock."""
  110. if self._lock_dp:
  111. await self._lock_dp.async_set_value(self._device, True)
  112. else:
  113. raise NotImplementedError()
  114. async def async_unlock(self, **kwargs):
  115. """Unlock the lock."""
  116. if self._lock_dp:
  117. await self._lock_dp.async_set_value(self._device, False)
  118. elif self._approve_unlock_dp:
  119. if self._req_unlock_dp and not self._req_unlock_dp.get_value(self._device):
  120. raise TimeoutError()
  121. await self._approve_unlock_dp.async_set_value(self._device, True)
  122. elif self._approve_intercom_dp:
  123. if self._req_intercom_dp and not self._req_intercom_dp.get_value(
  124. self._device
  125. ):
  126. raise TimeoutError()
  127. await self._approve_intercom_dp.async_set_value(self._device, True)
  128. else:
  129. raise NotImplementedError()