hass-core/homeassistant/components/weatherflow_cloud/sensor.py
Jeef de18be235d
Add Sensors to Weatherflow Cloud (#111651)
* continue

* Rebase dev

* signle function to generate attr_entity info

* rewrite icon determination as an if block

* handling PR

* Removing wind sensors for now - separate future PR

* ruff

* Update coordinator.py

Thought i already did this

* Update sensor.py

* Update icons.json

* Update sensor.py

* Update homeassistant/components/weatherflow_cloud/entity.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* working on a unified entity

* working on a unified entity

* sensor refactor

* addressing entity comment

* Update homeassistant/components/weatherflow_cloud/entity.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* Update homeassistant/components/weatherflow_cloud/sensor.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* doc

* pr comments again

* fixing PR

* fixing PR

* applying entity class in sensor

* Update homeassistant/components/weatherflow_cloud/sensor.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* Cleaning up weather.py

* station id cleanup for weather class

* rewrite adding sensors the correct way

* Adding snapshot testing

* snapshot update

* added total class

* updated snapshots

* minor tweak

* snapshot away

* adding more coverage

* switch back to total

* Apply suggestions from code review

* Apply suggestions from code review

* Apply suggestions from code review

---------

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
2024-07-19 10:30:01 +02:00

208 lines
7.6 KiB
Python

"""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
),
),
)
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
]
stations = coordinator.data.keys()
async_add_entities(
WeatherFlowCloudSensor(coordinator, sensor_description, station_id)
for station_id in stations
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])