2268 lines
75 KiB
Python
2268 lines
75 KiB
Python
"""The tests for the mqtt climate component."""
|
|
import copy
|
|
import json
|
|
from typing import Any
|
|
from unittest.mock import call, patch
|
|
|
|
import pytest
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components import climate, mqtt
|
|
from homeassistant.components.climate import (
|
|
ATTR_AUX_HEAT,
|
|
ATTR_CURRENT_HUMIDITY,
|
|
ATTR_CURRENT_TEMPERATURE,
|
|
ATTR_FAN_MODE,
|
|
ATTR_HUMIDITY,
|
|
ATTR_HVAC_ACTION,
|
|
ATTR_SWING_MODE,
|
|
ATTR_TARGET_TEMP_HIGH,
|
|
ATTR_TARGET_TEMP_LOW,
|
|
DEFAULT_MAX_HUMIDITY,
|
|
DEFAULT_MAX_TEMP,
|
|
DEFAULT_MIN_HUMIDITY,
|
|
DEFAULT_MIN_TEMP,
|
|
PRESET_ECO,
|
|
ClimateEntityFeature,
|
|
HVACAction,
|
|
HVACMode,
|
|
)
|
|
from homeassistant.components.mqtt.climate import MQTT_CLIMATE_ATTRIBUTES_BLOCKED
|
|
from homeassistant.const import ATTR_TEMPERATURE, Platform
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from .test_common import (
|
|
help_custom_config,
|
|
help_test_availability_when_connection_lost,
|
|
help_test_availability_without_topic,
|
|
help_test_custom_availability_payload,
|
|
help_test_default_availability_payload,
|
|
help_test_discovery_broken,
|
|
help_test_discovery_removal,
|
|
help_test_discovery_update,
|
|
help_test_discovery_update_attr,
|
|
help_test_discovery_update_unchanged,
|
|
help_test_encoding_subscribable_topics,
|
|
help_test_entity_debug_info_message,
|
|
help_test_entity_device_info_remove,
|
|
help_test_entity_device_info_update,
|
|
help_test_entity_device_info_with_connection,
|
|
help_test_entity_device_info_with_identifier,
|
|
help_test_entity_id_update_discovery_update,
|
|
help_test_entity_id_update_subscriptions,
|
|
help_test_publishing_with_custom_encoding,
|
|
help_test_reloadable,
|
|
help_test_setting_attribute_via_mqtt_json_message,
|
|
help_test_setting_attribute_with_template,
|
|
help_test_setting_blocked_attribute_via_mqtt_json_message,
|
|
help_test_unique_id,
|
|
help_test_unload_config_entry_with_platform,
|
|
help_test_update_with_json_attrs_bad_json,
|
|
help_test_update_with_json_attrs_not_dict,
|
|
)
|
|
|
|
from tests.common import async_fire_mqtt_message
|
|
from tests.components.climate import common
|
|
from tests.typing import MqttMockHAClientGenerator, MqttMockPahoClient
|
|
|
|
ENTITY_CLIMATE = "climate.test"
|
|
|
|
|
|
DEFAULT_CONFIG = {
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: {
|
|
"name": "test",
|
|
"mode_command_topic": "mode-topic",
|
|
"target_humidity_command_topic": "humidity-topic",
|
|
"temperature_command_topic": "temperature-topic",
|
|
"temperature_low_command_topic": "temperature-low-topic",
|
|
"temperature_high_command_topic": "temperature-high-topic",
|
|
"fan_mode_command_topic": "fan-mode-topic",
|
|
"swing_mode_command_topic": "swing-mode-topic",
|
|
"aux_command_topic": "aux-topic",
|
|
"preset_mode_command_topic": "preset-mode-topic",
|
|
"preset_modes": [
|
|
"eco",
|
|
"away",
|
|
"boost",
|
|
"comfort",
|
|
"home",
|
|
"sleep",
|
|
"activity",
|
|
],
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def climate_platform_only():
|
|
"""Only setup the climate platform to speed up tests."""
|
|
with patch("homeassistant.components.mqtt.PLATFORMS", [Platform.CLIMATE]):
|
|
yield
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_setup_params(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test the initial parameters."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 21
|
|
assert state.attributes.get("fan_mode") == "low"
|
|
assert state.attributes.get("swing_mode") == "off"
|
|
assert state.state == "off"
|
|
assert state.attributes.get("min_temp") == DEFAULT_MIN_TEMP
|
|
assert state.attributes.get("max_temp") == DEFAULT_MAX_TEMP
|
|
assert state.attributes.get("min_humidity") == DEFAULT_MIN_HUMIDITY
|
|
assert state.attributes.get("max_humidity") == DEFAULT_MAX_HUMIDITY
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"preset_modes": ["auto", "home", "none"]},),
|
|
)
|
|
],
|
|
)
|
|
async def test_preset_none_in_preset_modes(
|
|
hass: HomeAssistant,
|
|
caplog: pytest.LogCaptureFixture,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
) -> None:
|
|
"""Test the preset mode payload reset configuration."""
|
|
with pytest.raises(AssertionError):
|
|
await mqtt_mock_entry()
|
|
assert "Invalid config for [mqtt]: not a valid value" in caplog.text
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("hass_config", "parameter"),
|
|
[
|
|
(
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"away_mode_command_topic": "away-mode-command-topic"},),
|
|
),
|
|
"away_mode_command_topic",
|
|
),
|
|
(
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"away_mode_state_topic": "away-mode-state-topic"},),
|
|
),
|
|
"away_mode_state_topic",
|
|
),
|
|
(
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"away_mode_state_template": "{{ value_json }}"},),
|
|
),
|
|
"away_mode_state_template",
|
|
),
|
|
(
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"hold_mode_command_topic": "hold-mode-command-topic"},),
|
|
),
|
|
"hold_mode_command_topic",
|
|
),
|
|
(
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"hold_mode_command_template": "hold-mode-command-template"},),
|
|
),
|
|
"hold_mode_command_template",
|
|
),
|
|
(
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"hold_mode_state_topic": "hold-mode-state-topic"},),
|
|
),
|
|
"hold_mode_state_topic",
|
|
),
|
|
(
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"hold_mode_state_template": "{{ value_json }}"},),
|
|
),
|
|
"hold_mode_state_template",
|
|
),
|
|
],
|
|
)
|
|
async def test_preset_modes_deprecation_guard(
|
|
hass: HomeAssistant, caplog: pytest.LogCaptureFixture, parameter: str
|
|
) -> None:
|
|
"""Test the configuration for invalid legacy parameters."""
|
|
assert f"[{parameter}] is an invalid option for [mqtt]. Check: mqtt->mqtt->climate->0->{parameter}"
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_supported_features(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test the supported_features."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
support = (
|
|
ClimateEntityFeature.TARGET_TEMPERATURE
|
|
| ClimateEntityFeature.SWING_MODE
|
|
| ClimateEntityFeature.FAN_MODE
|
|
| ClimateEntityFeature.PRESET_MODE
|
|
| ClimateEntityFeature.AUX_HEAT
|
|
| ClimateEntityFeature.TARGET_TEMPERATURE_RANGE
|
|
| ClimateEntityFeature.TARGET_HUMIDITY
|
|
)
|
|
|
|
assert state.attributes.get("supported_features") == support
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_get_hvac_modes(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test that the operation list returns the correct modes."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
modes = state.attributes.get("hvac_modes")
|
|
assert [
|
|
HVACMode.AUTO,
|
|
HVACMode.OFF,
|
|
HVACMode.COOL,
|
|
HVACMode.HEAT,
|
|
HVACMode.DRY,
|
|
HVACMode.FAN_ONLY,
|
|
] == modes
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_operation_bad_attr_and_state(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test setting operation mode without required attribute.
|
|
|
|
Also check the state.
|
|
"""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "off"
|
|
with pytest.raises(vol.Invalid) as excinfo:
|
|
await common.async_set_hvac_mode(hass, None, ENTITY_CLIMATE)
|
|
assert (
|
|
"expected HVACMode or one of 'off', 'heat', 'cool', 'heat_cool', 'auto', 'dry',"
|
|
" 'fan_only' for dictionary value @ data['hvac_mode']" in str(excinfo.value)
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "off"
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_operation(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting of new operation mode."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "off"
|
|
await common.async_set_hvac_mode(hass, "cool", ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "cool"
|
|
assert state.state == "cool"
|
|
mqtt_mock.async_publish.assert_called_once_with("mode-topic", "cool", 0, False)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN, DEFAULT_CONFIG, ({"mode_state_topic": "mode-state"},)
|
|
)
|
|
],
|
|
)
|
|
async def test_set_operation_pessimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting operation mode in pessimistic mode."""
|
|
await hass.async_block_till_done()
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "unknown"
|
|
|
|
await common.async_set_hvac_mode(hass, "cool", ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "unknown"
|
|
|
|
async_fire_mqtt_message(hass, "mode-state", "cool")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "cool"
|
|
|
|
async_fire_mqtt_message(hass, "mode-state", "bogus mode")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "cool"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"mode_state_topic": "mode-state", "optimistic": True},),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_operation_optimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting operation mode in optimistic mode."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "off"
|
|
|
|
await common.async_set_hvac_mode(hass, "cool", ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "cool"
|
|
|
|
async_fire_mqtt_message(hass, "mode-state", "heat")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "heat"
|
|
|
|
async_fire_mqtt_message(hass, "mode-state", "bogus mode")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "heat"
|
|
|
|
|
|
# CONF_POWER_COMMAND_TOPIC, CONF_POWER_STATE_TOPIC and CONF_POWER_STATE_TEMPLATE are deprecated,
|
|
# support for CONF_POWER_STATE_TOPIC and CONF_POWER_STATE_TEMPLATE was already removed or never added
|
|
# support was deprecated with release 2023.2 and will be removed with release 2023.8
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN, DEFAULT_CONFIG, ({"power_command_topic": "power-command"},)
|
|
)
|
|
],
|
|
)
|
|
async def test_set_operation_with_power_command(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting of new operation mode with power command enabled."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "off"
|
|
await common.async_set_hvac_mode(hass, "cool", ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "cool"
|
|
mqtt_mock.async_publish.assert_has_calls(
|
|
[call("power-command", "ON", 0, False), call("mode-topic", "cool", 0, False)]
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
await common.async_set_hvac_mode(hass, "off", ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "off"
|
|
mqtt_mock.async_publish.assert_has_calls(
|
|
[call("power-command", "OFF", 0, False), call("mode-topic", "off", 0, False)]
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_fan_mode_bad_attr(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test setting fan mode without required attribute."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "low"
|
|
with pytest.raises(vol.Invalid) as excinfo:
|
|
await common.async_set_fan_mode(hass, None, ENTITY_CLIMATE)
|
|
assert "string value is None for dictionary value @ data['fan_mode']" in str(
|
|
excinfo.value
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "low"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN, DEFAULT_CONFIG, ({"fan_mode_state_topic": "fan-state"},)
|
|
)
|
|
],
|
|
)
|
|
async def test_set_fan_mode_pessimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting of new fan mode in pessimistic mode."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") is None
|
|
|
|
await common.async_set_fan_mode(hass, "high", ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") is None
|
|
|
|
async_fire_mqtt_message(hass, "fan-state", "high")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "high"
|
|
|
|
async_fire_mqtt_message(hass, "fan-state", "bogus mode")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "high"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"fan_mode_state_topic": "fan-state", "optimistic": True},),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_fan_mode_optimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting of new fan mode in optimistic mode."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "low"
|
|
|
|
await common.async_set_fan_mode(hass, "high", ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "high"
|
|
|
|
async_fire_mqtt_message(hass, "fan-state", "low")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "low"
|
|
|
|
async_fire_mqtt_message(hass, "fan-state", "bogus mode")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "low"
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_fan_mode(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting of new fan mode."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "low"
|
|
await common.async_set_fan_mode(hass, "high", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with("fan-mode-topic", "high", 0, False)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "high"
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_swing_mode_bad_attr(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test setting swing mode without required attribute."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "off"
|
|
with pytest.raises(vol.Invalid) as excinfo:
|
|
await common.async_set_swing_mode(hass, None, ENTITY_CLIMATE)
|
|
assert "string value is None for dictionary value @ data['swing_mode']" in str(
|
|
excinfo.value
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "off"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN, DEFAULT_CONFIG, ({"swing_mode_state_topic": "swing-state"},)
|
|
)
|
|
],
|
|
)
|
|
async def test_set_swing_pessimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting swing mode in pessimistic mode."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") is None
|
|
|
|
await common.async_set_swing_mode(hass, "on", ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") is None
|
|
|
|
async_fire_mqtt_message(hass, "swing-state", "on")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "on"
|
|
|
|
async_fire_mqtt_message(hass, "swing-state", "bogus state")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "on"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"swing_mode_state_topic": "swing-state", "optimistic": True},),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_swing_optimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting swing mode in optimistic mode."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "off"
|
|
|
|
await common.async_set_swing_mode(hass, "on", ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "on"
|
|
|
|
async_fire_mqtt_message(hass, "swing-state", "off")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "off"
|
|
|
|
async_fire_mqtt_message(hass, "swing-state", "bogus state")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "off"
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_swing(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting of new swing mode."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "off"
|
|
await common.async_set_swing_mode(hass, "on", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with("swing-mode-topic", "on", 0, False)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "on"
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_target_temperature(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting the target temperature."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 21
|
|
await common.async_set_hvac_mode(hass, "heat", ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "heat"
|
|
mqtt_mock.async_publish.assert_called_once_with("mode-topic", "heat", 0, False)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
await common.async_set_temperature(hass, temperature=47, entity_id=ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 47
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"temperature-topic", "47.0", 0, False
|
|
)
|
|
|
|
# also test directly supplying the operation mode to set_temperature
|
|
mqtt_mock.async_publish.reset_mock()
|
|
await common.async_set_temperature(
|
|
hass, temperature=21, hvac_mode="cool", entity_id=ENTITY_CLIMATE
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "cool"
|
|
assert state.attributes.get("temperature") == 21
|
|
mqtt_mock.async_publish.assert_has_calls(
|
|
[
|
|
call("mode-topic", "cool", 0, False),
|
|
call("temperature-topic", "21.0", 0, False),
|
|
]
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_target_humidity(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting the target humidity."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") is None
|
|
await common.async_set_humidity(hass, humidity=82, entity_id=ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") == 82
|
|
mqtt_mock.async_publish.assert_called_once_with("humidity-topic", "82", 0, False)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"temperature_state_topic": "temperature-state"},),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_target_temperature_pessimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting the target temperature."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") is None
|
|
await common.async_set_hvac_mode(hass, "heat", ENTITY_CLIMATE)
|
|
await common.async_set_temperature(hass, temperature=47, entity_id=ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") is None
|
|
|
|
async_fire_mqtt_message(hass, "temperature-state", "1701")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 1701
|
|
|
|
async_fire_mqtt_message(hass, "temperature-state", "not a number")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 1701
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"temperature_state_topic": "temperature-state", "optimistic": True},),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_target_temperature_optimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting the target temperature optimistic."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 21
|
|
await common.async_set_hvac_mode(hass, "heat", ENTITY_CLIMATE)
|
|
await common.async_set_temperature(hass, temperature=17, entity_id=ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 17
|
|
|
|
async_fire_mqtt_message(hass, "temperature-state", "18")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 18
|
|
|
|
async_fire_mqtt_message(hass, "temperature-state", "not a number")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 18
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_target_temperature_low_high(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting the low/high target temperature."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
await common.async_set_temperature(
|
|
hass, target_temp_low=20, target_temp_high=23, entity_id=ENTITY_CLIMATE
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 20
|
|
assert state.attributes.get("target_temp_high") == 23
|
|
mqtt_mock.async_publish.assert_any_call("temperature-low-topic", "20.0", 0, False)
|
|
mqtt_mock.async_publish.assert_any_call("temperature-high-topic", "23.0", 0, False)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
(
|
|
{
|
|
"temperature_low_state_topic": "temperature-low-state",
|
|
"temperature_high_state_topic": "temperature-high-state",
|
|
},
|
|
),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_target_temperature_low_highpessimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting the low/high target temperature."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") is None
|
|
assert state.attributes.get("target_temp_high") is None
|
|
await common.async_set_temperature(
|
|
hass, target_temp_low=20, target_temp_high=23, entity_id=ENTITY_CLIMATE
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") is None
|
|
assert state.attributes.get("target_temp_high") is None
|
|
|
|
async_fire_mqtt_message(hass, "temperature-low-state", "1701")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 1701
|
|
assert state.attributes.get("target_temp_high") is None
|
|
|
|
async_fire_mqtt_message(hass, "temperature-high-state", "1703")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 1701
|
|
assert state.attributes.get("target_temp_high") == 1703
|
|
|
|
async_fire_mqtt_message(hass, "temperature-low-state", "not a number")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 1701
|
|
|
|
async_fire_mqtt_message(hass, "temperature-high-state", "not a number")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_high") == 1703
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
(
|
|
{
|
|
"temperature_low_state_topic": "temperature-low-state",
|
|
"temperature_high_state_topic": "temperature-high-state",
|
|
"optimistic": True,
|
|
},
|
|
),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_target_temperature_low_high_optimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting the low/high target temperature optimistic."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 21
|
|
assert state.attributes.get("target_temp_high") == 21
|
|
await common.async_set_temperature(
|
|
hass, target_temp_low=20, target_temp_high=23, entity_id=ENTITY_CLIMATE
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 20
|
|
assert state.attributes.get("target_temp_high") == 23
|
|
|
|
async_fire_mqtt_message(hass, "temperature-low-state", "15")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 15
|
|
assert state.attributes.get("target_temp_high") == 23
|
|
|
|
async_fire_mqtt_message(hass, "temperature-high-state", "25")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 15
|
|
assert state.attributes.get("target_temp_high") == 25
|
|
|
|
async_fire_mqtt_message(hass, "temperature-low-state", "not a number")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 15
|
|
|
|
async_fire_mqtt_message(hass, "temperature-high-state", "not a number")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_high") == 25
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"target_humidity_state_topic": "humidity-state", "optimistic": True},),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_target_humidity_optimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting the target humidity optimistic."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") is None
|
|
await common.async_set_humidity(hass, humidity=52, entity_id=ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") == 52
|
|
|
|
async_fire_mqtt_message(hass, "humidity-state", "53")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") == 53
|
|
|
|
async_fire_mqtt_message(hass, "humidity-state", "not a number")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") == 53
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"target_humidity_state_topic": "humidity-state"},),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_target_humidity_pessimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting the target humidity."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") is None
|
|
await common.async_set_humidity(hass, humidity=50, entity_id=ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") is None
|
|
|
|
async_fire_mqtt_message(hass, "humidity-state", "80")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") == 80
|
|
|
|
async_fire_mqtt_message(hass, "humidity-state", "not a number")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") == 80
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"current_temperature_topic": "current_temperature"},),
|
|
)
|
|
],
|
|
)
|
|
async def test_receive_mqtt_temperature(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test getting the current temperature via MQTT."""
|
|
await mqtt_mock_entry()
|
|
|
|
async_fire_mqtt_message(hass, "current_temperature", "47")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("current_temperature") == 47
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"current_humidity_topic": "current_humidity"},),
|
|
)
|
|
],
|
|
)
|
|
async def test_receive_mqtt_humidity(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test getting the current humidity via MQTT."""
|
|
await mqtt_mock_entry()
|
|
|
|
async_fire_mqtt_message(hass, "current_humidity", "35")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("current_humidity") == 35
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"target_humidity_state_topic": "humidity-state"},),
|
|
)
|
|
],
|
|
)
|
|
async def test_handle_target_humidity_received(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting the target humidity via MQTT."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") is None
|
|
|
|
async_fire_mqtt_message(hass, "humidity-state", "65")
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") == 65
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[help_custom_config(climate.DOMAIN, DEFAULT_CONFIG, ({"action_topic": "action"},))],
|
|
)
|
|
async def test_handle_action_received(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test getting the action received via MQTT."""
|
|
await mqtt_mock_entry()
|
|
|
|
# Cycle through valid modes and also check for wrong input such as "None" (str(None))
|
|
async_fire_mqtt_message(hass, "action", "None")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
hvac_action = state.attributes.get(ATTR_HVAC_ACTION)
|
|
assert hvac_action is None
|
|
# Redefine actions according to https://developers.home-assistant.io/docs/core/entity/climate/#hvac-action
|
|
actions = ["off", "heating", "cooling", "drying", "idle", "fan"]
|
|
assert all(elem in actions for elem in HVACAction)
|
|
for action in actions:
|
|
async_fire_mqtt_message(hass, "action", action)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
hvac_action = state.attributes.get(ATTR_HVAC_ACTION)
|
|
assert hvac_action == action
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_preset_mode_optimistic(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test setting of the preset mode."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "none"
|
|
|
|
await common.async_set_preset_mode(hass, "away", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"preset-mode-topic", "away", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "away"
|
|
|
|
await common.async_set_preset_mode(hass, "eco", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"preset-mode-topic", "eco", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "eco"
|
|
|
|
await common.async_set_preset_mode(hass, "none", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"preset-mode-topic", "none", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "none"
|
|
|
|
await common.async_set_preset_mode(hass, "comfort", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"preset-mode-topic", "comfort", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "comfort"
|
|
|
|
await common.async_set_preset_mode(hass, "invalid", ENTITY_CLIMATE)
|
|
assert "'invalid' is not a valid preset mode" in caplog.text
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"preset_mode_state_topic": "preset-mode-state", "optimistic": True},),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_preset_mode_explicit_optimistic(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test setting of the preset mode."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "none"
|
|
|
|
await common.async_set_preset_mode(hass, "away", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"preset-mode-topic", "away", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "away"
|
|
|
|
await common.async_set_preset_mode(hass, "eco", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"preset-mode-topic", "eco", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "eco"
|
|
|
|
await common.async_set_preset_mode(hass, "none", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"preset-mode-topic", "none", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "none"
|
|
|
|
await common.async_set_preset_mode(hass, "comfort", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"preset-mode-topic", "comfort", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "comfort"
|
|
|
|
await common.async_set_preset_mode(hass, "invalid", ENTITY_CLIMATE)
|
|
assert "'invalid' is not a valid preset mode" in caplog.text
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
({"preset_mode_state_topic": "preset-mode-state"},),
|
|
)
|
|
],
|
|
)
|
|
async def test_set_preset_mode_pessimistic(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test setting of the preset mode."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "none"
|
|
|
|
async_fire_mqtt_message(hass, "preset-mode-state", "away")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "away"
|
|
|
|
async_fire_mqtt_message(hass, "preset-mode-state", "eco")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "eco"
|
|
|
|
async_fire_mqtt_message(hass, "preset-mode-state", "none")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "none"
|
|
|
|
async_fire_mqtt_message(hass, "preset-mode-state", "comfort")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "comfort"
|
|
|
|
async_fire_mqtt_message(hass, "preset-mode-state", "None")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "none"
|
|
|
|
async_fire_mqtt_message(hass, "preset-mode-state", "home")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "home"
|
|
|
|
async_fire_mqtt_message(hass, "preset-mode-state", "nonsense")
|
|
assert (
|
|
"'nonsense' received on topic preset-mode-state."
|
|
" 'nonsense' is not a valid preset mode" in caplog.text
|
|
)
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "home"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN, DEFAULT_CONFIG, ({"aux_state_topic": "aux-state"},)
|
|
)
|
|
],
|
|
)
|
|
async def test_set_aux_pessimistic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting of the aux heating in pessimistic mode."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("aux_heat") == "off"
|
|
|
|
await common.async_set_aux_heat(hass, True, ENTITY_CLIMATE)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("aux_heat") == "off"
|
|
|
|
async_fire_mqtt_message(hass, "aux-state", "ON")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("aux_heat") == "on"
|
|
|
|
async_fire_mqtt_message(hass, "aux-state", "OFF")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("aux_heat") == "off"
|
|
|
|
async_fire_mqtt_message(hass, "aux-state", "nonsense")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("aux_heat") == "off"
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_set_aux(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setting of the aux heating."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("aux_heat") == "off"
|
|
await common.async_set_aux_heat(hass, True, ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with("aux-topic", "ON", 0, False)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("aux_heat") == "on"
|
|
|
|
await common.async_set_aux_heat(hass, False, ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with("aux-topic", "OFF", 0, False)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("aux_heat") == "off"
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_availability_when_connection_lost(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test availability after MQTT disconnection."""
|
|
await help_test_availability_when_connection_lost(
|
|
hass, mqtt_mock_entry, climate.DOMAIN
|
|
)
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_availability_without_topic(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test availability without defined availability topic."""
|
|
await help_test_availability_without_topic(
|
|
hass, mqtt_mock_entry, climate.DOMAIN, DEFAULT_CONFIG
|
|
)
|
|
|
|
|
|
async def test_default_availability_payload(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test availability by default payload with defined topic."""
|
|
await help_test_default_availability_payload(
|
|
hass, mqtt_mock_entry, climate.DOMAIN, DEFAULT_CONFIG
|
|
)
|
|
|
|
|
|
async def test_custom_availability_payload(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test availability by custom payload with defined topic."""
|
|
await help_test_custom_availability_payload(
|
|
hass, mqtt_mock_entry, climate.DOMAIN, DEFAULT_CONFIG
|
|
)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
(
|
|
{
|
|
"temperature_low_state_topic": "temperature-state",
|
|
"temperature_high_state_topic": "temperature-state",
|
|
"temperature_low_state_template": "{{ value_json.temp_low }}",
|
|
"temperature_high_state_template": "{{ value_json.temp_high }}",
|
|
},
|
|
),
|
|
)
|
|
],
|
|
)
|
|
async def test_get_target_temperature_low_high_with_templates(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test getting temperature high/low with templates."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
|
|
# Temperature - with valid value
|
|
assert state.attributes.get("target_temp_low") is None
|
|
assert state.attributes.get("target_temp_high") is None
|
|
|
|
async_fire_mqtt_message(
|
|
hass, "temperature-state", '{"temp_low": "1031", "temp_high": "1032"}'
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 1031
|
|
assert state.attributes.get("target_temp_high") == 1032
|
|
|
|
# Temperature - with invalid value
|
|
async_fire_mqtt_message(
|
|
hass, "temperature-state", '{"temp_low": "INVALID", "temp_high": "INVALID"}'
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
# make sure, the invalid value gets logged...
|
|
assert "Could not parse temperature_low_state_template from" in caplog.text
|
|
assert "Could not parse temperature_high_state_template from" in caplog.text
|
|
# ... but the actual value stays unchanged.
|
|
assert state.attributes.get("target_temp_low") == 1031
|
|
assert state.attributes.get("target_temp_high") == 1032
|
|
|
|
# Reset the high and low values using a "None" of JSON null value
|
|
async_fire_mqtt_message(
|
|
hass, "temperature-state", '{"temp_low": "None", "temp_high": null}'
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
|
|
assert state.attributes.get("target_temp_low") is None
|
|
assert state.attributes.get("target_temp_high") is None
|
|
|
|
# Test ignoring an empty state works okay
|
|
caplog.clear()
|
|
async_fire_mqtt_message(
|
|
hass, "temperature-state", '{"temp_low": "", "temp_high": "21"}'
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") is None
|
|
assert state.attributes.get("target_temp_high") == 21.0
|
|
async_fire_mqtt_message(
|
|
hass, "temperature-state", '{"temp_low": "18", "temp_high": ""}'
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 18.0
|
|
assert state.attributes.get("target_temp_high") == 21.0
|
|
assert "Could not parse temperature_low_state_template from" not in caplog.text
|
|
assert "Could not parse temperature_high_state_template from" not in caplog.text
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
{
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: {
|
|
"name": "test",
|
|
"mode_command_topic": "mode-topic",
|
|
"target_humidity_command_topic": "humidity-topic",
|
|
"temperature_command_topic": "temperature-topic",
|
|
"temperature_low_command_topic": "temperature-low-topic",
|
|
"temperature_high_command_topic": "temperature-high-topic",
|
|
"fan_mode_command_topic": "fan-mode-topic",
|
|
"swing_mode_command_topic": "swing-mode-topic",
|
|
"aux_command_topic": "aux-topic",
|
|
"preset_mode_command_topic": "preset-mode-topic",
|
|
"preset_modes": [
|
|
"eco",
|
|
"away",
|
|
"boost",
|
|
"comfort",
|
|
"home",
|
|
"sleep",
|
|
"activity",
|
|
],
|
|
# By default, just unquote the JSON-strings
|
|
"value_template": "{{ value_json }}",
|
|
"action_template": "{{ value_json }}",
|
|
# Rendering to a bool for aux heat
|
|
"aux_state_template": "{{ value == 'switchmeon' }}",
|
|
# Rendering preset_mode
|
|
"preset_mode_value_template": "{{ value_json.attribute }}",
|
|
"action_topic": "action",
|
|
"mode_state_topic": "mode-state",
|
|
"fan_mode_state_topic": "fan-state",
|
|
"swing_mode_state_topic": "swing-state",
|
|
"temperature_state_topic": "temperature-state",
|
|
"target_humidity_state_topic": "humidity-state",
|
|
"aux_state_topic": "aux-state",
|
|
"current_temperature_topic": "current-temperature",
|
|
"current_humidity_topic": "current-humidity",
|
|
"preset_mode_state_topic": "current-preset-mode",
|
|
}
|
|
}
|
|
}
|
|
],
|
|
)
|
|
async def test_get_with_templates(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test getting various attributes with templates."""
|
|
await mqtt_mock_entry()
|
|
|
|
# Operation Mode
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
async_fire_mqtt_message(hass, "mode-state", '"cool"')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "cool"
|
|
|
|
# Fan Mode
|
|
assert state.attributes.get("fan_mode") is None
|
|
async_fire_mqtt_message(hass, "fan-state", '"high"')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "high"
|
|
|
|
# Swing Mode
|
|
assert state.attributes.get("swing_mode") is None
|
|
async_fire_mqtt_message(hass, "swing-state", '"on"')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "on"
|
|
|
|
# Temperature - with valid value
|
|
assert state.attributes.get("temperature") is None
|
|
async_fire_mqtt_message(hass, "temperature-state", '"1031"')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 1031
|
|
|
|
# Temperature - with invalid value
|
|
async_fire_mqtt_message(hass, "temperature-state", '"-INVALID-"')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
# make sure, the invalid value gets logged...
|
|
assert "Could not parse temperature_state_template from -INVALID-" in caplog.text
|
|
# ... but the actual value stays unchanged.
|
|
assert state.attributes.get("temperature") == 1031
|
|
|
|
# Temperature - with JSON null value
|
|
async_fire_mqtt_message(hass, "temperature-state", "null")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") is None
|
|
|
|
# Humidity - with valid value
|
|
assert state.attributes.get("humidity") is None
|
|
async_fire_mqtt_message(hass, "humidity-state", '"82"')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") == 82
|
|
|
|
# Humidity - with invalid value
|
|
async_fire_mqtt_message(hass, "humidity-state", '"-INVALID-"')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
# make sure, the invalid value gets logged...
|
|
assert (
|
|
"Could not parse target_humidity_state_template from -INVALID-" in caplog.text
|
|
)
|
|
# ... but the actual value stays unchanged.
|
|
assert state.attributes.get("humidity") == 82
|
|
|
|
# reset the humidity
|
|
async_fire_mqtt_message(hass, "humidity-state", "null")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") is None
|
|
|
|
# Preset Mode
|
|
assert state.attributes.get("preset_mode") == "none"
|
|
async_fire_mqtt_message(hass, "current-preset-mode", '{"attribute": "eco"}')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "eco"
|
|
# Test with an empty json
|
|
async_fire_mqtt_message(
|
|
hass, "current-preset-mode", '{"other_attribute": "some_value"}'
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == "eco"
|
|
|
|
# Aux mode
|
|
assert state.attributes.get("aux_heat") == "off"
|
|
async_fire_mqtt_message(hass, "aux-state", "switchmeon")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("aux_heat") == "on"
|
|
|
|
# anything other than 'switchmeon' should turn Aux mode off
|
|
async_fire_mqtt_message(hass, "aux-state", "somerandomstring")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("aux_heat") == "off"
|
|
|
|
# Current temperature
|
|
async_fire_mqtt_message(hass, "current-temperature", '"74656"')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("current_temperature") == 74656
|
|
# Test resetting the current temperature using a JSON null value
|
|
async_fire_mqtt_message(hass, "current-temperature", "null")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("current_temperature") is None
|
|
|
|
# Current humidity
|
|
async_fire_mqtt_message(hass, "current-humidity", '"35"')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("current_humidity") == 35
|
|
async_fire_mqtt_message(hass, "current-humidity", "null")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("current_humidity") is None
|
|
|
|
# Action
|
|
async_fire_mqtt_message(hass, "action", '"cooling"')
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("hvac_action") == "cooling"
|
|
|
|
# Test ignoring null values
|
|
async_fire_mqtt_message(hass, "action", "null")
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("hvac_action") == "cooling"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
{
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: {
|
|
"name": "test",
|
|
"mode_command_topic": "mode-topic",
|
|
"target_humidity_command_topic": "humidity-topic",
|
|
"temperature_command_topic": "temperature-topic",
|
|
"temperature_low_command_topic": "temperature-low-topic",
|
|
"temperature_high_command_topic": "temperature-high-topic",
|
|
"fan_mode_command_topic": "fan-mode-topic",
|
|
"swing_mode_command_topic": "swing-mode-topic",
|
|
"aux_command_topic": "aux-topic",
|
|
"preset_mode_command_topic": "preset-mode-topic",
|
|
"preset_modes": [
|
|
"eco",
|
|
"away",
|
|
"boost",
|
|
"comfort",
|
|
"home",
|
|
"sleep",
|
|
"activity",
|
|
],
|
|
# Create simple templates
|
|
"fan_mode_command_template": "fan_mode: {{ value }}",
|
|
"preset_mode_command_template": "preset_mode: {{ value }}",
|
|
"mode_command_template": "mode: {{ value }}",
|
|
"swing_mode_command_template": "swing_mode: {{ value }}",
|
|
"temperature_command_template": "temp: {{ value }}",
|
|
"temperature_high_command_template": "temp_hi: {{ value }}",
|
|
"temperature_low_command_template": "temp_lo: {{ value }}",
|
|
"target_humidity_command_template": "humidity: {{ value }}",
|
|
}
|
|
}
|
|
}
|
|
],
|
|
)
|
|
async def test_set_and_templates(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test setting various attributes with templates."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
# Fan Mode
|
|
await common.async_set_fan_mode(hass, "high", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"fan-mode-topic", "fan_mode: high", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("fan_mode") == "high"
|
|
|
|
# Preset Mode
|
|
await common.async_set_preset_mode(hass, PRESET_ECO, ENTITY_CLIMATE)
|
|
assert mqtt_mock.async_publish.call_count == 1
|
|
mqtt_mock.async_publish.assert_any_call(
|
|
"preset-mode-topic", "preset_mode: eco", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("preset_mode") == PRESET_ECO
|
|
|
|
# Mode
|
|
await common.async_set_hvac_mode(hass, "cool", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"mode-topic", "mode: cool", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.state == "cool"
|
|
|
|
# Swing Mode
|
|
await common.async_set_swing_mode(hass, "on", ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"swing-mode-topic", "swing_mode: on", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("swing_mode") == "on"
|
|
|
|
# Temperature
|
|
await common.async_set_temperature(hass, temperature=47, entity_id=ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"temperature-topic", "temp: 47.0", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 47
|
|
|
|
# Temperature Low/High
|
|
await common.async_set_temperature(
|
|
hass, target_temp_low=20, target_temp_high=23, entity_id=ENTITY_CLIMATE
|
|
)
|
|
mqtt_mock.async_publish.assert_any_call(
|
|
"temperature-low-topic", "temp_lo: 20.0", 0, False
|
|
)
|
|
mqtt_mock.async_publish.assert_any_call(
|
|
"temperature-high-topic", "temp_hi: 23.0", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("target_temp_low") == 20
|
|
assert state.attributes.get("target_temp_high") == 23
|
|
|
|
# Humidity
|
|
await common.async_set_humidity(hass, humidity=82, entity_id=ENTITY_CLIMATE)
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
"humidity-topic", "humidity: 82", 0, False
|
|
)
|
|
mqtt_mock.async_publish.reset_mock()
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("humidity") == 82
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[help_custom_config(climate.DOMAIN, DEFAULT_CONFIG, ({"min_temp": 26},))],
|
|
)
|
|
async def test_min_temp_custom(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test a custom min temp."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
min_temp = state.attributes.get("min_temp")
|
|
|
|
assert isinstance(min_temp, float)
|
|
assert state.attributes.get("min_temp") == 26
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[help_custom_config(climate.DOMAIN, DEFAULT_CONFIG, ({"max_temp": 60},))],
|
|
)
|
|
async def test_max_temp_custom(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test a custom max temp."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
max_temp = state.attributes.get("max_temp")
|
|
|
|
assert isinstance(max_temp, float)
|
|
assert max_temp == 60
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[help_custom_config(climate.DOMAIN, DEFAULT_CONFIG, ({"min_humidity": 42},))],
|
|
)
|
|
async def test_min_humidity_custom(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test a custom min humidity."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
min_humidity = state.attributes.get("min_humidity")
|
|
|
|
assert isinstance(min_humidity, float)
|
|
assert state.attributes.get("min_humidity") == 42
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[help_custom_config(climate.DOMAIN, DEFAULT_CONFIG, ({"max_humidity": 58},))],
|
|
)
|
|
async def test_max_humidity_custom(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test a custom max humidity."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
max_humidity = state.attributes.get("max_humidity")
|
|
|
|
assert isinstance(max_humidity, float)
|
|
assert max_humidity == 58
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[help_custom_config(climate.DOMAIN, DEFAULT_CONFIG, ({"temp_step": 0.01},))],
|
|
)
|
|
async def test_temp_step_custom(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test a custom temp step."""
|
|
await mqtt_mock_entry()
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
temp_step = state.attributes.get("target_temp_step")
|
|
|
|
assert isinstance(temp_step, float)
|
|
assert temp_step == 0.01
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
help_custom_config(
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
(
|
|
{
|
|
"temperature_unit": "F",
|
|
"current_temperature_topic": "current_temperature",
|
|
},
|
|
),
|
|
)
|
|
],
|
|
)
|
|
async def test_temperature_unit(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test that setting temperature unit converts temperature values."""
|
|
await mqtt_mock_entry()
|
|
|
|
async_fire_mqtt_message(hass, "current_temperature", "77")
|
|
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("current_temperature") == 25
|
|
|
|
|
|
async def test_setting_attribute_via_mqtt_json_message(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test the setting of attribute via MQTT with JSON payload."""
|
|
await help_test_setting_attribute_via_mqtt_json_message(
|
|
hass, mqtt_mock_entry, climate.DOMAIN, DEFAULT_CONFIG
|
|
)
|
|
|
|
|
|
async def test_setting_blocked_attribute_via_mqtt_json_message(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test the setting of attribute via MQTT with JSON payload."""
|
|
await help_test_setting_blocked_attribute_via_mqtt_json_message(
|
|
hass,
|
|
mqtt_mock_entry,
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
MQTT_CLIMATE_ATTRIBUTES_BLOCKED,
|
|
)
|
|
|
|
|
|
async def test_setting_attribute_with_template(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test the setting of attribute via MQTT with JSON payload."""
|
|
await help_test_setting_attribute_with_template(
|
|
hass, mqtt_mock_entry, climate.DOMAIN, DEFAULT_CONFIG
|
|
)
|
|
|
|
|
|
async def test_update_with_json_attrs_not_dict(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test attributes get extracted from a JSON result."""
|
|
await help_test_update_with_json_attrs_not_dict(
|
|
hass,
|
|
mqtt_mock_entry,
|
|
caplog,
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
)
|
|
|
|
|
|
async def test_update_with_json_attrs_bad_json(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test attributes get extracted from a JSON result."""
|
|
await help_test_update_with_json_attrs_bad_json(
|
|
hass,
|
|
mqtt_mock_entry,
|
|
caplog,
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
)
|
|
|
|
|
|
async def test_discovery_update_attr(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test update of discovered MQTTAttributes."""
|
|
await help_test_discovery_update_attr(
|
|
hass,
|
|
mqtt_mock_entry,
|
|
caplog,
|
|
climate.DOMAIN,
|
|
DEFAULT_CONFIG,
|
|
)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[
|
|
{
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: [
|
|
{
|
|
"name": "Test 1",
|
|
"mode_state_topic": "test_topic1/state",
|
|
"mode_command_topic": "test_topic1/command",
|
|
"unique_id": "TOTALLY_UNIQUE",
|
|
},
|
|
{
|
|
"name": "Test 2",
|
|
"mode_state_topic": "test_topic2/state",
|
|
"mode_command_topic": "test_topic2/command",
|
|
"unique_id": "TOTALLY_UNIQUE",
|
|
},
|
|
]
|
|
}
|
|
}
|
|
],
|
|
)
|
|
async def test_unique_id(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test unique id option only creates one climate per unique_id."""
|
|
await help_test_unique_id(hass, mqtt_mock_entry, climate.DOMAIN)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("topic", "value", "attribute", "attribute_value"),
|
|
[
|
|
("action_topic", "heating", ATTR_HVAC_ACTION, "heating"),
|
|
("action_topic", "cooling", ATTR_HVAC_ACTION, "cooling"),
|
|
("aux_state_topic", "ON", ATTR_AUX_HEAT, "on"),
|
|
("current_temperature_topic", "22.1", ATTR_CURRENT_TEMPERATURE, 22.1),
|
|
("current_humidity_topic", "60.4", ATTR_CURRENT_HUMIDITY, 60.4),
|
|
("fan_mode_state_topic", "low", ATTR_FAN_MODE, "low"),
|
|
("mode_state_topic", "cool", None, None),
|
|
("mode_state_topic", "fan_only", None, None),
|
|
("swing_mode_state_topic", "on", ATTR_SWING_MODE, "on"),
|
|
("temperature_low_state_topic", "19.1", ATTR_TARGET_TEMP_LOW, 19.1),
|
|
("temperature_high_state_topic", "22.9", ATTR_TARGET_TEMP_HIGH, 22.9),
|
|
("temperature_state_topic", "19.9", ATTR_TEMPERATURE, 19.9),
|
|
("target_humidity_state_topic", "82.6", ATTR_HUMIDITY, 82.6),
|
|
],
|
|
)
|
|
async def test_encoding_subscribable_topics(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
topic: str,
|
|
value: str,
|
|
attribute: str | None,
|
|
attribute_value: Any,
|
|
) -> None:
|
|
"""Test handling of incoming encoded payload."""
|
|
config = copy.deepcopy(DEFAULT_CONFIG[mqtt.DOMAIN][climate.DOMAIN])
|
|
await help_test_encoding_subscribable_topics(
|
|
hass,
|
|
mqtt_mock_entry,
|
|
climate.DOMAIN,
|
|
config,
|
|
topic,
|
|
value,
|
|
attribute,
|
|
attribute_value,
|
|
)
|
|
|
|
|
|
async def test_discovery_removal_climate(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test removal of discovered climate."""
|
|
data = json.dumps(DEFAULT_CONFIG[mqtt.DOMAIN][climate.DOMAIN])
|
|
await help_test_discovery_removal(
|
|
hass, mqtt_mock_entry, caplog, climate.DOMAIN, data
|
|
)
|
|
|
|
|
|
async def test_discovery_update_climate(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test update of discovered climate."""
|
|
config1 = {"name": "Beer"}
|
|
config2 = {"name": "Milk"}
|
|
await help_test_discovery_update(
|
|
hass, mqtt_mock_entry, caplog, climate.DOMAIN, config1, config2
|
|
)
|
|
|
|
|
|
async def test_discovery_update_unchanged_climate(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test update of discovered climate."""
|
|
data1 = '{ "name": "Beer" }'
|
|
with patch(
|
|
"homeassistant.components.mqtt.climate.MqttClimate.discovery_update"
|
|
) as discovery_update:
|
|
await help_test_discovery_update_unchanged(
|
|
hass,
|
|
mqtt_mock_entry,
|
|
caplog,
|
|
climate.DOMAIN,
|
|
data1,
|
|
discovery_update,
|
|
)
|
|
|
|
|
|
@pytest.mark.no_fail_on_log_exception
|
|
async def test_discovery_broken(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test handling of bad discovery message."""
|
|
data1 = '{ "name": "Beer", "power_command_topic": "test_topic#" }'
|
|
data2 = '{ "name": "Milk", "power_command_topic": "test_topic" }'
|
|
await help_test_discovery_broken(
|
|
hass, mqtt_mock_entry, caplog, climate.DOMAIN, data1, data2
|
|
)
|
|
|
|
|
|
async def test_entity_device_info_with_connection(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test MQTT climate device registry integration."""
|
|
await help_test_entity_device_info_with_connection(
|
|
hass, mqtt_mock_entry, climate.DOMAIN, DEFAULT_CONFIG
|
|
)
|
|
|
|
|
|
async def test_entity_device_info_with_identifier(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test MQTT climate device registry integration."""
|
|
await help_test_entity_device_info_with_identifier(
|
|
hass, mqtt_mock_entry, climate.DOMAIN, DEFAULT_CONFIG
|
|
)
|
|
|
|
|
|
async def test_entity_device_info_update(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test device registry update."""
|
|
await help_test_entity_device_info_update(
|
|
hass, mqtt_mock_entry, climate.DOMAIN, DEFAULT_CONFIG
|
|
)
|
|
|
|
|
|
async def test_entity_device_info_remove(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test device registry remove."""
|
|
await help_test_entity_device_info_remove(
|
|
hass, mqtt_mock_entry, climate.DOMAIN, DEFAULT_CONFIG
|
|
)
|
|
|
|
|
|
async def test_entity_id_update_subscriptions(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test MQTT subscriptions are managed when entity_id is updated."""
|
|
config = {
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: {
|
|
"name": "test",
|
|
"mode_state_topic": "test-topic",
|
|
"availability_topic": "avty-topic",
|
|
}
|
|
}
|
|
}
|
|
await help_test_entity_id_update_subscriptions(
|
|
hass,
|
|
mqtt_mock_entry,
|
|
climate.DOMAIN,
|
|
config,
|
|
["test-topic", "avty-topic"],
|
|
)
|
|
|
|
|
|
async def test_entity_id_update_discovery_update(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test MQTT discovery update when entity_id is updated."""
|
|
await help_test_entity_id_update_discovery_update(
|
|
hass, mqtt_mock_entry, climate.DOMAIN, DEFAULT_CONFIG
|
|
)
|
|
|
|
|
|
async def test_entity_debug_info_message(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test MQTT debug info."""
|
|
config = {
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: {
|
|
"name": "test",
|
|
"mode_command_topic": "command-topic",
|
|
"mode_state_topic": "test-topic",
|
|
}
|
|
}
|
|
}
|
|
await help_test_entity_debug_info_message(
|
|
hass,
|
|
mqtt_mock_entry,
|
|
climate.DOMAIN,
|
|
config,
|
|
climate.SERVICE_TURN_ON,
|
|
command_topic="command-topic",
|
|
command_payload="heat",
|
|
state_topic="test-topic",
|
|
)
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_precision_default(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test that setting precision to tenths works as intended."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
await common.async_set_temperature(
|
|
hass, temperature=23.67, entity_id=ENTITY_CLIMATE
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 23.7
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[help_custom_config(climate.DOMAIN, DEFAULT_CONFIG, ({"precision": 0.5},))],
|
|
)
|
|
async def test_precision_halves(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test that setting precision to halves works as intended."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
await common.async_set_temperature(
|
|
hass, temperature=23.67, entity_id=ENTITY_CLIMATE
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 23.5
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"hass_config",
|
|
[help_custom_config(climate.DOMAIN, DEFAULT_CONFIG, ({"precision": 1.0},))],
|
|
)
|
|
async def test_precision_whole(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test that setting precision to whole works as intended."""
|
|
mqtt_mock = await mqtt_mock_entry()
|
|
|
|
await common.async_set_temperature(
|
|
hass, temperature=23.67, entity_id=ENTITY_CLIMATE
|
|
)
|
|
state = hass.states.get(ENTITY_CLIMATE)
|
|
assert state.attributes.get("temperature") == 24.0
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("service", "topic", "parameters", "payload", "template"),
|
|
[
|
|
(
|
|
climate.SERVICE_TURN_ON,
|
|
"power_command_topic",
|
|
None,
|
|
"ON",
|
|
None,
|
|
),
|
|
(
|
|
climate.SERVICE_SET_HVAC_MODE,
|
|
"mode_command_topic",
|
|
{"hvac_mode": "cool"},
|
|
"cool",
|
|
"mode_command_template",
|
|
),
|
|
(
|
|
climate.SERVICE_SET_PRESET_MODE,
|
|
"preset_mode_command_topic",
|
|
{"preset_mode": "sleep"},
|
|
"sleep",
|
|
"preset_mode_command_template",
|
|
),
|
|
(
|
|
climate.SERVICE_SET_FAN_MODE,
|
|
"fan_mode_command_topic",
|
|
{"fan_mode": "medium"},
|
|
"medium",
|
|
"fan_mode_command_template",
|
|
),
|
|
(
|
|
climate.SERVICE_SET_SWING_MODE,
|
|
"swing_mode_command_topic",
|
|
{"swing_mode": "on"},
|
|
"on",
|
|
"swing_mode_command_template",
|
|
),
|
|
(
|
|
climate.SERVICE_SET_AUX_HEAT,
|
|
"aux_command_topic",
|
|
{"aux_heat": "on"},
|
|
"ON",
|
|
None,
|
|
),
|
|
(
|
|
climate.SERVICE_SET_TEMPERATURE,
|
|
"temperature_command_topic",
|
|
{"temperature": "20.1"},
|
|
20.1,
|
|
"temperature_command_template",
|
|
),
|
|
(
|
|
climate.SERVICE_SET_TEMPERATURE,
|
|
"temperature_low_command_topic",
|
|
{
|
|
"temperature": "20.1",
|
|
"target_temp_low": "15.1",
|
|
"target_temp_high": "29.8",
|
|
},
|
|
15.1,
|
|
"temperature_low_command_template",
|
|
),
|
|
(
|
|
climate.SERVICE_SET_TEMPERATURE,
|
|
"temperature_high_command_topic",
|
|
{
|
|
"temperature": "20.1",
|
|
"target_temp_low": "15.1",
|
|
"target_temp_high": "29.8",
|
|
},
|
|
29.8,
|
|
"temperature_high_command_template",
|
|
),
|
|
(
|
|
climate.SERVICE_SET_HUMIDITY,
|
|
"target_humidity_command_topic",
|
|
{"humidity": "82"},
|
|
82,
|
|
"target_humidity_command_template",
|
|
),
|
|
],
|
|
)
|
|
async def test_publishing_with_custom_encoding(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
service: str,
|
|
topic: str,
|
|
parameters: dict[str, Any],
|
|
payload: str,
|
|
template: str | None,
|
|
) -> None:
|
|
"""Test publishing MQTT payload with different encoding."""
|
|
domain = climate.DOMAIN
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
if topic != "preset_mode_command_topic":
|
|
del config[mqtt.DOMAIN][domain]["preset_mode_command_topic"]
|
|
del config[mqtt.DOMAIN][domain]["preset_modes"]
|
|
|
|
await help_test_publishing_with_custom_encoding(
|
|
hass,
|
|
mqtt_mock_entry,
|
|
caplog,
|
|
domain,
|
|
config,
|
|
service,
|
|
topic,
|
|
parameters,
|
|
payload,
|
|
template,
|
|
)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("hass_config", "valid"),
|
|
[
|
|
(
|
|
{
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: {
|
|
"name": "test_valid_humidity_min_max",
|
|
"min_humidity": 20,
|
|
"max_humidity": 80,
|
|
},
|
|
}
|
|
},
|
|
True,
|
|
),
|
|
(
|
|
{
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: {
|
|
"name": "test_invalid_humidity_min_max_1",
|
|
"min_humidity": 0,
|
|
"max_humidity": 101,
|
|
},
|
|
}
|
|
},
|
|
False,
|
|
),
|
|
(
|
|
{
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: {
|
|
"name": "test_invalid_humidity_min_max_2",
|
|
"max_humidity": 20,
|
|
"min_humidity": 40,
|
|
},
|
|
}
|
|
},
|
|
False,
|
|
),
|
|
(
|
|
{
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: {
|
|
"name": "test_valid_humidity_state",
|
|
"target_humidity_state_topic": "humidity-state",
|
|
"target_humidity_command_topic": "humidity-command",
|
|
},
|
|
}
|
|
},
|
|
True,
|
|
),
|
|
(
|
|
{
|
|
mqtt.DOMAIN: {
|
|
climate.DOMAIN: {
|
|
"name": "test_invalid_humidity_state",
|
|
"target_humidity_state_topic": "humidity-state",
|
|
},
|
|
}
|
|
},
|
|
False,
|
|
),
|
|
],
|
|
)
|
|
async def test_humidity_configuration_validity(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
valid: bool,
|
|
) -> None:
|
|
"""Test the validity of humidity configurations."""
|
|
if valid:
|
|
await mqtt_mock_entry()
|
|
return
|
|
with pytest.raises(AssertionError):
|
|
await mqtt_mock_entry()
|
|
|
|
|
|
async def test_reloadable(
|
|
hass: HomeAssistant,
|
|
mqtt_client_mock: MqttMockPahoClient,
|
|
) -> None:
|
|
"""Test reloading the MQTT platform."""
|
|
domain = climate.DOMAIN
|
|
config = DEFAULT_CONFIG
|
|
await help_test_reloadable(hass, mqtt_client_mock, domain, config)
|
|
|
|
|
|
@pytest.mark.parametrize("hass_config", [DEFAULT_CONFIG])
|
|
async def test_setup_manual_entity_from_yaml(
|
|
hass: HomeAssistant, mqtt_mock_entry: MqttMockHAClientGenerator
|
|
) -> None:
|
|
"""Test setup manual configured MQTT entity."""
|
|
await mqtt_mock_entry()
|
|
platform = climate.DOMAIN
|
|
assert hass.states.get(f"{platform}.test")
|
|
|
|
|
|
async def test_unload_entry(
|
|
hass: HomeAssistant,
|
|
mqtt_mock_entry: MqttMockHAClientGenerator,
|
|
) -> None:
|
|
"""Test unloading the config entry."""
|
|
domain = climate.DOMAIN
|
|
config = DEFAULT_CONFIG
|
|
await help_test_unload_config_entry_with_platform(
|
|
hass, mqtt_mock_entry, domain, config
|
|
)
|