Home Assistant Unofficial Reference 2024.12.1
climate.py
Go to the documentation of this file.
1 """Support for the EPH Controls Ember themostats."""
2 
3 from __future__ import annotations
4 
5 from datetime import timedelta
6 import logging
7 from typing import Any
8 
9 from pyephember.pyephember import (
10  EphEmber,
11  ZoneMode,
12  zone_current_temperature,
13  zone_is_active,
14  zone_is_boost_active,
15  zone_is_hot_water,
16  zone_mode,
17  zone_name,
18  zone_target_temperature,
19 )
20 import voluptuous as vol
21 
23  PLATFORM_SCHEMA as CLIMATE_PLATFORM_SCHEMA,
24  ClimateEntity,
25  ClimateEntityFeature,
26  HVACAction,
27  HVACMode,
28 )
29 from homeassistant.const import (
30  ATTR_TEMPERATURE,
31  CONF_PASSWORD,
32  CONF_USERNAME,
33  UnitOfTemperature,
34 )
35 from homeassistant.core import HomeAssistant
37 from homeassistant.helpers.entity_platform import AddEntitiesCallback
38 from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
39 
40 _LOGGER = logging.getLogger(__name__)
41 
42 # Return cached results if last scan was less then this time ago
43 SCAN_INTERVAL = timedelta(seconds=120)
44 
45 OPERATION_LIST = [HVACMode.HEAT_COOL, HVACMode.HEAT, HVACMode.OFF]
46 
47 PLATFORM_SCHEMA = CLIMATE_PLATFORM_SCHEMA.extend(
48  {vol.Required(CONF_USERNAME): cv.string, vol.Required(CONF_PASSWORD): cv.string}
49 )
50 
51 EPH_TO_HA_STATE = {
52  "AUTO": HVACMode.HEAT_COOL,
53  "ON": HVACMode.HEAT,
54  "OFF": HVACMode.OFF,
55 }
56 
57 HA_STATE_TO_EPH = {value: key for key, value in EPH_TO_HA_STATE.items()}
58 
59 
61  hass: HomeAssistant,
62  config: ConfigType,
63  add_entities: AddEntitiesCallback,
64  discovery_info: DiscoveryInfoType | None = None,
65 ) -> None:
66  """Set up the ephember thermostat."""
67  username = config.get(CONF_USERNAME)
68  password = config.get(CONF_PASSWORD)
69 
70  try:
71  ember = EphEmber(username, password)
72  zones = ember.get_zones()
73  for zone in zones:
74  add_entities([EphEmberThermostat(ember, zone)])
75  except RuntimeError:
76  _LOGGER.error("Cannot connect to EphEmber")
77  return
78 
79  return
80 
81 
83  """Representation of a EphEmber thermostat."""
84 
85  _attr_hvac_modes = OPERATION_LIST
86  _attr_temperature_unit = UnitOfTemperature.CELSIUS
87  _enable_turn_on_off_backwards_compatibility = False
88 
89  def __init__(self, ember, zone):
90  """Initialize the thermostat."""
91  self._ember_ember = ember
92  self._zone_name_zone_name = zone_name(zone)
93  self._zone_zone = zone
94  self._hot_water_hot_water = zone_is_hot_water(zone)
95 
96  self._attr_name_attr_name = self._zone_name_zone_name
97 
98  self._attr_supported_features_attr_supported_features = (
99  ClimateEntityFeature.TARGET_TEMPERATURE | ClimateEntityFeature.AUX_HEAT
100  )
101  self._attr_target_temperature_step_attr_target_temperature_step = 0.5
102  if self._hot_water_hot_water:
103  self._attr_supported_features_attr_supported_features = ClimateEntityFeature.AUX_HEAT
104  self._attr_target_temperature_step_attr_target_temperature_step = None
105  self._attr_supported_features_attr_supported_features |= (
106  ClimateEntityFeature.TURN_OFF | ClimateEntityFeature.TURN_ON
107  )
108 
109  @property
111  """Return the current temperature."""
112  return zone_current_temperature(self._zone_zone)
113 
114  @property
116  """Return the temperature we try to reach."""
117  return zone_target_temperature(self._zone_zone)
118 
119  @property
120  def hvac_action(self) -> HVACAction:
121  """Return current HVAC action."""
122  if zone_is_active(self._zone_zone):
123  return HVACAction.HEATING
124 
125  return HVACAction.IDLE
126 
127  @property
128  def hvac_mode(self) -> HVACMode:
129  """Return current operation ie. heat, cool, idle."""
130  mode = zone_mode(self._zone_zone)
131  return self.map_mode_eph_hassmap_mode_eph_hass(mode)
132 
133  def set_hvac_mode(self, hvac_mode: HVACMode) -> None:
134  """Set the operation mode."""
135  mode = self.map_mode_hass_ephmap_mode_hass_eph(hvac_mode)
136  if mode is not None:
137  self._ember_ember.set_mode_by_name(self._zone_name_zone_name, mode)
138  else:
139  _LOGGER.error("Invalid operation mode provided %s", hvac_mode)
140 
141  @property
142  def is_aux_heat(self):
143  """Return true if aux heater."""
144 
145  return zone_is_boost_active(self._zone_zone)
146 
147  def turn_aux_heat_on(self) -> None:
148  """Turn auxiliary heater on."""
149  self._ember_ember.activate_boost_by_name(
150  self._zone_name_zone_name, zone_target_temperature(self._zone_zone)
151  )
152 
153  def turn_aux_heat_off(self) -> None:
154  """Turn auxiliary heater off."""
155  self._ember_ember.deactivate_boost_by_name(self._zone_name_zone_name)
156 
157  def set_temperature(self, **kwargs: Any) -> None:
158  """Set new target temperature."""
159  if (temperature := kwargs.get(ATTR_TEMPERATURE)) is None:
160  return
161 
162  if self._hot_water_hot_water:
163  return
164 
165  if temperature == self.target_temperaturetarget_temperaturetarget_temperature:
166  return
167 
168  if temperature > self.max_tempmax_tempmax_temp or temperature < self.min_tempmin_tempmin_temp:
169  return
170 
171  self._ember_ember.set_target_temperture_by_name(self._zone_name_zone_name, temperature)
172 
173  @property
174  def min_temp(self):
175  """Return the minimum temperature."""
176  # Hot water temp doesn't support being changed
177  if self._hot_water_hot_water:
178  return zone_target_temperature(self._zone_zone)
179 
180  return 5.0
181 
182  @property
183  def max_temp(self):
184  """Return the maximum temperature."""
185  if self._hot_water_hot_water:
186  return zone_target_temperature(self._zone_zone)
187 
188  return 35.0
189 
190  def update(self) -> None:
191  """Get the latest data."""
192  self._zone_zone = self._ember_ember.get_zone(self._zone_name_zone_name)
193 
194  @staticmethod
195  def map_mode_hass_eph(operation_mode):
196  """Map from Home Assistant mode to eph mode."""
197  return getattr(ZoneMode, HA_STATE_TO_EPH.get(operation_mode), None)
198 
199  @staticmethod
200  def map_mode_eph_hass(operation_mode):
201  """Map from eph mode to Home Assistant mode."""
202  return EPH_TO_HA_STATE.get(operation_mode.name, HVACMode.HEAT_COOL)
None add_entities(AsusWrtRouter router, AddEntitiesCallback async_add_entities, set[str] tracked)
None setup_platform(HomeAssistant hass, ConfigType config, AddEntitiesCallback add_entities, DiscoveryInfoType|None discovery_info=None)
Definition: climate.py:65