1
0
mirror of https://github.com/home-assistant/core.git synced 2026-07-04 05:05:38 +01:00
Files
core/homeassistant/components/opensensemap/sensor.py
T
2026-06-11 14:57:43 +02:00

157 lines
5.6 KiB
Python

"""Support for openSenseMap sensors."""
from collections.abc import Callable
from dataclasses import dataclass
from homeassistant.components.sensor import (
SensorDeviceClass,
SensorEntity,
SensorEntityDescription,
SensorStateClass,
)
from homeassistant.const import (
CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
DEGREE,
LIGHT_LUX,
PERCENTAGE,
UnitOfPressure,
UnitOfSpeed,
UnitOfTemperature,
)
from homeassistant.core import HomeAssistant
from homeassistant.helpers.device_registry import DeviceEntryType, DeviceInfo
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
from homeassistant.helpers.update_coordinator import CoordinatorEntity
from .const import CONF_STATION_ID, DOMAIN, INTEGRATION_TITLE
from .coordinator import (
Measurement,
OpenSenseMapConfigEntry,
OpenSenseMapCoordinator,
OpenSenseMapStationData,
)
@dataclass(frozen=True, kw_only=True)
class OpenSenseMapSensorEntityDescription(SensorEntityDescription):
"""Describes openSenseMap sensor entities."""
value_fn: Callable[[OpenSenseMapStationData], Measurement]
SENSOR_DESCRIPTIONS: tuple[OpenSenseMapSensorEntityDescription, ...] = (
OpenSenseMapSensorEntityDescription(
key="pm2_5",
device_class=SensorDeviceClass.PM25,
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
state_class=SensorStateClass.MEASUREMENT,
value_fn=lambda data: data.pm2_5,
),
OpenSenseMapSensorEntityDescription(
key="pm10",
device_class=SensorDeviceClass.PM10,
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
state_class=SensorStateClass.MEASUREMENT,
value_fn=lambda data: data.pm10,
),
OpenSenseMapSensorEntityDescription(
key="pm1_0",
device_class=SensorDeviceClass.PM1,
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
state_class=SensorStateClass.MEASUREMENT,
value_fn=lambda data: data.pm1_0,
),
OpenSenseMapSensorEntityDescription(
key="temperature",
device_class=SensorDeviceClass.TEMPERATURE,
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
state_class=SensorStateClass.MEASUREMENT,
value_fn=lambda data: data.temperature,
),
OpenSenseMapSensorEntityDescription(
key="humidity",
device_class=SensorDeviceClass.HUMIDITY,
native_unit_of_measurement=PERCENTAGE,
state_class=SensorStateClass.MEASUREMENT,
value_fn=lambda data: data.humidity,
),
OpenSenseMapSensorEntityDescription(
key="air_pressure",
device_class=SensorDeviceClass.ATMOSPHERIC_PRESSURE,
native_unit_of_measurement=UnitOfPressure.HPA,
state_class=SensorStateClass.MEASUREMENT,
value_fn=lambda data: data.air_pressure,
),
OpenSenseMapSensorEntityDescription(
key="illuminance",
device_class=SensorDeviceClass.ILLUMINANCE,
native_unit_of_measurement=LIGHT_LUX,
state_class=SensorStateClass.MEASUREMENT,
value_fn=lambda data: data.illuminance,
),
OpenSenseMapSensorEntityDescription(
key="wind_speed",
device_class=SensorDeviceClass.WIND_SPEED,
native_unit_of_measurement=UnitOfSpeed.METERS_PER_SECOND,
state_class=SensorStateClass.MEASUREMENT,
value_fn=lambda data: data.wind_speed,
),
OpenSenseMapSensorEntityDescription(
key="wind_direction",
device_class=SensorDeviceClass.WIND_DIRECTION,
native_unit_of_measurement=DEGREE,
state_class=SensorStateClass.MEASUREMENT_ANGLE,
value_fn=lambda data: data.wind_direction,
),
)
async def async_setup_entry(
hass: HomeAssistant,
entry: OpenSenseMapConfigEntry,
async_add_entities: AddConfigEntryEntitiesCallback,
) -> None:
"""Set up openSenseMap sensors from a config entry."""
coordinator = entry.runtime_data
entities: list[OpenSenseMapSensor] = []
for description in SENSOR_DESCRIPTIONS:
measurement = description.value_fn(coordinator.data)
if measurement.value is None:
continue
native_unit = measurement.unit or description.native_unit_of_measurement
entities.append(OpenSenseMapSensor(coordinator, description, native_unit))
async_add_entities(entities)
class OpenSenseMapSensor(CoordinatorEntity[OpenSenseMapCoordinator], SensorEntity):
"""Sensor entity representing a single measurement from an openSenseMap station."""
_attr_attribution = "Data provided by openSenseMap"
_attr_has_entity_name = True
entity_description: OpenSenseMapSensorEntityDescription
def __init__(
self,
coordinator: OpenSenseMapCoordinator,
description: OpenSenseMapSensorEntityDescription,
native_unit: str | None,
) -> None:
"""Initialize the sensor."""
super().__init__(coordinator)
self.entity_description = description
self._attr_native_unit_of_measurement = native_unit
station_id = coordinator.config_entry.data[CONF_STATION_ID]
self._attr_unique_id = f"{station_id}_{description.key}"
self._attr_device_info = DeviceInfo(
entry_type=DeviceEntryType.SERVICE,
identifiers={(DOMAIN, station_id)},
manufacturer=INTEGRATION_TITLE,
configuration_url=f"https://opensensemap.org/explore/{station_id}",
)
@property
def native_value(self) -> float | str | None:
"""Return the latest value reported by the station."""
return self.entity_description.value_fn(self.coordinator.data).value