"""Support for PCA 301 smart switch.""" from __future__ import annotations import logging import pypca from serial import SerialException from homeassistant.components.switch import ATTR_CURRENT_POWER_W, SwitchEntity from homeassistant.const import EVENT_HOMEASSISTANT_STOP from homeassistant.core import HomeAssistant from homeassistant.helpers.entity_platform import AddEntitiesCallback from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType _LOGGER = logging.getLogger(__name__) ATTR_TOTAL_ENERGY_KWH = "total_energy_kwh" DEFAULT_NAME = "PCA 301" def setup_platform( hass: HomeAssistant, config: ConfigType, add_entities: AddEntitiesCallback, discovery_info: DiscoveryInfoType | None = None, ) -> None: """Set up the PCA switch platform.""" if discovery_info is None: return serial_device = discovery_info["device"] try: pca = pypca.PCA(serial_device) pca.open() entities = [SmartPlugSwitch(pca, device) for device in pca.get_devices()] add_entities(entities, True) except SerialException as exc: _LOGGER.warning("Unable to open serial port: %s", exc) return hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, pca.close) pca.start_scan() class SmartPlugSwitch(SwitchEntity): """Representation of a PCA Smart Plug switch.""" def __init__(self, pca, device_id): """Initialize the switch.""" self._device_id = device_id self._name = "PCA 301" self._state = None self._available = True self._emeter_params = {} self._pca = pca @property def name(self): """Return the name of the Smart Plug, if any.""" return self._name @property def available(self) -> bool: """Return if switch is available.""" return self._available @property def is_on(self): """Return true if switch is on.""" return self._state def turn_on(self, **kwargs): """Turn the switch on.""" self._pca.turn_on(self._device_id) def turn_off(self, **kwargs): """Turn the switch off.""" self._pca.turn_off(self._device_id) @property def extra_state_attributes(self): """Return the state attributes of the device.""" return self._emeter_params def update(self): """Update the PCA switch's state.""" try: self._emeter_params[ ATTR_CURRENT_POWER_W ] = f"{self._pca.get_current_power(self._device_id):.1f}" self._emeter_params[ ATTR_TOTAL_ENERGY_KWH ] = f"{self._pca.get_total_consumption(self._device_id):.2f}" self._available = True self._state = self._pca.get_state(self._device_id) except (OSError) as ex: if self._available: _LOGGER.warning("Could not read state for %s: %s", self.name, ex) self._available = False