346 lines
12 KiB
Python
346 lines
12 KiB
Python
"""Viessmann ViCare climate device."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from contextlib import suppress
|
|
import logging
|
|
from typing import Any
|
|
|
|
from PyViCare.PyViCareDevice import Device as PyViCareDevice
|
|
from PyViCare.PyViCareDeviceConfig import PyViCareDeviceConfig
|
|
from PyViCare.PyViCareHeatingDevice import HeatingCircuit as PyViCareHeatingCircuit
|
|
from PyViCare.PyViCareUtils import (
|
|
PyViCareCommandError,
|
|
PyViCareInvalidDataError,
|
|
PyViCareNotSupportedFeatureError,
|
|
PyViCareRateLimitError,
|
|
)
|
|
import requests
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.climate import (
|
|
ClimateEntity,
|
|
ClimateEntityFeature,
|
|
HVACAction,
|
|
HVACMode,
|
|
)
|
|
from homeassistant.const import (
|
|
ATTR_TEMPERATURE,
|
|
PRECISION_TENTHS,
|
|
PRECISION_WHOLE,
|
|
UnitOfTemperature,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.exceptions import ServiceValidationError
|
|
from homeassistant.helpers import entity_platform
|
|
import homeassistant.helpers.config_validation as cv
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from .const import DOMAIN
|
|
from .entity import ViCareEntity
|
|
from .types import HeatingProgram, ViCareConfigEntry, ViCareDevice
|
|
from .utils import get_burners, get_circuits, get_compressors, get_device_serial
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
SERVICE_SET_VICARE_MODE = "set_vicare_mode"
|
|
SERVICE_SET_VICARE_MODE_ATTR_MODE = "vicare_mode"
|
|
|
|
VICARE_MODE_DHW = "dhw"
|
|
VICARE_MODE_HEATING = "heating"
|
|
VICARE_MODE_HEATINGCOOLING = "heatingCooling"
|
|
VICARE_MODE_DHWANDHEATING = "dhwAndHeating"
|
|
VICARE_MODE_DHWANDHEATINGCOOLING = "dhwAndHeatingCooling"
|
|
VICARE_MODE_FORCEDREDUCED = "forcedReduced"
|
|
VICARE_MODE_FORCEDNORMAL = "forcedNormal"
|
|
VICARE_MODE_OFF = "standby"
|
|
|
|
VICARE_HOLD_MODE_AWAY = "away"
|
|
VICARE_HOLD_MODE_HOME = "home"
|
|
VICARE_HOLD_MODE_OFF = "off"
|
|
|
|
VICARE_TEMP_HEATING_MIN = 3
|
|
VICARE_TEMP_HEATING_MAX = 37
|
|
|
|
VICARE_TO_HA_HVAC_HEATING: dict[str, HVACMode] = {
|
|
VICARE_MODE_FORCEDREDUCED: HVACMode.OFF,
|
|
VICARE_MODE_OFF: HVACMode.OFF,
|
|
VICARE_MODE_DHW: HVACMode.OFF,
|
|
VICARE_MODE_DHWANDHEATINGCOOLING: HVACMode.AUTO,
|
|
VICARE_MODE_DHWANDHEATING: HVACMode.AUTO,
|
|
VICARE_MODE_HEATINGCOOLING: HVACMode.AUTO,
|
|
VICARE_MODE_HEATING: HVACMode.AUTO,
|
|
VICARE_MODE_FORCEDNORMAL: HVACMode.HEAT,
|
|
}
|
|
|
|
CHANGABLE_HEATING_PROGRAMS = [
|
|
HeatingProgram.COMFORT,
|
|
HeatingProgram.COMFORT_HEATING,
|
|
HeatingProgram.ECO,
|
|
]
|
|
|
|
|
|
def _build_entities(
|
|
device_list: list[ViCareDevice],
|
|
) -> list[ViCareClimate]:
|
|
"""Create ViCare climate entities for a device."""
|
|
return [
|
|
ViCareClimate(
|
|
get_device_serial(device.api),
|
|
device.config,
|
|
device.api,
|
|
circuit,
|
|
)
|
|
for device in device_list
|
|
for circuit in get_circuits(device.api)
|
|
]
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant,
|
|
config_entry: ViCareConfigEntry,
|
|
async_add_entities: AddEntitiesCallback,
|
|
) -> None:
|
|
"""Set up the ViCare climate platform."""
|
|
|
|
platform = entity_platform.async_get_current_platform()
|
|
platform.async_register_entity_service(
|
|
SERVICE_SET_VICARE_MODE,
|
|
{vol.Required(SERVICE_SET_VICARE_MODE_ATTR_MODE): cv.string},
|
|
"set_vicare_mode",
|
|
)
|
|
|
|
async_add_entities(
|
|
await hass.async_add_executor_job(
|
|
_build_entities,
|
|
config_entry.runtime_data.devices,
|
|
)
|
|
)
|
|
|
|
|
|
class ViCareClimate(ViCareEntity, ClimateEntity):
|
|
"""Representation of the ViCare heating climate device."""
|
|
|
|
_attr_precision = PRECISION_TENTHS
|
|
_attr_supported_features = (
|
|
ClimateEntityFeature.TARGET_TEMPERATURE
|
|
| ClimateEntityFeature.PRESET_MODE
|
|
| ClimateEntityFeature.TURN_OFF
|
|
| ClimateEntityFeature.TURN_ON
|
|
)
|
|
_attr_temperature_unit = UnitOfTemperature.CELSIUS
|
|
_attr_min_temp = VICARE_TEMP_HEATING_MIN
|
|
_attr_max_temp = VICARE_TEMP_HEATING_MAX
|
|
_attr_target_temperature_step = PRECISION_WHOLE
|
|
_attr_translation_key = "heating"
|
|
_current_action: bool | None = None
|
|
_current_mode: str | None = None
|
|
_current_program: str | None = None
|
|
|
|
def __init__(
|
|
self,
|
|
device_serial: str | None,
|
|
device_config: PyViCareDeviceConfig,
|
|
device: PyViCareDevice,
|
|
circuit: PyViCareHeatingCircuit,
|
|
) -> None:
|
|
"""Initialize the climate device."""
|
|
super().__init__(
|
|
self._attr_translation_key, device_serial, device_config, device, circuit
|
|
)
|
|
self._device = device
|
|
self._attributes: dict[str, Any] = {}
|
|
self._attributes["vicare_programs"] = self._api.getPrograms()
|
|
self._attr_preset_modes = [
|
|
preset
|
|
for heating_program in self._attributes["vicare_programs"]
|
|
if (preset := HeatingProgram.to_ha_preset(heating_program)) is not None
|
|
]
|
|
|
|
def update(self) -> None:
|
|
"""Let HA know there has been an update from the ViCare API."""
|
|
try:
|
|
_room_temperature = None
|
|
with suppress(PyViCareNotSupportedFeatureError):
|
|
self._attributes["room_temperature"] = _room_temperature = (
|
|
self._api.getRoomTemperature()
|
|
)
|
|
|
|
_supply_temperature = None
|
|
with suppress(PyViCareNotSupportedFeatureError):
|
|
_supply_temperature = self._api.getSupplyTemperature()
|
|
|
|
if _room_temperature is not None:
|
|
self._attr_current_temperature = _room_temperature
|
|
elif _supply_temperature is not None:
|
|
self._attr_current_temperature = _supply_temperature
|
|
else:
|
|
self._attr_current_temperature = None
|
|
|
|
with suppress(PyViCareNotSupportedFeatureError):
|
|
self._attributes["active_vicare_program"] = self._current_program = (
|
|
self._api.getActiveProgram()
|
|
)
|
|
|
|
with suppress(PyViCareNotSupportedFeatureError):
|
|
self._attr_target_temperature = self._api.getCurrentDesiredTemperature()
|
|
|
|
with suppress(PyViCareNotSupportedFeatureError):
|
|
self._attributes["active_vicare_mode"] = self._current_mode = (
|
|
self._api.getActiveMode()
|
|
)
|
|
|
|
with suppress(PyViCareNotSupportedFeatureError):
|
|
self._attributes["heating_curve_slope"] = (
|
|
self._api.getHeatingCurveSlope()
|
|
)
|
|
|
|
with suppress(PyViCareNotSupportedFeatureError):
|
|
self._attributes["heating_curve_shift"] = (
|
|
self._api.getHeatingCurveShift()
|
|
)
|
|
|
|
with suppress(PyViCareNotSupportedFeatureError):
|
|
self._attributes["vicare_modes"] = self._api.getModes()
|
|
|
|
self._current_action = False
|
|
# Update the specific device attributes
|
|
with suppress(PyViCareNotSupportedFeatureError):
|
|
for burner in get_burners(self._device):
|
|
self._current_action = self._current_action or burner.getActive()
|
|
|
|
with suppress(PyViCareNotSupportedFeatureError):
|
|
for compressor in get_compressors(self._device):
|
|
self._current_action = (
|
|
self._current_action or compressor.getActive()
|
|
)
|
|
|
|
except requests.exceptions.ConnectionError:
|
|
_LOGGER.error("Unable to retrieve data from ViCare server")
|
|
except PyViCareRateLimitError as limit_exception:
|
|
_LOGGER.error("Vicare API rate limit exceeded: %s", limit_exception)
|
|
except ValueError:
|
|
_LOGGER.error("Unable to decode data from ViCare server")
|
|
except PyViCareInvalidDataError as invalid_data_exception:
|
|
_LOGGER.error("Invalid data from Vicare server: %s", invalid_data_exception)
|
|
|
|
@property
|
|
def hvac_mode(self) -> HVACMode | None:
|
|
"""Return current hvac mode."""
|
|
if self._current_mode is None:
|
|
return None
|
|
return VICARE_TO_HA_HVAC_HEATING.get(self._current_mode, None)
|
|
|
|
def set_hvac_mode(self, hvac_mode: HVACMode) -> None:
|
|
"""Set a new hvac mode on the ViCare API."""
|
|
if "vicare_modes" not in self._attributes:
|
|
raise ValueError("Cannot set hvac mode when vicare_modes are not known")
|
|
|
|
vicare_mode = self.vicare_mode_from_hvac_mode(hvac_mode)
|
|
if vicare_mode is None:
|
|
raise ValueError(f"Cannot set invalid hvac mode: {hvac_mode}")
|
|
|
|
_LOGGER.debug("Setting hvac mode to %s / %s", hvac_mode, vicare_mode)
|
|
self._api.setMode(vicare_mode)
|
|
|
|
def vicare_mode_from_hvac_mode(self, hvac_mode) -> str | None:
|
|
"""Return the corresponding vicare mode for an hvac_mode."""
|
|
if "vicare_modes" not in self._attributes:
|
|
return None
|
|
|
|
supported_modes = self._attributes["vicare_modes"]
|
|
for key, value in VICARE_TO_HA_HVAC_HEATING.items():
|
|
if key in supported_modes and value == hvac_mode:
|
|
return key
|
|
return None
|
|
|
|
@property
|
|
def hvac_modes(self) -> list[HVACMode]:
|
|
"""Return the list of available hvac modes."""
|
|
if "vicare_modes" not in self._attributes:
|
|
return []
|
|
|
|
supported_modes = self._attributes["vicare_modes"]
|
|
hvac_modes = []
|
|
for key, value in VICARE_TO_HA_HVAC_HEATING.items():
|
|
if value in hvac_modes:
|
|
continue
|
|
if key in supported_modes:
|
|
hvac_modes.append(value)
|
|
return hvac_modes
|
|
|
|
@property
|
|
def hvac_action(self) -> HVACAction:
|
|
"""Return the current hvac action."""
|
|
if self._current_action:
|
|
return HVACAction.HEATING
|
|
return HVACAction.IDLE
|
|
|
|
def set_temperature(self, **kwargs: Any) -> None:
|
|
"""Set new target temperatures."""
|
|
if (temp := kwargs.get(ATTR_TEMPERATURE)) is not None:
|
|
self._api.setProgramTemperature(self._current_program, temp)
|
|
self._attr_target_temperature = temp
|
|
|
|
@property
|
|
def preset_mode(self):
|
|
"""Return the current preset mode, e.g., home, away, temp."""
|
|
return HeatingProgram.to_ha_preset(self._current_program)
|
|
|
|
def set_preset_mode(self, preset_mode: str) -> None:
|
|
"""Set new preset mode and deactivate any existing programs."""
|
|
target_program = HeatingProgram.from_ha_preset(
|
|
preset_mode, self._attributes["vicare_programs"]
|
|
)
|
|
if target_program is None:
|
|
raise ServiceValidationError(
|
|
translation_domain=DOMAIN,
|
|
translation_key="program_unknown",
|
|
translation_placeholders={
|
|
"preset": preset_mode,
|
|
},
|
|
)
|
|
|
|
_LOGGER.debug("Current preset %s", self._current_program)
|
|
if (
|
|
self._current_program
|
|
and self._current_program in CHANGABLE_HEATING_PROGRAMS
|
|
):
|
|
_LOGGER.debug("deactivating %s", self._current_program)
|
|
try:
|
|
self._api.deactivateProgram(self._current_program)
|
|
except PyViCareCommandError as err:
|
|
raise ServiceValidationError(
|
|
translation_domain=DOMAIN,
|
|
translation_key="program_not_deactivated",
|
|
translation_placeholders={
|
|
"program": self._current_program,
|
|
},
|
|
) from err
|
|
|
|
_LOGGER.debug("Setting preset to %s / %s", preset_mode, target_program)
|
|
if target_program in CHANGABLE_HEATING_PROGRAMS:
|
|
_LOGGER.debug("activating %s", target_program)
|
|
try:
|
|
self._api.activateProgram(target_program)
|
|
except PyViCareCommandError as err:
|
|
raise ServiceValidationError(
|
|
translation_domain=DOMAIN,
|
|
translation_key="program_not_activated",
|
|
translation_placeholders={
|
|
"program": target_program,
|
|
},
|
|
) from err
|
|
|
|
@property
|
|
def extra_state_attributes(self) -> dict[str, Any]:
|
|
"""Show Device Attributes."""
|
|
return self._attributes
|
|
|
|
def set_vicare_mode(self, vicare_mode) -> None:
|
|
"""Service function to set vicare modes directly."""
|
|
if vicare_mode not in self._attributes["vicare_modes"]:
|
|
raise ValueError(f"Cannot set invalid vicare mode: {vicare_mode}.")
|
|
|
|
self._api.setMode(vicare_mode)
|