"""Support for compensation sensor.""" from __future__ import annotations import logging from typing import Any import numpy as np from homeassistant.components.sensor import SensorEntity from homeassistant.const import ( ATTR_UNIT_OF_MEASUREMENT, CONF_ATTRIBUTE, CONF_SOURCE, CONF_UNIQUE_ID, CONF_UNIT_OF_MEASUREMENT, STATE_UNKNOWN, ) from homeassistant.core import Event, HomeAssistant, State, callback from homeassistant.helpers.entity_platform import AddEntitiesCallback from homeassistant.helpers.event import async_track_state_change_event from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType from .const import ( CONF_COMPENSATION, CONF_POLYNOMIAL, CONF_PRECISION, DATA_COMPENSATION, DEFAULT_NAME, ) _LOGGER = logging.getLogger(__name__) ATTR_COEFFICIENTS = "coefficients" ATTR_SOURCE = "source" ATTR_SOURCE_ATTRIBUTE = "source_attribute" async def async_setup_platform( hass: HomeAssistant, config: ConfigType, async_add_entities: AddEntitiesCallback, discovery_info: DiscoveryInfoType | None = None, ) -> None: """Set up the Compensation sensor.""" if discovery_info is None: return compensation: str = discovery_info[CONF_COMPENSATION] conf: dict[str, Any] = hass.data[DATA_COMPENSATION][compensation] source: str = conf[CONF_SOURCE] attribute: str | None = conf.get(CONF_ATTRIBUTE) name = f"{DEFAULT_NAME} {source}" if attribute is not None: name = f"{name} {attribute}" async_add_entities( [ CompensationSensor( conf.get(CONF_UNIQUE_ID), name, source, attribute, conf[CONF_PRECISION], conf[CONF_POLYNOMIAL], conf.get(CONF_UNIT_OF_MEASUREMENT), ) ] ) class CompensationSensor(SensorEntity): """Representation of a Compensation sensor.""" _attr_should_poll = False def __init__( self, unique_id: str | None, name: str, source: str, attribute: str | None, precision: int, polynomial: np.poly1d, unit_of_measurement: str | None, ) -> None: """Initialize the Compensation sensor.""" self._source_entity_id = source self._precision = precision self._source_attribute = attribute self._attr_native_unit_of_measurement = unit_of_measurement self._poly = polynomial self._coefficients = polynomial.coefficients.tolist() self._attr_unique_id = unique_id self._attr_name = name async def async_added_to_hass(self) -> None: """Handle added to Hass.""" self.async_on_remove( async_track_state_change_event( self.hass, [self._source_entity_id], self._async_compensation_sensor_state_listener, ) ) @property def extra_state_attributes(self) -> dict[str, Any]: """Return the state attributes of the sensor.""" ret = { ATTR_SOURCE: self._source_entity_id, ATTR_COEFFICIENTS: self._coefficients, } if self._source_attribute: ret[ATTR_SOURCE_ATTRIBUTE] = self._source_attribute return ret @callback def _async_compensation_sensor_state_listener(self, event: Event) -> None: """Handle sensor state changes.""" new_state: State | None if (new_state := event.data.get("new_state")) is None: return if self.native_unit_of_measurement is None and self._source_attribute is None: self._attr_native_unit_of_measurement = new_state.attributes.get( ATTR_UNIT_OF_MEASUREMENT ) if self._source_attribute: value = new_state.attributes.get(self._source_attribute) else: value = None if new_state.state == STATE_UNKNOWN else new_state.state try: self._attr_native_value = round(self._poly(float(value)), self._precision) except (ValueError, TypeError): self._attr_native_value = None if self._source_attribute: _LOGGER.warning( "%s attribute %s is not numerical", self._source_entity_id, self._source_attribute, ) else: _LOGGER.warning("%s state is not numerical", self._source_entity_id) self.async_write_ha_state()