Deprecate aux heat from Honeywell (#114110)

* Remove aux heat

* Add switch entity for emheat

* Optimized async_setup_entry

* Fix errors in comments

* Fix new ruff failuer

* Use constant for EM

* Protect EM mode - must be in heat to turn on/off

* Restore aux_heat

* Add repair issue

* Add missing place holder to issue

* Better placeholder "option"
pull/115032/head
mkmer 2024-04-06 11:22:56 -04:00 committed by GitHub
parent a5c82f3713
commit 7f8341e03a
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
7 changed files with 244 additions and 11 deletions

View File

@ -21,7 +21,7 @@ from .const import (
) )
UPDATE_LOOP_SLEEP_TIME = 5 UPDATE_LOOP_SLEEP_TIME = 5
PLATFORMS = [Platform.CLIMATE, Platform.SENSOR] PLATFORMS = [Platform.CLIMATE, Platform.SENSOR, Platform.SWITCH]
MIGRATE_OPTIONS_KEYS = {CONF_COOL_AWAY_TEMPERATURE, CONF_HEAT_AWAY_TEMPERATURE} MIGRATE_OPTIONS_KEYS = {CONF_COOL_AWAY_TEMPERATURE, CONF_HEAT_AWAY_TEMPERATURE}

View File

@ -35,7 +35,7 @@ from homeassistant.config_entries import ConfigEntry
from homeassistant.const import ATTR_TEMPERATURE, UnitOfTemperature from homeassistant.const import ATTR_TEMPERATURE, UnitOfTemperature
from homeassistant.core import HomeAssistant from homeassistant.core import HomeAssistant
from homeassistant.exceptions import HomeAssistantError from homeassistant.exceptions import HomeAssistantError
from homeassistant.helpers import device_registry as dr from homeassistant.helpers import device_registry as dr, issue_registry as ir
from homeassistant.helpers.device_registry import DeviceInfo from homeassistant.helpers.device_registry import DeviceInfo
from homeassistant.helpers.entity_platform import AddEntitiesCallback from homeassistant.helpers.entity_platform import AddEntitiesCallback
@ -169,6 +169,7 @@ class HoneywellUSThermostat(ClimateEntity):
manufacturer="Honeywell", manufacturer="Honeywell",
) )
self._attr_translation_placeholders = {"name": device.name}
self._attr_temperature_unit = UnitOfTemperature.FAHRENHEIT self._attr_temperature_unit = UnitOfTemperature.FAHRENHEIT
if device.temperature_unit == "C": if device.temperature_unit == "C":
self._attr_temperature_unit = UnitOfTemperature.CELSIUS self._attr_temperature_unit = UnitOfTemperature.CELSIUS
@ -480,6 +481,16 @@ class HoneywellUSThermostat(ClimateEntity):
async def async_turn_aux_heat_on(self) -> None: async def async_turn_aux_heat_on(self) -> None:
"""Turn auxiliary heater on.""" """Turn auxiliary heater on."""
ir.async_create_issue(
self.hass,
DOMAIN,
"service_deprecation",
breaks_in_ha_version="2024.10.0",
is_fixable=True,
is_persistent=True,
severity=ir.IssueSeverity.WARNING,
translation_key="service_deprecation",
)
try: try:
await self._device.set_system_mode("emheat") await self._device.set_system_mode("emheat")
except SomeComfortError as err: except SomeComfortError as err:
@ -489,6 +500,18 @@ class HoneywellUSThermostat(ClimateEntity):
async def async_turn_aux_heat_off(self) -> None: async def async_turn_aux_heat_off(self) -> None:
"""Turn auxiliary heater off.""" """Turn auxiliary heater off."""
ir.async_create_issue(
self.hass,
DOMAIN,
"service_deprecation",
breaks_in_ha_version="2024.10.0",
is_fixable=True,
is_persistent=True,
severity=ir.IssueSeverity.WARNING,
translation_key="service_deprecation",
)
try: try:
if HVACMode.HEAT in self.hvac_modes: if HVACMode.HEAT in self.hvac_modes:
await self.async_set_hvac_mode(HVACMode.HEAT) await self.async_set_hvac_mode(HVACMode.HEAT)

View File

@ -41,6 +41,11 @@
"name": "Outdoor humidity" "name": "Outdoor humidity"
} }
}, },
"switch": {
"emergency_heat": {
"name": "Emergency heat"
}
},
"climate": { "climate": {
"honeywell": { "honeywell": {
"state_attributes": { "state_attributes": {
@ -54,5 +59,26 @@
} }
} }
} }
},
"exceptions": {
"switch_failed_off": {
"message": "Honeywell could turn off emergency heat mode."
},
"switch_failed_on": {
"message": "Honeywell could not set system mode to emergency heat mode."
}
},
"issues": {
"service_deprecation": {
"title": "Honeywell aux heat is being removed",
"fix_flow": {
"step": {
"confirm": {
"title": "[%key:component::honeywell::issues::service_deprecation::title%]",
"description": "Use `switch.{name}_emergency_heat` instead to change mode.\n\nPlease adjust your automations and scripts and select **submit** to fix this issue."
}
}
}
}
} }
} }

View File

@ -0,0 +1,97 @@
"""Support for Honeywell switches."""
from __future__ import annotations
from typing import Any
from aiosomecomfort import SomeComfortError
from aiosomecomfort.device import Device as SomeComfortDevice
from homeassistant.components.switch import (
SwitchDeviceClass,
SwitchEntity,
SwitchEntityDescription,
)
from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant
from homeassistant.exceptions import HomeAssistantError
from homeassistant.helpers.device_registry import DeviceInfo
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from . import HoneywellData
from .const import DOMAIN
EMERGENCY_HEAT_KEY = "emergency_heat"
SWITCH_TYPES: tuple[SwitchEntityDescription, ...] = (
SwitchEntityDescription(
key=EMERGENCY_HEAT_KEY,
translation_key=EMERGENCY_HEAT_KEY,
device_class=SwitchDeviceClass.SWITCH,
),
)
async def async_setup_entry(
hass: HomeAssistant,
config_entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up the Honeywell switches."""
data: HoneywellData = hass.data[DOMAIN][config_entry.entry_id]
async_add_entities(
HoneywellSwitch(hass, config_entry, device, description)
for device in data.devices.values()
if device.raw_ui_data.get("SwitchEmergencyHeatAllowed")
for description in SWITCH_TYPES
)
class HoneywellSwitch(SwitchEntity):
"""Representation of a honeywell switch."""
_attr_has_entity_name = True
def __init__(
self,
hass: HomeAssistant,
config_entry: ConfigEntry,
device: SomeComfortDevice,
description: SwitchEntityDescription,
) -> None:
"""Initialize the switch."""
self._data = hass.data[DOMAIN][config_entry.entry_id]
self._device = device
self.entity_description = description
self._attr_unique_id = f"{device.deviceid}_{description.key}"
self._attr_device_info = DeviceInfo(
identifiers={(DOMAIN, device.deviceid)},
name=device.name,
manufacturer="Honeywell",
)
async def async_turn_on(self, **kwargs: Any) -> None:
"""Turn the switch on if heat mode is enabled."""
if self._device.system_mode == "heat":
try:
await self._device.set_system_mode("emheat")
except SomeComfortError as err:
raise HomeAssistantError(
translation_domain=DOMAIN, translation_key="switch_failed_on"
) from err
async def async_turn_off(self, **kwargs: Any) -> None:
"""Turn the switch off if on."""
if self.is_on:
try:
await self._device.set_system_mode("off")
except SomeComfortError as err:
raise HomeAssistantError(
translation_domain=DOMAIN, translation_key="switch_failed_off"
) from err
@property
def is_on(self) -> bool:
"""Return true if Emergency heat is enabled."""
return self._device.system_mode == "emheat"

View File

@ -29,7 +29,7 @@ async def test_entry_diagnostics(
await hass.config_entries.async_setup(config_entry.entry_id) await hass.config_entries.async_setup(config_entry.entry_id)
await hass.async_block_till_done() await hass.async_block_till_done()
assert config_entry.state is ConfigEntryState.LOADED assert config_entry.state is ConfigEntryState.LOADED
assert hass.states.async_entity_ids_count() == 6 assert hass.states.async_entity_ids_count() == 8
result = await get_diagnostics_for_config_entry(hass, hass_client, config_entry) result = await get_diagnostics_for_config_entry(hass, hass_client, config_entry)

View File

@ -30,7 +30,7 @@ async def test_setup_entry(hass: HomeAssistant, config_entry: MockConfigEntry) -
await hass.async_block_till_done() await hass.async_block_till_done()
assert config_entry.state is ConfigEntryState.LOADED assert config_entry.state is ConfigEntryState.LOADED
assert ( assert (
hass.states.async_entity_ids_count() == 3 hass.states.async_entity_ids_count() == 4
) # 1 climate entity; 2 sensor entities ) # 1 climate entity; 2 sensor entities
@ -63,8 +63,8 @@ async def test_setup_multiple_thermostats(
await hass.async_block_till_done() await hass.async_block_till_done()
assert config_entry.state is ConfigEntryState.LOADED assert config_entry.state is ConfigEntryState.LOADED
assert ( assert (
hass.states.async_entity_ids_count() == 6 hass.states.async_entity_ids_count() == 8
) # 2 climate entities; 4 sensor entities ) # 2 climate entities; 4 sensor entities; 2 switch entities
async def test_setup_multiple_thermostats_with_same_deviceid( async def test_setup_multiple_thermostats_with_same_deviceid(
@ -84,8 +84,8 @@ async def test_setup_multiple_thermostats_with_same_deviceid(
await hass.async_block_till_done() await hass.async_block_till_done()
assert config_entry.state is ConfigEntryState.LOADED assert config_entry.state is ConfigEntryState.LOADED
assert ( assert (
hass.states.async_entity_ids_count() == 3 hass.states.async_entity_ids_count() == 4
) # 1 climate entity; 2 sensor entities ) # 1 climate entity; 2 sensor entities; 1 switch enitiy
assert "Platform honeywell does not generate unique IDs" not in caplog.text assert "Platform honeywell does not generate unique IDs" not in caplog.text
@ -171,7 +171,7 @@ async def test_remove_stale_device(
await hass.config_entries.async_setup(config_entry.entry_id) await hass.config_entries.async_setup(config_entry.entry_id)
await hass.async_block_till_done() await hass.async_block_till_done()
assert config_entry.state is ConfigEntryState.LOADED assert config_entry.state is ConfigEntryState.LOADED
assert hass.states.async_entity_ids_count() == 6 assert hass.states.async_entity_ids_count() == 8
device_entries = dr.async_entries_for_config_entry( device_entries = dr.async_entries_for_config_entry(
device_registry, config_entry.entry_id device_registry, config_entry.entry_id
@ -209,8 +209,8 @@ async def test_remove_stale_device(
assert config_entry.state is ConfigEntryState.LOADED assert config_entry.state is ConfigEntryState.LOADED
assert ( assert (
hass.states.async_entity_ids_count() == 3 hass.states.async_entity_ids_count() == 4
) # 1 climate entities; 2 sensor entities ) # 1 climate entities; 2 sensor entities; 1 switch entity
device_entries = dr.async_entries_for_config_entry( device_entries = dr.async_entries_for_config_entry(
device_registry, config_entry.entry_id device_registry, config_entry.entry_id

View File

@ -0,0 +1,87 @@
"""Tests for Honeywell switch component."""
from unittest.mock import MagicMock
from aiosomecomfort.exceptions import SomeComfortError
import pytest
from homeassistant.components.switch import DOMAIN as SWITCH_DOMAIN
from homeassistant.const import ATTR_ENTITY_ID, SERVICE_TURN_OFF, SERVICE_TURN_ON
from homeassistant.core import HomeAssistant
from homeassistant.exceptions import HomeAssistantError
from . import init_integration
from tests.common import MockConfigEntry
async def test_emheat_switch(
hass: HomeAssistant,
config_entry: MockConfigEntry,
device: MagicMock,
) -> None:
"""Test emergency heat switch."""
await init_integration(hass, config_entry)
entity_id = f"switch.{device.name}_emergency_heat"
await hass.services.async_call(
SWITCH_DOMAIN,
SERVICE_TURN_ON,
{ATTR_ENTITY_ID: entity_id},
blocking=True,
)
device.set_system_mode.assert_not_called()
device.set_system_mode.reset_mock()
await hass.services.async_call(
SWITCH_DOMAIN,
SERVICE_TURN_OFF,
{ATTR_ENTITY_ID: entity_id},
blocking=True,
)
device.set_system_mode.assert_not_called()
device.system_mode = "heat"
await hass.services.async_call(
SWITCH_DOMAIN,
SERVICE_TURN_ON,
{ATTR_ENTITY_ID: entity_id},
blocking=True,
)
device.set_system_mode.assert_called_once_with("emheat")
device.set_system_mode.reset_mock()
device.system_mode = "emheat"
await hass.services.async_call(
SWITCH_DOMAIN,
SERVICE_TURN_OFF,
{ATTR_ENTITY_ID: entity_id},
blocking=True,
)
device.set_system_mode.assert_called_once_with("off")
device.set_system_mode.reset_mock()
device.system_mode = "heat"
device.set_system_mode.side_effect = SomeComfortError
with pytest.raises(HomeAssistantError):
await hass.services.async_call(
SWITCH_DOMAIN,
SERVICE_TURN_ON,
{ATTR_ENTITY_ID: entity_id},
blocking=True,
)
device.set_system_mode.assert_called_once_with("emheat")
device.set_system_mode.reset_mock()
device.system_mode = "emheat"
device.set_system_mode.side_effect = SomeComfortError
with pytest.raises(HomeAssistantError):
await hass.services.async_call(
SWITCH_DOMAIN,
SERVICE_TURN_OFF,
{ATTR_ENTITY_ID: entity_id},
blocking=True,
)
device.set_system_mode.assert_called_once_with("off")