device.py 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  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. def _sb_uuid(comms_type: str = "service") -> UUID | str:
  31. """Return Switchbot UUID."""
  32. _uuid = {"tx": "002", "rx": "003", "service": "d00"}
  33. if comms_type in _uuid:
  34. return UUID(f"cba20{_uuid[comms_type]}-224d-11e6-9fb8-0002a5d5c51b")
  35. return "Incorrect type, choose between: tx, rx or service"
  36. class SwitchbotDevice:
  37. """Base Representation of a Switchbot Device."""
  38. def __init__(
  39. self,
  40. device: BLEDevice,
  41. password: str | None = None,
  42. interface: int = 0,
  43. **kwargs: Any,
  44. ) -> None:
  45. """Switchbot base class constructor."""
  46. self._interface = f"hci{interface}"
  47. self._device = device
  48. self._sb_adv_data: SwitchBotAdvertisement | None = None
  49. self._scan_timeout: int = kwargs.pop("scan_timeout", DEFAULT_SCAN_TIMEOUT)
  50. self._retry_count: int = kwargs.pop("retry_count", DEFAULT_RETRY_COUNT)
  51. self._connect_lock = asyncio.Lock()
  52. self._operation_lock = asyncio.Lock()
  53. if password is None or password == "":
  54. self._password_encoded = None
  55. else:
  56. self._password_encoded = "%08x" % (
  57. binascii.crc32(password.encode("ascii")) & 0xFFFFFFFF
  58. )
  59. self._client: BleakClientWithServiceCache | None = None
  60. self._cached_services: BleakGATTServiceCollection | None = None
  61. self._read_char: BleakGATTCharacteristic | None = None
  62. self._write_char: BleakGATTCharacteristic | None = None
  63. def _commandkey(self, key: str) -> str:
  64. """Add password to key if set."""
  65. if self._password_encoded is None:
  66. return key
  67. key_action = key[3]
  68. key_suffix = key[4:]
  69. return KEY_PASSWORD_PREFIX + key_action + self._password_encoded + key_suffix
  70. async def _sendcommand(self, key: str, retry: int) -> bytes:
  71. """Send command to device and read response."""
  72. command = bytearray.fromhex(self._commandkey(key))
  73. _LOGGER.debug("Sending command to switchbot %s", command)
  74. max_attempts = retry + 1
  75. async with self._operation_lock:
  76. for attempt in range(max_attempts):
  77. try:
  78. return await self._send_command_locked(key, command)
  79. except BLEAK_EXCEPTIONS(
  80. bleak.BleakError, asyncio.exceptions.TimeoutError
  81. ):
  82. if attempt == retry:
  83. _LOGGER.error(
  84. "Switchbot communication failed. Stopping trying",
  85. exc_info=True,
  86. )
  87. return b"\x00"
  88. _LOGGER.debug("Switchbot communication failed with:", exc_info=True)
  89. raise RuntimeError("Unreachable")
  90. @property
  91. def name(self) -> str:
  92. """Return device name."""
  93. return f"{self._device.name} ({self._device.address})"
  94. async def _ensure_connected(self):
  95. if self._client and self._client.is_connected:
  96. return
  97. async with self._connect_lock:
  98. # Check again while holding the lock
  99. if self._client and self._client.is_connected:
  100. return
  101. client = await establish_connection(
  102. BleakClientWithServiceCache,
  103. self._device,
  104. self.name,
  105. cached_services=self._cached_services,
  106. )
  107. self._cached_services = client.services
  108. _LOGGER.debug("%s: Connected to SwitchBot Device", self.name)
  109. services = client.services
  110. self._read_char = services.get_characteristic(_sb_uuid(comms_type="rx"))
  111. self._write_char = services.get_characteristic(_sb_uuid(comms_type="tx"))
  112. self._client = client
  113. async def _send_command_locked(self, key: str, command: bytes) -> bytes:
  114. """Send command to device and read response."""
  115. client: BleakClientWithServiceCache | None = None
  116. await self._ensure_connected()
  117. client = self._client
  118. future: asyncio.Future[bytearray] = asyncio.Future()
  119. def _notification_handler(_sender: int, data: bytearray) -> None:
  120. """Handle notification responses."""
  121. if future.done():
  122. _LOGGER.debug("%s: Notification handler already done", self.name)
  123. return
  124. future.set_result(data)
  125. _LOGGER.debug("%s: Subscribe to notifications", self.name)
  126. await client.start_notify(self._read_char, _notification_handler)
  127. _LOGGER.debug("%s: Sending command, %s", self.name, key)
  128. await client.write_gatt_char(self._write_char, command, False)
  129. async with async_timeout.timeout(5):
  130. notify_msg = await future
  131. _LOGGER.info("%s: Notification received: %s", self.name, notify_msg)
  132. _LOGGER.debug("%s: UnSubscribe to notifications", self.name)
  133. await client.stop_notify(self._read_char)
  134. if notify_msg == b"\x07":
  135. _LOGGER.error("Password required")
  136. elif notify_msg == b"\t":
  137. _LOGGER.error("Password incorrect")
  138. return notify_msg
  139. def get_address(self) -> str:
  140. """Return address of device."""
  141. return self._device.address
  142. def _get_adv_value(self, key: str) -> Any:
  143. """Return value from advertisement data."""
  144. if not self._sb_adv_data:
  145. return None
  146. return self._sb_adv_data.data["data"][key]
  147. def get_battery_percent(self) -> Any:
  148. """Return device battery level in percent."""
  149. return self._get_adv_value("battery")
  150. def update_from_advertisement(self, advertisement: SwitchBotAdvertisement) -> None:
  151. """Update device data from advertisement."""
  152. self._sb_adv_data = advertisement
  153. if self._device and ble_device_has_changed(self._device, advertisement.device):
  154. self._cached_services = None
  155. self._device = advertisement.device
  156. async def get_device_data(
  157. self, retry: int = DEFAULT_RETRY_COUNT, interface: int | None = None
  158. ) -> dict | None:
  159. """Find switchbot devices and their advertisement data."""
  160. if interface:
  161. _interface: int = interface
  162. else:
  163. _interface = int(self._interface.replace("hci", ""))
  164. _data = await GetSwitchbotDevices(interface=_interface).discover(
  165. retry=retry, scan_timeout=self._scan_timeout
  166. )
  167. if self._device.address in _data:
  168. self._sb_adv_data = _data[self._device.address]
  169. return self._sb_adv_data
  170. async def _get_basic_info(self) -> dict | None:
  171. """Return basic info of device."""
  172. _data = await self._sendcommand(
  173. key=DEVICE_GET_BASIC_SETTINGS_KEY, retry=self._retry_count
  174. )
  175. if _data in (b"\x07", b"\x00"):
  176. _LOGGER.error("Unsuccessful, please try again")
  177. return None
  178. return _data