2017-11-11 06:22:37 +00:00
|
|
|
"""
|
|
|
|
Support for NuHeat thermostats.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
2017-11-11 22:31:35 +00:00
|
|
|
https://home-assistant.io/components/climate.nuheat/
|
2017-11-11 06:22:37 +00:00
|
|
|
"""
|
|
|
|
import logging
|
|
|
|
from datetime import timedelta
|
|
|
|
|
|
|
|
from homeassistant.components.climate import (
|
|
|
|
ClimateDevice,
|
2017-12-07 04:24:54 +00:00
|
|
|
SUPPORT_HOLD_MODE,
|
|
|
|
SUPPORT_OPERATION_MODE,
|
|
|
|
SUPPORT_TARGET_TEMPERATURE,
|
2017-12-24 17:09:27 +00:00
|
|
|
STATE_AUTO,
|
2017-11-11 06:22:37 +00:00
|
|
|
STATE_HEAT,
|
|
|
|
STATE_IDLE)
|
2017-11-11 16:18:32 +00:00
|
|
|
from homeassistant.components.nuheat import DATA_NUHEAT
|
2017-11-11 06:22:37 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_TEMPERATURE,
|
|
|
|
TEMP_CELSIUS,
|
|
|
|
TEMP_FAHRENHEIT)
|
|
|
|
from homeassistant.util import Throttle
|
|
|
|
|
|
|
|
DEPENDENCIES = ["nuheat"]
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2017-11-14 05:13:04 +00:00
|
|
|
ICON = "mdi:thermometer"
|
|
|
|
|
2017-11-11 06:22:37 +00:00
|
|
|
MIN_TIME_BETWEEN_UPDATES = timedelta(minutes=5)
|
|
|
|
|
2017-11-13 17:38:08 +00:00
|
|
|
# Hold modes
|
2017-12-24 17:09:27 +00:00
|
|
|
MODE_AUTO = STATE_AUTO # Run device schedule
|
2017-11-11 06:22:37 +00:00
|
|
|
MODE_HOLD_TEMPERATURE = "temperature"
|
|
|
|
MODE_TEMPORARY_HOLD = "temporary_temperature"
|
|
|
|
|
|
|
|
OPERATION_LIST = [STATE_HEAT, STATE_IDLE]
|
|
|
|
|
|
|
|
SCHEDULE_HOLD = 3
|
|
|
|
SCHEDULE_RUN = 1
|
|
|
|
SCHEDULE_TEMPORARY_HOLD = 2
|
|
|
|
|
2017-12-07 04:24:54 +00:00
|
|
|
SUPPORT_FLAGS = (SUPPORT_TARGET_TEMPERATURE | SUPPORT_HOLD_MODE |
|
2017-12-07 04:34:13 +00:00
|
|
|
SUPPORT_OPERATION_MODE)
|
2017-12-07 04:24:54 +00:00
|
|
|
|
2017-11-11 06:22:37 +00:00
|
|
|
|
|
|
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
2017-11-11 16:18:32 +00:00
|
|
|
"""Set up the NuHeat thermostat(s)."""
|
2017-11-11 06:22:37 +00:00
|
|
|
if discovery_info is None:
|
|
|
|
return
|
|
|
|
|
|
|
|
temperature_unit = hass.config.units.temperature_unit
|
2017-12-07 04:34:13 +00:00
|
|
|
api, serial_numbers = hass.data[DATA_NUHEAT]
|
2017-11-11 06:22:37 +00:00
|
|
|
thermostats = [
|
2017-12-07 04:34:13 +00:00
|
|
|
NuHeatThermostat(api, serial_number, temperature_unit)
|
2017-11-11 06:22:37 +00:00
|
|
|
for serial_number in serial_numbers
|
|
|
|
]
|
|
|
|
add_devices(thermostats, True)
|
|
|
|
|
|
|
|
|
|
|
|
class NuHeatThermostat(ClimateDevice):
|
|
|
|
"""Representation of a NuHeat Thermostat."""
|
2017-11-14 05:13:04 +00:00
|
|
|
|
2017-12-07 04:34:13 +00:00
|
|
|
def __init__(self, api, serial_number, temperature_unit):
|
2017-11-14 05:13:04 +00:00
|
|
|
"""Initialize the thermostat."""
|
2017-11-11 06:22:37 +00:00
|
|
|
self._thermostat = api.get_thermostat(serial_number)
|
|
|
|
self._temperature_unit = temperature_unit
|
|
|
|
self._force_update = False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the thermostat."""
|
|
|
|
return self._thermostat.room
|
|
|
|
|
2017-11-14 05:13:04 +00:00
|
|
|
@property
|
|
|
|
def icon(self):
|
|
|
|
"""Return the icon to use in the frontend."""
|
|
|
|
return ICON
|
|
|
|
|
2017-12-07 04:24:54 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Return the list of supported features."""
|
|
|
|
return SUPPORT_FLAGS
|
|
|
|
|
2017-11-11 06:22:37 +00:00
|
|
|
@property
|
|
|
|
def temperature_unit(self):
|
|
|
|
"""Return the unit of measurement."""
|
|
|
|
if self._temperature_unit == "C":
|
|
|
|
return TEMP_CELSIUS
|
|
|
|
|
|
|
|
return TEMP_FAHRENHEIT
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_temperature(self):
|
|
|
|
"""Return the current temperature."""
|
|
|
|
if self._temperature_unit == "C":
|
|
|
|
return self._thermostat.celsius
|
2017-11-11 16:18:32 +00:00
|
|
|
|
2017-11-11 06:22:37 +00:00
|
|
|
return self._thermostat.fahrenheit
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_operation(self):
|
|
|
|
"""Return current operation. ie. heat, idle."""
|
|
|
|
if self._thermostat.heating:
|
|
|
|
return STATE_HEAT
|
|
|
|
|
|
|
|
return STATE_IDLE
|
|
|
|
|
|
|
|
@property
|
|
|
|
def min_temp(self):
|
|
|
|
"""Return the minimum supported temperature for the thermostat."""
|
|
|
|
if self._temperature_unit == "C":
|
|
|
|
return self._thermostat.min_celsius
|
|
|
|
|
|
|
|
return self._thermostat.min_fahrenheit
|
|
|
|
|
|
|
|
@property
|
|
|
|
def max_temp(self):
|
|
|
|
"""Return the maximum supported temperature for the thermostat."""
|
|
|
|
if self._temperature_unit == "C":
|
|
|
|
return self._thermostat.max_celsius
|
|
|
|
|
|
|
|
return self._thermostat.max_fahrenheit
|
|
|
|
|
|
|
|
@property
|
|
|
|
def target_temperature(self):
|
|
|
|
"""Return the currently programmed temperature."""
|
|
|
|
if self._temperature_unit == "C":
|
|
|
|
return self._thermostat.target_celsius
|
|
|
|
|
|
|
|
return self._thermostat.target_fahrenheit
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_hold_mode(self):
|
|
|
|
"""Return current hold mode."""
|
|
|
|
schedule_mode = self._thermostat.schedule_mode
|
|
|
|
if schedule_mode == SCHEDULE_RUN:
|
|
|
|
return MODE_AUTO
|
|
|
|
|
|
|
|
if schedule_mode == SCHEDULE_HOLD:
|
|
|
|
return MODE_HOLD_TEMPERATURE
|
|
|
|
|
|
|
|
if schedule_mode == SCHEDULE_TEMPORARY_HOLD:
|
|
|
|
return MODE_TEMPORARY_HOLD
|
|
|
|
|
|
|
|
return MODE_AUTO
|
|
|
|
|
|
|
|
@property
|
|
|
|
def operation_list(self):
|
|
|
|
"""Return list of possible operation modes."""
|
|
|
|
return OPERATION_LIST
|
|
|
|
|
2017-11-11 22:21:14 +00:00
|
|
|
def resume_program(self):
|
|
|
|
"""Resume the thermostat's programmed schedule."""
|
2017-11-11 06:22:37 +00:00
|
|
|
self._thermostat.resume_schedule()
|
|
|
|
self._force_update = True
|
|
|
|
|
2017-12-24 17:09:27 +00:00
|
|
|
def set_hold_mode(self, hold_mode, **kwargs):
|
|
|
|
"""Update the hold mode of the thermostat."""
|
|
|
|
if hold_mode == MODE_AUTO:
|
|
|
|
schedule_mode = SCHEDULE_RUN
|
|
|
|
|
|
|
|
if hold_mode == MODE_HOLD_TEMPERATURE:
|
|
|
|
schedule_mode = SCHEDULE_HOLD
|
|
|
|
|
|
|
|
if hold_mode == MODE_TEMPORARY_HOLD:
|
|
|
|
schedule_mode = SCHEDULE_TEMPORARY_HOLD
|
|
|
|
|
|
|
|
self._thermostat.schedule_mode = schedule_mode
|
|
|
|
self._force_update = True
|
|
|
|
|
2017-11-11 06:22:37 +00:00
|
|
|
def set_temperature(self, **kwargs):
|
|
|
|
"""Set a new target temperature."""
|
|
|
|
temperature = kwargs.get(ATTR_TEMPERATURE)
|
|
|
|
if self._temperature_unit == "C":
|
|
|
|
self._thermostat.target_celsius = temperature
|
|
|
|
else:
|
|
|
|
self._thermostat.target_fahrenheit = temperature
|
|
|
|
|
2017-12-07 03:52:44 +00:00
|
|
|
_LOGGER.debug(
|
2017-11-11 06:22:37 +00:00
|
|
|
"Setting NuHeat thermostat temperature to %s %s",
|
|
|
|
temperature, self.temperature_unit)
|
|
|
|
|
|
|
|
self._force_update = True
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Get the latest state from the thermostat."""
|
|
|
|
if self._force_update:
|
|
|
|
self._throttled_update(no_throttle=True)
|
|
|
|
self._force_update = False
|
|
|
|
else:
|
|
|
|
self._throttled_update()
|
|
|
|
|
|
|
|
@Throttle(MIN_TIME_BETWEEN_UPDATES)
|
2017-11-11 16:18:32 +00:00
|
|
|
def _throttled_update(self, **kwargs):
|
2017-11-14 05:13:04 +00:00
|
|
|
"""Get the latest state from the thermostat with a throttle."""
|
2017-11-11 06:22:37 +00:00
|
|
|
self._thermostat.get_data()
|