Refactor SwitchBot Cloud make_device_data (#135698)

This commit is contained in:
Mick Montorier-Aberman 2025-01-19 14:22:21 +01:00 committed by GitHub
parent dfc4cdf785
commit 41fe863b72
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 78 additions and 57 deletions

View File

@ -8,7 +8,7 @@ from switchbot_api import CannotConnect, Device, InvalidAuth, Remote, SwitchBotA
from homeassistant.config_entries import ConfigEntry from homeassistant.config_entries import ConfigEntry
from homeassistant.const import CONF_API_KEY, CONF_API_TOKEN, Platform from homeassistant.const import CONF_API_KEY, CONF_API_TOKEN, Platform
from homeassistant.core import HomeAssistant, callback from homeassistant.core import HomeAssistant
from homeassistant.exceptions import ConfigEntryNotReady from homeassistant.exceptions import ConfigEntryNotReady
from .const import DOMAIN from .const import DOMAIN
@ -43,36 +43,54 @@ class SwitchbotCloudData:
devices: SwitchbotDevices devices: SwitchbotDevices
@callback async def coordinator_for_device(
def prepare_device(
hass: HomeAssistant, hass: HomeAssistant,
api: SwitchBotAPI, api: SwitchBotAPI,
device: Device | Remote, device: Device | Remote,
coordinators_by_id: dict[str, SwitchBotCoordinator], coordinators_by_id: dict[str, SwitchBotCoordinator],
) -> tuple[Device | Remote, SwitchBotCoordinator]: ) -> SwitchBotCoordinator:
"""Instantiate coordinator and adds to list for gathering.""" """Instantiate coordinator and adds to list for gathering."""
coordinator = coordinators_by_id.setdefault( coordinator = coordinators_by_id.setdefault(
device.device_id, SwitchBotCoordinator(hass, api, device) device.device_id, SwitchBotCoordinator(hass, api, device)
) )
return (device, coordinator)
if coordinator.data is None:
await coordinator.async_config_entry_first_refresh()
return coordinator
@callback async def make_switchbot_devices(
def make_device_data(
hass: HomeAssistant, hass: HomeAssistant,
api: SwitchBotAPI, api: SwitchBotAPI,
devices: list[Device | Remote], devices: list[Device | Remote],
coordinators_by_id: dict[str, SwitchBotCoordinator], coordinators_by_id: dict[str, SwitchBotCoordinator],
) -> SwitchbotDevices: ) -> SwitchbotDevices:
"""Make device data.""" """Make SwitchBot devices."""
devices_data = SwitchbotDevices() devices_data = SwitchbotDevices()
for device in devices: await gather(
if isinstance(device, Remote) and device.device_type.endswith( *[
"Air Conditioner" make_device_data(hass, api, device, devices_data, coordinators_by_id)
): for device in devices
devices_data.climates.append( ]
prepare_device(hass, api, device, coordinators_by_id)
) )
return devices_data
async def make_device_data(
hass: HomeAssistant,
api: SwitchBotAPI,
device: Device | Remote,
devices_data: SwitchbotDevices,
coordinators_by_id: dict[str, SwitchBotCoordinator],
) -> None:
"""Make device data."""
if isinstance(device, Remote) and device.device_type.endswith("Air Conditioner"):
coordinator = await coordinator_for_device(
hass, api, device, coordinators_by_id
)
devices_data.climates.append((device, coordinator))
if ( if (
isinstance(device, Device) isinstance(device, Device)
and ( and (
@ -80,9 +98,11 @@ def make_device_data(
or device.device_type in ["Relay Switch 1PM", "Relay Switch 1"] or device.device_type in ["Relay Switch 1PM", "Relay Switch 1"]
) )
) or isinstance(device, Remote): ) or isinstance(device, Remote):
devices_data.switches.append( coordinator = await coordinator_for_device(
prepare_device(hass, api, device, coordinators_by_id) hass, api, device, coordinators_by_id
) )
devices_data.switches.append((device, coordinator))
if isinstance(device, Device) and device.device_type in [ if isinstance(device, Device) and device.device_type in [
"Meter", "Meter",
"MeterPlus", "MeterPlus",
@ -94,24 +114,24 @@ def make_device_data(
"Plug Mini (US)", "Plug Mini (US)",
"Plug Mini (JP)", "Plug Mini (JP)",
]: ]:
devices_data.sensors.append( devices_data.sensors.append((device, coordinator))
prepare_device(hass, api, device, coordinators_by_id)
)
if isinstance(device, Device) and device.device_type in [ if isinstance(device, Device) and device.device_type in [
"K10+", "K10+",
"K10+ Pro", "K10+ Pro",
"Robot Vacuum Cleaner S1", "Robot Vacuum Cleaner S1",
"Robot Vacuum Cleaner S1 Plus", "Robot Vacuum Cleaner S1 Plus",
]: ]:
devices_data.vacuums.append( coordinator = await coordinator_for_device(
prepare_device(hass, api, device, coordinators_by_id) hass, api, device, coordinators_by_id
) )
devices_data.vacuums.append((device, coordinator))
if isinstance(device, Device) and device.device_type.startswith("Smart Lock"): if isinstance(device, Device) and device.device_type.startswith("Smart Lock"):
devices_data.locks.append( coordinator = await coordinator_for_device(
prepare_device(hass, api, device, coordinators_by_id) hass, api, device, coordinators_by_id
) )
return devices_data devices_data.locks.append((device, coordinator))
async def async_setup_entry(hass: HomeAssistant, config: ConfigEntry) -> bool: async def async_setup_entry(hass: HomeAssistant, config: ConfigEntry) -> bool:
@ -131,12 +151,13 @@ async def async_setup_entry(hass: HomeAssistant, config: ConfigEntry) -> bool:
raise ConfigEntryNotReady from ex raise ConfigEntryNotReady from ex
_LOGGER.debug("Devices: %s", devices) _LOGGER.debug("Devices: %s", devices)
coordinators_by_id: dict[str, SwitchBotCoordinator] = {} coordinators_by_id: dict[str, SwitchBotCoordinator] = {}
switchbot_devices = await make_switchbot_devices(
hass, api, devices, coordinators_by_id
)
hass.data.setdefault(DOMAIN, {}) hass.data.setdefault(DOMAIN, {})
hass.data[DOMAIN][config.entry_id] = SwitchbotCloudData( hass.data[DOMAIN][config.entry_id] = SwitchbotCloudData(
api=api, devices=make_device_data(hass, api, devices, coordinators_by_id) api=api, devices=switchbot_devices
)
await gather(
*[coordinator.async_refresh() for coordinator in coordinators_by_id.values()]
) )
await hass.config_entries.async_forward_entry_setups(config, PLATFORMS) await hass.config_entries.async_forward_entry_setups(config, PLATFORMS)
return True return True

View File

@ -116,7 +116,7 @@ async def test_setup_entry_fails_when_refreshing(
mock_get_status.side_effect = CannotConnect mock_get_status.side_effect = CannotConnect
entry = configure_integration(hass) entry = configure_integration(hass)
await hass.config_entries.async_setup(entry.entry_id) await hass.config_entries.async_setup(entry.entry_id)
assert entry.state is ConfigEntryState.LOADED assert entry.state is ConfigEntryState.SETUP_RETRY
hass.bus.async_fire(EVENT_HOMEASSISTANT_START) hass.bus.async_fire(EVENT_HOMEASSISTANT_START)
await hass.async_block_till_done() await hass.async_block_till_done()