2019-05-07 15:37:47 +00:00
|
|
|
"""Support for Genius Hub climate devices."""
|
2019-04-16 21:54:46 +00:00
|
|
|
import asyncio
|
|
|
|
import logging
|
|
|
|
|
|
|
|
from homeassistant.components.climate import ClimateDevice
|
|
|
|
from homeassistant.components.climate.const import (
|
|
|
|
STATE_AUTO, STATE_ECO, STATE_HEAT, STATE_MANUAL,
|
|
|
|
SUPPORT_TARGET_TEMPERATURE, SUPPORT_OPERATION_MODE, SUPPORT_ON_OFF)
|
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_TEMPERATURE, TEMP_CELSIUS)
|
|
|
|
|
|
|
|
from . import DOMAIN
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-05-07 15:37:47 +00:00
|
|
|
GH_CLIMATE_DEVICES = ['radiator']
|
|
|
|
|
2019-04-16 21:54:46 +00:00
|
|
|
GENIUSHUB_SUPPORT_FLAGS = \
|
|
|
|
SUPPORT_TARGET_TEMPERATURE | \
|
|
|
|
SUPPORT_ON_OFF | \
|
|
|
|
SUPPORT_OPERATION_MODE
|
|
|
|
|
|
|
|
GENIUSHUB_MAX_TEMP = 28.0
|
|
|
|
GENIUSHUB_MIN_TEMP = 4.0
|
|
|
|
|
2019-05-07 15:37:47 +00:00
|
|
|
# Genius Hub Zones support only Off, Override/Boost, Footprint & Timer modes
|
2019-04-16 21:54:46 +00:00
|
|
|
HA_OPMODE_TO_GH = {
|
|
|
|
STATE_AUTO: 'timer',
|
|
|
|
STATE_ECO: 'footprint',
|
|
|
|
STATE_MANUAL: 'override',
|
|
|
|
}
|
|
|
|
GH_OPMODE_OFF = 'off'
|
|
|
|
GH_STATE_TO_HA = {
|
|
|
|
'timer': STATE_AUTO,
|
|
|
|
'footprint': STATE_ECO,
|
|
|
|
'away': None,
|
|
|
|
'override': STATE_MANUAL,
|
|
|
|
'early': STATE_HEAT,
|
|
|
|
'test': None,
|
|
|
|
'linked': None,
|
|
|
|
'other': None,
|
|
|
|
} # intentionally missing 'off': None
|
2019-05-07 15:37:47 +00:00
|
|
|
|
|
|
|
# temperature is repeated here, as it gives access to high-precision temps
|
2019-04-16 21:54:46 +00:00
|
|
|
GH_DEVICE_STATE_ATTRS = ['temperature', 'type', 'occupied', 'override']
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_platform(hass, hass_config, async_add_entities,
|
|
|
|
discovery_info=None):
|
2019-05-07 15:37:47 +00:00
|
|
|
"""Set up the Genius Hub climate entities."""
|
2019-04-16 21:54:46 +00:00
|
|
|
client = hass.data[DOMAIN]['client']
|
|
|
|
|
2019-05-07 15:37:47 +00:00
|
|
|
entities = [GeniusClimate(client, z)
|
|
|
|
for z in client.hub.zone_objs if z.type in GH_CLIMATE_DEVICES]
|
2019-04-16 21:54:46 +00:00
|
|
|
|
2019-05-07 15:37:47 +00:00
|
|
|
async_add_entities(entities)
|
2019-04-16 21:54:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
class GeniusClimate(ClimateDevice):
|
|
|
|
"""Representation of a Genius Hub climate device."""
|
|
|
|
|
|
|
|
def __init__(self, client, zone):
|
|
|
|
"""Initialize the climate device."""
|
|
|
|
self._client = client
|
|
|
|
self._objref = zone
|
|
|
|
self._id = zone.id
|
|
|
|
self._name = zone.name
|
|
|
|
|
|
|
|
# Only some zones have movement detectors, which allows footprint mode
|
|
|
|
op_list = list(HA_OPMODE_TO_GH)
|
|
|
|
if not hasattr(self._objref, 'occupied'):
|
|
|
|
op_list.remove(STATE_ECO)
|
|
|
|
self._operation_list = op_list
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the climate device."""
|
|
|
|
return self._objref.name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the device state attributes."""
|
|
|
|
tmp = self._objref.__dict__.items()
|
|
|
|
state = {k: v for k, v in tmp if k in GH_DEVICE_STATE_ATTRS}
|
|
|
|
|
|
|
|
return {'status': state}
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_temperature(self):
|
|
|
|
"""Return the current temperature."""
|
|
|
|
return self._objref.temperature
|
|
|
|
|
|
|
|
@property
|
|
|
|
def target_temperature(self):
|
|
|
|
"""Return the temperature we try to reach."""
|
|
|
|
return self._objref.setpoint
|
|
|
|
|
|
|
|
@property
|
|
|
|
def min_temp(self):
|
|
|
|
"""Return max valid temperature that can be set."""
|
|
|
|
return GENIUSHUB_MIN_TEMP
|
|
|
|
|
|
|
|
@property
|
|
|
|
def max_temp(self):
|
|
|
|
"""Return max valid temperature that can be set."""
|
|
|
|
return GENIUSHUB_MAX_TEMP
|
|
|
|
|
|
|
|
@property
|
|
|
|
def temperature_unit(self):
|
|
|
|
"""Return the unit of measurement."""
|
|
|
|
return TEMP_CELSIUS
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Return the list of supported features."""
|
|
|
|
return GENIUSHUB_SUPPORT_FLAGS
|
|
|
|
|
|
|
|
@property
|
|
|
|
def operation_list(self):
|
|
|
|
"""Return the list of available operation modes."""
|
|
|
|
return self._operation_list
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_operation(self):
|
|
|
|
"""Return the current operation mode."""
|
|
|
|
return GH_STATE_TO_HA.get(self._objref.mode)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return True if the device is on."""
|
|
|
|
return self._objref.mode in GH_STATE_TO_HA
|
|
|
|
|
|
|
|
async def async_set_operation_mode(self, operation_mode):
|
|
|
|
"""Set a new operation mode for this zone."""
|
|
|
|
await self._objref.set_mode(HA_OPMODE_TO_GH.get(operation_mode))
|
|
|
|
|
|
|
|
async def async_set_temperature(self, **kwargs):
|
|
|
|
"""Set a new target temperature for this zone."""
|
|
|
|
temperature = kwargs.get(ATTR_TEMPERATURE)
|
|
|
|
await self._objref.set_override(temperature, 3600) # 1 hour
|
|
|
|
|
|
|
|
async def async_turn_on(self):
|
|
|
|
"""Turn on this heating zone."""
|
|
|
|
await self._objref.set_mode(HA_OPMODE_TO_GH.get(STATE_AUTO))
|
|
|
|
|
|
|
|
async def async_turn_off(self):
|
|
|
|
"""Turn off this heating zone (i.e. to frost protect)."""
|
|
|
|
await self._objref.set_mode(GH_OPMODE_OFF)
|
|
|
|
|
|
|
|
async def async_update(self):
|
|
|
|
"""Get the latest data from the hub."""
|
|
|
|
try:
|
|
|
|
await self._objref.update()
|
|
|
|
except (AssertionError, asyncio.TimeoutError) as err:
|
|
|
|
_LOGGER.warning("Update for %s failed, message: %s",
|
|
|
|
self._id, err)
|