2019-02-14 15:42:03 +00:00
|
|
|
"""Support for Buienradar.nl weather service."""
|
2024-03-08 18:15:59 +00:00
|
|
|
|
2017-06-05 06:48:11 +00:00
|
|
|
import logging
|
2018-02-11 17:20:28 +00:00
|
|
|
|
2019-10-18 23:02:54 +00:00
|
|
|
from buienradar.constants import (
|
|
|
|
CONDCODE,
|
|
|
|
CONDITION,
|
|
|
|
DATETIME,
|
|
|
|
MAX_TEMP,
|
|
|
|
MIN_TEMP,
|
|
|
|
RAIN,
|
|
|
|
WINDAZIMUTH,
|
|
|
|
WINDSPEED,
|
|
|
|
)
|
2018-02-11 17:20:28 +00:00
|
|
|
|
2017-06-05 06:48:11 +00:00
|
|
|
from homeassistant.components.weather import (
|
2020-11-20 20:04:03 +00:00
|
|
|
ATTR_CONDITION_CLOUDY,
|
|
|
|
ATTR_CONDITION_EXCEPTIONAL,
|
|
|
|
ATTR_CONDITION_FOG,
|
|
|
|
ATTR_CONDITION_HAIL,
|
|
|
|
ATTR_CONDITION_LIGHTNING,
|
|
|
|
ATTR_CONDITION_LIGHTNING_RAINY,
|
|
|
|
ATTR_CONDITION_PARTLYCLOUDY,
|
|
|
|
ATTR_CONDITION_POURING,
|
|
|
|
ATTR_CONDITION_RAINY,
|
|
|
|
ATTR_CONDITION_SNOWY,
|
|
|
|
ATTR_CONDITION_SNOWY_RAINY,
|
|
|
|
ATTR_CONDITION_SUNNY,
|
|
|
|
ATTR_CONDITION_WINDY,
|
|
|
|
ATTR_CONDITION_WINDY_VARIANT,
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_FORECAST_CONDITION,
|
2022-06-27 23:57:46 +00:00
|
|
|
ATTR_FORECAST_NATIVE_PRECIPITATION,
|
|
|
|
ATTR_FORECAST_NATIVE_TEMP,
|
|
|
|
ATTR_FORECAST_NATIVE_TEMP_LOW,
|
|
|
|
ATTR_FORECAST_NATIVE_WIND_SPEED,
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_FORECAST_TIME,
|
|
|
|
ATTR_FORECAST_WIND_BEARING,
|
2023-08-16 20:19:22 +00:00
|
|
|
Forecast,
|
2019-10-18 23:02:54 +00:00
|
|
|
WeatherEntity,
|
2023-08-16 20:19:22 +00:00
|
|
|
WeatherEntityFeature,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2021-05-04 11:49:16 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2022-06-27 23:57:46 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_LATITUDE,
|
|
|
|
CONF_LONGITUDE,
|
|
|
|
CONF_NAME,
|
2023-04-16 12:20:54 +00:00
|
|
|
Platform,
|
2022-11-29 17:13:05 +00:00
|
|
|
UnitOfLength,
|
|
|
|
UnitOfPrecipitationDepth,
|
|
|
|
UnitOfPressure,
|
|
|
|
UnitOfSpeed,
|
|
|
|
UnitOfTemperature,
|
2022-06-27 23:57:46 +00:00
|
|
|
)
|
2023-08-15 11:28:43 +00:00
|
|
|
from homeassistant.core import HomeAssistant, callback
|
2021-05-04 11:49:16 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2017-06-05 06:48:11 +00:00
|
|
|
|
2019-03-21 05:56:46 +00:00
|
|
|
# Reuse data and API logic from the sensor implementation
|
2021-05-04 11:49:16 +00:00
|
|
|
from .const import DEFAULT_TIMEFRAME, DOMAIN
|
2019-12-08 13:59:21 +00:00
|
|
|
from .util import BrData
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2017-06-05 06:48:11 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_FORECAST = "forecast"
|
2017-06-05 06:48:11 +00:00
|
|
|
|
2021-05-04 11:49:16 +00:00
|
|
|
DATA_CONDITION = "buienradar_condition"
|
2017-08-16 06:07:04 +00:00
|
|
|
|
|
|
|
CONDITION_CLASSES = {
|
2021-05-04 11:49:16 +00:00
|
|
|
ATTR_CONDITION_CLOUDY: ("c", "p"),
|
|
|
|
ATTR_CONDITION_FOG: ("d", "n"),
|
|
|
|
ATTR_CONDITION_HAIL: (),
|
|
|
|
ATTR_CONDITION_LIGHTNING: ("g",),
|
|
|
|
ATTR_CONDITION_LIGHTNING_RAINY: ("s",),
|
|
|
|
ATTR_CONDITION_PARTLYCLOUDY: (
|
|
|
|
"b",
|
|
|
|
"j",
|
|
|
|
"o",
|
|
|
|
"r",
|
|
|
|
),
|
|
|
|
ATTR_CONDITION_POURING: ("l", "q"),
|
|
|
|
ATTR_CONDITION_RAINY: ("f", "h", "k", "m"),
|
|
|
|
ATTR_CONDITION_SNOWY: ("u", "i", "v", "t"),
|
|
|
|
ATTR_CONDITION_SNOWY_RAINY: ("w",),
|
|
|
|
ATTR_CONDITION_SUNNY: ("a",),
|
|
|
|
ATTR_CONDITION_WINDY: (),
|
|
|
|
ATTR_CONDITION_WINDY_VARIANT: (),
|
|
|
|
ATTR_CONDITION_EXCEPTIONAL: (),
|
2017-08-16 06:07:04 +00:00
|
|
|
}
|
2023-08-15 16:15:23 +00:00
|
|
|
CONDITION_MAP = {
|
|
|
|
cond_code: cond_ha
|
|
|
|
for cond_ha, cond_codes in CONDITION_CLASSES.items()
|
|
|
|
for cond_code in cond_codes
|
|
|
|
}
|
2017-08-16 06:07:04 +00:00
|
|
|
|
2021-05-04 11:49:16 +00:00
|
|
|
|
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
|
|
|
) -> None:
|
2017-06-05 06:48:11 +00:00
|
|
|
"""Set up the buienradar platform."""
|
2021-05-04 11:49:16 +00:00
|
|
|
config = entry.data
|
|
|
|
|
2017-06-05 06:48:11 +00:00
|
|
|
latitude = config.get(CONF_LATITUDE, hass.config.latitude)
|
|
|
|
longitude = config.get(CONF_LONGITUDE, hass.config.longitude)
|
|
|
|
|
|
|
|
if None in (latitude, longitude):
|
|
|
|
_LOGGER.error("Latitude or longitude not set in Home Assistant config")
|
2021-05-04 11:49:16 +00:00
|
|
|
return
|
2017-06-05 06:48:11 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
coordinates = {CONF_LATITUDE: float(latitude), CONF_LONGITUDE: float(longitude)}
|
2017-06-05 06:48:11 +00:00
|
|
|
|
2023-08-15 11:28:43 +00:00
|
|
|
# create weather entity:
|
|
|
|
_LOGGER.debug("Initializing buienradar weather: coordinates %s", coordinates)
|
|
|
|
entities = [BrWeather(config, coordinates)]
|
|
|
|
|
2017-06-05 06:48:11 +00:00
|
|
|
# create weather data:
|
2023-08-15 11:28:43 +00:00
|
|
|
data = BrData(hass, coordinates, DEFAULT_TIMEFRAME, entities)
|
2023-04-16 12:20:54 +00:00
|
|
|
hass.data[DOMAIN][entry.entry_id][Platform.WEATHER] = data
|
2023-08-15 16:15:23 +00:00
|
|
|
await data.async_update()
|
2017-08-16 06:07:04 +00:00
|
|
|
|
2023-08-15 11:28:43 +00:00
|
|
|
async_add_entities(entities)
|
2017-06-05 06:48:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
class BrWeather(WeatherEntity):
|
|
|
|
"""Representation of a weather condition."""
|
|
|
|
|
2022-11-29 17:13:05 +00:00
|
|
|
_attr_native_precipitation_unit = UnitOfPrecipitationDepth.MILLIMETERS
|
|
|
|
_attr_native_pressure_unit = UnitOfPressure.HPA
|
|
|
|
_attr_native_temperature_unit = UnitOfTemperature.CELSIUS
|
|
|
|
_attr_native_visibility_unit = UnitOfLength.METERS
|
|
|
|
_attr_native_wind_speed_unit = UnitOfSpeed.METERS_PER_SECOND
|
2023-08-15 11:28:43 +00:00
|
|
|
_attr_should_poll = False
|
2023-08-16 20:19:22 +00:00
|
|
|
_attr_supported_features = WeatherEntityFeature.FORECAST_DAILY
|
2021-07-19 19:22:20 +00:00
|
|
|
|
2024-09-03 14:50:30 +00:00
|
|
|
def __init__(self, config, coordinates) -> None:
|
2021-07-19 19:22:20 +00:00
|
|
|
"""Initialize the platform with a data instance and station name."""
|
2021-05-04 11:49:16 +00:00
|
|
|
self._stationname = config.get(CONF_NAME, "Buienradar")
|
2023-08-15 11:28:43 +00:00
|
|
|
self._attr_name = self._stationname or f"BR {'(unknown station)'}"
|
2017-06-05 06:48:11 +00:00
|
|
|
|
2024-03-08 17:44:42 +00:00
|
|
|
self._attr_unique_id = (
|
|
|
|
f"{coordinates[CONF_LATITUDE]:2.6f}{coordinates[CONF_LONGITUDE]:2.6f}"
|
2020-08-01 07:13:17 +00:00
|
|
|
)
|
2024-03-27 15:51:29 +00:00
|
|
|
self._forecast: list | None = None
|
2020-08-01 07:13:17 +00:00
|
|
|
|
2023-08-15 11:28:43 +00:00
|
|
|
@callback
|
|
|
|
def data_updated(self, data: BrData) -> None:
|
|
|
|
"""Update data."""
|
|
|
|
self._attr_attribution = data.attribution
|
|
|
|
self._attr_condition = self._calc_condition(data)
|
2024-03-27 15:51:29 +00:00
|
|
|
self._forecast = self._calc_forecast(data)
|
2023-08-15 11:28:43 +00:00
|
|
|
self._attr_humidity = data.humidity
|
|
|
|
self._attr_name = (
|
|
|
|
self._stationname or f"BR {data.stationname or '(unknown station)'}"
|
|
|
|
)
|
|
|
|
self._attr_native_pressure = data.pressure
|
|
|
|
self._attr_native_temperature = data.temperature
|
|
|
|
self._attr_native_visibility = data.visibility
|
|
|
|
self._attr_native_wind_speed = data.wind_speed
|
|
|
|
self._attr_wind_bearing = data.wind_bearing
|
2023-08-15 16:15:23 +00:00
|
|
|
|
|
|
|
if not self.hass:
|
|
|
|
return
|
2023-08-15 11:28:43 +00:00
|
|
|
self.async_write_ha_state()
|
2023-08-16 20:19:22 +00:00
|
|
|
assert self.platform.config_entry
|
|
|
|
self.platform.config_entry.async_create_task(
|
|
|
|
self.hass, self.async_update_listeners(("daily",))
|
|
|
|
)
|
2023-08-15 11:28:43 +00:00
|
|
|
|
|
|
|
def _calc_condition(self, data: BrData):
|
2017-08-16 06:07:04 +00:00
|
|
|
"""Return the current condition."""
|
2023-08-15 16:15:23 +00:00
|
|
|
if data.condition and (ccode := data.condition.get(CONDCODE)):
|
|
|
|
return CONDITION_MAP.get(ccode)
|
2023-08-15 11:28:43 +00:00
|
|
|
return None
|
2017-08-16 06:07:04 +00:00
|
|
|
|
2023-08-15 11:28:43 +00:00
|
|
|
def _calc_forecast(self, data: BrData):
|
2017-08-16 06:07:04 +00:00
|
|
|
"""Return the forecast array."""
|
2019-07-28 20:08:20 +00:00
|
|
|
fcdata_out = []
|
|
|
|
|
2023-08-15 11:28:43 +00:00
|
|
|
if not data.forecast:
|
2019-07-28 20:08:20 +00:00
|
|
|
return None
|
|
|
|
|
2023-08-15 11:28:43 +00:00
|
|
|
for data_in in data.forecast:
|
2019-07-28 20:08:20 +00:00
|
|
|
# remap keys from external library to
|
|
|
|
# keys understood by the weather component:
|
2023-08-15 16:15:23 +00:00
|
|
|
condcode = data_in.get(CONDITION, {}).get(CONDCODE)
|
2019-07-28 20:08:20 +00:00
|
|
|
data_out = {
|
2021-03-23 19:19:47 +00:00
|
|
|
ATTR_FORECAST_TIME: data_in.get(DATETIME).isoformat(),
|
2023-08-15 16:15:23 +00:00
|
|
|
ATTR_FORECAST_CONDITION: CONDITION_MAP.get(condcode),
|
2022-06-27 23:57:46 +00:00
|
|
|
ATTR_FORECAST_NATIVE_TEMP_LOW: data_in.get(MIN_TEMP),
|
|
|
|
ATTR_FORECAST_NATIVE_TEMP: data_in.get(MAX_TEMP),
|
|
|
|
ATTR_FORECAST_NATIVE_PRECIPITATION: data_in.get(RAIN),
|
2019-07-28 20:08:20 +00:00
|
|
|
ATTR_FORECAST_WIND_BEARING: data_in.get(WINDAZIMUTH),
|
2022-06-27 23:57:46 +00:00
|
|
|
ATTR_FORECAST_NATIVE_WIND_SPEED: data_in.get(WINDSPEED),
|
2019-07-28 20:08:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fcdata_out.append(data_out)
|
|
|
|
|
|
|
|
return fcdata_out
|
2023-08-16 20:19:22 +00:00
|
|
|
|
|
|
|
async def async_forecast_daily(self) -> list[Forecast] | None:
|
|
|
|
"""Return the daily forecast in native units."""
|
2024-03-27 15:51:29 +00:00
|
|
|
return self._forecast
|