2021-09-18 19:25:05 +00:00
|
|
|
"""Provides the switchbot DataUpdateCoordinator."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
2022-07-24 16:38:45 +00:00
|
|
|
import asyncio
|
2022-08-12 01:10:59 +00:00
|
|
|
import contextlib
|
2021-09-18 19:25:05 +00:00
|
|
|
import logging
|
2023-01-03 07:19:53 +00:00
|
|
|
from typing import TYPE_CHECKING
|
2021-09-18 19:25:05 +00:00
|
|
|
|
2022-08-12 01:10:59 +00:00
|
|
|
import async_timeout
|
2022-06-27 11:56:51 +00:00
|
|
|
import switchbot
|
2022-12-19 00:54:29 +00:00
|
|
|
from switchbot import SwitchbotModel
|
2021-09-18 19:25:05 +00:00
|
|
|
|
2022-07-24 16:38:45 +00:00
|
|
|
from homeassistant.components import bluetooth
|
2022-12-20 02:55:18 +00:00
|
|
|
from homeassistant.components.bluetooth.active_update_coordinator import (
|
|
|
|
ActiveBluetoothDataUpdateCoordinator,
|
2022-07-24 16:38:45 +00:00
|
|
|
)
|
2022-12-20 02:55:18 +00:00
|
|
|
from homeassistant.core import CoreState, HomeAssistant, callback
|
2021-09-18 19:25:05 +00:00
|
|
|
|
2022-07-29 05:26:37 +00:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from bleak.backends.device import BLEDevice
|
|
|
|
|
|
|
|
|
2021-09-18 19:25:05 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2022-08-29 14:30:09 +00:00
|
|
|
DEVICE_STARTUP_TIMEOUT = 30
|
|
|
|
|
2021-09-18 19:25:05 +00:00
|
|
|
|
2023-01-03 07:19:53 +00:00
|
|
|
class SwitchbotDataUpdateCoordinator(ActiveBluetoothDataUpdateCoordinator[None]):
|
2021-09-18 19:25:05 +00:00
|
|
|
"""Class to manage fetching switchbot data."""
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
hass: HomeAssistant,
|
2022-07-24 16:38:45 +00:00
|
|
|
logger: logging.Logger,
|
|
|
|
ble_device: BLEDevice,
|
|
|
|
device: switchbot.SwitchbotDevice,
|
2022-08-10 19:02:08 +00:00
|
|
|
base_unique_id: str,
|
|
|
|
device_name: str,
|
2022-08-22 18:02:26 +00:00
|
|
|
connectable: bool,
|
2022-12-19 00:54:29 +00:00
|
|
|
model: SwitchbotModel,
|
2021-09-18 19:25:05 +00:00
|
|
|
) -> None:
|
|
|
|
"""Initialize global switchbot data updater."""
|
2022-07-30 00:53:33 +00:00
|
|
|
super().__init__(
|
2022-12-20 02:55:18 +00:00
|
|
|
hass=hass,
|
|
|
|
logger=logger,
|
|
|
|
address=ble_device.address,
|
|
|
|
needs_poll_method=self._needs_poll,
|
|
|
|
poll_method=self._async_update,
|
|
|
|
mode=bluetooth.BluetoothScanningMode.ACTIVE,
|
|
|
|
connectable=connectable,
|
2022-07-30 00:53:33 +00:00
|
|
|
)
|
2022-07-24 16:38:45 +00:00
|
|
|
self.ble_device = ble_device
|
|
|
|
self.device = device
|
2022-08-10 19:02:08 +00:00
|
|
|
self.device_name = device_name
|
|
|
|
self.base_unique_id = base_unique_id
|
2022-08-29 14:30:09 +00:00
|
|
|
self.model = model
|
2022-07-24 16:38:45 +00:00
|
|
|
self._ready_event = asyncio.Event()
|
2022-11-09 07:09:06 +00:00
|
|
|
self._was_unavailable = True
|
|
|
|
|
2022-12-20 02:55:18 +00:00
|
|
|
@callback
|
|
|
|
def _needs_poll(
|
|
|
|
self,
|
|
|
|
service_info: bluetooth.BluetoothServiceInfoBleak,
|
|
|
|
seconds_since_last_poll: float | None,
|
|
|
|
) -> bool:
|
|
|
|
# Only poll if hass is running, we need to poll,
|
|
|
|
# and we actually have a way to connect to the device
|
|
|
|
return (
|
|
|
|
self.hass.state == CoreState.running
|
|
|
|
and self.device.poll_needed(seconds_since_last_poll)
|
|
|
|
and bool(
|
|
|
|
bluetooth.async_ble_device_from_address(
|
|
|
|
self.hass, service_info.device.address, connectable=True
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
async def _async_update(
|
|
|
|
self, service_info: bluetooth.BluetoothServiceInfoBleak
|
2023-01-03 07:19:53 +00:00
|
|
|
) -> None:
|
2022-12-20 02:55:18 +00:00
|
|
|
"""Poll the device."""
|
2023-01-03 07:19:53 +00:00
|
|
|
await self.device.update()
|
2022-12-20 02:55:18 +00:00
|
|
|
|
2022-11-09 07:09:06 +00:00
|
|
|
@callback
|
|
|
|
def _async_handle_unavailable(
|
|
|
|
self, service_info: bluetooth.BluetoothServiceInfoBleak
|
|
|
|
) -> None:
|
|
|
|
"""Handle the device going unavailable."""
|
|
|
|
super()._async_handle_unavailable(service_info)
|
|
|
|
self._was_unavailable = True
|
2022-07-24 16:38:45 +00:00
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_handle_bluetooth_event(
|
|
|
|
self,
|
2022-07-30 00:53:33 +00:00
|
|
|
service_info: bluetooth.BluetoothServiceInfoBleak,
|
2022-07-24 16:38:45 +00:00
|
|
|
change: bluetooth.BluetoothChange,
|
|
|
|
) -> None:
|
|
|
|
"""Handle a Bluetooth event."""
|
2022-09-09 23:13:27 +00:00
|
|
|
self.ble_device = service_info.device
|
2022-11-09 07:09:06 +00:00
|
|
|
if not (
|
|
|
|
adv := switchbot.parse_advertisement_data(
|
2022-12-19 00:54:29 +00:00
|
|
|
service_info.device, service_info.advertisement, self.model
|
2022-11-09 07:09:06 +00:00
|
|
|
)
|
2022-07-24 16:38:45 +00:00
|
|
|
):
|
2022-11-09 07:09:06 +00:00
|
|
|
return
|
|
|
|
if "modelName" in adv.data:
|
|
|
|
self._ready_event.set()
|
2022-12-19 17:47:42 +00:00
|
|
|
_LOGGER.debug(
|
|
|
|
"%s: Switchbot data: %s", self.ble_device.address, self.device.data
|
|
|
|
)
|
2022-11-09 07:09:06 +00:00
|
|
|
if not self.device.advertisement_changed(adv) and not self._was_unavailable:
|
|
|
|
return
|
|
|
|
self._was_unavailable = False
|
|
|
|
self.device.update_from_advertisement(adv)
|
2022-08-29 14:30:09 +00:00
|
|
|
super()._async_handle_bluetooth_event(service_info, change)
|
2022-07-24 16:38:45 +00:00
|
|
|
|
|
|
|
async def async_wait_ready(self) -> bool:
|
|
|
|
"""Wait for the device to be ready."""
|
2022-08-12 01:10:59 +00:00
|
|
|
with contextlib.suppress(asyncio.TimeoutError):
|
2022-08-29 14:30:09 +00:00
|
|
|
async with async_timeout.timeout(DEVICE_STARTUP_TIMEOUT):
|
2022-08-12 01:10:59 +00:00
|
|
|
await self._ready_event.wait()
|
|
|
|
return True
|
|
|
|
return False
|