"""Sensors for cloud based weatherflow.""" from __future__ import annotations from collections.abc import Callable from dataclasses import dataclass from datetime import UTC, datetime from weatherflow4py.models.rest.observation import Observation from homeassistant.components.sensor import ( SensorDeviceClass, SensorEntity, SensorEntityDescription, SensorStateClass, ) from homeassistant.config_entries import ConfigEntry from homeassistant.const import UnitOfLength, UnitOfPressure, UnitOfTemperature from homeassistant.core import HomeAssistant from homeassistant.helpers.entity_platform import AddEntitiesCallback from homeassistant.helpers.typing import StateType from .const import DOMAIN from .coordinator import WeatherFlowCloudDataUpdateCoordinator from .entity import WeatherFlowCloudEntity @dataclass(frozen=True, kw_only=True) class WeatherFlowCloudSensorEntityDescription( SensorEntityDescription, ): """Describes a weatherflow sensor.""" value_fn: Callable[[Observation], StateType | datetime] WF_SENSORS: tuple[WeatherFlowCloudSensorEntityDescription, ...] = ( # Air Sensors WeatherFlowCloudSensorEntityDescription( key="air_density", translation_key="air_density", state_class=SensorStateClass.MEASUREMENT, suggested_display_precision=5, value_fn=lambda data: data.air_density, native_unit_of_measurement="kg/m³", ), # Temp Sensors WeatherFlowCloudSensorEntityDescription( key="air_temperature", translation_key="air_temperature", device_class=SensorDeviceClass.TEMPERATURE, state_class=SensorStateClass.MEASUREMENT, suggested_display_precision=1, value_fn=lambda data: data.air_temperature, native_unit_of_measurement=UnitOfTemperature.CELSIUS, ), WeatherFlowCloudSensorEntityDescription( key="dew_point", translation_key="dew_point", value_fn=lambda data: data.dew_point, native_unit_of_measurement=UnitOfTemperature.CELSIUS, device_class=SensorDeviceClass.TEMPERATURE, state_class=SensorStateClass.MEASUREMENT, suggested_display_precision=1, ), WeatherFlowCloudSensorEntityDescription( key="feels_like", translation_key="feels_like", device_class=SensorDeviceClass.TEMPERATURE, state_class=SensorStateClass.MEASUREMENT, suggested_display_precision=1, value_fn=lambda data: data.feels_like, native_unit_of_measurement=UnitOfTemperature.CELSIUS, ), WeatherFlowCloudSensorEntityDescription( key="heat_index", translation_key="heat_index", device_class=SensorDeviceClass.TEMPERATURE, state_class=SensorStateClass.MEASUREMENT, suggested_display_precision=1, value_fn=lambda data: data.heat_index, native_unit_of_measurement=UnitOfTemperature.CELSIUS, ), WeatherFlowCloudSensorEntityDescription( key="wind_chill", translation_key="wind_chill", device_class=SensorDeviceClass.TEMPERATURE, state_class=SensorStateClass.MEASUREMENT, suggested_display_precision=1, value_fn=lambda data: data.wind_chill, native_unit_of_measurement=UnitOfTemperature.CELSIUS, ), WeatherFlowCloudSensorEntityDescription( key="wet_bulb_temperature", translation_key="wet_bulb_temperature", device_class=SensorDeviceClass.TEMPERATURE, state_class=SensorStateClass.MEASUREMENT, suggested_display_precision=1, value_fn=lambda data: data.wet_bulb_temperature, native_unit_of_measurement=UnitOfTemperature.CELSIUS, ), WeatherFlowCloudSensorEntityDescription( key="wet_bulb_globe_temperature", translation_key="wet_bulb_globe_temperature", device_class=SensorDeviceClass.TEMPERATURE, state_class=SensorStateClass.MEASUREMENT, suggested_display_precision=1, value_fn=lambda data: data.wet_bulb_globe_temperature, native_unit_of_measurement=UnitOfTemperature.CELSIUS, ), # Pressure Sensors WeatherFlowCloudSensorEntityDescription( key="barometric_pressure", translation_key="barometric_pressure", value_fn=lambda data: data.barometric_pressure, native_unit_of_measurement=UnitOfPressure.MBAR, device_class=SensorDeviceClass.ATMOSPHERIC_PRESSURE, state_class=SensorStateClass.MEASUREMENT, suggested_display_precision=3, ), WeatherFlowCloudSensorEntityDescription( key="sea_level_pressure", translation_key="sea_level_pressure", value_fn=lambda data: data.sea_level_pressure, native_unit_of_measurement=UnitOfPressure.MBAR, device_class=SensorDeviceClass.ATMOSPHERIC_PRESSURE, state_class=SensorStateClass.MEASUREMENT, suggested_display_precision=3, ), # Lightning Sensors WeatherFlowCloudSensorEntityDescription( key="lightning_strike_count", translation_key="lightning_strike_count", state_class=SensorStateClass.TOTAL, value_fn=lambda data: data.lightning_strike_count, ), WeatherFlowCloudSensorEntityDescription( key="lightning_strike_count_last_1hr", translation_key="lightning_strike_count_last_1hr", state_class=SensorStateClass.TOTAL, value_fn=lambda data: data.lightning_strike_count_last_1hr, ), WeatherFlowCloudSensorEntityDescription( key="lightning_strike_count_last_3hr", translation_key="lightning_strike_count_last_3hr", state_class=SensorStateClass.TOTAL, value_fn=lambda data: data.lightning_strike_count_last_3hr, ), WeatherFlowCloudSensorEntityDescription( key="lightning_strike_last_distance", translation_key="lightning_strike_last_distance", state_class=SensorStateClass.MEASUREMENT, device_class=SensorDeviceClass.DISTANCE, native_unit_of_measurement=UnitOfLength.KILOMETERS, value_fn=lambda data: data.lightning_strike_last_distance, ), WeatherFlowCloudSensorEntityDescription( key="lightning_strike_last_epoch", translation_key="lightning_strike_last_epoch", device_class=SensorDeviceClass.TIMESTAMP, value_fn=( lambda data: datetime.fromtimestamp( data.lightning_strike_last_epoch, tz=UTC ) if data.lightning_strike_last_epoch is not None else None ), ), ) async def async_setup_entry( hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback, ) -> None: """Set up WeatherFlow sensors based on a config entry.""" coordinator: WeatherFlowCloudDataUpdateCoordinator = hass.data[DOMAIN][ entry.entry_id ] async_add_entities( WeatherFlowCloudSensor(coordinator, sensor_description, station_id) for station_id in coordinator.data for sensor_description in WF_SENSORS ) class WeatherFlowCloudSensor(WeatherFlowCloudEntity, SensorEntity): """Implementation of a WeatherFlow sensor.""" entity_description: WeatherFlowCloudSensorEntityDescription def __init__( self, coordinator: WeatherFlowCloudDataUpdateCoordinator, description: WeatherFlowCloudSensorEntityDescription, station_id: int, ) -> None: """Initialize the sensor.""" # Initialize the Entity Class super().__init__(coordinator, station_id) self.entity_description = description self._attr_unique_id = f"{station_id}_{description.key}" @property def native_value(self) -> StateType | datetime: """Return the state of the sensor.""" return self.entity_description.value_fn(self.station.observation.obs[0])