2020-11-25 16:40:32 +00:00
|
|
|
"""The tests for the Tasmota fan platform."""
|
|
|
|
import copy
|
|
|
|
import json
|
2021-01-01 21:31:56 +00:00
|
|
|
from unittest.mock import patch
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
from hatasmota.utils import (
|
|
|
|
get_topic_stat_result,
|
|
|
|
get_topic_tele_state,
|
|
|
|
get_topic_tele_will,
|
|
|
|
)
|
2020-12-10 21:17:58 +00:00
|
|
|
import pytest
|
2021-08-25 11:54:36 +00:00
|
|
|
from voluptuous import MultipleInvalid
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
from homeassistant.components import fan
|
|
|
|
from homeassistant.components.tasmota.const import DEFAULT_PREFIX
|
2021-12-21 22:01:07 +00:00
|
|
|
from homeassistant.const import ATTR_ASSUMED_STATE, STATE_OFF, STATE_ON, Platform
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
from .test_common import (
|
|
|
|
DEFAULT_CONFIG,
|
|
|
|
help_test_availability,
|
|
|
|
help_test_availability_discovery_update,
|
|
|
|
help_test_availability_poll_state,
|
|
|
|
help_test_availability_when_connection_lost,
|
|
|
|
help_test_discovery_device_remove,
|
|
|
|
help_test_discovery_removal,
|
|
|
|
help_test_discovery_update_unchanged,
|
|
|
|
help_test_entity_id_update_discovery_update,
|
|
|
|
help_test_entity_id_update_subscriptions,
|
|
|
|
)
|
|
|
|
|
|
|
|
from tests.common import async_fire_mqtt_message
|
|
|
|
from tests.components.fan import common
|
|
|
|
|
|
|
|
|
|
|
|
async def test_controlling_state_via_mqtt(hass, mqtt_mock, setup_tasmota):
|
|
|
|
"""Test state update via MQTT."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["if"] = 1
|
|
|
|
mac = config["mac"]
|
|
|
|
|
|
|
|
async_fire_mqtt_message(
|
|
|
|
hass,
|
|
|
|
f"{DEFAULT_PREFIX}/{mac}/config",
|
|
|
|
json.dumps(config),
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == "unavailable"
|
|
|
|
assert not state.attributes.get(ATTR_ASSUMED_STATE)
|
|
|
|
|
|
|
|
async_fire_mqtt_message(hass, "tasmota_49A3BC/tele/LWT", "Online")
|
2021-10-28 12:23:26 +00:00
|
|
|
await hass.async_block_till_done()
|
2020-11-25 16:40:32 +00:00
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == STATE_OFF
|
2021-02-23 03:27:21 +00:00
|
|
|
assert state.attributes["percentage"] is None
|
2020-11-25 16:40:32 +00:00
|
|
|
assert state.attributes["supported_features"] == fan.SUPPORT_SET_SPEED
|
|
|
|
assert not state.attributes.get(ATTR_ASSUMED_STATE)
|
|
|
|
|
|
|
|
async_fire_mqtt_message(hass, "tasmota_49A3BC/tele/STATE", '{"FanSpeed":1}')
|
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == STATE_ON
|
2021-02-23 03:27:21 +00:00
|
|
|
assert state.attributes["percentage"] == 33
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
async_fire_mqtt_message(hass, "tasmota_49A3BC/tele/STATE", '{"FanSpeed":2}')
|
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == STATE_ON
|
2021-02-23 03:27:21 +00:00
|
|
|
assert state.attributes["percentage"] == 66
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
async_fire_mqtt_message(hass, "tasmota_49A3BC/tele/STATE", '{"FanSpeed":3}')
|
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == STATE_ON
|
2021-02-23 03:27:21 +00:00
|
|
|
assert state.attributes["percentage"] == 100
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
async_fire_mqtt_message(hass, "tasmota_49A3BC/tele/STATE", '{"FanSpeed":0}')
|
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == STATE_OFF
|
2021-02-23 03:27:21 +00:00
|
|
|
assert state.attributes["percentage"] == 0
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
async_fire_mqtt_message(hass, "tasmota_49A3BC/stat/RESULT", '{"FanSpeed":1}')
|
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == STATE_ON
|
2021-02-23 03:27:21 +00:00
|
|
|
assert state.attributes["percentage"] == 33
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
async_fire_mqtt_message(hass, "tasmota_49A3BC/stat/RESULT", '{"FanSpeed":0}')
|
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == STATE_OFF
|
2021-02-23 03:27:21 +00:00
|
|
|
assert state.attributes["percentage"] == 0
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_sending_mqtt_commands(hass, mqtt_mock, setup_tasmota):
|
|
|
|
"""Test the sending MQTT commands."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["if"] = 1
|
|
|
|
mac = config["mac"]
|
|
|
|
|
|
|
|
async_fire_mqtt_message(
|
|
|
|
hass,
|
|
|
|
f"{DEFAULT_PREFIX}/{mac}/config",
|
|
|
|
json.dumps(config),
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
async_fire_mqtt_message(hass, "tasmota_49A3BC/tele/LWT", "Online")
|
2021-10-28 12:23:26 +00:00
|
|
|
await hass.async_block_till_done()
|
2020-11-25 16:40:32 +00:00
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == STATE_OFF
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
|
|
# Turn the fan on and verify MQTT message is sent
|
|
|
|
await common.async_turn_on(hass, "fan.tasmota")
|
|
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
|
|
"tasmota_49A3BC/cmnd/FanSpeed", "2", 0, False
|
|
|
|
)
|
|
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
|
|
# Tasmota is not optimistic, the state should still be off
|
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == STATE_OFF
|
|
|
|
|
|
|
|
# Turn the fan off and verify MQTT message is sent
|
|
|
|
await common.async_turn_off(hass, "fan.tasmota")
|
|
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
|
|
"tasmota_49A3BC/cmnd/FanSpeed", "0", 0, False
|
|
|
|
)
|
|
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
2021-02-23 03:27:21 +00:00
|
|
|
# Set speed percentage and verify MQTT message is sent
|
|
|
|
await common.async_set_percentage(hass, "fan.tasmota", 0)
|
|
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
|
|
"tasmota_49A3BC/cmnd/FanSpeed", "0", 0, False
|
|
|
|
)
|
|
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
|
|
# Set speed percentage and verify MQTT message is sent
|
|
|
|
await common.async_set_percentage(hass, "fan.tasmota", 15)
|
|
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
|
|
"tasmota_49A3BC/cmnd/FanSpeed", "1", 0, False
|
|
|
|
)
|
|
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
|
|
# Set speed percentage and verify MQTT message is sent
|
|
|
|
await common.async_set_percentage(hass, "fan.tasmota", 50)
|
|
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
|
|
"tasmota_49A3BC/cmnd/FanSpeed", "2", 0, False
|
|
|
|
)
|
|
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
|
|
# Set speed percentage and verify MQTT message is sent
|
|
|
|
await common.async_set_percentage(hass, "fan.tasmota", 90)
|
|
|
|
mqtt_mock.async_publish.assert_called_once_with(
|
|
|
|
"tasmota_49A3BC/cmnd/FanSpeed", "3", 0, False
|
|
|
|
)
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
|
2021-08-25 11:54:36 +00:00
|
|
|
async def test_invalid_fan_speed_percentage(hass, mqtt_mock, setup_tasmota):
|
2020-12-10 21:17:58 +00:00
|
|
|
"""Test the sending MQTT commands."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["if"] = 1
|
|
|
|
mac = config["mac"]
|
|
|
|
|
|
|
|
async_fire_mqtt_message(
|
|
|
|
hass,
|
|
|
|
f"{DEFAULT_PREFIX}/{mac}/config",
|
|
|
|
json.dumps(config),
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
async_fire_mqtt_message(hass, "tasmota_49A3BC/tele/LWT", "Online")
|
2021-10-28 12:23:26 +00:00
|
|
|
await hass.async_block_till_done()
|
2020-12-10 21:17:58 +00:00
|
|
|
state = hass.states.get("fan.tasmota")
|
|
|
|
assert state.state == STATE_OFF
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
mqtt_mock.async_publish.reset_mock()
|
|
|
|
|
|
|
|
# Set an unsupported speed and verify MQTT message is not sent
|
2021-08-25 11:54:36 +00:00
|
|
|
with pytest.raises(MultipleInvalid) as excinfo:
|
|
|
|
await common.async_set_percentage(hass, "fan.tasmota", 101)
|
|
|
|
assert "value must be at most 100" in str(excinfo.value)
|
2020-12-10 21:17:58 +00:00
|
|
|
mqtt_mock.async_publish.assert_not_called()
|
|
|
|
|
|
|
|
|
2020-11-25 16:40:32 +00:00
|
|
|
async def test_availability_when_connection_lost(
|
|
|
|
hass, mqtt_client_mock, mqtt_mock, setup_tasmota
|
|
|
|
):
|
|
|
|
"""Test availability after MQTT disconnection."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["dn"] = "Test"
|
|
|
|
config["if"] = 1
|
|
|
|
await help_test_availability_when_connection_lost(
|
2021-12-21 22:01:07 +00:00
|
|
|
hass, mqtt_client_mock, mqtt_mock, Platform.FAN, config
|
2020-11-25 16:40:32 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def test_availability(hass, mqtt_mock, setup_tasmota):
|
|
|
|
"""Test availability."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["dn"] = "Test"
|
|
|
|
config["if"] = 1
|
2021-12-21 22:01:07 +00:00
|
|
|
await help_test_availability(hass, mqtt_mock, Platform.FAN, config)
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_availability_discovery_update(hass, mqtt_mock, setup_tasmota):
|
|
|
|
"""Test availability discovery update."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["dn"] = "Test"
|
|
|
|
config["if"] = 1
|
2021-12-21 22:01:07 +00:00
|
|
|
await help_test_availability_discovery_update(hass, mqtt_mock, Platform.FAN, config)
|
2020-11-25 16:40:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_availability_poll_state(
|
|
|
|
hass, mqtt_client_mock, mqtt_mock, setup_tasmota
|
|
|
|
):
|
|
|
|
"""Test polling after MQTT connection (re)established."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["if"] = 1
|
|
|
|
poll_topic = "tasmota_49A3BC/cmnd/STATE"
|
|
|
|
await help_test_availability_poll_state(
|
2021-12-21 22:01:07 +00:00
|
|
|
hass, mqtt_client_mock, mqtt_mock, Platform.FAN, config, poll_topic, ""
|
2020-11-25 16:40:32 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def test_discovery_removal_fan(hass, mqtt_mock, caplog, setup_tasmota):
|
|
|
|
"""Test removal of discovered fan."""
|
|
|
|
config1 = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config1["dn"] = "Test"
|
|
|
|
config1["if"] = 1
|
|
|
|
config2 = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config2["dn"] = "Test"
|
|
|
|
config2["if"] = 0
|
|
|
|
|
|
|
|
await help_test_discovery_removal(
|
2021-12-21 22:01:07 +00:00
|
|
|
hass, mqtt_mock, caplog, Platform.FAN, config1, config2
|
2020-11-25 16:40:32 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def test_discovery_update_unchanged_fan(hass, mqtt_mock, caplog, setup_tasmota):
|
|
|
|
"""Test update of discovered fan."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["dn"] = "Test"
|
|
|
|
config["if"] = 1
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.tasmota.fan.TasmotaFan.discovery_update"
|
|
|
|
) as discovery_update:
|
|
|
|
await help_test_discovery_update_unchanged(
|
2021-12-21 22:01:07 +00:00
|
|
|
hass, mqtt_mock, caplog, Platform.FAN, config, discovery_update
|
2020-11-25 16:40:32 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def test_discovery_device_remove(hass, mqtt_mock, setup_tasmota):
|
|
|
|
"""Test device registry remove."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["dn"] = "Test"
|
|
|
|
config["if"] = 1
|
|
|
|
unique_id = f"{DEFAULT_CONFIG['mac']}_fan_fan_ifan"
|
|
|
|
await help_test_discovery_device_remove(
|
2021-12-21 22:01:07 +00:00
|
|
|
hass, mqtt_mock, Platform.FAN, unique_id, config
|
2020-11-25 16:40:32 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def test_entity_id_update_subscriptions(hass, mqtt_mock, setup_tasmota):
|
|
|
|
"""Test MQTT subscriptions are managed when entity_id is updated."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["dn"] = "Test"
|
|
|
|
config["if"] = 1
|
|
|
|
topics = [
|
|
|
|
get_topic_stat_result(config),
|
|
|
|
get_topic_tele_state(config),
|
|
|
|
get_topic_tele_will(config),
|
|
|
|
]
|
|
|
|
await help_test_entity_id_update_subscriptions(
|
2021-12-21 22:01:07 +00:00
|
|
|
hass, mqtt_mock, Platform.FAN, config, topics
|
2020-11-25 16:40:32 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def test_entity_id_update_discovery_update(hass, mqtt_mock, setup_tasmota):
|
|
|
|
"""Test MQTT discovery update when entity_id is updated."""
|
|
|
|
config = copy.deepcopy(DEFAULT_CONFIG)
|
|
|
|
config["dn"] = "Test"
|
|
|
|
config["if"] = 1
|
|
|
|
await help_test_entity_id_update_discovery_update(
|
2021-12-21 22:01:07 +00:00
|
|
|
hass, mqtt_mock, Platform.FAN, config
|
2020-11-25 16:40:32 +00:00
|
|
|
)
|