2019-02-15 09:57:47 +00:00
|
|
|
"""Support for Meteo-France weather service."""
|
2019-02-14 15:42:03 +00:00
|
|
|
import logging
|
2020-08-01 20:56:00 +00:00
|
|
|
import time
|
2020-02-04 21:37:59 +00:00
|
|
|
|
2019-02-14 13:40:27 +00:00
|
|
|
from homeassistant.components.weather import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_FORECAST_CONDITION,
|
2020-08-01 20:56:00 +00:00
|
|
|
ATTR_FORECAST_PRECIPITATION,
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_FORECAST_TEMP,
|
|
|
|
ATTR_FORECAST_TEMP_LOW,
|
|
|
|
ATTR_FORECAST_TIME,
|
2020-08-01 20:56:00 +00:00
|
|
|
ATTR_FORECAST_WIND_BEARING,
|
|
|
|
ATTR_FORECAST_WIND_SPEED,
|
2019-07-31 19:25:30 +00:00
|
|
|
WeatherEntity,
|
|
|
|
)
|
2020-02-04 21:37:59 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2020-08-01 20:56:00 +00:00
|
|
|
from homeassistant.const import CONF_MODE, TEMP_CELSIUS
|
2020-02-04 21:37:59 +00:00
|
|
|
from homeassistant.helpers.typing import HomeAssistantType
|
2020-08-01 20:56:00 +00:00
|
|
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
|
|
|
|
|
|
|
from .const import (
|
|
|
|
ATTRIBUTION,
|
|
|
|
CONDITION_CLASSES,
|
|
|
|
COORDINATOR_FORECAST,
|
|
|
|
DOMAIN,
|
|
|
|
FORECAST_MODE_DAILY,
|
|
|
|
FORECAST_MODE_HOURLY,
|
|
|
|
)
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2019-02-14 13:40:27 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2020-08-01 20:56:00 +00:00
|
|
|
def format_condition(condition: str):
|
|
|
|
"""Return condition from dict CONDITION_CLASSES."""
|
|
|
|
for key, value in CONDITION_CLASSES.items():
|
|
|
|
if condition in value:
|
|
|
|
return key
|
|
|
|
return condition
|
|
|
|
|
|
|
|
|
2020-02-04 21:37:59 +00:00
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities
|
|
|
|
) -> None:
|
2019-02-14 13:40:27 +00:00
|
|
|
"""Set up the Meteo-France weather platform."""
|
2020-08-01 20:56:00 +00:00
|
|
|
coordinator = hass.data[DOMAIN][entry.entry_id][COORDINATOR_FORECAST]
|
|
|
|
|
|
|
|
async_add_entities(
|
|
|
|
[
|
|
|
|
MeteoFranceWeather(
|
|
|
|
coordinator, entry.options.get(CONF_MODE, FORECAST_MODE_DAILY),
|
|
|
|
)
|
|
|
|
],
|
|
|
|
True,
|
|
|
|
)
|
|
|
|
_LOGGER.debug(
|
|
|
|
"Weather entity (%s) added for %s.",
|
|
|
|
entry.options.get(CONF_MODE, FORECAST_MODE_DAILY),
|
|
|
|
coordinator.data.position["name"],
|
|
|
|
)
|
2019-02-14 13:40:27 +00:00
|
|
|
|
|
|
|
|
|
|
|
class MeteoFranceWeather(WeatherEntity):
|
|
|
|
"""Representation of a weather condition."""
|
|
|
|
|
2020-08-01 20:56:00 +00:00
|
|
|
def __init__(self, coordinator: DataUpdateCoordinator, mode: str):
|
2019-02-14 13:40:27 +00:00
|
|
|
"""Initialise the platform with a data instance and station name."""
|
2020-08-01 20:56:00 +00:00
|
|
|
self.coordinator = coordinator
|
|
|
|
self._city_name = self.coordinator.data.position["name"]
|
|
|
|
self._mode = mode
|
|
|
|
self._unique_id = f"{self.coordinator.data.position['lat']},{self.coordinator.data.position['lon']}"
|
2019-02-14 13:40:27 +00:00
|
|
|
|
2020-08-01 20:56:00 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return the unique id of the sensor."""
|
|
|
|
return self._unique_id
|
2019-02-14 13:40:27 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
2020-08-01 20:56:00 +00:00
|
|
|
return self._city_name
|
2020-02-04 21:37:59 +00:00
|
|
|
|
2019-02-14 13:40:27 +00:00
|
|
|
@property
|
|
|
|
def condition(self):
|
|
|
|
"""Return the current condition."""
|
2020-08-01 20:56:00 +00:00
|
|
|
return format_condition(
|
|
|
|
self.coordinator.data.current_forecast["weather"]["desc"]
|
|
|
|
)
|
2019-02-14 13:40:27 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def temperature(self):
|
2019-02-15 09:57:47 +00:00
|
|
|
"""Return the temperature."""
|
2020-08-01 20:56:00 +00:00
|
|
|
return self.coordinator.data.current_forecast["T"]["value"]
|
2019-02-14 13:40:27 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def temperature_unit(self):
|
|
|
|
"""Return the unit of measurement."""
|
|
|
|
return TEMP_CELSIUS
|
|
|
|
|
2020-08-01 20:56:00 +00:00
|
|
|
@property
|
|
|
|
def pressure(self):
|
|
|
|
"""Return the pressure."""
|
|
|
|
return self.coordinator.data.current_forecast["sea_level"]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def humidity(self):
|
|
|
|
"""Return the humidity."""
|
|
|
|
return self.coordinator.data.current_forecast["humidity"]
|
|
|
|
|
2019-02-14 13:40:27 +00:00
|
|
|
@property
|
|
|
|
def wind_speed(self):
|
|
|
|
"""Return the wind speed."""
|
2020-08-01 20:56:00 +00:00
|
|
|
# convert from API m/s to km/h
|
|
|
|
return round(self.coordinator.data.current_forecast["wind"]["speed"] * 3.6)
|
2019-02-14 13:40:27 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def wind_bearing(self):
|
|
|
|
"""Return the wind bearing."""
|
2020-08-01 20:56:00 +00:00
|
|
|
wind_bearing = self.coordinator.data.current_forecast["wind"]["direction"]
|
|
|
|
if wind_bearing != -1:
|
|
|
|
return wind_bearing
|
2019-02-14 13:40:27 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def forecast(self):
|
|
|
|
"""Return the forecast."""
|
|
|
|
forecast_data = []
|
2020-08-01 20:56:00 +00:00
|
|
|
|
|
|
|
if self._mode == FORECAST_MODE_HOURLY:
|
|
|
|
today = time.time()
|
|
|
|
for forecast in self.coordinator.data.forecast:
|
|
|
|
# Can have data in the past
|
|
|
|
if forecast["dt"] < today:
|
|
|
|
_LOGGER.debug(
|
|
|
|
"remove forecast in the past: %s %s", self._mode, forecast
|
|
|
|
)
|
|
|
|
continue
|
|
|
|
forecast_data.append(
|
|
|
|
{
|
|
|
|
ATTR_FORECAST_TIME: self.coordinator.data.timestamp_to_locale_time(
|
|
|
|
forecast["dt"]
|
|
|
|
),
|
|
|
|
ATTR_FORECAST_CONDITION: format_condition(
|
|
|
|
forecast["weather"]["desc"]
|
|
|
|
),
|
|
|
|
ATTR_FORECAST_TEMP: forecast["T"]["value"],
|
|
|
|
ATTR_FORECAST_PRECIPITATION: forecast["rain"].get("1h"),
|
|
|
|
ATTR_FORECAST_WIND_SPEED: forecast["wind"]["speed"],
|
|
|
|
ATTR_FORECAST_WIND_BEARING: forecast["wind"]["direction"]
|
|
|
|
if forecast["wind"]["direction"] != -1
|
|
|
|
else None,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
for forecast in self.coordinator.data.daily_forecast:
|
|
|
|
# stop when we don't have a weather condition (can happen around last days of forcast, max 14)
|
|
|
|
if not forecast.get("weather12H"):
|
|
|
|
break
|
|
|
|
forecast_data.append(
|
|
|
|
{
|
|
|
|
ATTR_FORECAST_TIME: self.coordinator.data.timestamp_to_locale_time(
|
|
|
|
forecast["dt"]
|
|
|
|
),
|
|
|
|
ATTR_FORECAST_CONDITION: format_condition(
|
|
|
|
forecast["weather12H"]["desc"]
|
|
|
|
),
|
|
|
|
ATTR_FORECAST_TEMP: forecast["T"]["max"],
|
|
|
|
ATTR_FORECAST_TEMP_LOW: forecast["T"]["min"],
|
|
|
|
ATTR_FORECAST_PRECIPITATION: forecast["precipitation"]["24h"],
|
|
|
|
}
|
|
|
|
)
|
2019-02-14 13:40:27 +00:00
|
|
|
return forecast_data
|
|
|
|
|
2020-08-01 20:56:00 +00:00
|
|
|
@property
|
|
|
|
def attribution(self):
|
|
|
|
"""Return the attribution."""
|
|
|
|
return ATTRIBUTION
|
|
|
|
|
|
|
|
@property
|
|
|
|
def available(self):
|
|
|
|
"""Return if state is available."""
|
|
|
|
return self.coordinator.last_update_success
|
2019-06-19 21:39:13 +00:00
|
|
|
|
|
|
|
@property
|
2020-08-01 20:56:00 +00:00
|
|
|
def should_poll(self) -> bool:
|
|
|
|
"""No polling needed."""
|
|
|
|
return False
|
|
|
|
|
|
|
|
async def async_update(self):
|
|
|
|
"""Only used by the generic entity update service."""
|
|
|
|
if not self.enabled:
|
|
|
|
return
|
|
|
|
|
|
|
|
await self.coordinator.async_request_refresh()
|
|
|
|
|
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Subscribe to updates."""
|
|
|
|
self.async_on_remove(
|
|
|
|
self.coordinator.async_add_listener(self.async_write_ha_state)
|
|
|
|
)
|