2019-02-13 20:21:14 +00:00
|
|
|
"""Support for the (unofficial) Tado API."""
|
2020-04-12 18:42:36 +00:00
|
|
|
import asyncio
|
2019-11-24 21:47:31 +00:00
|
|
|
from datetime import timedelta
|
2017-03-22 12:18:13 +00:00
|
|
|
import logging
|
2017-03-26 13:50:40 +00:00
|
|
|
|
2019-11-24 21:47:31 +00:00
|
|
|
from PyTado.interface import Tado
|
2020-03-30 14:06:26 +00:00
|
|
|
from requests import RequestException
|
2020-04-12 18:42:36 +00:00
|
|
|
import requests.exceptions
|
2017-03-22 12:18:13 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2020-03-23 15:40:15 +00:00
|
|
|
from homeassistant.components.climate.const import PRESET_AWAY, PRESET_HOME
|
2020-04-12 18:42:36 +00:00
|
|
|
from homeassistant.config_entries import SOURCE_IMPORT, ConfigEntry
|
2019-11-24 21:47:31 +00:00
|
|
|
from homeassistant.const import CONF_PASSWORD, CONF_USERNAME
|
2020-04-12 18:42:36 +00:00
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
from homeassistant.exceptions import ConfigEntryNotReady
|
2017-03-22 12:18:13 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv
|
2019-12-20 12:24:43 +00:00
|
|
|
from homeassistant.helpers.dispatcher import dispatcher_send
|
2020-04-12 18:42:36 +00:00
|
|
|
from homeassistant.helpers.event import async_track_time_interval
|
2017-03-22 12:18:13 +00:00
|
|
|
from homeassistant.util import Throttle
|
|
|
|
|
2020-04-12 18:42:36 +00:00
|
|
|
from .const import (
|
|
|
|
CONF_FALLBACK,
|
|
|
|
DATA,
|
|
|
|
DOMAIN,
|
|
|
|
SIGNAL_TADO_UPDATE_RECEIVED,
|
|
|
|
UPDATE_LISTENER,
|
|
|
|
UPDATE_TRACK,
|
|
|
|
)
|
2019-12-20 12:24:43 +00:00
|
|
|
|
2017-03-22 12:18:13 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2020-02-15 16:08:21 +00:00
|
|
|
TADO_COMPONENTS = ["sensor", "climate", "water_heater"]
|
2017-03-22 12:18:13 +00:00
|
|
|
|
2019-12-20 12:24:43 +00:00
|
|
|
MIN_TIME_BETWEEN_UPDATES = timedelta(seconds=10)
|
|
|
|
SCAN_INTERVAL = timedelta(seconds=15)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
|
|
{
|
2020-02-11 16:46:02 +00:00
|
|
|
DOMAIN: vol.All(
|
|
|
|
cv.ensure_list,
|
|
|
|
[
|
|
|
|
{
|
|
|
|
vol.Required(CONF_USERNAME): cv.string,
|
|
|
|
vol.Required(CONF_PASSWORD): cv.string,
|
|
|
|
vol.Optional(CONF_FALLBACK, default=True): cv.boolean,
|
|
|
|
}
|
|
|
|
],
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
2017-03-22 12:18:13 +00:00
|
|
|
|
|
|
|
|
2020-04-12 18:42:36 +00:00
|
|
|
async def async_setup(hass: HomeAssistant, config: dict):
|
|
|
|
"""Set up the Tado component."""
|
|
|
|
|
|
|
|
hass.data.setdefault(DOMAIN, {})
|
|
|
|
|
|
|
|
if DOMAIN not in config:
|
|
|
|
return True
|
2017-03-22 12:18:13 +00:00
|
|
|
|
2020-04-12 18:42:36 +00:00
|
|
|
for conf in config[DOMAIN]:
|
|
|
|
hass.async_create_task(
|
|
|
|
hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": SOURCE_IMPORT}, data=conf,
|
|
|
|
)
|
|
|
|
)
|
2017-03-22 12:18:13 +00:00
|
|
|
|
2020-04-12 18:42:36 +00:00
|
|
|
return True
|
2017-03-22 12:18:13 +00:00
|
|
|
|
2020-02-11 16:46:02 +00:00
|
|
|
|
2020-04-12 18:42:36 +00:00
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry):
|
|
|
|
"""Set up Tado from a config entry."""
|
2020-02-11 16:46:02 +00:00
|
|
|
|
2020-04-12 18:42:36 +00:00
|
|
|
_async_import_options_from_data_if_missing(hass, entry)
|
|
|
|
|
|
|
|
username = entry.data[CONF_USERNAME]
|
|
|
|
password = entry.data[CONF_PASSWORD]
|
|
|
|
fallback = entry.options.get(CONF_FALLBACK, True)
|
|
|
|
|
|
|
|
tadoconnector = TadoConnector(hass, username, password, fallback)
|
|
|
|
|
|
|
|
try:
|
|
|
|
await hass.async_add_executor_job(tadoconnector.setup)
|
|
|
|
except KeyError:
|
|
|
|
_LOGGER.error("Failed to login to tado")
|
|
|
|
return False
|
|
|
|
except RuntimeError as exc:
|
|
|
|
_LOGGER.error("Failed to setup tado: %s", exc)
|
|
|
|
return ConfigEntryNotReady
|
|
|
|
except requests.exceptions.HTTPError as ex:
|
|
|
|
if ex.response.status_code > 400 and ex.response.status_code < 500:
|
|
|
|
_LOGGER.error("Failed to login to tado: %s", ex)
|
|
|
|
return False
|
|
|
|
raise ConfigEntryNotReady
|
|
|
|
|
|
|
|
# Do first update
|
|
|
|
await hass.async_add_executor_job(tadoconnector.update)
|
|
|
|
|
|
|
|
# Poll for updates in the background
|
|
|
|
update_track = async_track_time_interval(
|
|
|
|
hass, lambda now: tadoconnector.update(), SCAN_INTERVAL,
|
|
|
|
)
|
|
|
|
|
|
|
|
update_listener = entry.add_update_listener(_async_update_listener)
|
2020-02-11 16:46:02 +00:00
|
|
|
|
2020-04-12 18:42:36 +00:00
|
|
|
hass.data[DOMAIN][entry.entry_id] = {
|
|
|
|
DATA: tadoconnector,
|
|
|
|
UPDATE_TRACK: update_track,
|
|
|
|
UPDATE_LISTENER: update_listener,
|
|
|
|
}
|
2019-12-20 12:24:43 +00:00
|
|
|
|
2017-03-22 12:18:13 +00:00
|
|
|
for component in TADO_COMPONENTS:
|
2020-04-12 18:42:36 +00:00
|
|
|
hass.async_create_task(
|
|
|
|
hass.config_entries.async_forward_entry_setup(entry, component)
|
2019-12-20 12:24:43 +00:00
|
|
|
)
|
|
|
|
|
2017-03-22 12:18:13 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2020-04-12 18:42:36 +00:00
|
|
|
@callback
|
|
|
|
def _async_import_options_from_data_if_missing(hass: HomeAssistant, entry: ConfigEntry):
|
|
|
|
options = dict(entry.options)
|
|
|
|
if CONF_FALLBACK not in options:
|
|
|
|
options[CONF_FALLBACK] = entry.data.get(CONF_FALLBACK, True)
|
|
|
|
hass.config_entries.async_update_entry(entry, options=options)
|
|
|
|
|
|
|
|
|
|
|
|
async def _async_update_listener(hass: HomeAssistant, entry: ConfigEntry):
|
|
|
|
"""Handle options update."""
|
|
|
|
await hass.config_entries.async_reload(entry.entry_id)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry):
|
|
|
|
"""Unload a config entry."""
|
|
|
|
unload_ok = all(
|
|
|
|
await asyncio.gather(
|
|
|
|
*[
|
|
|
|
hass.config_entries.async_forward_entry_unload(entry, component)
|
|
|
|
for component in TADO_COMPONENTS
|
|
|
|
]
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
hass.data[DOMAIN][entry.entry_id][UPDATE_TRACK]()
|
|
|
|
hass.data[DOMAIN][entry.entry_id][UPDATE_LISTENER]()
|
|
|
|
|
|
|
|
if unload_ok:
|
|
|
|
hass.data[DOMAIN].pop(entry.entry_id)
|
|
|
|
|
|
|
|
return unload_ok
|
|
|
|
|
|
|
|
|
2019-12-20 12:24:43 +00:00
|
|
|
class TadoConnector:
|
2017-03-26 13:50:40 +00:00
|
|
|
"""An object to store the Tado data."""
|
2017-03-22 12:18:13 +00:00
|
|
|
|
2020-02-11 16:46:02 +00:00
|
|
|
def __init__(self, hass, username, password, fallback):
|
2019-12-20 12:24:43 +00:00
|
|
|
"""Initialize Tado Connector."""
|
|
|
|
self.hass = hass
|
|
|
|
self._username = username
|
|
|
|
self._password = password
|
2020-02-11 16:46:02 +00:00
|
|
|
self._fallback = fallback
|
2017-03-22 12:18:13 +00:00
|
|
|
|
2020-02-11 16:46:02 +00:00
|
|
|
self.device_id = None
|
2019-12-20 12:24:43 +00:00
|
|
|
self.tado = None
|
|
|
|
self.zones = None
|
|
|
|
self.devices = None
|
|
|
|
self.data = {
|
|
|
|
"zone": {},
|
|
|
|
"device": {},
|
|
|
|
}
|
|
|
|
|
2020-02-11 16:46:02 +00:00
|
|
|
@property
|
|
|
|
def fallback(self):
|
|
|
|
"""Return fallback flag to Smart Schedule."""
|
|
|
|
return self._fallback
|
|
|
|
|
2019-12-20 12:24:43 +00:00
|
|
|
def setup(self):
|
|
|
|
"""Connect to Tado and fetch the zones."""
|
2020-04-12 18:42:36 +00:00
|
|
|
self.tado = Tado(self._username, self._password)
|
2019-12-20 12:24:43 +00:00
|
|
|
self.tado.setDebugging(True)
|
|
|
|
# Load zones and devices
|
|
|
|
self.zones = self.tado.getZones()
|
|
|
|
self.devices = self.tado.getMe()["homes"]
|
2020-02-11 16:46:02 +00:00
|
|
|
self.device_id = self.devices[0]["id"]
|
2017-03-22 12:18:13 +00:00
|
|
|
|
|
|
|
@Throttle(MIN_TIME_BETWEEN_UPDATES)
|
|
|
|
def update(self):
|
2019-12-20 12:24:43 +00:00
|
|
|
"""Update the registered zones."""
|
|
|
|
for zone in self.zones:
|
|
|
|
self.update_sensor("zone", zone["id"])
|
|
|
|
for device in self.devices:
|
|
|
|
self.update_sensor("device", device["id"])
|
|
|
|
|
|
|
|
def update_sensor(self, sensor_type, sensor):
|
|
|
|
"""Update the internal data from Tado."""
|
|
|
|
_LOGGER.debug("Updating %s %s", sensor_type, sensor)
|
|
|
|
try:
|
|
|
|
if sensor_type == "zone":
|
2020-03-30 14:06:26 +00:00
|
|
|
data = self.tado.getZoneState(sensor)
|
2019-12-20 12:24:43 +00:00
|
|
|
elif sensor_type == "device":
|
2020-03-30 14:06:26 +00:00
|
|
|
devices_data = self.tado.getDevices()
|
|
|
|
if not devices_data:
|
2020-07-05 21:04:19 +00:00
|
|
|
_LOGGER.info("There are no devices to setup on this tado account")
|
2020-03-30 14:06:26 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
data = devices_data[0]
|
2019-12-20 12:24:43 +00:00
|
|
|
else:
|
|
|
|
_LOGGER.debug("Unknown sensor: %s", sensor_type)
|
|
|
|
return
|
|
|
|
except RuntimeError:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Unable to connect to Tado while updating %s %s", sensor_type, sensor,
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
self.data[sensor_type][sensor] = data
|
|
|
|
|
2020-04-18 22:39:43 +00:00
|
|
|
_LOGGER.debug(
|
|
|
|
"Dispatching update to %s %s %s: %s",
|
|
|
|
self.device_id,
|
|
|
|
sensor_type,
|
|
|
|
sensor,
|
|
|
|
data,
|
|
|
|
)
|
2019-12-20 12:24:43 +00:00
|
|
|
dispatcher_send(
|
2020-04-18 22:39:43 +00:00
|
|
|
self.hass,
|
|
|
|
SIGNAL_TADO_UPDATE_RECEIVED.format(self.device_id, sensor_type, sensor),
|
2019-12-20 12:24:43 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
def get_capabilities(self, zone_id):
|
|
|
|
"""Return the capabilities of the devices."""
|
|
|
|
return self.tado.getCapabilities(zone_id)
|
2017-03-22 12:18:13 +00:00
|
|
|
|
|
|
|
def reset_zone_overlay(self, zone_id):
|
2019-12-20 12:24:43 +00:00
|
|
|
"""Reset the zone back to the default operation."""
|
2018-02-03 01:28:54 +00:00
|
|
|
self.tado.resetZoneOverlay(zone_id)
|
2019-12-20 12:24:43 +00:00
|
|
|
self.update_sensor("zone", zone_id)
|
2017-03-22 12:18:13 +00:00
|
|
|
|
2020-03-23 15:40:15 +00:00
|
|
|
def set_presence(
|
|
|
|
self, presence=PRESET_HOME,
|
|
|
|
):
|
|
|
|
"""Set the presence to home or away."""
|
|
|
|
if presence == PRESET_AWAY:
|
|
|
|
self.tado.setAway()
|
|
|
|
elif presence == PRESET_HOME:
|
|
|
|
self.tado.setHome()
|
|
|
|
|
2019-08-12 04:02:16 +00:00
|
|
|
def set_zone_overlay(
|
|
|
|
self,
|
2020-03-30 14:06:26 +00:00
|
|
|
zone_id=None,
|
|
|
|
overlay_mode=None,
|
2019-08-12 04:02:16 +00:00
|
|
|
temperature=None,
|
|
|
|
duration=None,
|
|
|
|
device_type="HEATING",
|
|
|
|
mode=None,
|
2020-03-30 14:06:26 +00:00
|
|
|
fan_speed=None,
|
2020-03-31 22:29:45 +00:00
|
|
|
swing=None,
|
2019-08-12 04:02:16 +00:00
|
|
|
):
|
2019-12-20 12:24:43 +00:00
|
|
|
"""Set a zone overlay."""
|
|
|
|
_LOGGER.debug(
|
2020-03-31 22:29:45 +00:00
|
|
|
"Set overlay for zone %s: overlay_mode=%s, temp=%s, duration=%s, type=%s, mode=%s fan_speed=%s swing=%s",
|
2019-12-20 12:24:43 +00:00
|
|
|
zone_id,
|
|
|
|
overlay_mode,
|
|
|
|
temperature,
|
|
|
|
duration,
|
|
|
|
device_type,
|
|
|
|
mode,
|
2020-03-30 14:06:26 +00:00
|
|
|
fan_speed,
|
2020-03-31 22:29:45 +00:00
|
|
|
swing,
|
2019-08-12 04:02:16 +00:00
|
|
|
)
|
2020-03-30 14:06:26 +00:00
|
|
|
|
2019-12-20 12:24:43 +00:00
|
|
|
try:
|
|
|
|
self.tado.setZoneOverlay(
|
2020-03-30 14:06:26 +00:00
|
|
|
zone_id,
|
|
|
|
overlay_mode,
|
|
|
|
temperature,
|
|
|
|
duration,
|
|
|
|
device_type,
|
|
|
|
"ON",
|
|
|
|
mode,
|
2020-03-31 22:29:45 +00:00
|
|
|
fanSpeed=fan_speed,
|
|
|
|
swing=swing,
|
2019-12-20 12:24:43 +00:00
|
|
|
)
|
2020-03-30 14:06:26 +00:00
|
|
|
|
|
|
|
except RequestException as exc:
|
|
|
|
_LOGGER.error("Could not set zone overlay: %s", exc)
|
2019-12-20 12:24:43 +00:00
|
|
|
|
|
|
|
self.update_sensor("zone", zone_id)
|
2019-04-05 00:52:06 +00:00
|
|
|
|
2019-08-12 04:02:16 +00:00
|
|
|
def set_zone_off(self, zone_id, overlay_mode, device_type="HEATING"):
|
2019-04-05 00:52:06 +00:00
|
|
|
"""Set a zone to off."""
|
2019-12-20 12:24:43 +00:00
|
|
|
try:
|
|
|
|
self.tado.setZoneOverlay(
|
|
|
|
zone_id, overlay_mode, None, None, device_type, "OFF"
|
|
|
|
)
|
2020-03-30 14:06:26 +00:00
|
|
|
except RequestException as exc:
|
|
|
|
_LOGGER.error("Could not set zone overlay: %s", exc)
|
2019-12-20 12:24:43 +00:00
|
|
|
|
|
|
|
self.update_sensor("zone", zone_id)
|