mirror of
https://github.com/home-assistant/core.git
synced 2025-07-22 20:57:21 +00:00
Add support for Dyson Pure HP04 purifying heater + fan (#34537)
* fix unnecesary checks * change ClimateDevice to ClimateEntity * Clean up * Formatting * Fix tests * Clean tests * Clean up tests * Fix device mock * Use safer patch target path * Extract constant * Remove not needed property * Guard for missing target temperature * Use async_mock mocks Co-authored-by: Martin Hjelmare <marhje52@gmail.com>
This commit is contained in:
parent
8541ae0360
commit
f8de0594b9
@ -1,19 +1,36 @@
|
|||||||
"""Support for Dyson Pure Hot+Cool link fan."""
|
"""Support for Dyson Pure Hot+Cool link fan."""
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
from libpurecool.const import FocusMode, HeatMode, HeatState, HeatTarget
|
from libpurecool.const import (
|
||||||
|
FanPower,
|
||||||
|
FanSpeed,
|
||||||
|
FanState,
|
||||||
|
FocusMode,
|
||||||
|
HeatMode,
|
||||||
|
HeatState,
|
||||||
|
HeatTarget,
|
||||||
|
)
|
||||||
|
from libpurecool.dyson_pure_hotcool import DysonPureHotCool
|
||||||
from libpurecool.dyson_pure_hotcool_link import DysonPureHotCoolLink
|
from libpurecool.dyson_pure_hotcool_link import DysonPureHotCoolLink
|
||||||
from libpurecool.dyson_pure_state import DysonPureHotCoolState
|
from libpurecool.dyson_pure_state import DysonPureHotCoolState
|
||||||
|
from libpurecool.dyson_pure_state_v2 import DysonPureHotCoolV2State
|
||||||
|
|
||||||
from homeassistant.components.climate import ClimateEntity
|
from homeassistant.components.climate import ClimateEntity
|
||||||
from homeassistant.components.climate.const import (
|
from homeassistant.components.climate.const import (
|
||||||
CURRENT_HVAC_COOL,
|
CURRENT_HVAC_COOL,
|
||||||
CURRENT_HVAC_HEAT,
|
CURRENT_HVAC_HEAT,
|
||||||
CURRENT_HVAC_IDLE,
|
CURRENT_HVAC_IDLE,
|
||||||
|
CURRENT_HVAC_OFF,
|
||||||
|
FAN_AUTO,
|
||||||
FAN_DIFFUSE,
|
FAN_DIFFUSE,
|
||||||
FAN_FOCUS,
|
FAN_FOCUS,
|
||||||
|
FAN_HIGH,
|
||||||
|
FAN_LOW,
|
||||||
|
FAN_MEDIUM,
|
||||||
|
FAN_OFF,
|
||||||
HVAC_MODE_COOL,
|
HVAC_MODE_COOL,
|
||||||
HVAC_MODE_HEAT,
|
HVAC_MODE_HEAT,
|
||||||
|
HVAC_MODE_OFF,
|
||||||
SUPPORT_FAN_MODE,
|
SUPPORT_FAN_MODE,
|
||||||
SUPPORT_TARGET_TEMPERATURE,
|
SUPPORT_TARGET_TEMPERATURE,
|
||||||
)
|
)
|
||||||
@ -24,26 +41,53 @@ from . import DYSON_DEVICES
|
|||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
SUPPORT_FAN = [FAN_FOCUS, FAN_DIFFUSE]
|
SUPPORT_FAN = [FAN_FOCUS, FAN_DIFFUSE]
|
||||||
|
SUPPORT_FAN_PCOOL = [FAN_OFF, FAN_AUTO, FAN_LOW, FAN_MEDIUM, FAN_HIGH]
|
||||||
SUPPORT_HVAG = [HVAC_MODE_COOL, HVAC_MODE_HEAT]
|
SUPPORT_HVAG = [HVAC_MODE_COOL, HVAC_MODE_HEAT]
|
||||||
|
SUPPORT_HVAC_PCOOL = [HVAC_MODE_COOL, HVAC_MODE_HEAT, HVAC_MODE_OFF]
|
||||||
SUPPORT_FLAGS = SUPPORT_TARGET_TEMPERATURE | SUPPORT_FAN_MODE
|
SUPPORT_FLAGS = SUPPORT_TARGET_TEMPERATURE | SUPPORT_FAN_MODE
|
||||||
|
|
||||||
|
DYSON_KNOWN_CLIMATE_DEVICES = "dyson_known_climate_devices"
|
||||||
|
|
||||||
def setup_platform(hass, config, add_devices, discovery_info=None):
|
SPEED_MAP = {
|
||||||
|
FanSpeed.FAN_SPEED_1.value: FAN_LOW,
|
||||||
|
FanSpeed.FAN_SPEED_2.value: FAN_LOW,
|
||||||
|
FanSpeed.FAN_SPEED_3.value: FAN_LOW,
|
||||||
|
FanSpeed.FAN_SPEED_4.value: FAN_LOW,
|
||||||
|
FanSpeed.FAN_SPEED_AUTO.value: FAN_AUTO,
|
||||||
|
FanSpeed.FAN_SPEED_5.value: FAN_MEDIUM,
|
||||||
|
FanSpeed.FAN_SPEED_6.value: FAN_MEDIUM,
|
||||||
|
FanSpeed.FAN_SPEED_7.value: FAN_MEDIUM,
|
||||||
|
FanSpeed.FAN_SPEED_8.value: FAN_HIGH,
|
||||||
|
FanSpeed.FAN_SPEED_9.value: FAN_HIGH,
|
||||||
|
FanSpeed.FAN_SPEED_10.value: FAN_HIGH,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
||||||
"""Set up the Dyson fan components."""
|
"""Set up the Dyson fan components."""
|
||||||
if discovery_info is None:
|
if discovery_info is None:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Get Dyson Devices from parent component.
|
known_devices = hass.data.setdefault(DYSON_KNOWN_CLIMATE_DEVICES, set())
|
||||||
add_devices(
|
|
||||||
[
|
# Get Dyson Devices from parent component
|
||||||
DysonPureHotCoolLinkDevice(device)
|
new_entities = []
|
||||||
for device in hass.data[DYSON_DEVICES]
|
|
||||||
if isinstance(device, DysonPureHotCoolLink)
|
for device in hass.data[DYSON_DEVICES]:
|
||||||
]
|
if device.serial not in known_devices:
|
||||||
)
|
if isinstance(device, DysonPureHotCool):
|
||||||
|
dyson_entity = DysonPureHotCoolEntity(device)
|
||||||
|
new_entities.append(dyson_entity)
|
||||||
|
known_devices.add(device.serial)
|
||||||
|
elif isinstance(device, DysonPureHotCoolLink):
|
||||||
|
dyson_entity = DysonPureHotCoolLinkEntity(device)
|
||||||
|
new_entities.append(dyson_entity)
|
||||||
|
known_devices.add(device.serial)
|
||||||
|
|
||||||
|
add_entities(new_entities)
|
||||||
|
|
||||||
|
|
||||||
class DysonPureHotCoolLinkDevice(ClimateEntity):
|
class DysonPureHotCoolLinkEntity(ClimateEntity):
|
||||||
"""Representation of a Dyson climate fan."""
|
"""Representation of a Dyson climate fan."""
|
||||||
|
|
||||||
def __init__(self, device):
|
def __init__(self, device):
|
||||||
@ -57,11 +101,11 @@ class DysonPureHotCoolLinkDevice(ClimateEntity):
|
|||||||
|
|
||||||
def on_message(self, message):
|
def on_message(self, message):
|
||||||
"""Call when new messages received from the climate."""
|
"""Call when new messages received from the climate."""
|
||||||
if not isinstance(message, DysonPureHotCoolState):
|
if isinstance(message, DysonPureHotCoolState):
|
||||||
return
|
_LOGGER.debug(
|
||||||
|
"Message received for climate device %s : %s", self.name, message
|
||||||
_LOGGER.debug("Message received for climate device %s : %s", self.name, message)
|
)
|
||||||
self.schedule_update_ha_state()
|
self.schedule_update_ha_state()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def should_poll(self):
|
def should_poll(self):
|
||||||
@ -188,3 +232,164 @@ class DysonPureHotCoolLinkDevice(ClimateEntity):
|
|||||||
def max_temp(self):
|
def max_temp(self):
|
||||||
"""Return the maximum temperature."""
|
"""Return the maximum temperature."""
|
||||||
return 37
|
return 37
|
||||||
|
|
||||||
|
|
||||||
|
class DysonPureHotCoolEntity(ClimateEntity):
|
||||||
|
"""Representation of a Dyson climate hot+cool fan."""
|
||||||
|
|
||||||
|
def __init__(self, device):
|
||||||
|
"""Initialize the fan."""
|
||||||
|
self._device = device
|
||||||
|
|
||||||
|
async def async_added_to_hass(self):
|
||||||
|
"""Call when entity is added to hass."""
|
||||||
|
self.hass.async_add_executor_job(
|
||||||
|
self._device.add_message_listener, self.on_message
|
||||||
|
)
|
||||||
|
|
||||||
|
def on_message(self, message):
|
||||||
|
"""Call when new messages received from the climate device."""
|
||||||
|
if isinstance(message, DysonPureHotCoolV2State):
|
||||||
|
_LOGGER.debug(
|
||||||
|
"Message received for climate device %s : %s", self.name, message
|
||||||
|
)
|
||||||
|
self.schedule_update_ha_state()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def should_poll(self):
|
||||||
|
"""No polling needed."""
|
||||||
|
return False
|
||||||
|
|
||||||
|
@property
|
||||||
|
def supported_features(self):
|
||||||
|
"""Return the list of supported features."""
|
||||||
|
return SUPPORT_FLAGS
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""Return the display name of this climate."""
|
||||||
|
return self._device.name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def temperature_unit(self):
|
||||||
|
"""Return the unit of measurement."""
|
||||||
|
return TEMP_CELSIUS
|
||||||
|
|
||||||
|
@property
|
||||||
|
def current_temperature(self):
|
||||||
|
"""Return the current temperature."""
|
||||||
|
if self._device.environmental_state.temperature is not None:
|
||||||
|
temperature_kelvin = self._device.environmental_state.temperature
|
||||||
|
if temperature_kelvin != 0:
|
||||||
|
return float("{:.1f}".format(temperature_kelvin - 273))
|
||||||
|
return None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def target_temperature(self):
|
||||||
|
"""Return the target temperature."""
|
||||||
|
heat_target = int(self._device.state.heat_target) / 10
|
||||||
|
return int(heat_target - 273)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def current_humidity(self):
|
||||||
|
"""Return the current humidity."""
|
||||||
|
if self._device.environmental_state.humidity is not None:
|
||||||
|
if self._device.environmental_state.humidity != 0:
|
||||||
|
return self._device.environmental_state.humidity
|
||||||
|
return None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def hvac_mode(self):
|
||||||
|
"""Return hvac operation ie. heat, cool mode.
|
||||||
|
|
||||||
|
Need to be one of HVAC_MODE_*.
|
||||||
|
"""
|
||||||
|
if self._device.state.fan_power == FanPower.POWER_OFF.value:
|
||||||
|
return HVAC_MODE_OFF
|
||||||
|
if self._device.state.heat_mode == HeatMode.HEAT_ON.value:
|
||||||
|
return HVAC_MODE_HEAT
|
||||||
|
return HVAC_MODE_COOL
|
||||||
|
|
||||||
|
@property
|
||||||
|
def hvac_modes(self):
|
||||||
|
"""Return the list of available hvac operation modes.
|
||||||
|
|
||||||
|
Need to be a subset of HVAC_MODES.
|
||||||
|
"""
|
||||||
|
return SUPPORT_HVAC_PCOOL
|
||||||
|
|
||||||
|
@property
|
||||||
|
def hvac_action(self):
|
||||||
|
"""Return the current running hvac operation if supported.
|
||||||
|
|
||||||
|
Need to be one of CURRENT_HVAC_*.
|
||||||
|
"""
|
||||||
|
if self._device.state.fan_power == FanPower.POWER_OFF.value:
|
||||||
|
return CURRENT_HVAC_OFF
|
||||||
|
if self._device.state.heat_mode == HeatMode.HEAT_ON.value:
|
||||||
|
if self._device.state.heat_state == HeatState.HEAT_STATE_ON.value:
|
||||||
|
return CURRENT_HVAC_HEAT
|
||||||
|
return CURRENT_HVAC_IDLE
|
||||||
|
return CURRENT_HVAC_COOL
|
||||||
|
|
||||||
|
@property
|
||||||
|
def fan_mode(self):
|
||||||
|
"""Return the fan setting."""
|
||||||
|
if self._device.state.fan_state == FanState.FAN_OFF.value:
|
||||||
|
return FAN_OFF
|
||||||
|
|
||||||
|
return SPEED_MAP[self._device.state.speed]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def fan_modes(self):
|
||||||
|
"""Return the list of available fan modes."""
|
||||||
|
return SUPPORT_FAN_PCOOL
|
||||||
|
|
||||||
|
def set_temperature(self, **kwargs):
|
||||||
|
"""Set new target temperature."""
|
||||||
|
target_temp = kwargs.get(ATTR_TEMPERATURE)
|
||||||
|
if target_temp is None:
|
||||||
|
_LOGGER.error("Missing target temperature %s", kwargs)
|
||||||
|
return
|
||||||
|
target_temp = int(target_temp)
|
||||||
|
_LOGGER.debug("Set %s temperature %s", self.name, target_temp)
|
||||||
|
# Limit the target temperature into acceptable range.
|
||||||
|
target_temp = min(self.max_temp, target_temp)
|
||||||
|
target_temp = max(self.min_temp, target_temp)
|
||||||
|
self._device.set_heat_target(HeatTarget.celsius(target_temp))
|
||||||
|
|
||||||
|
def set_fan_mode(self, fan_mode):
|
||||||
|
"""Set new fan mode."""
|
||||||
|
_LOGGER.debug("Set %s focus mode %s", self.name, fan_mode)
|
||||||
|
if fan_mode == FAN_OFF:
|
||||||
|
self._device.turn_off()
|
||||||
|
elif fan_mode == FAN_LOW:
|
||||||
|
self._device.set_fan_speed(FanSpeed.FAN_SPEED_4)
|
||||||
|
elif fan_mode == FAN_MEDIUM:
|
||||||
|
self._device.set_fan_speed(FanSpeed.FAN_SPEED_7)
|
||||||
|
elif fan_mode == FAN_HIGH:
|
||||||
|
self._device.set_fan_speed(FanSpeed.FAN_SPEED_10)
|
||||||
|
elif fan_mode == FAN_AUTO:
|
||||||
|
self._device.set_fan_speed(FanSpeed.FAN_SPEED_AUTO)
|
||||||
|
|
||||||
|
def set_hvac_mode(self, hvac_mode):
|
||||||
|
"""Set new target hvac mode."""
|
||||||
|
_LOGGER.debug("Set %s heat mode %s", self.name, hvac_mode)
|
||||||
|
if hvac_mode == HVAC_MODE_OFF:
|
||||||
|
self._device.turn_off()
|
||||||
|
elif self._device.state.fan_power == FanPower.POWER_OFF.value:
|
||||||
|
self._device.turn_on()
|
||||||
|
if hvac_mode == HVAC_MODE_HEAT:
|
||||||
|
self._device.enable_heat_mode()
|
||||||
|
elif hvac_mode == HVAC_MODE_COOL:
|
||||||
|
self._device.disable_heat_mode()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def min_temp(self):
|
||||||
|
"""Return the minimum temperature."""
|
||||||
|
return 1
|
||||||
|
|
||||||
|
@property
|
||||||
|
def max_temp(self):
|
||||||
|
"""Return the maximum temperature."""
|
||||||
|
return 37
|
||||||
|
@ -23,3 +23,4 @@ def load_mock_device(device):
|
|||||||
device.state.oscillation_angle_low = "000"
|
device.state.oscillation_angle_low = "000"
|
||||||
device.state.oscillation_angle_high = "000"
|
device.state.oscillation_angle_high = "000"
|
||||||
device.state.filter_life = "000"
|
device.state.filter_life = "000"
|
||||||
|
device.state.heat_target = 200
|
||||||
|
@ -1,19 +1,52 @@
|
|||||||
"""Test the Dyson fan component."""
|
"""Test the Dyson fan component."""
|
||||||
|
import json
|
||||||
import unittest
|
import unittest
|
||||||
from unittest import mock
|
|
||||||
|
|
||||||
from libpurecool.const import FocusMode, HeatMode, HeatState, HeatTarget
|
from libpurecool.const import (
|
||||||
|
FanPower,
|
||||||
|
FanSpeed,
|
||||||
|
FanState,
|
||||||
|
FocusMode,
|
||||||
|
HeatMode,
|
||||||
|
HeatState,
|
||||||
|
HeatTarget,
|
||||||
|
)
|
||||||
|
from libpurecool.dyson_pure_hotcool import DysonPureHotCool
|
||||||
from libpurecool.dyson_pure_hotcool_link import DysonPureHotCoolLink
|
from libpurecool.dyson_pure_hotcool_link import DysonPureHotCoolLink
|
||||||
from libpurecool.dyson_pure_state import DysonPureHotCoolState
|
from libpurecool.dyson_pure_state import DysonPureHotCoolState
|
||||||
|
from libpurecool.dyson_pure_state_v2 import DysonPureHotCoolV2State
|
||||||
|
|
||||||
from homeassistant.components import dyson as dyson_parent
|
from homeassistant.components import dyson as dyson_parent
|
||||||
|
from homeassistant.components.climate import (
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_FAN_MODE,
|
||||||
|
SERVICE_SET_HVAC_MODE,
|
||||||
|
SERVICE_SET_TEMPERATURE,
|
||||||
|
)
|
||||||
|
from homeassistant.components.climate.const import (
|
||||||
|
ATTR_CURRENT_HUMIDITY,
|
||||||
|
ATTR_FAN_MODE,
|
||||||
|
ATTR_HVAC_ACTION,
|
||||||
|
ATTR_HVAC_MODE,
|
||||||
|
CURRENT_HVAC_COOL,
|
||||||
|
CURRENT_HVAC_HEAT,
|
||||||
|
CURRENT_HVAC_IDLE,
|
||||||
|
FAN_AUTO,
|
||||||
|
FAN_HIGH,
|
||||||
|
FAN_LOW,
|
||||||
|
FAN_MEDIUM,
|
||||||
|
FAN_OFF,
|
||||||
|
HVAC_MODE_COOL,
|
||||||
|
HVAC_MODE_HEAT,
|
||||||
|
HVAC_MODE_OFF,
|
||||||
|
)
|
||||||
from homeassistant.components.dyson import climate as dyson
|
from homeassistant.components.dyson import climate as dyson
|
||||||
from homeassistant.const import ATTR_TEMPERATURE, TEMP_CELSIUS
|
from homeassistant.const import ATTR_ENTITY_ID, ATTR_TEMPERATURE, TEMP_CELSIUS
|
||||||
from homeassistant.setup import async_setup_component
|
from homeassistant.setup import async_setup_component
|
||||||
|
|
||||||
from .common import load_mock_device
|
from .common import load_mock_device
|
||||||
|
|
||||||
from tests.async_mock import patch
|
from tests.async_mock import MagicMock, Mock, patch
|
||||||
from tests.common import get_test_home_assistant
|
from tests.common import get_test_home_assistant
|
||||||
|
|
||||||
|
|
||||||
@ -22,7 +55,6 @@ class MockDysonState(DysonPureHotCoolState):
|
|||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
"""Create new Mock Dyson State."""
|
"""Create new Mock Dyson State."""
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
def _get_config():
|
def _get_config():
|
||||||
@ -40,9 +72,22 @@ def _get_config():
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def _get_dyson_purehotcool_device():
|
||||||
|
"""Return a valid device as provided by the Dyson web services."""
|
||||||
|
device = Mock(spec=DysonPureHotCool)
|
||||||
|
load_mock_device(device)
|
||||||
|
device.name = "Living room"
|
||||||
|
device.state.heat_target = "0000"
|
||||||
|
device.state.heat_mode = HeatMode.HEAT_OFF.value
|
||||||
|
device.state.fan_power = FanPower.POWER_OFF.value
|
||||||
|
device.environmental_state.humidity = 42
|
||||||
|
device.environmental_state.temperature = 298
|
||||||
|
return device
|
||||||
|
|
||||||
|
|
||||||
def _get_device_with_no_state():
|
def _get_device_with_no_state():
|
||||||
"""Return a device with no state."""
|
"""Return a device with no state."""
|
||||||
device = mock.Mock(spec=DysonPureHotCoolLink)
|
device = Mock(spec=DysonPureHotCoolLink)
|
||||||
load_mock_device(device)
|
load_mock_device(device)
|
||||||
device.state = None
|
device.state = None
|
||||||
device.environmental_state = None
|
device.environmental_state = None
|
||||||
@ -51,14 +96,14 @@ def _get_device_with_no_state():
|
|||||||
|
|
||||||
def _get_device_off():
|
def _get_device_off():
|
||||||
"""Return a device with state off."""
|
"""Return a device with state off."""
|
||||||
device = mock.Mock(spec=DysonPureHotCoolLink)
|
device = Mock(spec=DysonPureHotCoolLink)
|
||||||
load_mock_device(device)
|
load_mock_device(device)
|
||||||
return device
|
return device
|
||||||
|
|
||||||
|
|
||||||
def _get_device_focus():
|
def _get_device_focus():
|
||||||
"""Return a device with fan state of focus mode."""
|
"""Return a device with fan state of focus mode."""
|
||||||
device = mock.Mock(spec=DysonPureHotCoolLink)
|
device = Mock(spec=DysonPureHotCoolLink)
|
||||||
load_mock_device(device)
|
load_mock_device(device)
|
||||||
device.state.focus_mode = FocusMode.FOCUS_ON.value
|
device.state.focus_mode = FocusMode.FOCUS_ON.value
|
||||||
return device
|
return device
|
||||||
@ -66,7 +111,7 @@ def _get_device_focus():
|
|||||||
|
|
||||||
def _get_device_diffuse():
|
def _get_device_diffuse():
|
||||||
"""Return a device with fan state of diffuse mode."""
|
"""Return a device with fan state of diffuse mode."""
|
||||||
device = mock.Mock(spec=DysonPureHotCoolLink)
|
device = Mock(spec=DysonPureHotCoolLink)
|
||||||
load_mock_device(device)
|
load_mock_device(device)
|
||||||
device.state.focus_mode = FocusMode.FOCUS_OFF.value
|
device.state.focus_mode = FocusMode.FOCUS_OFF.value
|
||||||
return device
|
return device
|
||||||
@ -74,7 +119,7 @@ def _get_device_diffuse():
|
|||||||
|
|
||||||
def _get_device_cool():
|
def _get_device_cool():
|
||||||
"""Return a device with state of cooling."""
|
"""Return a device with state of cooling."""
|
||||||
device = mock.Mock(spec=DysonPureHotCoolLink)
|
device = Mock(spec=DysonPureHotCoolLink)
|
||||||
load_mock_device(device)
|
load_mock_device(device)
|
||||||
device.state.focus_mode = FocusMode.FOCUS_OFF.value
|
device.state.focus_mode = FocusMode.FOCUS_OFF.value
|
||||||
device.state.heat_target = HeatTarget.celsius(12)
|
device.state.heat_target = HeatTarget.celsius(12)
|
||||||
@ -85,7 +130,7 @@ def _get_device_cool():
|
|||||||
|
|
||||||
def _get_device_heat_off():
|
def _get_device_heat_off():
|
||||||
"""Return a device with state of heat reached target."""
|
"""Return a device with state of heat reached target."""
|
||||||
device = mock.Mock(spec=DysonPureHotCoolLink)
|
device = Mock(spec=DysonPureHotCoolLink)
|
||||||
load_mock_device(device)
|
load_mock_device(device)
|
||||||
device.state.heat_mode = HeatMode.HEAT_ON.value
|
device.state.heat_mode = HeatMode.HEAT_ON.value
|
||||||
device.state.heat_state = HeatState.HEAT_STATE_OFF.value
|
device.state.heat_state = HeatState.HEAT_STATE_OFF.value
|
||||||
@ -94,7 +139,7 @@ def _get_device_heat_off():
|
|||||||
|
|
||||||
def _get_device_heat_on():
|
def _get_device_heat_on():
|
||||||
"""Return a device with state of heating."""
|
"""Return a device with state of heating."""
|
||||||
device = mock.Mock(spec=DysonPureHotCoolLink)
|
device = Mock(spec=DysonPureHotCoolLink)
|
||||||
load_mock_device(device)
|
load_mock_device(device)
|
||||||
device.serial = "YY-YYYYY-YY"
|
device.serial = "YY-YYYYY-YY"
|
||||||
device.state.heat_target = HeatTarget.celsius(23)
|
device.state.heat_target = HeatTarget.celsius(23)
|
||||||
@ -120,7 +165,7 @@ class DysonTest(unittest.TestCase):
|
|||||||
def test_setup_component_without_devices(self):
|
def test_setup_component_without_devices(self):
|
||||||
"""Test setup component with no devices."""
|
"""Test setup component with no devices."""
|
||||||
self.hass.data[dyson.DYSON_DEVICES] = []
|
self.hass.data[dyson.DYSON_DEVICES] = []
|
||||||
add_devices = mock.MagicMock()
|
add_devices = MagicMock()
|
||||||
dyson.setup_platform(self.hass, None, add_devices)
|
dyson.setup_platform(self.hass, None, add_devices)
|
||||||
add_devices.assert_not_called()
|
add_devices.assert_not_called()
|
||||||
|
|
||||||
@ -132,18 +177,10 @@ class DysonTest(unittest.TestCase):
|
|||||||
_get_device_heat_on(),
|
_get_device_heat_on(),
|
||||||
]
|
]
|
||||||
self.hass.data[dyson.DYSON_DEVICES] = devices
|
self.hass.data[dyson.DYSON_DEVICES] = devices
|
||||||
add_devices = mock.MagicMock()
|
add_devices = MagicMock()
|
||||||
dyson.setup_platform(self.hass, None, add_devices, discovery_info={})
|
dyson.setup_platform(self.hass, None, add_devices, discovery_info={})
|
||||||
assert add_devices.called
|
assert add_devices.called
|
||||||
|
|
||||||
def test_setup_component_with_invalid_devices(self):
|
|
||||||
"""Test setup component with invalid devices."""
|
|
||||||
devices = [None, "foo_bar"]
|
|
||||||
self.hass.data[dyson.DYSON_DEVICES] = devices
|
|
||||||
add_devices = mock.MagicMock()
|
|
||||||
dyson.setup_platform(self.hass, None, add_devices, discovery_info={})
|
|
||||||
add_devices.assert_called_with([])
|
|
||||||
|
|
||||||
def test_setup_component(self):
|
def test_setup_component(self):
|
||||||
"""Test setup component with devices."""
|
"""Test setup component with devices."""
|
||||||
device_fan = _get_device_heat_on()
|
device_fan = _get_device_heat_on()
|
||||||
@ -160,7 +197,7 @@ class DysonTest(unittest.TestCase):
|
|||||||
"""Test set climate temperature."""
|
"""Test set climate temperature."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
device.temp_unit = TEMP_CELSIUS
|
device.temp_unit = TEMP_CELSIUS
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert not entity.should_poll
|
assert not entity.should_poll
|
||||||
|
|
||||||
# Without target temp.
|
# Without target temp.
|
||||||
@ -195,8 +232,8 @@ class DysonTest(unittest.TestCase):
|
|||||||
"""Test set climate temperature when heating is off."""
|
"""Test set climate temperature when heating is off."""
|
||||||
device = _get_device_cool()
|
device = _get_device_cool()
|
||||||
device.temp_unit = TEMP_CELSIUS
|
device.temp_unit = TEMP_CELSIUS
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
entity.schedule_update_ha_state = mock.Mock()
|
entity.schedule_update_ha_state = Mock()
|
||||||
|
|
||||||
kwargs = {ATTR_TEMPERATURE: 23}
|
kwargs = {ATTR_TEMPERATURE: 23}
|
||||||
entity.set_temperature(**kwargs)
|
entity.set_temperature(**kwargs)
|
||||||
@ -208,7 +245,7 @@ class DysonTest(unittest.TestCase):
|
|||||||
def test_dyson_set_fan_mode(self):
|
def test_dyson_set_fan_mode(self):
|
||||||
"""Test set fan mode."""
|
"""Test set fan mode."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert not entity.should_poll
|
assert not entity.should_poll
|
||||||
|
|
||||||
entity.set_fan_mode(dyson.FAN_FOCUS)
|
entity.set_fan_mode(dyson.FAN_FOCUS)
|
||||||
@ -222,7 +259,7 @@ class DysonTest(unittest.TestCase):
|
|||||||
def test_dyson_fan_modes(self):
|
def test_dyson_fan_modes(self):
|
||||||
"""Test get fan list."""
|
"""Test get fan list."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert len(entity.fan_modes) == 2
|
assert len(entity.fan_modes) == 2
|
||||||
assert dyson.FAN_FOCUS in entity.fan_modes
|
assert dyson.FAN_FOCUS in entity.fan_modes
|
||||||
assert dyson.FAN_DIFFUSE in entity.fan_modes
|
assert dyson.FAN_DIFFUSE in entity.fan_modes
|
||||||
@ -230,19 +267,19 @@ class DysonTest(unittest.TestCase):
|
|||||||
def test_dyson_fan_mode_focus(self):
|
def test_dyson_fan_mode_focus(self):
|
||||||
"""Test fan focus mode."""
|
"""Test fan focus mode."""
|
||||||
device = _get_device_focus()
|
device = _get_device_focus()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert entity.fan_mode == dyson.FAN_FOCUS
|
assert entity.fan_mode == dyson.FAN_FOCUS
|
||||||
|
|
||||||
def test_dyson_fan_mode_diffuse(self):
|
def test_dyson_fan_mode_diffuse(self):
|
||||||
"""Test fan diffuse mode."""
|
"""Test fan diffuse mode."""
|
||||||
device = _get_device_diffuse()
|
device = _get_device_diffuse()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert entity.fan_mode == dyson.FAN_DIFFUSE
|
assert entity.fan_mode == dyson.FAN_DIFFUSE
|
||||||
|
|
||||||
def test_dyson_set_hvac_mode(self):
|
def test_dyson_set_hvac_mode(self):
|
||||||
"""Test set operation mode."""
|
"""Test set operation mode."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert not entity.should_poll
|
assert not entity.should_poll
|
||||||
|
|
||||||
entity.set_hvac_mode(dyson.HVAC_MODE_HEAT)
|
entity.set_hvac_mode(dyson.HVAC_MODE_HEAT)
|
||||||
@ -256,7 +293,7 @@ class DysonTest(unittest.TestCase):
|
|||||||
def test_dyson_operation_list(self):
|
def test_dyson_operation_list(self):
|
||||||
"""Test get operation list."""
|
"""Test get operation list."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert len(entity.hvac_modes) == 2
|
assert len(entity.hvac_modes) == 2
|
||||||
assert dyson.HVAC_MODE_HEAT in entity.hvac_modes
|
assert dyson.HVAC_MODE_HEAT in entity.hvac_modes
|
||||||
assert dyson.HVAC_MODE_COOL in entity.hvac_modes
|
assert dyson.HVAC_MODE_COOL in entity.hvac_modes
|
||||||
@ -264,7 +301,7 @@ class DysonTest(unittest.TestCase):
|
|||||||
def test_dyson_heat_off(self):
|
def test_dyson_heat_off(self):
|
||||||
"""Test turn off heat."""
|
"""Test turn off heat."""
|
||||||
device = _get_device_heat_off()
|
device = _get_device_heat_off()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
entity.set_hvac_mode(dyson.HVAC_MODE_COOL)
|
entity.set_hvac_mode(dyson.HVAC_MODE_COOL)
|
||||||
set_config = device.set_configuration
|
set_config = device.set_configuration
|
||||||
set_config.assert_called_with(heat_mode=HeatMode.HEAT_OFF)
|
set_config.assert_called_with(heat_mode=HeatMode.HEAT_OFF)
|
||||||
@ -272,7 +309,7 @@ class DysonTest(unittest.TestCase):
|
|||||||
def test_dyson_heat_on(self):
|
def test_dyson_heat_on(self):
|
||||||
"""Test turn on heat."""
|
"""Test turn on heat."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
entity.set_hvac_mode(dyson.HVAC_MODE_HEAT)
|
entity.set_hvac_mode(dyson.HVAC_MODE_HEAT)
|
||||||
set_config = device.set_configuration
|
set_config = device.set_configuration
|
||||||
set_config.assert_called_with(heat_mode=HeatMode.HEAT_ON)
|
set_config.assert_called_with(heat_mode=HeatMode.HEAT_ON)
|
||||||
@ -280,34 +317,34 @@ class DysonTest(unittest.TestCase):
|
|||||||
def test_dyson_heat_value_on(self):
|
def test_dyson_heat_value_on(self):
|
||||||
"""Test get heat value on."""
|
"""Test get heat value on."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert entity.hvac_mode == dyson.HVAC_MODE_HEAT
|
assert entity.hvac_mode == dyson.HVAC_MODE_HEAT
|
||||||
|
|
||||||
def test_dyson_heat_value_off(self):
|
def test_dyson_heat_value_off(self):
|
||||||
"""Test get heat value off."""
|
"""Test get heat value off."""
|
||||||
device = _get_device_cool()
|
device = _get_device_cool()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert entity.hvac_mode == dyson.HVAC_MODE_COOL
|
assert entity.hvac_mode == dyson.HVAC_MODE_COOL
|
||||||
|
|
||||||
def test_dyson_heat_value_idle(self):
|
def test_dyson_heat_value_idle(self):
|
||||||
"""Test get heat value idle."""
|
"""Test get heat value idle."""
|
||||||
device = _get_device_heat_off()
|
device = _get_device_heat_off()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert entity.hvac_mode == dyson.HVAC_MODE_HEAT
|
assert entity.hvac_mode == dyson.HVAC_MODE_HEAT
|
||||||
assert entity.hvac_action == dyson.CURRENT_HVAC_IDLE
|
assert entity.hvac_action == dyson.CURRENT_HVAC_IDLE
|
||||||
|
|
||||||
def test_on_message(self):
|
def test_on_message(self):
|
||||||
"""Test when message is received."""
|
"""Test when message is received."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
entity.schedule_update_ha_state = mock.Mock()
|
entity.schedule_update_ha_state = Mock()
|
||||||
entity.on_message(MockDysonState())
|
entity.on_message(MockDysonState())
|
||||||
entity.schedule_update_ha_state.assert_called_with()
|
entity.schedule_update_ha_state.assert_called_with()
|
||||||
|
|
||||||
def test_general_properties(self):
|
def test_general_properties(self):
|
||||||
"""Test properties of entity."""
|
"""Test properties of entity."""
|
||||||
device = _get_device_with_no_state()
|
device = _get_device_with_no_state()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert entity.should_poll is False
|
assert entity.should_poll is False
|
||||||
assert entity.supported_features == dyson.SUPPORT_FLAGS
|
assert entity.supported_features == dyson.SUPPORT_FLAGS
|
||||||
assert entity.temperature_unit == TEMP_CELSIUS
|
assert entity.temperature_unit == TEMP_CELSIUS
|
||||||
@ -315,41 +352,41 @@ class DysonTest(unittest.TestCase):
|
|||||||
def test_property_current_humidity(self):
|
def test_property_current_humidity(self):
|
||||||
"""Test properties of current humidity."""
|
"""Test properties of current humidity."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert entity.current_humidity == 53
|
assert entity.current_humidity == 53
|
||||||
|
|
||||||
def test_property_current_humidity_with_invalid_env_state(self):
|
def test_property_current_humidity_with_invalid_env_state(self):
|
||||||
"""Test properties of current humidity with invalid env state."""
|
"""Test properties of current humidity with invalid env state."""
|
||||||
device = _get_device_off()
|
device = _get_device_off()
|
||||||
device.environmental_state.humidity = 0
|
device.environmental_state.humidity = 0
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert entity.current_humidity is None
|
assert entity.current_humidity is None
|
||||||
|
|
||||||
def test_property_current_humidity_without_env_state(self):
|
def test_property_current_humidity_without_env_state(self):
|
||||||
"""Test properties of current humidity without env state."""
|
"""Test properties of current humidity without env state."""
|
||||||
device = _get_device_with_no_state()
|
device = _get_device_with_no_state()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert entity.current_humidity is None
|
assert entity.current_humidity is None
|
||||||
|
|
||||||
def test_property_current_temperature(self):
|
def test_property_current_temperature(self):
|
||||||
"""Test properties of current temperature."""
|
"""Test properties of current temperature."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
# Result should be in celsius, hence then subtraction of 273.
|
# Result should be in celsius, hence then subtraction of 273.
|
||||||
assert entity.current_temperature == 289 - 273
|
assert entity.current_temperature == 289 - 273
|
||||||
|
|
||||||
def test_property_target_temperature(self):
|
def test_property_target_temperature(self):
|
||||||
"""Test properties of target temperature."""
|
"""Test properties of target temperature."""
|
||||||
device = _get_device_heat_on()
|
device = _get_device_heat_on()
|
||||||
entity = dyson.DysonPureHotCoolLinkDevice(device)
|
entity = dyson.DysonPureHotCoolLinkEntity(device)
|
||||||
assert entity.target_temperature == 23
|
assert entity.target_temperature == 23
|
||||||
|
|
||||||
|
|
||||||
@patch(
|
@patch(
|
||||||
"libpurecool.dyson.DysonAccount.devices",
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
return_value=[_get_device_heat_on(), _get_device_cool()],
|
return_value=[_get_device_heat_on(), _get_device_cool()],
|
||||||
)
|
)
|
||||||
@patch("libpurecool.dyson.DysonAccount.login", return_value=True)
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
async def test_setup_component_with_parent_discovery(
|
async def test_setup_component_with_parent_discovery(
|
||||||
mocked_login, mocked_devices, hass
|
mocked_login, mocked_devices, hass
|
||||||
):
|
):
|
||||||
@ -357,4 +394,328 @@ async def test_setup_component_with_parent_discovery(
|
|||||||
await async_setup_component(hass, dyson_parent.DOMAIN, _get_config())
|
await async_setup_component(hass, dyson_parent.DOMAIN, _get_config())
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
assert len(hass.data[dyson.DYSON_DEVICES]) == 2
|
entity_ids = hass.states.async_entity_ids("climate")
|
||||||
|
assert len(entity_ids) == 2
|
||||||
|
|
||||||
|
|
||||||
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_dyson_purehotcool_device()],
|
||||||
|
)
|
||||||
|
async def test_purehotcool_component_setup_only_once(devices, login, hass):
|
||||||
|
"""Test if entities are created only once."""
|
||||||
|
config = _get_config()
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, config)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
entity_ids = hass.states.async_entity_ids("climate")
|
||||||
|
assert len(entity_ids) == 1
|
||||||
|
state = hass.states.get(entity_ids[0])
|
||||||
|
assert state.name == "Living room"
|
||||||
|
|
||||||
|
|
||||||
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_device_off()],
|
||||||
|
)
|
||||||
|
async def test_purehotcoollink_component_setup_only_once(devices, login, hass):
|
||||||
|
"""Test if entities are created only once."""
|
||||||
|
config = _get_config()
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, config)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
entity_ids = hass.states.async_entity_ids("climate")
|
||||||
|
assert len(entity_ids) == 1
|
||||||
|
state = hass.states.get(entity_ids[0])
|
||||||
|
assert state.name == "Temp Name"
|
||||||
|
|
||||||
|
|
||||||
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_dyson_purehotcool_device()],
|
||||||
|
)
|
||||||
|
async def test_purehotcool_update_state(devices, login, hass):
|
||||||
|
"""Test state update."""
|
||||||
|
device = devices.return_value[0]
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, _get_config())
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
event = {
|
||||||
|
"msg": "CURRENT-STATE",
|
||||||
|
"product-state": {
|
||||||
|
"fpwr": "ON",
|
||||||
|
"fdir": "OFF",
|
||||||
|
"auto": "OFF",
|
||||||
|
"oscs": "ON",
|
||||||
|
"oson": "ON",
|
||||||
|
"nmod": "OFF",
|
||||||
|
"rhtm": "ON",
|
||||||
|
"fnst": "FAN",
|
||||||
|
"ercd": "11E1",
|
||||||
|
"wacd": "NONE",
|
||||||
|
"nmdv": "0004",
|
||||||
|
"fnsp": "0002",
|
||||||
|
"bril": "0002",
|
||||||
|
"corf": "ON",
|
||||||
|
"cflr": "0085",
|
||||||
|
"hflr": "0095",
|
||||||
|
"sltm": "OFF",
|
||||||
|
"osal": "0045",
|
||||||
|
"osau": "0095",
|
||||||
|
"ancp": "CUST",
|
||||||
|
"tilt": "OK",
|
||||||
|
"hsta": "HEAT",
|
||||||
|
"hmax": "2986",
|
||||||
|
"hmod": "HEAT",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
device.state = DysonPureHotCoolV2State(json.dumps(event))
|
||||||
|
|
||||||
|
for call in device.add_message_listener.call_args_list:
|
||||||
|
callback = call[0][0]
|
||||||
|
if type(callback.__self__) == dyson.DysonPureHotCoolEntity:
|
||||||
|
callback(device.state)
|
||||||
|
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
state = hass.states.get("climate.living_room")
|
||||||
|
attributes = state.attributes
|
||||||
|
|
||||||
|
assert attributes[ATTR_TEMPERATURE] == 25
|
||||||
|
assert attributes[ATTR_HVAC_ACTION] == CURRENT_HVAC_HEAT
|
||||||
|
|
||||||
|
|
||||||
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_dyson_purehotcool_device()],
|
||||||
|
)
|
||||||
|
async def test_purehotcool_empty_env_attributes(devices, login, hass):
|
||||||
|
"""Test empty environmental state update."""
|
||||||
|
device = devices.return_value[0]
|
||||||
|
device.environmental_state.temperature = None
|
||||||
|
device.environmental_state.humidity = None
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, _get_config())
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
state = hass.states.get("climate.living_room")
|
||||||
|
attributes = state.attributes
|
||||||
|
|
||||||
|
assert ATTR_CURRENT_HUMIDITY not in attributes
|
||||||
|
|
||||||
|
|
||||||
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_dyson_purehotcool_device()],
|
||||||
|
)
|
||||||
|
async def test_purehotcool_fan_state_off(devices, login, hass):
|
||||||
|
"""Test device fan state off."""
|
||||||
|
device = devices.return_value[0]
|
||||||
|
device.state.fan_state = FanState.FAN_OFF.value
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, _get_config())
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
state = hass.states.get("climate.living_room")
|
||||||
|
attributes = state.attributes
|
||||||
|
|
||||||
|
assert attributes[ATTR_FAN_MODE] == FAN_OFF
|
||||||
|
|
||||||
|
|
||||||
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_dyson_purehotcool_device()],
|
||||||
|
)
|
||||||
|
async def test_purehotcool_hvac_action_cool(devices, login, hass):
|
||||||
|
"""Test device HVAC action cool."""
|
||||||
|
device = devices.return_value[0]
|
||||||
|
device.state.fan_power = FanPower.POWER_ON.value
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, _get_config())
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
state = hass.states.get("climate.living_room")
|
||||||
|
attributes = state.attributes
|
||||||
|
|
||||||
|
assert attributes[ATTR_HVAC_ACTION] == CURRENT_HVAC_COOL
|
||||||
|
|
||||||
|
|
||||||
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_dyson_purehotcool_device()],
|
||||||
|
)
|
||||||
|
async def test_purehotcool_hvac_action_idle(devices, login, hass):
|
||||||
|
"""Test device HVAC action idle."""
|
||||||
|
device = devices.return_value[0]
|
||||||
|
device.state.fan_power = FanPower.POWER_ON.value
|
||||||
|
device.state.heat_mode = HeatMode.HEAT_ON.value
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, _get_config())
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
state = hass.states.get("climate.living_room")
|
||||||
|
attributes = state.attributes
|
||||||
|
|
||||||
|
assert attributes[ATTR_HVAC_ACTION] == CURRENT_HVAC_IDLE
|
||||||
|
|
||||||
|
|
||||||
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_dyson_purehotcool_device()],
|
||||||
|
)
|
||||||
|
async def test_purehotcool_set_temperature(devices, login, hass):
|
||||||
|
"""Test set temperature."""
|
||||||
|
device = devices.return_value[0]
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, _get_config())
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
state = hass.states.get("climate.living_room")
|
||||||
|
attributes = state.attributes
|
||||||
|
min_temp = attributes["min_temp"]
|
||||||
|
max_temp = attributes["max_temp"]
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_TEMPERATURE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.bed_room", ATTR_TEMPERATURE: 23},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
device.set_heat_target.assert_not_called()
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_TEMPERATURE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_TEMPERATURE: 23},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.set_heat_target.call_count == 1
|
||||||
|
device.set_heat_target.assert_called_with("2960")
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_TEMPERATURE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_TEMPERATURE: min_temp - 1},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.set_heat_target.call_count == 2
|
||||||
|
device.set_heat_target.assert_called_with(HeatTarget.celsius(min_temp))
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_TEMPERATURE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_TEMPERATURE: max_temp + 1},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.set_heat_target.call_count == 3
|
||||||
|
device.set_heat_target.assert_called_with(HeatTarget.celsius(max_temp))
|
||||||
|
|
||||||
|
|
||||||
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_dyson_purehotcool_device()],
|
||||||
|
)
|
||||||
|
async def test_purehotcool_set_fan_mode(devices, login, hass):
|
||||||
|
"""Test set fan mode."""
|
||||||
|
device = devices.return_value[0]
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, _get_config())
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_FAN_MODE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.bed_room", ATTR_FAN_MODE: FAN_OFF},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
device.turn_off.assert_not_called()
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_FAN_MODE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_FAN_MODE: FAN_OFF},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.turn_off.call_count == 1
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_FAN_MODE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_FAN_MODE: FAN_LOW},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.set_fan_speed.call_count == 1
|
||||||
|
device.set_fan_speed.assert_called_with(FanSpeed.FAN_SPEED_4)
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_FAN_MODE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_FAN_MODE: FAN_MEDIUM},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.set_fan_speed.call_count == 2
|
||||||
|
device.set_fan_speed.assert_called_with(FanSpeed.FAN_SPEED_7)
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_FAN_MODE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_FAN_MODE: FAN_HIGH},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.set_fan_speed.call_count == 3
|
||||||
|
device.set_fan_speed.assert_called_with(FanSpeed.FAN_SPEED_10)
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_FAN_MODE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_FAN_MODE: FAN_AUTO},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.set_fan_speed.call_count == 4
|
||||||
|
device.set_fan_speed.assert_called_with(FanSpeed.FAN_SPEED_AUTO)
|
||||||
|
|
||||||
|
|
||||||
|
@patch("homeassistant.components.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"homeassistant.components.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_dyson_purehotcool_device()],
|
||||||
|
)
|
||||||
|
async def test_purehotcool_set_hvac_mode(devices, login, hass):
|
||||||
|
"""Test set HVAC mode."""
|
||||||
|
device = devices.return_value[0]
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, _get_config())
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_HVAC_MODE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.bed_room", ATTR_HVAC_MODE: HVAC_MODE_OFF},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
device.turn_off.assert_not_called()
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_HVAC_MODE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_HVAC_MODE: HVAC_MODE_OFF},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.turn_off.call_count == 1
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_HVAC_MODE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_HVAC_MODE: HVAC_MODE_HEAT},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.turn_on.call_count == 1
|
||||||
|
assert device.enable_heat_mode.call_count == 1
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
DOMAIN,
|
||||||
|
SERVICE_SET_HVAC_MODE,
|
||||||
|
{ATTR_ENTITY_ID: "climate.living_room", ATTR_HVAC_MODE: HVAC_MODE_COOL},
|
||||||
|
True,
|
||||||
|
)
|
||||||
|
assert device.turn_on.call_count == 2
|
||||||
|
assert device.disable_heat_mode.call_count == 1
|
||||||
|
Loading…
x
Reference in New Issue
Block a user