2019-02-13 20:21:14 +00:00
|
|
|
"""Support for displaying weather info from Ecobee API."""
|
2020-09-14 19:29:51 +00:00
|
|
|
from datetime import timedelta
|
2018-11-12 21:32:30 +00:00
|
|
|
|
2019-09-25 20:38:21 +00:00
|
|
|
from pyecobee.const import ECOBEE_STATE_UNKNOWN
|
|
|
|
|
2017-12-01 11:30:45 +00:00
|
|
|
from homeassistant.components.weather import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_FORECAST_CONDITION,
|
|
|
|
ATTR_FORECAST_TEMP,
|
|
|
|
ATTR_FORECAST_TEMP_LOW,
|
|
|
|
ATTR_FORECAST_TIME,
|
2019-10-10 07:05:46 +00:00
|
|
|
ATTR_FORECAST_WIND_BEARING,
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_FORECAST_WIND_SPEED,
|
|
|
|
WeatherEntity,
|
|
|
|
)
|
2021-06-09 12:06:24 +00:00
|
|
|
from homeassistant.const import PRESSURE_HPA, PRESSURE_INHG, TEMP_FAHRENHEIT
|
2020-09-14 19:29:51 +00:00
|
|
|
from homeassistant.util import dt as dt_util
|
2021-06-09 12:06:24 +00:00
|
|
|
from homeassistant.util.pressure import convert as pressure_convert
|
2017-12-01 11:30:45 +00:00
|
|
|
|
2019-10-10 07:05:46 +00:00
|
|
|
from .const import (
|
|
|
|
DOMAIN,
|
|
|
|
ECOBEE_MODEL_TO_NAME,
|
|
|
|
ECOBEE_WEATHER_SYMBOL_TO_HASS,
|
|
|
|
MANUFACTURER,
|
|
|
|
)
|
2017-12-01 11:30:45 +00:00
|
|
|
|
2019-09-25 20:38:21 +00:00
|
|
|
|
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
|
|
|
"""Set up the ecobee weather platform."""
|
|
|
|
data = hass.data[DOMAIN]
|
2020-04-04 21:14:47 +00:00
|
|
|
dev = []
|
2017-12-01 11:30:45 +00:00
|
|
|
for index in range(len(data.ecobee.thermostats)):
|
|
|
|
thermostat = data.ecobee.get_thermostat(index)
|
2019-07-31 19:25:30 +00:00
|
|
|
if "weather" in thermostat:
|
2019-09-25 20:38:21 +00:00
|
|
|
dev.append(EcobeeWeather(data, thermostat["name"], index))
|
2017-12-01 11:30:45 +00:00
|
|
|
|
2019-09-25 20:38:21 +00:00
|
|
|
async_add_entities(dev, True)
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
class EcobeeWeather(WeatherEntity):
|
|
|
|
"""Representation of Ecobee weather data."""
|
|
|
|
|
2019-09-25 20:38:21 +00:00
|
|
|
def __init__(self, data, name, index):
|
2018-11-12 21:32:30 +00:00
|
|
|
"""Initialize the Ecobee weather platform."""
|
2019-09-25 20:38:21 +00:00
|
|
|
self.data = data
|
2017-12-01 11:30:45 +00:00
|
|
|
self._name = name
|
|
|
|
self._index = index
|
|
|
|
self.weather = None
|
|
|
|
|
|
|
|
def get_forecast(self, index, param):
|
|
|
|
"""Retrieve forecast parameter."""
|
|
|
|
try:
|
2019-07-31 19:25:30 +00:00
|
|
|
forecast = self.weather["forecasts"][index]
|
2017-12-01 11:30:45 +00:00
|
|
|
return forecast[param]
|
2020-08-28 11:50:32 +00:00
|
|
|
except (IndexError, KeyError) as err:
|
|
|
|
raise ValueError from err
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._name
|
|
|
|
|
2019-10-01 21:28:13 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return a unique identifier for the weather platform."""
|
|
|
|
return self.data.ecobee.get_thermostat(self._index)["identifier"]
|
|
|
|
|
2019-10-04 06:31:45 +00:00
|
|
|
@property
|
|
|
|
def device_info(self):
|
|
|
|
"""Return device information for the ecobee weather platform."""
|
|
|
|
thermostat = self.data.ecobee.get_thermostat(self._index)
|
|
|
|
try:
|
|
|
|
model = f"{ECOBEE_MODEL_TO_NAME[thermostat['modelNumber']]} Thermostat"
|
|
|
|
except KeyError:
|
2021-06-17 08:59:13 +00:00
|
|
|
# Ecobee model is not in our list
|
|
|
|
model = None
|
2019-10-04 06:31:45 +00:00
|
|
|
|
|
|
|
return {
|
|
|
|
"identifiers": {(DOMAIN, thermostat["identifier"])},
|
|
|
|
"name": self.name,
|
|
|
|
"manufacturer": MANUFACTURER,
|
|
|
|
"model": model,
|
|
|
|
}
|
|
|
|
|
2017-12-01 11:30:45 +00:00
|
|
|
@property
|
|
|
|
def condition(self):
|
|
|
|
"""Return the current condition."""
|
2017-12-02 21:44:55 +00:00
|
|
|
try:
|
2019-10-10 07:05:46 +00:00
|
|
|
return ECOBEE_WEATHER_SYMBOL_TO_HASS[self.get_forecast(0, "weatherSymbol")]
|
2017-12-02 21:44:55 +00:00
|
|
|
except ValueError:
|
|
|
|
return None
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def temperature(self):
|
|
|
|
"""Return the temperature."""
|
2017-12-02 21:44:55 +00:00
|
|
|
try:
|
2019-07-31 19:25:30 +00:00
|
|
|
return float(self.get_forecast(0, "temperature")) / 10
|
2017-12-02 21:44:55 +00:00
|
|
|
except ValueError:
|
|
|
|
return None
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def temperature_unit(self):
|
|
|
|
"""Return the unit of measurement."""
|
|
|
|
return TEMP_FAHRENHEIT
|
|
|
|
|
|
|
|
@property
|
|
|
|
def pressure(self):
|
|
|
|
"""Return the pressure."""
|
2017-12-02 21:44:55 +00:00
|
|
|
try:
|
2021-06-09 12:06:24 +00:00
|
|
|
pressure = self.get_forecast(0, "pressure")
|
|
|
|
if not self.hass.config.units.is_metric:
|
|
|
|
pressure = pressure_convert(pressure, PRESSURE_HPA, PRESSURE_INHG)
|
|
|
|
return round(pressure, 2)
|
|
|
|
return round(pressure)
|
2017-12-02 21:44:55 +00:00
|
|
|
except ValueError:
|
|
|
|
return None
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def humidity(self):
|
|
|
|
"""Return the humidity."""
|
2017-12-02 21:44:55 +00:00
|
|
|
try:
|
2019-07-31 19:25:30 +00:00
|
|
|
return int(self.get_forecast(0, "relativeHumidity"))
|
2017-12-02 21:44:55 +00:00
|
|
|
except ValueError:
|
|
|
|
return None
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def visibility(self):
|
|
|
|
"""Return the visibility."""
|
2017-12-02 21:44:55 +00:00
|
|
|
try:
|
2019-10-10 07:05:46 +00:00
|
|
|
return int(self.get_forecast(0, "visibility")) / 1000
|
2017-12-02 21:44:55 +00:00
|
|
|
except ValueError:
|
|
|
|
return None
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def wind_speed(self):
|
|
|
|
"""Return the wind speed."""
|
2017-12-02 21:44:55 +00:00
|
|
|
try:
|
2019-07-31 19:25:30 +00:00
|
|
|
return int(self.get_forecast(0, "windSpeed"))
|
2017-12-02 21:44:55 +00:00
|
|
|
except ValueError:
|
|
|
|
return None
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def wind_bearing(self):
|
|
|
|
"""Return the wind direction."""
|
2017-12-02 21:44:55 +00:00
|
|
|
try:
|
2019-07-31 19:25:30 +00:00
|
|
|
return int(self.get_forecast(0, "windBearing"))
|
2017-12-02 21:44:55 +00:00
|
|
|
except ValueError:
|
|
|
|
return None
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def attribution(self):
|
|
|
|
"""Return the attribution."""
|
2019-10-10 07:05:46 +00:00
|
|
|
if not self.weather:
|
|
|
|
return None
|
|
|
|
|
|
|
|
station = self.weather.get("weatherStation", "UNKNOWN")
|
|
|
|
time = self.weather.get("timestamp", "UNKNOWN")
|
|
|
|
return f"Ecobee weather provided by {station} at {time} UTC"
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def forecast(self):
|
|
|
|
"""Return the forecast array."""
|
2019-10-10 07:05:46 +00:00
|
|
|
if "forecasts" not in self.weather:
|
2017-12-02 21:44:55 +00:00
|
|
|
return None
|
2017-12-01 11:30:45 +00:00
|
|
|
|
2020-04-04 21:14:47 +00:00
|
|
|
forecasts = []
|
2020-09-14 19:29:51 +00:00
|
|
|
date = dt_util.utcnow()
|
|
|
|
for day in range(0, 5):
|
2019-10-10 07:05:46 +00:00
|
|
|
forecast = _process_forecast(self.weather["forecasts"][day])
|
|
|
|
if forecast is None:
|
|
|
|
continue
|
2020-09-14 19:29:51 +00:00
|
|
|
forecast[ATTR_FORECAST_TIME] = date.isoformat()
|
|
|
|
date += timedelta(days=1)
|
2019-10-10 07:05:46 +00:00
|
|
|
forecasts.append(forecast)
|
|
|
|
|
|
|
|
if forecasts:
|
|
|
|
return forecasts
|
|
|
|
return None
|
|
|
|
|
2019-09-25 20:38:21 +00:00
|
|
|
async def async_update(self):
|
|
|
|
"""Get the latest weather data."""
|
|
|
|
await self.data.update()
|
|
|
|
thermostat = self.data.ecobee.get_thermostat(self._index)
|
2020-04-07 19:06:05 +00:00
|
|
|
self.weather = thermostat.get("weather")
|
2019-10-10 07:05:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _process_forecast(json):
|
|
|
|
"""Process a single ecobee API forecast to return expected values."""
|
2020-04-04 20:31:56 +00:00
|
|
|
forecast = {}
|
2019-10-10 07:05:46 +00:00
|
|
|
try:
|
|
|
|
forecast[ATTR_FORECAST_CONDITION] = ECOBEE_WEATHER_SYMBOL_TO_HASS[
|
|
|
|
json["weatherSymbol"]
|
|
|
|
]
|
|
|
|
if json["tempHigh"] != ECOBEE_STATE_UNKNOWN:
|
|
|
|
forecast[ATTR_FORECAST_TEMP] = float(json["tempHigh"]) / 10
|
|
|
|
if json["tempLow"] != ECOBEE_STATE_UNKNOWN:
|
|
|
|
forecast[ATTR_FORECAST_TEMP_LOW] = float(json["tempLow"]) / 10
|
|
|
|
if json["windBearing"] != ECOBEE_STATE_UNKNOWN:
|
|
|
|
forecast[ATTR_FORECAST_WIND_BEARING] = int(json["windBearing"])
|
|
|
|
if json["windSpeed"] != ECOBEE_STATE_UNKNOWN:
|
|
|
|
forecast[ATTR_FORECAST_WIND_SPEED] = int(json["windSpeed"])
|
|
|
|
|
|
|
|
except (ValueError, IndexError, KeyError):
|
|
|
|
return None
|
|
|
|
|
|
|
|
if forecast:
|
|
|
|
return forecast
|
|
|
|
return None
|