2016-08-19 07:17:28 +00:00
|
|
|
"""
|
|
|
|
Support for Radio Thermostat wifi-enabled home thermostats.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/climate.radiotherm/
|
|
|
|
"""
|
|
|
|
import datetime
|
|
|
|
import logging
|
|
|
|
|
2016-09-11 09:38:43 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
from homeassistant.components.climate import (
|
|
|
|
STATE_AUTO, STATE_COOL, STATE_HEAT, STATE_IDLE, STATE_OFF,
|
2016-09-11 09:38:43 +00:00
|
|
|
ClimateDevice, PLATFORM_SCHEMA)
|
2016-09-09 17:06:53 +00:00
|
|
|
from homeassistant.const import CONF_HOST, TEMP_FAHRENHEIT, ATTR_TEMPERATURE
|
2016-09-11 09:38:43 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-08-19 07:17:28 +00:00
|
|
|
|
2017-06-23 09:08:23 +00:00
|
|
|
REQUIREMENTS = ['radiotherm==1.3']
|
2016-09-11 09:38:43 +00:00
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2016-09-11 09:38:43 +00:00
|
|
|
ATTR_FAN = 'fan'
|
|
|
|
ATTR_MODE = 'mode'
|
|
|
|
|
|
|
|
CONF_HOLD_TEMP = 'hold_temp'
|
2016-12-09 07:26:02 +00:00
|
|
|
CONF_AWAY_TEMPERATURE_HEAT = 'away_temperature_heat'
|
|
|
|
CONF_AWAY_TEMPERATURE_COOL = 'away_temperature_cool'
|
|
|
|
|
|
|
|
DEFAULT_AWAY_TEMPERATURE_HEAT = 60
|
|
|
|
DEFAULT_AWAY_TEMPERATURE_COOL = 85
|
2016-09-11 09:38:43 +00:00
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Optional(CONF_HOST): vol.All(cv.ensure_list, [cv.string]),
|
|
|
|
vol.Optional(CONF_HOLD_TEMP, default=False): cv.boolean,
|
2016-12-09 07:26:02 +00:00
|
|
|
vol.Optional(CONF_AWAY_TEMPERATURE_HEAT,
|
|
|
|
default=DEFAULT_AWAY_TEMPERATURE_HEAT): vol.Coerce(float),
|
|
|
|
vol.Optional(CONF_AWAY_TEMPERATURE_COOL,
|
|
|
|
default=DEFAULT_AWAY_TEMPERATURE_COOL): vol.Coerce(float),
|
2016-09-11 09:38:43 +00:00
|
|
|
})
|
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up the Radio Thermostat."""
|
2016-08-19 07:17:28 +00:00
|
|
|
import radiotherm
|
|
|
|
|
|
|
|
hosts = []
|
|
|
|
if CONF_HOST in config:
|
|
|
|
hosts = config[CONF_HOST]
|
|
|
|
else:
|
|
|
|
hosts.append(radiotherm.discover.discover_address())
|
|
|
|
|
|
|
|
if hosts is None:
|
2016-09-11 09:38:43 +00:00
|
|
|
_LOGGER.error("No Radiotherm Thermostats detected")
|
2016-08-19 07:17:28 +00:00
|
|
|
return False
|
|
|
|
|
2016-09-11 09:38:43 +00:00
|
|
|
hold_temp = config.get(CONF_HOLD_TEMP)
|
2016-12-09 07:26:02 +00:00
|
|
|
away_temps = [
|
|
|
|
config.get(CONF_AWAY_TEMPERATURE_HEAT),
|
|
|
|
config.get(CONF_AWAY_TEMPERATURE_COOL)
|
|
|
|
]
|
2016-08-19 07:17:28 +00:00
|
|
|
tstats = []
|
|
|
|
|
|
|
|
for host in hosts:
|
|
|
|
try:
|
|
|
|
tstat = radiotherm.get_thermostat(host)
|
2016-12-09 07:26:02 +00:00
|
|
|
tstats.append(RadioThermostat(tstat, hold_temp, away_temps))
|
2016-10-29 22:33:56 +00:00
|
|
|
except OSError:
|
2016-08-19 07:17:28 +00:00
|
|
|
_LOGGER.exception("Unable to connect to Radio Thermostat: %s",
|
|
|
|
host)
|
|
|
|
|
2017-08-08 18:21:33 +00:00
|
|
|
add_devices(tstats, True)
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RadioThermostat(ClimateDevice):
|
|
|
|
"""Representation of a Radio Thermostat."""
|
|
|
|
|
2016-12-09 07:26:02 +00:00
|
|
|
def __init__(self, device, hold_temp, away_temps):
|
2016-08-19 07:17:28 +00:00
|
|
|
"""Initialize the thermostat."""
|
|
|
|
self.device = device
|
|
|
|
self.set_time()
|
|
|
|
self._target_temperature = None
|
|
|
|
self._current_temperature = None
|
|
|
|
self._current_operation = STATE_IDLE
|
|
|
|
self._name = None
|
2016-11-05 20:28:11 +00:00
|
|
|
self._fmode = None
|
|
|
|
self._tmode = None
|
2017-06-24 05:53:10 +00:00
|
|
|
self._tstate = None
|
2016-12-09 07:26:02 +00:00
|
|
|
self._hold_temp = hold_temp
|
|
|
|
self._away = False
|
|
|
|
self._away_temps = away_temps
|
|
|
|
self._prev_temp = None
|
2016-09-15 01:14:39 +00:00
|
|
|
self._operation_list = [STATE_AUTO, STATE_COOL, STATE_HEAT, STATE_OFF]
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the Radio Thermostat."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
2016-10-11 07:00:29 +00:00
|
|
|
def temperature_unit(self):
|
2016-08-19 07:17:28 +00:00
|
|
|
"""Return the unit of measurement."""
|
|
|
|
return TEMP_FAHRENHEIT
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the device specific state attributes."""
|
|
|
|
return {
|
2016-11-05 20:28:11 +00:00
|
|
|
ATTR_FAN: self._fmode,
|
|
|
|
ATTR_MODE: self._tmode,
|
2016-08-19 07:17:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_temperature(self):
|
|
|
|
"""Return the current temperature."""
|
|
|
|
return self._current_temperature
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_operation(self):
|
|
|
|
"""Return the current operation. head, cool idle."""
|
|
|
|
return self._current_operation
|
|
|
|
|
2016-09-15 01:14:39 +00:00
|
|
|
@property
|
|
|
|
def operation_list(self):
|
|
|
|
"""Return the operation modes list."""
|
|
|
|
return self._operation_list
|
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
@property
|
|
|
|
def target_temperature(self):
|
|
|
|
"""Return the temperature we try to reach."""
|
|
|
|
return self._target_temperature
|
|
|
|
|
2016-12-09 07:26:02 +00:00
|
|
|
@property
|
|
|
|
def is_away_mode_on(self):
|
|
|
|
"""Return true if away mode is on."""
|
|
|
|
return self._away
|
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
def update(self):
|
2017-07-11 08:12:51 +00:00
|
|
|
"""Update and validate the data from the thermostat."""
|
|
|
|
current_temp = self.device.temp['raw']
|
|
|
|
if current_temp == -1:
|
|
|
|
_LOGGER.error("Couldn't get valid temperature reading")
|
|
|
|
return
|
|
|
|
self._current_temperature = current_temp
|
2016-08-19 07:17:28 +00:00
|
|
|
self._name = self.device.name['raw']
|
2017-07-11 08:12:51 +00:00
|
|
|
try:
|
|
|
|
self._fmode = self.device.fmode['human']
|
|
|
|
except AttributeError:
|
|
|
|
_LOGGER.error("Couldn't get valid fan mode reading")
|
|
|
|
try:
|
|
|
|
self._tmode = self.device.tmode['human']
|
|
|
|
except AttributeError:
|
|
|
|
_LOGGER.error("Couldn't get valid thermostat mode reading")
|
|
|
|
try:
|
|
|
|
self._tstate = self.device.tstate['human']
|
|
|
|
except AttributeError:
|
|
|
|
_LOGGER.error("Couldn't get valid thermostat state reading")
|
2016-11-05 20:28:11 +00:00
|
|
|
|
|
|
|
if self._tmode == 'Cool':
|
2017-07-11 08:12:51 +00:00
|
|
|
target_temp = self.device.t_cool['raw']
|
|
|
|
if target_temp == -1:
|
|
|
|
_LOGGER.error("Couldn't get target reading")
|
|
|
|
return
|
|
|
|
self._target_temperature = target_temp
|
2016-08-19 07:17:28 +00:00
|
|
|
self._current_operation = STATE_COOL
|
2016-11-05 20:28:11 +00:00
|
|
|
elif self._tmode == 'Heat':
|
2017-07-11 08:12:51 +00:00
|
|
|
target_temp = self.device.t_heat['raw']
|
|
|
|
if target_temp == -1:
|
|
|
|
_LOGGER.error("Couldn't get valid target reading")
|
|
|
|
return
|
|
|
|
self._target_temperature = target_temp
|
2016-08-19 07:17:28 +00:00
|
|
|
self._current_operation = STATE_HEAT
|
2017-06-24 05:53:10 +00:00
|
|
|
elif self._tmode == 'Auto':
|
|
|
|
if self._tstate == 'Cool':
|
2017-07-11 08:12:51 +00:00
|
|
|
target_temp = self.device.t_cool['raw']
|
|
|
|
if target_temp == -1:
|
|
|
|
_LOGGER.error("Couldn't get valid target reading")
|
|
|
|
return
|
|
|
|
self._target_temperature = target_temp
|
2017-06-24 05:53:10 +00:00
|
|
|
elif self._tstate == 'Heat':
|
2017-07-11 08:12:51 +00:00
|
|
|
target_temp = self.device.t_heat['raw']
|
|
|
|
if target_temp == -1:
|
|
|
|
_LOGGER.error("Couldn't get valid target reading")
|
|
|
|
return
|
|
|
|
self._target_temperature = target_temp
|
2017-06-24 05:53:10 +00:00
|
|
|
self._current_operation = STATE_AUTO
|
2016-08-19 07:17:28 +00:00
|
|
|
else:
|
|
|
|
self._current_operation = STATE_IDLE
|
|
|
|
|
2016-09-09 17:06:53 +00:00
|
|
|
def set_temperature(self, **kwargs):
|
2016-08-19 07:17:28 +00:00
|
|
|
"""Set new target temperature."""
|
2016-09-09 17:06:53 +00:00
|
|
|
temperature = kwargs.get(ATTR_TEMPERATURE)
|
|
|
|
if temperature is None:
|
|
|
|
return
|
2016-08-19 07:17:28 +00:00
|
|
|
if self._current_operation == STATE_COOL:
|
2016-11-07 07:18:06 +00:00
|
|
|
self.device.t_cool = round(temperature * 2.0) / 2.0
|
2016-08-19 07:17:28 +00:00
|
|
|
elif self._current_operation == STATE_HEAT:
|
2016-11-07 07:18:06 +00:00
|
|
|
self.device.t_heat = round(temperature * 2.0) / 2.0
|
2017-06-24 05:53:10 +00:00
|
|
|
elif self._current_operation == STATE_AUTO:
|
|
|
|
if self._tstate == 'Cool':
|
|
|
|
self.device.t_cool = round(temperature * 2.0) / 2.0
|
|
|
|
elif self._tstate == 'Heat':
|
|
|
|
self.device.t_heat = round(temperature * 2.0) / 2.0
|
|
|
|
|
2016-12-09 07:26:02 +00:00
|
|
|
if self._hold_temp or self._away:
|
2016-08-19 07:17:28 +00:00
|
|
|
self.device.hold = 1
|
|
|
|
else:
|
|
|
|
self.device.hold = 0
|
|
|
|
|
|
|
|
def set_time(self):
|
|
|
|
"""Set device time."""
|
|
|
|
now = datetime.datetime.now()
|
2016-09-11 09:38:43 +00:00
|
|
|
self.device.time = {
|
|
|
|
'day': now.weekday(),
|
|
|
|
'hour': now.hour,
|
|
|
|
'minute': now.minute
|
|
|
|
}
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
def set_operation_mode(self, operation_mode):
|
|
|
|
"""Set operation mode (auto, cool, heat, off)."""
|
|
|
|
if operation_mode == STATE_OFF:
|
|
|
|
self.device.tmode = 0
|
|
|
|
elif operation_mode == STATE_AUTO:
|
|
|
|
self.device.tmode = 3
|
|
|
|
elif operation_mode == STATE_COOL:
|
2016-11-07 07:18:06 +00:00
|
|
|
self.device.t_cool = round(self._target_temperature * 2.0) / 2.0
|
2016-08-19 07:17:28 +00:00
|
|
|
elif operation_mode == STATE_HEAT:
|
2016-11-07 07:18:06 +00:00
|
|
|
self.device.t_heat = round(self._target_temperature * 2.0) / 2.0
|
2016-12-09 07:26:02 +00:00
|
|
|
|
|
|
|
def turn_away_mode_on(self):
|
|
|
|
"""Turn away on.
|
|
|
|
|
|
|
|
The RTCOA app simulates away mode by using a hold.
|
|
|
|
"""
|
|
|
|
away_temp = None
|
|
|
|
if not self._away:
|
|
|
|
self._prev_temp = self._target_temperature
|
|
|
|
if self._current_operation == STATE_HEAT:
|
|
|
|
away_temp = self._away_temps[0]
|
|
|
|
elif self._current_operation == STATE_COOL:
|
|
|
|
away_temp = self._away_temps[1]
|
|
|
|
self._away = True
|
|
|
|
self.set_temperature(temperature=away_temp)
|
|
|
|
|
|
|
|
def turn_away_mode_off(self):
|
|
|
|
"""Turn away off."""
|
|
|
|
self._away = False
|
|
|
|
self.set_temperature(temperature=self._prev_temp)
|