mirror of
https://github.com/home-assistant/core.git
synced 2025-07-22 12:47:08 +00:00
Use external temp if needed in Broadlink (#118375)
* Use external temp for current temp depends on the sensor state * Add SensorMode enum * Add tests for Broadlink climate * Check is the sensor included in the data * Use IntEnum as parent of SensorMode * Use SensorMode enum value for sensor test data * Parametrizing tests * Readd accidentally removed assert * Use local sensor variable Co-authored-by: Robert Resch <robert@resch.dev> * Refactor test_climate. Check call_counts. * Add parameter types Co-authored-by: Robert Resch <robert@resch.dev> * Update homeassistant/components/broadlink/climate.py --------- Co-authored-by: Robert Resch <robert@resch.dev> Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
This commit is contained in:
parent
632dec614a
commit
77282ed4b0
@ -1,5 +1,6 @@
|
|||||||
"""Support for Broadlink climate devices."""
|
"""Support for Broadlink climate devices."""
|
||||||
|
|
||||||
|
from enum import IntEnum
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
from homeassistant.components.climate import (
|
from homeassistant.components.climate import (
|
||||||
@ -19,6 +20,14 @@ from .device import BroadlinkDevice
|
|||||||
from .entity import BroadlinkEntity
|
from .entity import BroadlinkEntity
|
||||||
|
|
||||||
|
|
||||||
|
class SensorMode(IntEnum):
|
||||||
|
"""Thermostat sensor modes."""
|
||||||
|
|
||||||
|
INNER_SENSOR_CONTROL = 0
|
||||||
|
OUTER_SENSOR_CONTROL = 1
|
||||||
|
INNER_SENSOR_CONTROL_OUTER_LIMIT = 2
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(
|
async def async_setup_entry(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
config_entry: ConfigEntry,
|
config_entry: ConfigEntry,
|
||||||
@ -50,6 +59,7 @@ class BroadlinkThermostat(BroadlinkEntity, ClimateEntity):
|
|||||||
super().__init__(device)
|
super().__init__(device)
|
||||||
self._attr_unique_id = device.unique_id
|
self._attr_unique_id = device.unique_id
|
||||||
self._attr_hvac_mode = None
|
self._attr_hvac_mode = None
|
||||||
|
self.sensor_mode = SensorMode.INNER_SENSOR_CONTROL
|
||||||
|
|
||||||
async def async_set_temperature(self, **kwargs: Any) -> None:
|
async def async_set_temperature(self, **kwargs: Any) -> None:
|
||||||
"""Set new target temperature."""
|
"""Set new target temperature."""
|
||||||
@ -61,6 +71,8 @@ class BroadlinkThermostat(BroadlinkEntity, ClimateEntity):
|
|||||||
@callback
|
@callback
|
||||||
def _update_state(self, data: dict[str, Any]) -> None:
|
def _update_state(self, data: dict[str, Any]) -> None:
|
||||||
"""Update data."""
|
"""Update data."""
|
||||||
|
if (sensor := data.get("sensor")) is not None:
|
||||||
|
self.sensor_mode = SensorMode(sensor)
|
||||||
if data.get("power"):
|
if data.get("power"):
|
||||||
if data.get("auto_mode"):
|
if data.get("auto_mode"):
|
||||||
self._attr_hvac_mode = HVACMode.AUTO
|
self._attr_hvac_mode = HVACMode.AUTO
|
||||||
@ -74,8 +86,10 @@ class BroadlinkThermostat(BroadlinkEntity, ClimateEntity):
|
|||||||
else:
|
else:
|
||||||
self._attr_hvac_mode = HVACMode.OFF
|
self._attr_hvac_mode = HVACMode.OFF
|
||||||
self._attr_hvac_action = HVACAction.OFF
|
self._attr_hvac_action = HVACAction.OFF
|
||||||
|
if self.sensor_mode is SensorMode.OUTER_SENSOR_CONTROL:
|
||||||
self._attr_current_temperature = data.get("room_temp")
|
self._attr_current_temperature = data.get("external_temp")
|
||||||
|
else:
|
||||||
|
self._attr_current_temperature = data.get("room_temp")
|
||||||
self._attr_target_temperature = data.get("thermostat_temp")
|
self._attr_target_temperature = data.get("thermostat_temp")
|
||||||
|
|
||||||
async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None:
|
async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None:
|
||||||
@ -85,7 +99,9 @@ class BroadlinkThermostat(BroadlinkEntity, ClimateEntity):
|
|||||||
else:
|
else:
|
||||||
await self._device.async_request(self._device.api.set_power, 1)
|
await self._device.async_request(self._device.api.set_power, 1)
|
||||||
mode = 0 if hvac_mode == HVACMode.HEAT else 1
|
mode = 0 if hvac_mode == HVACMode.HEAT else 1
|
||||||
await self._device.async_request(self._device.api.set_mode, mode, 0)
|
await self._device.async_request(
|
||||||
|
self._device.api.set_mode, mode, 0, self.sensor_mode.value
|
||||||
|
)
|
||||||
|
|
||||||
self._attr_hvac_mode = hvac_mode
|
self._attr_hvac_mode = hvac_mode
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
180
tests/components/broadlink/test_climate.py
Normal file
180
tests/components/broadlink/test_climate.py
Normal file
@ -0,0 +1,180 @@
|
|||||||
|
"""Tests for Broadlink climate."""
|
||||||
|
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from homeassistant.components.broadlink.climate import SensorMode
|
||||||
|
from homeassistant.components.broadlink.const import DOMAIN
|
||||||
|
from homeassistant.components.climate import (
|
||||||
|
ATTR_TEMPERATURE,
|
||||||
|
DOMAIN as CLIMATE_DOMAIN,
|
||||||
|
SERVICE_SET_TEMPERATURE,
|
||||||
|
SERVICE_TURN_OFF,
|
||||||
|
SERVICE_TURN_ON,
|
||||||
|
HVACAction,
|
||||||
|
HVACMode,
|
||||||
|
)
|
||||||
|
from homeassistant.const import ATTR_ENTITY_ID, Platform
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
from homeassistant.helpers import device_registry as dr, entity_registry as er
|
||||||
|
from homeassistant.helpers.entity_component import async_update_entity
|
||||||
|
|
||||||
|
from . import get_device
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
(
|
||||||
|
"api_return_value",
|
||||||
|
"expected_state",
|
||||||
|
"expected_current_temperature",
|
||||||
|
"expected_temperature",
|
||||||
|
"expected_hvac_action",
|
||||||
|
),
|
||||||
|
[
|
||||||
|
(
|
||||||
|
{
|
||||||
|
"sensor": SensorMode.INNER_SENSOR_CONTROL.value,
|
||||||
|
"power": 1,
|
||||||
|
"auto_mode": 0,
|
||||||
|
"active": 1,
|
||||||
|
"room_temp": 22,
|
||||||
|
"thermostat_temp": 23,
|
||||||
|
"external_temp": 30,
|
||||||
|
},
|
||||||
|
HVACMode.HEAT,
|
||||||
|
22,
|
||||||
|
23,
|
||||||
|
HVACAction.HEATING,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
{
|
||||||
|
"sensor": SensorMode.OUTER_SENSOR_CONTROL.value,
|
||||||
|
"power": 1,
|
||||||
|
"auto_mode": 1,
|
||||||
|
"active": 0,
|
||||||
|
"room_temp": 22,
|
||||||
|
"thermostat_temp": 23,
|
||||||
|
"external_temp": 30,
|
||||||
|
},
|
||||||
|
HVACMode.AUTO,
|
||||||
|
30,
|
||||||
|
23,
|
||||||
|
HVACAction.IDLE,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
{
|
||||||
|
"sensor": SensorMode.INNER_SENSOR_CONTROL.value,
|
||||||
|
"power": 0,
|
||||||
|
"auto_mode": 0,
|
||||||
|
"active": 0,
|
||||||
|
"room_temp": 22,
|
||||||
|
"thermostat_temp": 23,
|
||||||
|
"external_temp": 30,
|
||||||
|
},
|
||||||
|
HVACMode.OFF,
|
||||||
|
22,
|
||||||
|
23,
|
||||||
|
HVACAction.OFF,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
async def test_climate(
|
||||||
|
api_return_value: dict[str, Any],
|
||||||
|
expected_state: HVACMode,
|
||||||
|
expected_current_temperature: int,
|
||||||
|
expected_temperature: int,
|
||||||
|
expected_hvac_action: HVACAction,
|
||||||
|
hass: HomeAssistant,
|
||||||
|
device_registry: dr.DeviceRegistry,
|
||||||
|
entity_registry: er.EntityRegistry,
|
||||||
|
) -> None:
|
||||||
|
"""Test Broadlink climate."""
|
||||||
|
|
||||||
|
device = get_device("Guest room")
|
||||||
|
mock_setup = await device.setup_entry(hass)
|
||||||
|
|
||||||
|
device_entry = device_registry.async_get_device(
|
||||||
|
identifiers={(DOMAIN, mock_setup.entry.unique_id)}
|
||||||
|
)
|
||||||
|
entries = er.async_entries_for_device(entity_registry, device_entry.id)
|
||||||
|
climates = [entry for entry in entries if entry.domain == Platform.CLIMATE]
|
||||||
|
assert len(climates) == 1
|
||||||
|
|
||||||
|
climate = climates[0]
|
||||||
|
|
||||||
|
mock_setup.api.get_full_status.return_value = api_return_value
|
||||||
|
|
||||||
|
await async_update_entity(hass, climate.entity_id)
|
||||||
|
assert mock_setup.api.get_full_status.call_count == 2
|
||||||
|
state = hass.states.get(climate.entity_id)
|
||||||
|
assert state.state == expected_state
|
||||||
|
assert state.attributes["current_temperature"] == expected_current_temperature
|
||||||
|
assert state.attributes["temperature"] == expected_temperature
|
||||||
|
assert state.attributes["hvac_action"] == expected_hvac_action
|
||||||
|
|
||||||
|
|
||||||
|
async def test_climate_set_temperature_turn_off_turn_on(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
device_registry: dr.DeviceRegistry,
|
||||||
|
entity_registry: er.EntityRegistry,
|
||||||
|
) -> None:
|
||||||
|
"""Test Broadlink climate."""
|
||||||
|
|
||||||
|
device = get_device("Guest room")
|
||||||
|
mock_setup = await device.setup_entry(hass)
|
||||||
|
|
||||||
|
device_entry = device_registry.async_get_device(
|
||||||
|
identifiers={(DOMAIN, mock_setup.entry.unique_id)}
|
||||||
|
)
|
||||||
|
entries = er.async_entries_for_device(entity_registry, device_entry.id)
|
||||||
|
climates = [entry for entry in entries if entry.domain == Platform.CLIMATE]
|
||||||
|
assert len(climates) == 1
|
||||||
|
|
||||||
|
climate = climates[0]
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
CLIMATE_DOMAIN,
|
||||||
|
SERVICE_SET_TEMPERATURE,
|
||||||
|
{
|
||||||
|
ATTR_ENTITY_ID: climate.entity_id,
|
||||||
|
ATTR_TEMPERATURE: "24",
|
||||||
|
},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
state = hass.states.get(climate.entity_id)
|
||||||
|
|
||||||
|
assert mock_setup.api.set_temp.call_count == 1
|
||||||
|
assert mock_setup.api.set_power.call_count == 0
|
||||||
|
assert mock_setup.api.set_mode.call_count == 0
|
||||||
|
assert state.attributes["temperature"] == 24
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
CLIMATE_DOMAIN,
|
||||||
|
SERVICE_TURN_OFF,
|
||||||
|
{
|
||||||
|
ATTR_ENTITY_ID: climate.entity_id,
|
||||||
|
},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
state = hass.states.get(climate.entity_id)
|
||||||
|
|
||||||
|
assert mock_setup.api.set_temp.call_count == 1
|
||||||
|
assert mock_setup.api.set_power.call_count == 1
|
||||||
|
assert mock_setup.api.set_mode.call_count == 0
|
||||||
|
assert state.state == HVACMode.OFF
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
CLIMATE_DOMAIN,
|
||||||
|
SERVICE_TURN_ON,
|
||||||
|
{
|
||||||
|
ATTR_ENTITY_ID: climate.entity_id,
|
||||||
|
},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
state = hass.states.get(climate.entity_id)
|
||||||
|
|
||||||
|
assert mock_setup.api.set_temp.call_count == 1
|
||||||
|
assert mock_setup.api.set_power.call_count == 2
|
||||||
|
assert mock_setup.api.set_mode.call_count == 1
|
||||||
|
assert state.state == HVACMode.HEAT
|
Loading…
x
Reference in New Issue
Block a user