device.py 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  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("%s: Sending command %s", self.name, 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. if attempt == retry:
  92. _LOGGER.error(
  93. "%s: communication failed. Stopping trying",
  94. self.name,
  95. exc_info=True,
  96. )
  97. return b"\x00"
  98. _LOGGER.debug(
  99. "%s: communication failed with:", self.name, exc_info=True
  100. )
  101. raise RuntimeError("Unreachable")
  102. @property
  103. def name(self) -> str:
  104. """Return device name."""
  105. return f"{self._device.name} ({self._device.address})"
  106. async def _ensure_connected(self):
  107. """Ensure connection to device is established."""
  108. if self._connect_lock.locked():
  109. _LOGGER.debug(
  110. "%s: Connection already in progress, waiting for it to complete.",
  111. self.name,
  112. )
  113. if self._client and self._client.is_connected:
  114. self._reset_disconnect_timer()
  115. return
  116. async with self._connect_lock:
  117. # Check again while holding the lock
  118. if self._client and self._client.is_connected:
  119. self._reset_disconnect_timer()
  120. return
  121. client = await establish_connection(
  122. BleakClientWithServiceCache,
  123. self._device,
  124. self.name,
  125. cached_services=self._cached_services,
  126. )
  127. self._cached_services = client.services
  128. _LOGGER.debug("%s: Connected", self.name)
  129. services = client.services
  130. self._read_char = services.get_characteristic(_sb_uuid(comms_type="rx"))
  131. self._write_char = services.get_characteristic(_sb_uuid(comms_type="tx"))
  132. self._client = client
  133. self._reset_disconnect_timer()
  134. def _reset_disconnect_timer(self):
  135. """Reset disconnect timer."""
  136. if self._disconnect_timer:
  137. self._disconnect_timer.cancel()
  138. self._disconnect_timer = self.loop.call_later(
  139. DISCONNECT_DELAY, self._disconnect
  140. )
  141. def _disconnect(self):
  142. """Disconnect from device."""
  143. self._disconnect_timer = None
  144. asyncio.create_task(self._execute_disconnect())
  145. async def _execute_disconnect(self):
  146. """Execute disconnection."""
  147. _LOGGER.debug(
  148. "%s: Disconnecting after timeout of %s",
  149. self.name,
  150. DISCONNECT_DELAY,
  151. )
  152. async with self._connect_lock:
  153. if not self._client or not self._client.is_connected:
  154. return
  155. await self._client.disconnect()
  156. self._client = None
  157. self._read_char = None
  158. self._write_char = None
  159. async def _send_command_locked(self, key: str, command: bytes) -> bytes:
  160. """Send command to device and read response."""
  161. await self._ensure_connected()
  162. assert self._client is not None
  163. assert self._read_char is not None
  164. assert self._write_char is not None
  165. future: asyncio.Future[bytearray] = asyncio.Future()
  166. client = self._client
  167. def _notification_handler(_sender: int, data: bytearray) -> None:
  168. """Handle notification responses."""
  169. if future.done():
  170. _LOGGER.debug("%s: Notification handler already done", self.name)
  171. return
  172. future.set_result(data)
  173. _LOGGER.debug("%s: Subscribe to notifications", self.name)
  174. await client.start_notify(self._read_char, _notification_handler)
  175. _LOGGER.debug("%s: Sending command: %s", self.name, key)
  176. await client.write_gatt_char(self._write_char, command, False)
  177. async with async_timeout.timeout(5):
  178. notify_msg = await future
  179. _LOGGER.debug("%s: Notification received: %s", self.name, notify_msg)
  180. _LOGGER.debug("%s: UnSubscribe to notifications", self.name)
  181. await client.stop_notify(self._read_char)
  182. if notify_msg == b"\x07":
  183. _LOGGER.error("Password required")
  184. elif notify_msg == b"\t":
  185. _LOGGER.error("Password incorrect")
  186. return notify_msg
  187. def get_address(self) -> str:
  188. """Return address of device."""
  189. return self._device.address
  190. def _get_adv_value(self, key: str) -> Any:
  191. """Return value from advertisement data."""
  192. if not self._sb_adv_data:
  193. return None
  194. return self._sb_adv_data.data["data"][key]
  195. def get_battery_percent(self) -> Any:
  196. """Return device battery level in percent."""
  197. return self._get_adv_value("battery")
  198. def update_from_advertisement(self, advertisement: SwitchBotAdvertisement) -> None:
  199. """Update device data from advertisement."""
  200. self._sb_adv_data = advertisement
  201. if self._device and ble_device_has_changed(self._device, advertisement.device):
  202. self._cached_services = None
  203. self._device = advertisement.device
  204. async def get_device_data(
  205. self, retry: int = DEFAULT_RETRY_COUNT, interface: int | None = None
  206. ) -> SwitchBotAdvertisement | None:
  207. """Find switchbot devices and their advertisement data."""
  208. if interface:
  209. _interface: int = interface
  210. else:
  211. _interface = int(self._interface.replace("hci", ""))
  212. _data = await GetSwitchbotDevices(interface=_interface).discover(
  213. retry=retry, scan_timeout=self._scan_timeout
  214. )
  215. if self._device.address in _data:
  216. self._sb_adv_data = _data[self._device.address]
  217. return self._sb_adv_data
  218. async def _get_basic_info(self) -> bytes | None:
  219. """Return basic info of device."""
  220. _data = await self._sendcommand(
  221. key=DEVICE_GET_BASIC_SETTINGS_KEY, retry=self._retry_count
  222. )
  223. if _data in (b"\x07", b"\x00"):
  224. _LOGGER.error("Unsuccessful, please try again")
  225. return None
  226. return _data