2019-02-14 15:42:03 +00:00
|
|
|
"""Support for the OpenWeatherMap (OWM) service."""
|
2016-10-25 04:53:03 +00:00
|
|
|
from datetime import timedelta
|
2017-12-29 09:06:52 +00:00
|
|
|
import logging
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2017-02-20 00:42:12 +00:00
|
|
|
from homeassistant.components.weather import (
|
2018-04-29 15:50:49 +00:00
|
|
|
ATTR_FORECAST_CONDITION, ATTR_FORECAST_PRECIPITATION, ATTR_FORECAST_TEMP,
|
2019-02-14 15:42:03 +00:00
|
|
|
ATTR_FORECAST_TEMP_LOW, ATTR_FORECAST_TIME, ATTR_FORECAST_WIND_BEARING,
|
|
|
|
ATTR_FORECAST_WIND_SPEED, PLATFORM_SCHEMA, WeatherEntity)
|
2017-12-29 09:06:52 +00:00
|
|
|
from homeassistant.const import (
|
2019-02-14 15:42:03 +00:00
|
|
|
CONF_API_KEY, CONF_LATITUDE, CONF_LONGITUDE, CONF_MODE, CONF_NAME,
|
2019-03-24 17:37:31 +00:00
|
|
|
PRESSURE_HPA, PRESSURE_INHG, STATE_UNKNOWN, TEMP_CELSIUS)
|
2016-10-25 04:53:03 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
from homeassistant.util import Throttle
|
2019-03-24 17:37:31 +00:00
|
|
|
from homeassistant.util.pressure import convert as convert_pressure
|
2016-10-25 04:53:03 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
ATTRIBUTION = 'Data provided by OpenWeatherMap'
|
|
|
|
|
2019-02-23 14:52:08 +00:00
|
|
|
FORECAST_MODE = ['hourly', 'daily', 'freedaily']
|
2018-06-10 10:35:10 +00:00
|
|
|
|
2017-12-29 09:06:52 +00:00
|
|
|
DEFAULT_NAME = 'OpenWeatherMap'
|
|
|
|
|
2017-02-20 00:42:12 +00:00
|
|
|
MIN_TIME_BETWEEN_FORECAST_UPDATES = timedelta(minutes=30)
|
2017-12-29 09:06:52 +00:00
|
|
|
MIN_TIME_BETWEEN_UPDATES = timedelta(minutes=10)
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
CONDITION_CLASSES = {
|
2018-06-10 10:35:10 +00:00
|
|
|
'cloudy': [803, 804],
|
2016-10-25 04:53:03 +00:00
|
|
|
'fog': [701, 741],
|
|
|
|
'hail': [906],
|
|
|
|
'lightning': [210, 211, 212, 221],
|
|
|
|
'lightning-rainy': [200, 201, 202, 230, 231, 232],
|
2018-06-10 10:35:10 +00:00
|
|
|
'partlycloudy': [801, 802],
|
2016-10-25 04:53:03 +00:00
|
|
|
'pouring': [504, 314, 502, 503, 522],
|
|
|
|
'rainy': [300, 301, 302, 310, 311, 312, 313, 500, 501, 520, 521],
|
|
|
|
'snowy': [600, 601, 602, 611, 612, 620, 621, 622],
|
|
|
|
'snowy-rainy': [511, 615, 616],
|
|
|
|
'sunny': [800],
|
|
|
|
'windy': [905, 951, 952, 953, 954, 955, 956, 957],
|
|
|
|
'windy-variant': [958, 959, 960, 961],
|
|
|
|
'exceptional': [711, 721, 731, 751, 761, 762, 771, 900, 901, 962, 903,
|
|
|
|
904],
|
|
|
|
}
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Required(CONF_API_KEY): cv.string,
|
|
|
|
vol.Optional(CONF_LATITUDE): cv.latitude,
|
|
|
|
vol.Optional(CONF_LONGITUDE): cv.longitude,
|
2018-06-10 10:35:10 +00:00
|
|
|
vol.Optional(CONF_MODE, default='hourly'): vol.In(FORECAST_MODE),
|
2016-10-25 04:53:03 +00:00
|
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Set up the OpenWeatherMap weather platform."""
|
2016-10-25 04:53:03 +00:00
|
|
|
import pyowm
|
|
|
|
|
|
|
|
longitude = config.get(CONF_LONGITUDE, round(hass.config.longitude, 5))
|
|
|
|
latitude = config.get(CONF_LATITUDE, round(hass.config.latitude, 5))
|
|
|
|
name = config.get(CONF_NAME)
|
2018-06-10 10:35:10 +00:00
|
|
|
mode = config.get(CONF_MODE)
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
owm = pyowm.OWM(config.get(CONF_API_KEY))
|
|
|
|
except pyowm.exceptions.api_call_error.APICallError:
|
|
|
|
_LOGGER.error("Error while connecting to OpenWeatherMap")
|
|
|
|
return False
|
|
|
|
|
2018-06-10 10:35:10 +00:00
|
|
|
data = WeatherData(owm, latitude, longitude, mode)
|
2016-10-25 04:53:03 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities([OpenWeatherMapWeather(
|
2018-06-10 10:35:10 +00:00
|
|
|
name, data, hass.config.units.temperature_unit, mode)], True)
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
class OpenWeatherMapWeather(WeatherEntity):
|
|
|
|
"""Implementation of an OpenWeatherMap sensor."""
|
|
|
|
|
2018-06-10 10:35:10 +00:00
|
|
|
def __init__(self, name, owm, temperature_unit, mode):
|
2016-10-25 04:53:03 +00:00
|
|
|
"""Initialize the sensor."""
|
|
|
|
self._name = name
|
|
|
|
self._owm = owm
|
|
|
|
self._temperature_unit = temperature_unit
|
2018-06-10 10:35:10 +00:00
|
|
|
self._mode = mode
|
2016-11-09 02:57:56 +00:00
|
|
|
self.data = None
|
2017-02-20 00:42:12 +00:00
|
|
|
self.forecast_data = None
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def condition(self):
|
|
|
|
"""Return the current condition."""
|
|
|
|
try:
|
|
|
|
return [k for k, v in CONDITION_CLASSES.items() if
|
|
|
|
self.data.get_weather_code() in v][0]
|
|
|
|
except IndexError:
|
|
|
|
return STATE_UNKNOWN
|
|
|
|
|
|
|
|
@property
|
|
|
|
def temperature(self):
|
|
|
|
"""Return the temperature."""
|
2016-11-05 08:15:59 +00:00
|
|
|
return self.data.get_temperature('celsius').get('temp')
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def temperature_unit(self):
|
|
|
|
"""Return the unit of measurement."""
|
2017-02-20 00:42:12 +00:00
|
|
|
return TEMP_CELSIUS
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def pressure(self):
|
|
|
|
"""Return the pressure."""
|
2019-03-24 17:37:31 +00:00
|
|
|
pressure = self.data.get_pressure().get('press')
|
|
|
|
if self.hass.config.units.name == 'imperial':
|
|
|
|
return round(
|
|
|
|
convert_pressure(pressure, PRESSURE_HPA, PRESSURE_INHG), 2)
|
|
|
|
return pressure
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def humidity(self):
|
|
|
|
"""Return the humidity."""
|
|
|
|
return self.data.get_humidity()
|
|
|
|
|
|
|
|
@property
|
|
|
|
def wind_speed(self):
|
|
|
|
"""Return the wind speed."""
|
2018-10-01 18:38:41 +00:00
|
|
|
if self.hass.config.units.name == 'imperial':
|
|
|
|
return round(self.data.get_wind().get('speed') * 2.24, 2)
|
|
|
|
|
2018-07-31 17:11:29 +00:00
|
|
|
return round(self.data.get_wind().get('speed') * 3.6, 2)
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def wind_bearing(self):
|
|
|
|
"""Return the wind bearing."""
|
2016-11-05 08:15:59 +00:00
|
|
|
return self.data.get_wind().get('deg')
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def attribution(self):
|
|
|
|
"""Return the attribution."""
|
|
|
|
return ATTRIBUTION
|
|
|
|
|
2017-02-20 00:42:12 +00:00
|
|
|
@property
|
|
|
|
def forecast(self):
|
|
|
|
"""Return the forecast array."""
|
2018-01-31 12:05:15 +00:00
|
|
|
data = []
|
2018-10-26 13:50:30 +00:00
|
|
|
|
|
|
|
def calc_precipitation(rain, snow):
|
|
|
|
"""Calculate the precipitation."""
|
|
|
|
rain_value = 0 if rain is None else rain
|
|
|
|
snow_value = 0 if snow is None else snow
|
|
|
|
if round(rain_value + snow_value, 1) == 0:
|
|
|
|
return None
|
|
|
|
return round(rain_value + snow_value, 1)
|
|
|
|
|
2019-02-23 14:52:08 +00:00
|
|
|
if self._mode == 'freedaily':
|
|
|
|
weather = self.forecast_data.get_weathers()[::8]
|
|
|
|
else:
|
|
|
|
weather = self.forecast_data.get_weathers()
|
|
|
|
|
|
|
|
for entry in weather:
|
2018-06-10 10:35:10 +00:00
|
|
|
if self._mode == 'daily':
|
|
|
|
data.append({
|
|
|
|
ATTR_FORECAST_TIME:
|
|
|
|
entry.get_reference_time('unix') * 1000,
|
|
|
|
ATTR_FORECAST_TEMP:
|
|
|
|
entry.get_temperature('celsius').get('day'),
|
|
|
|
ATTR_FORECAST_TEMP_LOW:
|
|
|
|
entry.get_temperature('celsius').get('night'),
|
2018-07-01 09:54:24 +00:00
|
|
|
ATTR_FORECAST_PRECIPITATION:
|
2018-10-26 13:50:30 +00:00
|
|
|
calc_precipitation(
|
|
|
|
entry.get_rain().get('all'),
|
|
|
|
entry.get_snow().get('all')),
|
2018-06-10 10:35:10 +00:00
|
|
|
ATTR_FORECAST_WIND_SPEED:
|
|
|
|
entry.get_wind().get('speed'),
|
|
|
|
ATTR_FORECAST_WIND_BEARING:
|
|
|
|
entry.get_wind().get('deg'),
|
|
|
|
ATTR_FORECAST_CONDITION:
|
|
|
|
[k for k, v in CONDITION_CLASSES.items()
|
|
|
|
if entry.get_weather_code() in v][0]
|
|
|
|
})
|
|
|
|
else:
|
|
|
|
data.append({
|
|
|
|
ATTR_FORECAST_TIME:
|
|
|
|
entry.get_reference_time('unix') * 1000,
|
|
|
|
ATTR_FORECAST_TEMP:
|
|
|
|
entry.get_temperature('celsius').get('temp'),
|
|
|
|
ATTR_FORECAST_PRECIPITATION:
|
2018-07-31 17:18:11 +00:00
|
|
|
(round(entry.get_rain().get('3h'), 1)
|
|
|
|
if entry.get_rain().get('3h') is not None
|
|
|
|
and (round(entry.get_rain().get('3h'), 1) > 0)
|
|
|
|
else None),
|
2018-06-10 10:35:10 +00:00
|
|
|
ATTR_FORECAST_CONDITION:
|
|
|
|
[k for k, v in CONDITION_CLASSES.items()
|
|
|
|
if entry.get_weather_code() in v][0]
|
|
|
|
})
|
2018-01-31 12:05:15 +00:00
|
|
|
return data
|
2017-02-20 00:42:12 +00:00
|
|
|
|
2016-10-25 04:53:03 +00:00
|
|
|
def update(self):
|
|
|
|
"""Get the latest data from OWM and updates the states."""
|
2017-10-15 08:31:34 +00:00
|
|
|
from pyowm.exceptions.api_call_error import APICallError
|
|
|
|
|
|
|
|
try:
|
|
|
|
self._owm.update()
|
|
|
|
self._owm.update_forecast()
|
|
|
|
except APICallError:
|
|
|
|
_LOGGER.error("Exception when calling OWM web API to update data")
|
|
|
|
return
|
|
|
|
|
2016-10-25 04:53:03 +00:00
|
|
|
self.data = self._owm.data
|
2017-02-20 00:42:12 +00:00
|
|
|
self.forecast_data = self._owm.forecast_data
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
|
2018-07-20 08:45:20 +00:00
|
|
|
class WeatherData:
|
2016-10-25 04:53:03 +00:00
|
|
|
"""Get the latest data from OpenWeatherMap."""
|
|
|
|
|
2018-06-10 10:35:10 +00:00
|
|
|
def __init__(self, owm, latitude, longitude, mode):
|
2016-10-25 04:53:03 +00:00
|
|
|
"""Initialize the data object."""
|
2018-06-10 10:35:10 +00:00
|
|
|
self._mode = mode
|
2016-10-25 04:53:03 +00:00
|
|
|
self.owm = owm
|
|
|
|
self.latitude = latitude
|
|
|
|
self.longitude = longitude
|
|
|
|
self.data = None
|
2017-02-20 00:42:12 +00:00
|
|
|
self.forecast_data = None
|
2016-10-25 04:53:03 +00:00
|
|
|
|
|
|
|
@Throttle(MIN_TIME_BETWEEN_UPDATES)
|
|
|
|
def update(self):
|
|
|
|
"""Get the latest data from OpenWeatherMap."""
|
|
|
|
obs = self.owm.weather_at_coords(self.latitude, self.longitude)
|
|
|
|
if obs is None:
|
|
|
|
_LOGGER.warning("Failed to fetch data from OWM")
|
|
|
|
return
|
|
|
|
|
|
|
|
self.data = obs.get_weather()
|
2017-02-20 00:42:12 +00:00
|
|
|
|
|
|
|
@Throttle(MIN_TIME_BETWEEN_FORECAST_UPDATES)
|
|
|
|
def update_forecast(self):
|
2018-01-29 22:37:19 +00:00
|
|
|
"""Get the latest forecast from OpenWeatherMap."""
|
2017-10-15 08:31:34 +00:00
|
|
|
from pyowm.exceptions.api_call_error import APICallError
|
|
|
|
|
|
|
|
try:
|
2018-06-10 10:35:10 +00:00
|
|
|
if self._mode == 'daily':
|
|
|
|
fcd = self.owm.daily_forecast_at_coords(
|
2018-07-01 09:54:24 +00:00
|
|
|
self.latitude, self.longitude, 15)
|
2018-06-10 10:35:10 +00:00
|
|
|
else:
|
|
|
|
fcd = self.owm.three_hours_forecast_at_coords(
|
2018-07-01 09:54:24 +00:00
|
|
|
self.latitude, self.longitude)
|
2017-10-15 08:31:34 +00:00
|
|
|
except APICallError:
|
|
|
|
_LOGGER.error("Exception when calling OWM web API "
|
|
|
|
"to update forecast")
|
|
|
|
return
|
2017-02-20 00:42:12 +00:00
|
|
|
|
|
|
|
if fcd is None:
|
|
|
|
_LOGGER.warning("Failed to fetch forecast data from OWM")
|
|
|
|
return
|
|
|
|
|
|
|
|
self.forecast_data = fcd.get_forecast()
|