core/homeassistant/components/iotty/switch.py

195 lines
6.3 KiB
Python

"""Implement a iotty Light Switch Device."""
from __future__ import annotations
import logging
from typing import TYPE_CHECKING, Any
from iottycloud.lightswitch import LightSwitch
from iottycloud.outlet import Outlet
from iottycloud.verbs import (
COMMAND_TURNOFF,
COMMAND_TURNON,
LS_DEVICE_TYPE_UID,
OU_DEVICE_TYPE_UID,
)
from homeassistant.components.switch import (
SwitchDeviceClass,
SwitchEntity,
SwitchEntityDescription,
)
from homeassistant.core import HomeAssistant, callback
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from . import IottyConfigEntry
from .api import IottyProxy
from .coordinator import IottyDataUpdateCoordinator
from .entity import IottyEntity
_LOGGER = logging.getLogger(__name__)
ENTITIES: dict[str, SwitchEntityDescription] = {
LS_DEVICE_TYPE_UID: SwitchEntityDescription(
key="light",
name=None,
device_class=SwitchDeviceClass.SWITCH,
),
OU_DEVICE_TYPE_UID: SwitchEntityDescription(
key="outlet",
name=None,
device_class=SwitchDeviceClass.OUTLET,
),
}
async def async_setup_entry(
hass: HomeAssistant,
config_entry: IottyConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Activate the iotty Switch component."""
_LOGGER.debug("Setup SWITCH entry id is %s", config_entry.entry_id)
coordinator = config_entry.runtime_data.coordinator
lightswitch_entities = [
IottySwitch(
coordinator=coordinator,
iotty_cloud=coordinator.iotty,
iotty_device=d,
entity_description=ENTITIES[LS_DEVICE_TYPE_UID],
)
for d in coordinator.data.devices
if d.device_type == LS_DEVICE_TYPE_UID
if (isinstance(d, LightSwitch))
]
_LOGGER.debug("Found %d LightSwitches", len(lightswitch_entities))
outlet_entities = [
IottySwitch(
coordinator=coordinator,
iotty_cloud=coordinator.iotty,
iotty_device=d,
entity_description=ENTITIES[OU_DEVICE_TYPE_UID],
)
for d in coordinator.data.devices
if d.device_type == OU_DEVICE_TYPE_UID
if (isinstance(d, Outlet))
]
_LOGGER.debug("Found %d Outlets", len(outlet_entities))
entities = lightswitch_entities + outlet_entities
async_add_entities(entities)
known_devices: set = config_entry.runtime_data.known_devices
for known_device in coordinator.data.devices:
if known_device.device_type in {LS_DEVICE_TYPE_UID, OU_DEVICE_TYPE_UID}:
known_devices.add(known_device)
@callback
def async_update_data() -> None:
"""Handle updated data from the API endpoint."""
if not coordinator.last_update_success:
return
devices = coordinator.data.devices
entities = []
known_devices: set = config_entry.runtime_data.known_devices
# Add entities for devices which we've not yet seen
for device in devices:
if any(d.device_id == device.device_id for d in known_devices) or (
device.device_type not in {LS_DEVICE_TYPE_UID, OU_DEVICE_TYPE_UID}
):
continue
iotty_entity: SwitchEntity
iotty_device: LightSwitch | Outlet
if device.device_type == LS_DEVICE_TYPE_UID:
if TYPE_CHECKING:
assert isinstance(device, LightSwitch)
iotty_device = LightSwitch(
device.device_id,
device.serial_number,
device.device_type,
device.device_name,
)
else:
if TYPE_CHECKING:
assert isinstance(device, Outlet)
iotty_device = Outlet(
device.device_id,
device.serial_number,
device.device_type,
device.device_name,
)
iotty_entity = IottySwitch(
coordinator=coordinator,
iotty_cloud=coordinator.iotty,
iotty_device=iotty_device,
entity_description=ENTITIES[device.device_type],
)
entities.extend([iotty_entity])
known_devices.add(device)
async_add_entities(entities)
# Add a subscriber to the coordinator to discover new devices
coordinator.async_add_listener(async_update_data)
class IottySwitch(IottyEntity, SwitchEntity):
"""Haas entity class for iotty switch."""
_attr_device_class: SwitchDeviceClass | None
_iotty_device: LightSwitch | Outlet
def __init__(
self,
coordinator: IottyDataUpdateCoordinator,
iotty_cloud: IottyProxy,
iotty_device: LightSwitch | Outlet,
entity_description: SwitchEntityDescription,
) -> None:
"""Initialize the Switch device."""
super().__init__(coordinator, iotty_cloud, iotty_device)
self.entity_description = entity_description
self._attr_device_class = entity_description.device_class
@property
def is_on(self) -> bool:
"""Return true if the Switch is on."""
_LOGGER.debug(
"Retrieve device status for %s ? %s",
self._iotty_device.device_id,
self._iotty_device.is_on,
)
return self._iotty_device.is_on
async def async_turn_on(self, **kwargs: Any) -> None:
"""Turn the Switch on."""
_LOGGER.debug("[%s] Turning on", self._iotty_device.device_id)
await self._iotty_cloud.command(self._iotty_device.device_id, COMMAND_TURNON)
await self.coordinator.async_request_refresh()
async def async_turn_off(self, **kwargs: Any) -> None:
"""Turn the Switch off."""
_LOGGER.debug("[%s] Turning off", self._iotty_device.device_id)
await self._iotty_cloud.command(self._iotty_device.device_id, COMMAND_TURNOFF)
await self.coordinator.async_request_refresh()
@callback
def _handle_coordinator_update(self) -> None:
"""Handle updated data from the coordinator."""
device: LightSwitch | Outlet = next( # type: ignore[assignment]
device
for device in self.coordinator.data.devices
if device.device_id == self._iotty_device.device_id
)
self._iotty_device.is_on = device.is_on
self.async_write_ha_state()