device.py 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. """Library to handle connection with Switchbot."""
  2. from __future__ import annotations
  3. import asyncio
  4. import binascii
  5. import logging
  6. from ctypes import cast
  7. from typing import Any, Callable, TypeVar
  8. from uuid import UUID
  9. import async_timeout
  10. import bleak
  11. from bleak import BleakError
  12. from bleak.backends.device import BLEDevice
  13. from bleak.backends.service import BleakGATTCharacteristic, BleakGATTServiceCollection
  14. from bleak_retry_connector import (
  15. BleakClientWithServiceCache,
  16. ble_device_has_changed,
  17. establish_connection,
  18. )
  19. from ..const import DEFAULT_RETRY_COUNT, DEFAULT_SCAN_TIMEOUT
  20. from ..discovery import GetSwitchbotDevices
  21. from ..models import SwitchBotAdvertisement
  22. _LOGGER = logging.getLogger(__name__)
  23. # Keys common to all device types
  24. DEVICE_GET_BASIC_SETTINGS_KEY = "5702"
  25. DEVICE_SET_MODE_KEY = "5703"
  26. DEVICE_SET_EXTENDED_KEY = "570f"
  27. # Base key when encryption is set
  28. KEY_PASSWORD_PREFIX = "571"
  29. BLEAK_EXCEPTIONS = (AttributeError, BleakError, asyncio.exceptions.TimeoutError)
  30. # How long to hold the connection
  31. # to wait for additional commands for
  32. # disconnecting the device.
  33. DISCONNECT_DELAY = 59
  34. def _sb_uuid(comms_type: str = "service") -> UUID | str:
  35. """Return Switchbot UUID."""
  36. _uuid = {"tx": "002", "rx": "003", "service": "d00"}
  37. if comms_type in _uuid:
  38. return UUID(f"cba20{_uuid[comms_type]}-224d-11e6-9fb8-0002a5d5c51b")
  39. return "Incorrect type, choose between: tx, rx or service"
  40. class SwitchbotDevice:
  41. """Base Representation of a Switchbot Device."""
  42. def __init__(
  43. self,
  44. device: BLEDevice,
  45. password: str | None = None,
  46. interface: int = 0,
  47. **kwargs: Any,
  48. ) -> None:
  49. """Switchbot base class constructor."""
  50. self._interface = f"hci{interface}"
  51. self._device = device
  52. self._sb_adv_data: SwitchBotAdvertisement | None = None
  53. self._scan_timeout: int = kwargs.pop("scan_timeout", DEFAULT_SCAN_TIMEOUT)
  54. self._retry_count: int = kwargs.pop("retry_count", DEFAULT_RETRY_COUNT)
  55. self._connect_lock = asyncio.Lock()
  56. self._operation_lock = asyncio.Lock()
  57. if password is None or password == "":
  58. self._password_encoded = None
  59. else:
  60. self._password_encoded = "%08x" % (
  61. binascii.crc32(password.encode("ascii")) & 0xFFFFFFFF
  62. )
  63. self._client: BleakClientWithServiceCache | None = None
  64. self._cached_services: BleakGATTServiceCollection | None = None
  65. self._read_char: BleakGATTCharacteristic | None = None
  66. self._write_char: BleakGATTCharacteristic | None = None
  67. self._disconnect_timer: asyncio.TimerHandle | None = None
  68. self.loop = asyncio.get_event_loop()
  69. def _commandkey(self, key: str) -> str:
  70. """Add password to key if set."""
  71. if self._password_encoded is None:
  72. return key
  73. key_action = key[3]
  74. key_suffix = key[4:]
  75. return KEY_PASSWORD_PREFIX + key_action + self._password_encoded + key_suffix
  76. async def _sendcommand(self, key: str, retry: int) -> bytes:
  77. """Send command to device and read response."""
  78. command = bytearray.fromhex(self._commandkey(key))
  79. _LOGGER.debug("Sending command to switchbot %s", command)
  80. if self._operation_lock.locked():
  81. _LOGGER.debug(
  82. "%s: Operation already in progress, waiting for it to complete.",
  83. self.name,
  84. )
  85. max_attempts = retry + 1
  86. async with self._operation_lock:
  87. for attempt in range(max_attempts):
  88. try:
  89. return await self._send_command_locked(key, command)
  90. except BLEAK_EXCEPTIONS(
  91. bleak.BleakError, asyncio.exceptions.TimeoutError
  92. ):
  93. if attempt == retry:
  94. _LOGGER.error(
  95. "Switchbot communication failed. Stopping trying",
  96. exc_info=True,
  97. )
  98. return b"\x00"
  99. _LOGGER.debug("Switchbot communication failed with:", exc_info=True)
  100. raise RuntimeError("Unreachable")
  101. @property
  102. def name(self) -> str:
  103. """Return device name."""
  104. return f"{self._device.name} ({self._device.address})"
  105. async def _ensure_connected(self):
  106. """Ensure connection to device is established."""
  107. if self._connect_lock.locked():
  108. _LOGGER.debug(
  109. "%s: Connection already in progress, waiting for it to complete.",
  110. self.name,
  111. )
  112. if self._client and self._client.is_connected:
  113. self._reset_disconnect_timer()
  114. return
  115. async with self._connect_lock:
  116. # Check again while holding the lock
  117. if self._client and self._client.is_connected:
  118. self._reset_disconnect_timer()
  119. return
  120. client = await establish_connection(
  121. BleakClientWithServiceCache,
  122. self._device,
  123. self.name,
  124. cached_services=self._cached_services,
  125. )
  126. self._cached_services = client.services
  127. _LOGGER.debug("%s: Connected to SwitchBot Device", self.name)
  128. services = client.services
  129. self._read_char = services.get_characteristic(_sb_uuid(comms_type="rx"))
  130. self._write_char = services.get_characteristic(_sb_uuid(comms_type="tx"))
  131. self._client = client
  132. self._reset_disconnect_timer()
  133. def _reset_disconnect_timer(self):
  134. """Reset disconnect timer."""
  135. if self._disconnect_timer:
  136. self._disconnect_timer.cancel()
  137. self._disconnect_timer = self.loop.call_later(
  138. DISCONNECT_DELAY, self._disconnect
  139. )
  140. def _disconnect(self):
  141. """Disconnect from device."""
  142. self._disconnect_timer = None
  143. asyncio.create_task(self._execute_disconnect())
  144. async def _execute_disconnect(self):
  145. """Execute disconnection."""
  146. _LOGGER.debug(
  147. "%s: Disconnecting from SwitchBot Device after timeout of %s",
  148. self.name,
  149. DISCONNECT_DELAY,
  150. )
  151. async with self._connect_lock:
  152. if not self._client or not self._client.is_connected:
  153. return
  154. await self._client.disconnect()
  155. self._client = None
  156. self._read_char = None
  157. self._write_char = None
  158. async def _send_command_locked(self, key: str, command: bytes) -> bytes:
  159. """Send command to device and read response."""
  160. client: BleakClientWithServiceCache | None = None
  161. await self._ensure_connected()
  162. client = self._client
  163. future: asyncio.Future[bytearray] = asyncio.Future()
  164. def _notification_handler(_sender: int, data: bytearray) -> None:
  165. """Handle notification responses."""
  166. if future.done():
  167. _LOGGER.debug("%s: Notification handler already done", self.name)
  168. return
  169. future.set_result(data)
  170. _LOGGER.debug("%s: Subscribe to notifications", self.name)
  171. await client.start_notify(self._read_char, _notification_handler)
  172. _LOGGER.debug("%s: Sending command, %s", self.name, key)
  173. await client.write_gatt_char(self._write_char, command, False)
  174. async with async_timeout.timeout(5):
  175. notify_msg = await future
  176. _LOGGER.info("%s: Notification received: %s", self.name, notify_msg)
  177. _LOGGER.debug("%s: UnSubscribe to notifications", self.name)
  178. await client.stop_notify(self._read_char)
  179. if notify_msg == b"\x07":
  180. _LOGGER.error("Password required")
  181. elif notify_msg == b"\t":
  182. _LOGGER.error("Password incorrect")
  183. return notify_msg
  184. def get_address(self) -> str:
  185. """Return address of device."""
  186. return self._device.address
  187. def _get_adv_value(self, key: str) -> Any:
  188. """Return value from advertisement data."""
  189. if not self._sb_adv_data:
  190. return None
  191. return self._sb_adv_data.data["data"][key]
  192. def get_battery_percent(self) -> Any:
  193. """Return device battery level in percent."""
  194. return self._get_adv_value("battery")
  195. def update_from_advertisement(self, advertisement: SwitchBotAdvertisement) -> None:
  196. """Update device data from advertisement."""
  197. self._sb_adv_data = advertisement
  198. if self._device and ble_device_has_changed(self._device, advertisement.device):
  199. self._cached_services = None
  200. self._device = advertisement.device
  201. async def get_device_data(
  202. self, retry: int = DEFAULT_RETRY_COUNT, interface: int | None = None
  203. ) -> dict | None:
  204. """Find switchbot devices and their advertisement data."""
  205. if interface:
  206. _interface: int = interface
  207. else:
  208. _interface = int(self._interface.replace("hci", ""))
  209. _data = await GetSwitchbotDevices(interface=_interface).discover(
  210. retry=retry, scan_timeout=self._scan_timeout
  211. )
  212. if self._device.address in _data:
  213. self._sb_adv_data = _data[self._device.address]
  214. return self._sb_adv_data
  215. async def _get_basic_info(self) -> dict | None:
  216. """Return basic info of device."""
  217. _data = await self._sendcommand(
  218. key=DEVICE_GET_BASIC_SETTINGS_KEY, retry=self._retry_count
  219. )
  220. if _data in (b"\x07", b"\x00"):
  221. _LOGGER.error("Unsuccessful, please try again")
  222. return None
  223. return _data