2020-03-27 17:53:36 +00:00
|
|
|
"""Support for Schluter thermostats."""
|
|
|
|
import logging
|
|
|
|
|
|
|
|
from requests import RequestException
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.components.climate import (
|
|
|
|
PLATFORM_SCHEMA,
|
|
|
|
SCAN_INTERVAL,
|
2020-04-14 21:10:44 +00:00
|
|
|
TEMP_CELSIUS,
|
2020-04-25 16:04:03 +00:00
|
|
|
ClimateEntity,
|
2020-03-27 17:53:36 +00:00
|
|
|
)
|
|
|
|
from homeassistant.components.climate.const import (
|
|
|
|
CURRENT_HVAC_HEAT,
|
|
|
|
CURRENT_HVAC_IDLE,
|
|
|
|
HVAC_MODE_HEAT,
|
|
|
|
SUPPORT_TARGET_TEMPERATURE,
|
|
|
|
)
|
|
|
|
from homeassistant.const import ATTR_TEMPERATURE, CONF_SCAN_INTERVAL
|
2020-08-30 19:19:48 +00:00
|
|
|
from homeassistant.helpers.update_coordinator import (
|
|
|
|
CoordinatorEntity,
|
|
|
|
DataUpdateCoordinator,
|
|
|
|
UpdateFailed,
|
|
|
|
)
|
2020-03-27 17:53:36 +00:00
|
|
|
|
|
|
|
from . import DATA_SCHLUTER_API, DATA_SCHLUTER_SESSION, DOMAIN
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{vol.Optional(CONF_SCAN_INTERVAL): vol.All(vol.Coerce(int), vol.Range(min=1))}
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_platform(hass, config, async_add_entities, discovery_info=None):
|
|
|
|
"""Set up the Schluter thermostats."""
|
|
|
|
if discovery_info is None:
|
|
|
|
return
|
|
|
|
session_id = hass.data[DOMAIN][DATA_SCHLUTER_SESSION]
|
|
|
|
api = hass.data[DOMAIN][DATA_SCHLUTER_API]
|
|
|
|
|
|
|
|
async def async_update_data():
|
|
|
|
try:
|
|
|
|
thermostats = await hass.async_add_executor_job(
|
|
|
|
api.get_thermostats, session_id
|
|
|
|
)
|
|
|
|
except RequestException as err:
|
2020-08-28 11:50:32 +00:00
|
|
|
raise UpdateFailed(f"Error communicating with Schluter API: {err}") from err
|
2020-03-27 17:53:36 +00:00
|
|
|
|
|
|
|
if thermostats is None:
|
|
|
|
return {}
|
|
|
|
|
|
|
|
return {thermo.serial_number: thermo for thermo in thermostats}
|
|
|
|
|
|
|
|
coordinator = DataUpdateCoordinator(
|
|
|
|
hass,
|
|
|
|
_LOGGER,
|
|
|
|
name="schluter",
|
|
|
|
update_method=async_update_data,
|
|
|
|
update_interval=SCAN_INTERVAL,
|
|
|
|
)
|
|
|
|
|
|
|
|
await coordinator.async_refresh()
|
|
|
|
|
|
|
|
async_add_entities(
|
2020-04-14 21:10:44 +00:00
|
|
|
SchluterThermostat(coordinator, serial_number, api, session_id)
|
2020-03-27 17:53:36 +00:00
|
|
|
for serial_number, thermostat in coordinator.data.items()
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2020-08-30 19:19:48 +00:00
|
|
|
class SchluterThermostat(CoordinatorEntity, ClimateEntity):
|
2020-03-27 17:53:36 +00:00
|
|
|
"""Representation of a Schluter thermostat."""
|
|
|
|
|
2020-04-14 21:10:44 +00:00
|
|
|
def __init__(self, coordinator, serial_number, api, session_id):
|
2020-03-27 17:53:36 +00:00
|
|
|
"""Initialize the thermostat."""
|
2020-08-30 19:19:48 +00:00
|
|
|
super().__init__(coordinator)
|
2020-03-27 17:53:36 +00:00
|
|
|
self._serial_number = serial_number
|
|
|
|
self._api = api
|
|
|
|
self._session_id = session_id
|
|
|
|
self._support_flags = SUPPORT_TARGET_TEMPERATURE
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Return the list of supported features."""
|
|
|
|
return self._support_flags
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return unique ID for this device."""
|
|
|
|
return self._serial_number
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the thermostat."""
|
2020-08-30 19:19:48 +00:00
|
|
|
return self.coordinator.data[self._serial_number].name
|
2020-03-27 17:53:36 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def temperature_unit(self):
|
2020-04-14 21:10:44 +00:00
|
|
|
"""Schluter API always uses celsius."""
|
|
|
|
return TEMP_CELSIUS
|
2020-03-27 17:53:36 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def current_temperature(self):
|
|
|
|
"""Return the current temperature."""
|
2020-08-30 19:19:48 +00:00
|
|
|
return self.coordinator.data[self._serial_number].temperature
|
2020-03-27 17:53:36 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def hvac_mode(self):
|
|
|
|
"""Return current mode. Only heat available for floor thermostat."""
|
|
|
|
return HVAC_MODE_HEAT
|
|
|
|
|
|
|
|
@property
|
|
|
|
def hvac_action(self):
|
|
|
|
"""Return current operation. Can only be heating or idle."""
|
|
|
|
return (
|
|
|
|
CURRENT_HVAC_HEAT
|
2020-08-30 19:19:48 +00:00
|
|
|
if self.coordinator.data[self._serial_number].is_heating
|
2020-03-27 17:53:36 +00:00
|
|
|
else CURRENT_HVAC_IDLE
|
|
|
|
)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def target_temperature(self):
|
|
|
|
"""Return the temperature we try to reach."""
|
2020-08-30 19:19:48 +00:00
|
|
|
return self.coordinator.data[self._serial_number].set_point_temp
|
2020-03-27 17:53:36 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def hvac_modes(self):
|
|
|
|
"""List of available operation modes."""
|
|
|
|
return [HVAC_MODE_HEAT]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def min_temp(self):
|
|
|
|
"""Identify min_temp in Schluter API."""
|
2020-08-30 19:19:48 +00:00
|
|
|
return self.coordinator.data[self._serial_number].min_temp
|
2020-03-27 17:53:36 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def max_temp(self):
|
|
|
|
"""Identify max_temp in Schluter API."""
|
2020-08-30 19:19:48 +00:00
|
|
|
return self.coordinator.data[self._serial_number].max_temp
|
2020-03-27 17:53:36 +00:00
|
|
|
|
|
|
|
async def async_set_hvac_mode(self, hvac_mode):
|
|
|
|
"""Mode is always heating, so do nothing."""
|
|
|
|
|
|
|
|
def set_temperature(self, **kwargs):
|
|
|
|
"""Set new target temperature."""
|
|
|
|
target_temp = None
|
|
|
|
target_temp = kwargs.get(ATTR_TEMPERATURE)
|
2020-08-30 19:19:48 +00:00
|
|
|
serial_number = self.coordinator.data[self._serial_number].serial_number
|
2020-03-27 17:53:36 +00:00
|
|
|
_LOGGER.debug("Setting thermostat temperature: %s", target_temp)
|
|
|
|
|
|
|
|
try:
|
|
|
|
if target_temp is not None:
|
|
|
|
self._api.set_temperature(self._session_id, serial_number, target_temp)
|
|
|
|
except RequestException as ex:
|
|
|
|
_LOGGER.error("An error occurred while setting temperature: %s", ex)
|