mirror of
https://github.com/jdejaegh/irm-kmi-ha.git
synced 2025-06-27 11:39:26 +02:00
229 lines
No EOL
9.1 KiB
Python
229 lines
No EOL
9.1 KiB
Python
"""Sensor for pollen from the IRM KMI"""
|
|
import logging
|
|
from datetime import datetime
|
|
|
|
from homeassistant.components import sensor
|
|
from homeassistant.components.sensor import SensorDeviceClass, SensorEntity
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
|
from homeassistant.util import dt
|
|
|
|
from custom_components.irm_kmi import DOMAIN, IrmKmiCoordinator
|
|
from custom_components.irm_kmi.const import POLLEN_NAMES, POLLEN_TO_ICON_MAP, CURRENT_WEATHER_SENSOR_UNITS, \
|
|
CURRENT_WEATHER_SENSOR_CLASS, CURRENT_WEATHER_SENSORS, CURRENT_WEATHER_SENSOR_ICON
|
|
from custom_components.irm_kmi.data import IrmKmiForecast
|
|
from custom_components.irm_kmi.pollen import PollenParser
|
|
from custom_components.irm_kmi.radar_data import IrmKmiRadarForecast
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback):
|
|
"""Set up the sensor platform"""
|
|
coordinator = hass.data[DOMAIN][entry.entry_id]
|
|
async_add_entities([IrmKmiPollen(coordinator, entry, pollen.lower()) for pollen in POLLEN_NAMES])
|
|
async_add_entities([IrmKmiCurrentWeather(coordinator, entry, name) for name in CURRENT_WEATHER_SENSORS])
|
|
async_add_entities([IrmKmiNextWarning(coordinator, entry),
|
|
IrmKmiCurrentRainfall(coordinator, entry)])
|
|
|
|
if coordinator.data.get('country') != 'NL':
|
|
async_add_entities([IrmKmiNextSunMove(coordinator, entry, move) for move in ['sunset', 'sunrise']])
|
|
|
|
|
|
class IrmKmiPollen(CoordinatorEntity, SensorEntity):
|
|
"""Representation of a pollen sensor"""
|
|
_attr_has_entity_name = True
|
|
_attr_device_class = SensorDeviceClass.ENUM
|
|
_attr_attribution = "Weather data from the Royal Meteorological Institute of Belgium meteo.be"
|
|
|
|
def __init__(self,
|
|
coordinator: IrmKmiCoordinator,
|
|
entry: ConfigEntry,
|
|
pollen: str
|
|
) -> None:
|
|
super().__init__(coordinator)
|
|
SensorEntity.__init__(self)
|
|
self._attr_unique_id = f"{entry.entry_id}-pollen-{pollen}"
|
|
self.entity_id = sensor.ENTITY_ID_FORMAT.format(f"{str(entry.title).lower()}_{pollen}_level")
|
|
self._attr_options = PollenParser.get_option_values()
|
|
self._attr_device_info = coordinator.shared_device_info
|
|
self._pollen = pollen
|
|
self._attr_translation_key = f"pollen_{pollen}"
|
|
self._attr_icon = POLLEN_TO_ICON_MAP[pollen]
|
|
|
|
@property
|
|
def native_value(self) -> str | None:
|
|
"""Return the state of the sensor."""
|
|
return self.coordinator.data.get('pollen', {}).get(self._pollen, None)
|
|
|
|
|
|
class IrmKmiNextWarning(CoordinatorEntity, SensorEntity):
|
|
"""Representation of the next weather warning"""
|
|
|
|
_attr_has_entity_name = True
|
|
_attr_device_class = SensorDeviceClass.TIMESTAMP
|
|
_attr_attribution = "Weather data from the Royal Meteorological Institute of Belgium meteo.be"
|
|
|
|
def __init__(self,
|
|
coordinator: IrmKmiCoordinator,
|
|
entry: ConfigEntry,
|
|
) -> None:
|
|
super().__init__(coordinator)
|
|
SensorEntity.__init__(self)
|
|
self._attr_unique_id = f"{entry.entry_id}-next-warning"
|
|
self.entity_id = sensor.ENTITY_ID_FORMAT.format(f"{str(entry.title).lower()}_next_warning")
|
|
self._attr_device_info = coordinator.shared_device_info
|
|
self._attr_translation_key = f"next_warning"
|
|
|
|
@property
|
|
def native_value(self) -> datetime | None:
|
|
"""Return the timestamp for the start of the next warning. Is None when no future warning are available"""
|
|
if self.coordinator.data.get('warnings') is None:
|
|
return None
|
|
|
|
now = dt.now()
|
|
earliest_next = None
|
|
for item in self.coordinator.data.get('warnings'):
|
|
if now < item.get('starts_at'):
|
|
if earliest_next is None:
|
|
earliest_next = item.get('starts_at')
|
|
else:
|
|
earliest_next = min(earliest_next, item.get('starts_at'))
|
|
|
|
return earliest_next
|
|
|
|
@property
|
|
def extra_state_attributes(self) -> dict:
|
|
"""Return the attributes related to all the future warnings."""
|
|
now = dt.now()
|
|
attrs = {"next_warnings": [w for w in self.coordinator.data.get('warnings', []) if now < w.get('starts_at')]}
|
|
|
|
attrs["next_warnings_friendly_names"] = ", ".join(
|
|
[warning['friendly_name'] for warning in attrs['next_warnings'] if warning['friendly_name'] != ''])
|
|
|
|
return attrs
|
|
|
|
|
|
class IrmKmiNextSunMove(CoordinatorEntity, SensorEntity):
|
|
"""Representation of the next sunrise or sunset"""
|
|
|
|
_attr_has_entity_name = True
|
|
_attr_device_class = SensorDeviceClass.TIMESTAMP
|
|
_attr_attribution = "Weather data from the Royal Meteorological Institute of Belgium meteo.be"
|
|
|
|
def __init__(self,
|
|
coordinator: IrmKmiCoordinator,
|
|
entry: ConfigEntry,
|
|
move: str) -> None:
|
|
assert move in ['sunset', 'sunrise']
|
|
super().__init__(coordinator)
|
|
SensorEntity.__init__(self)
|
|
self._attr_unique_id = f"{entry.entry_id}-next-{move}"
|
|
self.entity_id = sensor.ENTITY_ID_FORMAT.format(f"{str(entry.title).lower()}_next_{move}")
|
|
self._attr_device_info = coordinator.shared_device_info
|
|
self._attr_translation_key = f"next_{move}"
|
|
self._move: str = move
|
|
self._attr_icon = 'mdi:weather-sunset-down' if move == 'sunset' else 'mdi:weather-sunset-up'
|
|
|
|
@property
|
|
def native_value(self) -> datetime | None:
|
|
"""Return the timestamp for the next sunrise or sunset"""
|
|
now = dt.now()
|
|
data: list[IrmKmiForecast] = self.coordinator.data.get('daily_forecast')
|
|
|
|
upcoming = [datetime.fromisoformat(f.get(self._move)) for f in data
|
|
if f.get(self._move) is not None and datetime.fromisoformat(f.get(self._move)) >= now]
|
|
|
|
if len(upcoming) > 0:
|
|
return upcoming[0]
|
|
return None
|
|
|
|
|
|
class IrmKmiCurrentWeather(CoordinatorEntity, SensorEntity):
|
|
"""Representation of a current weather sensor"""
|
|
|
|
_attr_has_entity_name = True
|
|
_attr_attribution = "Weather data from the Royal Meteorological Institute of Belgium meteo.be"
|
|
|
|
def __init__(self,
|
|
coordinator: IrmKmiCoordinator,
|
|
entry: ConfigEntry,
|
|
sensor_name: str) -> None:
|
|
super().__init__(coordinator)
|
|
SensorEntity.__init__(self)
|
|
self._attr_unique_id = f"{entry.entry_id}-current-{sensor_name}"
|
|
self.entity_id = sensor.ENTITY_ID_FORMAT.format(f"{str(entry.title).lower()}_current_{sensor_name}")
|
|
self._attr_device_info = coordinator.shared_device_info
|
|
self._attr_translation_key = f"current_{sensor_name}"
|
|
self._sensor_name: str = sensor_name
|
|
|
|
@property
|
|
def native_value(self) -> float | None:
|
|
"""Return the current value of the sensor"""
|
|
return self.coordinator.data.get('current_weather', {}).get(self._sensor_name, None)
|
|
|
|
@property
|
|
def native_unit_of_measurement(self) -> str | None:
|
|
return CURRENT_WEATHER_SENSOR_UNITS[self._sensor_name]
|
|
|
|
@property
|
|
def device_class(self) -> SensorDeviceClass | None:
|
|
return CURRENT_WEATHER_SENSOR_CLASS[self._sensor_name]
|
|
|
|
@property
|
|
def icon(self) -> str | None:
|
|
return CURRENT_WEATHER_SENSOR_ICON[self._sensor_name]
|
|
|
|
|
|
class IrmKmiCurrentRainfall(CoordinatorEntity, SensorEntity):
|
|
"""Representation of a current rainfall sensor"""
|
|
|
|
_attr_has_entity_name = True
|
|
_attr_attribution = "Weather data from the Royal Meteorological Institute of Belgium meteo.be"
|
|
|
|
def __init__(self,
|
|
coordinator: IrmKmiCoordinator,
|
|
entry: ConfigEntry) -> None:
|
|
super().__init__(coordinator)
|
|
SensorEntity.__init__(self)
|
|
self._attr_unique_id = f"{entry.entry_id}-current-rainfall"
|
|
self.entity_id = sensor.ENTITY_ID_FORMAT.format(f"{str(entry.title).lower()}_current_rainfall")
|
|
self._attr_device_info = coordinator.shared_device_info
|
|
self._attr_translation_key = "current_rainfall"
|
|
self._attr_icon = 'mdi:weather-pouring'
|
|
|
|
def _current_forecast(self) -> IrmKmiRadarForecast | None:
|
|
now = dt.now()
|
|
forecasts = self.coordinator.data.get('radar_forecast', None)
|
|
|
|
if forecasts is None:
|
|
return None
|
|
|
|
prev = forecasts[0]
|
|
for f in forecasts:
|
|
if datetime.fromisoformat(f.get('datetime')) > now:
|
|
return prev
|
|
prev = f
|
|
|
|
return forecasts[-1]
|
|
|
|
@property
|
|
def native_value(self) -> float | None:
|
|
"""Return the current value of the sensor"""
|
|
current = self._current_forecast()
|
|
|
|
if current is None:
|
|
return None
|
|
|
|
return current.get('native_precipitation', None)
|
|
|
|
@property
|
|
def native_unit_of_measurement(self) -> str | None:
|
|
current = self._current_forecast()
|
|
|
|
if current is None:
|
|
return None
|
|
|
|
return current.get('unit', None) |