2019-02-13 20:21:14 +00:00
|
|
|
"""Support for displaying weather info from Ecobee API."""
|
2018-06-15 21:09:01 +00:00
|
|
|
from datetime import datetime
|
2018-11-12 21:32:30 +00:00
|
|
|
|
2017-12-01 11:30:45 +00:00
|
|
|
from homeassistant.components import ecobee
|
|
|
|
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,
|
|
|
|
ATTR_FORECAST_WIND_SPEED,
|
|
|
|
WeatherEntity,
|
|
|
|
)
|
2018-11-12 21:32:30 +00:00
|
|
|
from homeassistant.const import TEMP_FAHRENHEIT
|
2017-12-01 11:30:45 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_FORECAST_TEMP_HIGH = "temphigh"
|
|
|
|
ATTR_FORECAST_PRESSURE = "pressure"
|
|
|
|
ATTR_FORECAST_VISIBILITY = "visibility"
|
|
|
|
ATTR_FORECAST_HUMIDITY = "humidity"
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
MISSING_DATA = -5002
|
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2019-02-13 20:21:14 +00:00
|
|
|
"""Set up the Ecobee weather platform."""
|
2017-12-01 11:30:45 +00:00
|
|
|
if discovery_info is None:
|
|
|
|
return
|
|
|
|
dev = list()
|
|
|
|
data = ecobee.NETWORK
|
|
|
|
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:
|
|
|
|
dev.append(EcobeeWeather(thermostat["name"], index))
|
2017-12-01 11:30:45 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(dev, True)
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
class EcobeeWeather(WeatherEntity):
|
|
|
|
"""Representation of Ecobee weather data."""
|
|
|
|
|
|
|
|
def __init__(self, name, index):
|
2018-11-12 21:32:30 +00:00
|
|
|
"""Initialize the Ecobee weather platform."""
|
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]
|
2017-12-02 21:44:55 +00:00
|
|
|
except (ValueError, IndexError, KeyError):
|
|
|
|
raise ValueError
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def condition(self):
|
|
|
|
"""Return the current condition."""
|
2017-12-02 21:44:55 +00:00
|
|
|
try:
|
2019-07-31 19:25:30 +00:00
|
|
|
return self.get_forecast(0, "condition")
|
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:
|
2019-07-31 19:25:30 +00:00
|
|
|
return int(self.get_forecast(0, "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-07-31 19:25:30 +00:00
|
|
|
return int(self.get_forecast(0, "visibility"))
|
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."""
|
2017-12-02 21:44:55 +00:00
|
|
|
if self.weather:
|
2019-07-31 19:25:30 +00:00
|
|
|
station = self.weather.get("weatherStation", "UNKNOWN")
|
|
|
|
time = self.weather.get("timestamp", "UNKNOWN")
|
2019-09-03 15:10:56 +00:00
|
|
|
return f"Ecobee weather provided by {station} at {time}"
|
2017-12-02 21:44:55 +00:00
|
|
|
return None
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def forecast(self):
|
|
|
|
"""Return the forecast array."""
|
|
|
|
try:
|
|
|
|
forecasts = []
|
2019-07-31 19:25:30 +00:00
|
|
|
for day in self.weather["forecasts"]:
|
|
|
|
date_time = datetime.strptime(
|
|
|
|
day["dateTime"], "%Y-%m-%d %H:%M:%S"
|
|
|
|
).isoformat()
|
2017-12-01 11:30:45 +00:00
|
|
|
forecast = {
|
2018-06-15 21:09:01 +00:00
|
|
|
ATTR_FORECAST_TIME: date_time,
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_FORECAST_CONDITION: day["condition"],
|
|
|
|
ATTR_FORECAST_TEMP: float(day["tempHigh"]) / 10,
|
2017-12-01 11:30:45 +00:00
|
|
|
}
|
2019-07-31 19:25:30 +00:00
|
|
|
if day["tempHigh"] == MISSING_DATA:
|
2017-12-01 11:30:45 +00:00
|
|
|
break
|
2019-07-31 19:25:30 +00:00
|
|
|
if day["tempLow"] != MISSING_DATA:
|
|
|
|
forecast[ATTR_FORECAST_TEMP_LOW] = float(day["tempLow"]) / 10
|
|
|
|
if day["pressure"] != MISSING_DATA:
|
|
|
|
forecast[ATTR_FORECAST_PRESSURE] = int(day["pressure"])
|
|
|
|
if day["windSpeed"] != MISSING_DATA:
|
|
|
|
forecast[ATTR_FORECAST_WIND_SPEED] = int(day["windSpeed"])
|
|
|
|
if day["visibility"] != MISSING_DATA:
|
|
|
|
forecast[ATTR_FORECAST_WIND_SPEED] = int(day["visibility"])
|
|
|
|
if day["relativeHumidity"] != MISSING_DATA:
|
|
|
|
forecast[ATTR_FORECAST_HUMIDITY] = int(day["relativeHumidity"])
|
2017-12-01 11:30:45 +00:00
|
|
|
forecasts.append(forecast)
|
|
|
|
return forecasts
|
2017-12-02 21:44:55 +00:00
|
|
|
except (ValueError, IndexError, KeyError):
|
|
|
|
return None
|
2017-12-01 11:30:45 +00:00
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Get the latest state of the sensor."""
|
|
|
|
data = ecobee.NETWORK
|
|
|
|
data.update()
|
|
|
|
thermostat = data.ecobee.get_thermostat(self._index)
|
2019-07-31 19:25:30 +00:00
|
|
|
self.weather = thermostat.get("weather", None)
|