2016-08-19 07:17:28 +00:00
|
|
|
"""
|
2016-11-18 22:05:03 +00:00
|
|
|
Support for Z-Wave climate devices.
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/climate.zwave/
|
|
|
|
"""
|
|
|
|
# Because we do not compile openzwave on CI
|
|
|
|
# pylint: disable=import-error
|
|
|
|
import logging
|
|
|
|
from homeassistant.components.climate import DOMAIN
|
2016-11-15 12:14:29 +00:00
|
|
|
from homeassistant.components.climate import ClimateDevice
|
2016-09-30 15:43:18 +00:00
|
|
|
from homeassistant.components.zwave import ZWaveDeviceEntity
|
2016-08-19 07:17:28 +00:00
|
|
|
from homeassistant.components import zwave
|
2016-09-09 17:06:53 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
TEMP_CELSIUS, TEMP_FAHRENHEIT, ATTR_TEMPERATURE)
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
CONF_NAME = 'name'
|
2016-11-18 22:05:03 +00:00
|
|
|
DEFAULT_NAME = 'Z-Wave Climate'
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
REMOTEC = 0x5254
|
|
|
|
REMOTEC_ZXT_120 = 0x8377
|
|
|
|
REMOTEC_ZXT_120_THERMOSTAT = (REMOTEC, REMOTEC_ZXT_120)
|
2016-11-18 20:42:30 +00:00
|
|
|
ATTR_OPERATING_STATE = 'operating_state'
|
|
|
|
ATTR_FAN_STATE = 'fan_state'
|
2016-09-07 09:22:51 +00:00
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
WORKAROUND_ZXT_120 = 'zxt_120'
|
|
|
|
|
|
|
|
DEVICE_MAPPINGS = {
|
2016-11-15 12:14:29 +00:00
|
|
|
REMOTEC_ZXT_120_THERMOSTAT: WORKAROUND_ZXT_120
|
2016-08-19 07:17:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
2016-11-18 22:05:03 +00:00
|
|
|
"""Set up the Z-Wave Climate devices."""
|
2016-08-19 07:17:28 +00:00
|
|
|
if discovery_info is None or zwave.NETWORK is None:
|
|
|
|
_LOGGER.debug("No discovery_info=%s or no NETWORK=%s",
|
|
|
|
discovery_info, zwave.NETWORK)
|
|
|
|
return
|
2016-08-31 19:50:03 +00:00
|
|
|
temp_unit = hass.config.units.temperature_unit
|
2016-09-30 15:43:18 +00:00
|
|
|
node = zwave.NETWORK.nodes[discovery_info[zwave.const.ATTR_NODE_ID]]
|
|
|
|
value = node.values[discovery_info[zwave.const.ATTR_VALUE_ID]]
|
2016-08-19 07:17:28 +00:00
|
|
|
value.set_change_verified(False)
|
2016-08-31 19:50:03 +00:00
|
|
|
add_devices([ZWaveClimate(value, temp_unit)])
|
2016-08-19 07:17:28 +00:00
|
|
|
_LOGGER.debug("discovery_info=%s and zwave.NETWORK=%s",
|
|
|
|
discovery_info, zwave.NETWORK)
|
|
|
|
|
|
|
|
|
|
|
|
class ZWaveClimate(ZWaveDeviceEntity, ClimateDevice):
|
2016-11-18 22:05:03 +00:00
|
|
|
"""Representation of a Z-Wave Climate device."""
|
2016-08-19 07:17:28 +00:00
|
|
|
|
2016-08-31 19:50:03 +00:00
|
|
|
def __init__(self, value, temp_unit):
|
2016-11-18 22:05:03 +00:00
|
|
|
"""Initialize the Z-Wave climate device."""
|
2016-08-19 07:17:28 +00:00
|
|
|
ZWaveDeviceEntity.__init__(self, value, DOMAIN)
|
2016-11-15 12:14:29 +00:00
|
|
|
self._index = value.index
|
2016-08-19 07:17:28 +00:00
|
|
|
self._node = value.node
|
|
|
|
self._target_temperature = None
|
|
|
|
self._current_temperature = None
|
|
|
|
self._current_operation = None
|
|
|
|
self._operation_list = None
|
2016-10-28 05:25:17 +00:00
|
|
|
self._operating_state = None
|
2016-08-19 07:17:28 +00:00
|
|
|
self._current_fan_mode = None
|
|
|
|
self._fan_list = None
|
2016-11-15 12:14:29 +00:00
|
|
|
self._fan_state = None
|
2016-08-19 07:17:28 +00:00
|
|
|
self._current_swing_mode = None
|
|
|
|
self._swing_list = None
|
2016-08-31 19:50:03 +00:00
|
|
|
self._unit = temp_unit
|
|
|
|
_LOGGER.debug("temp_unit is %s", self._unit)
|
2016-08-19 07:17:28 +00:00
|
|
|
self._zxt_120 = None
|
|
|
|
self.update_properties()
|
|
|
|
# Make sure that we have values for the key before converting to int
|
|
|
|
if (value.node.manufacturer_id.strip() and
|
|
|
|
value.node.product_id.strip()):
|
|
|
|
specific_sensor_key = (int(value.node.manufacturer_id, 16),
|
|
|
|
int(value.node.product_id, 16))
|
|
|
|
if specific_sensor_key in DEVICE_MAPPINGS:
|
|
|
|
if DEVICE_MAPPINGS[specific_sensor_key] == WORKAROUND_ZXT_120:
|
|
|
|
_LOGGER.debug("Remotec ZXT-120 Zwave Thermostat"
|
|
|
|
" workaround")
|
|
|
|
self._zxt_120 = 1
|
|
|
|
|
|
|
|
def update_properties(self):
|
2017-01-27 06:21:33 +00:00
|
|
|
"""Callback on data changes for node values."""
|
2016-08-19 07:17:28 +00:00
|
|
|
# Operation Mode
|
2017-02-09 12:40:35 +00:00
|
|
|
self._current_operation = self.get_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_THERMOSTAT_MODE, member='data')
|
|
|
|
operation_list = self.get_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_THERMOSTAT_MODE,
|
|
|
|
member='data_items')
|
|
|
|
if operation_list:
|
|
|
|
self._operation_list = list(operation_list)
|
|
|
|
_LOGGER.debug("self._operation_list=%s", self._operation_list)
|
|
|
|
_LOGGER.debug("self._current_operation=%s", self._current_operation)
|
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
# Current Temp
|
2017-02-09 12:40:35 +00:00
|
|
|
self._current_temperature = self.get_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_SENSOR_MULTILEVEL,
|
|
|
|
label=['Temperature'], member='data')
|
|
|
|
self._unit = self.get_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_SENSOR_MULTILEVEL,
|
|
|
|
label=['Temperature'], member='units')
|
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
# Fan Mode
|
2017-02-09 12:40:35 +00:00
|
|
|
self._current_fan_mode = self.get_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_THERMOSTAT_FAN_MODE,
|
|
|
|
member='data')
|
|
|
|
fan_list = self.get_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_THERMOSTAT_FAN_MODE,
|
|
|
|
member='data_items')
|
|
|
|
if fan_list:
|
|
|
|
self._fan_list = list(fan_list)
|
|
|
|
_LOGGER.debug("self._fan_list=%s", self._fan_list)
|
|
|
|
_LOGGER.debug("self._current_fan_mode=%s",
|
|
|
|
self._current_fan_mode)
|
2016-08-19 07:17:28 +00:00
|
|
|
# Swing mode
|
|
|
|
if self._zxt_120 == 1:
|
2017-02-09 12:40:35 +00:00
|
|
|
self._current_swing_mode = (
|
|
|
|
self.get_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_CONFIGURATION,
|
|
|
|
index=33,
|
|
|
|
member='data'))
|
|
|
|
swing_list = self.get_value(class_id=zwave.const
|
|
|
|
.COMMAND_CLASS_CONFIGURATION,
|
|
|
|
index=33,
|
|
|
|
member='data_items')
|
|
|
|
if swing_list:
|
|
|
|
self._swing_list = list(swing_list)
|
|
|
|
_LOGGER.debug("self._swing_list=%s", self._swing_list)
|
|
|
|
_LOGGER.debug("self._current_swing_mode=%s",
|
|
|
|
self._current_swing_mode)
|
2016-08-31 19:50:03 +00:00
|
|
|
# Set point
|
2016-11-15 12:14:29 +00:00
|
|
|
temps = []
|
2016-11-19 09:19:22 +00:00
|
|
|
for value in (
|
|
|
|
self._node.get_values(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_THERMOSTAT_SETPOINT)
|
|
|
|
.values()):
|
2016-11-15 12:14:29 +00:00
|
|
|
temps.append((round(float(value.data)), 1))
|
|
|
|
if value.index == self._index:
|
|
|
|
if value.data == 0:
|
|
|
|
_LOGGER.debug("Setpoint is 0, setting default to "
|
|
|
|
"current_temperature=%s",
|
|
|
|
self._current_temperature)
|
|
|
|
self._target_temperature = (
|
|
|
|
round((float(self._current_temperature)), 1))
|
2016-09-09 17:06:53 +00:00
|
|
|
break
|
2016-11-15 12:14:29 +00:00
|
|
|
else:
|
|
|
|
self._target_temperature = round((float(value.data)), 1)
|
2017-02-09 12:40:35 +00:00
|
|
|
|
2016-10-28 05:25:17 +00:00
|
|
|
# Operating state
|
2017-02-09 12:40:35 +00:00
|
|
|
self._operating_state = self.get_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_THERMOSTAT_OPERATING_STATE,
|
|
|
|
member='data')
|
2016-08-19 07:17:28 +00:00
|
|
|
|
2016-11-15 12:14:29 +00:00
|
|
|
# Fan operating state
|
2017-02-09 12:40:35 +00:00
|
|
|
self._fan_state = self.get_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_THERMOSTAT_FAN_STATE,
|
|
|
|
member='data')
|
2016-11-15 12:14:29 +00:00
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
@property
|
|
|
|
def should_poll(self):
|
2016-11-18 22:05:03 +00:00
|
|
|
"""No polling on Z-Wave."""
|
2016-08-19 07:17:28 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_fan_mode(self):
|
|
|
|
"""Return the fan speed set."""
|
|
|
|
return self._current_fan_mode
|
|
|
|
|
|
|
|
@property
|
|
|
|
def fan_list(self):
|
|
|
|
"""List of available fan modes."""
|
|
|
|
return self._fan_list
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_swing_mode(self):
|
|
|
|
"""Return the swing mode set."""
|
|
|
|
return self._current_swing_mode
|
|
|
|
|
|
|
|
@property
|
|
|
|
def swing_list(self):
|
|
|
|
"""List of available swing modes."""
|
|
|
|
return self._swing_list
|
|
|
|
|
|
|
|
@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."""
|
2016-09-02 05:17:41 +00:00
|
|
|
if self._unit == 'C':
|
|
|
|
return TEMP_CELSIUS
|
|
|
|
elif self._unit == 'F':
|
|
|
|
return TEMP_FAHRENHEIT
|
|
|
|
else:
|
|
|
|
return self._unit
|
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 mode."""
|
|
|
|
return self._current_operation
|
|
|
|
|
|
|
|
@property
|
|
|
|
def operation_list(self):
|
|
|
|
"""List of available operation modes."""
|
|
|
|
return self._operation_list
|
|
|
|
|
|
|
|
@property
|
|
|
|
def target_temperature(self):
|
|
|
|
"""Return the temperature we try to reach."""
|
|
|
|
return self._target_temperature
|
|
|
|
|
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
|
|
|
if kwargs.get(ATTR_TEMPERATURE) is not None:
|
|
|
|
temperature = kwargs.get(ATTR_TEMPERATURE)
|
|
|
|
else:
|
|
|
|
return
|
|
|
|
|
2017-02-09 12:40:35 +00:00
|
|
|
self.set_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_THERMOSTAT_SETPOINT,
|
|
|
|
index=self._index, data=temperature)
|
|
|
|
self.update_ha_state()
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
def set_fan_mode(self, fan):
|
|
|
|
"""Set new target fan mode."""
|
2017-02-09 12:40:35 +00:00
|
|
|
self.set_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_THERMOSTAT_FAN_MODE,
|
|
|
|
index=0, data=bytes(fan, 'utf-8'))
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
def set_operation_mode(self, operation_mode):
|
|
|
|
"""Set new target operation mode."""
|
2017-02-09 12:40:35 +00:00
|
|
|
self.set_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_THERMOSTAT_MODE,
|
|
|
|
index=0, data=bytes(operation_mode, 'utf-8'))
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
def set_swing_mode(self, swing_mode):
|
|
|
|
"""Set new target swing mode."""
|
|
|
|
if self._zxt_120 == 1:
|
2017-02-09 12:40:35 +00:00
|
|
|
self.set_value(
|
|
|
|
class_id=zwave.const.COMMAND_CLASS_CONFIGURATION,
|
|
|
|
index=33, data=bytes(swing_mode, 'utf-8'))
|
2016-10-28 05:25:17 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the device specific state attributes."""
|
2016-11-18 20:42:30 +00:00
|
|
|
data = super().device_state_attributes
|
2016-10-28 05:25:17 +00:00
|
|
|
if self._operating_state:
|
2016-11-18 20:42:30 +00:00
|
|
|
data[ATTR_OPERATING_STATE] = self._operating_state,
|
2016-11-15 12:14:29 +00:00
|
|
|
if self._fan_state:
|
2016-11-18 20:42:30 +00:00
|
|
|
data[ATTR_FAN_STATE] = self._fan_state
|
2016-11-15 12:14:29 +00:00
|
|
|
return data
|