197 lines
5.7 KiB
Python
Raw Normal View History

2022-04-01 15:02:13 +02:00
"""Support for VeSync humidifiers."""
from __future__ import annotations
import logging
from typing import Any, Mapping
2022-04-01 15:02:13 +02:00
from homeassistant.components.humidifier import HumidifierEntity
from homeassistant.components.humidifier.const import (
MODE_AUTO,
2022-04-03 23:22:01 +02:00
MODE_NORMAL,
MODE_SLEEP,
SUPPORT_MODES,
)
2022-04-01 15:02:13 +02:00
from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant, callback
from homeassistant.helpers.dispatcher import async_dispatcher_connect
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from pyvesync.vesyncfan import VeSyncHumid200300S
2022-04-01 15:02:13 +02:00
from .common import VeSyncDevice
from .const import (
DOMAIN,
VS_DISCOVERY,
VS_HUMIDIFIERS,
VS_MODE_AUTO,
VS_MODE_HUMIDITY,
VS_MODE_MANUAL,
VS_MODE_SLEEP,
VS_TO_HA_ATTRIBUTES,
)
2022-04-01 15:02:13 +02:00
_LOGGER = logging.getLogger(__name__)
2022-04-01 15:02:13 +02:00
MAX_HUMIDITY = 80
MIN_HUMIDITY = 30
VS_TO_HA_MODE_MAP = {
VS_MODE_AUTO: MODE_AUTO,
VS_MODE_HUMIDITY: MODE_AUTO,
VS_MODE_MANUAL: MODE_NORMAL,
VS_MODE_SLEEP: MODE_SLEEP,
}
HA_TO_VS_MODE_MAP = {v: k for k, v in VS_TO_HA_MODE_MAP.items()}
2022-04-01 15:02:13 +02:00
async def async_setup_entry(
hass: HomeAssistant,
config_entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up the VeSync humidifier platform."""
coordinator = hass.data[DOMAIN][config_entry.entry_id]["coordinator"]
2022-04-01 15:02:13 +02:00
@callback
def discover(devices):
"""Add new devices to platform."""
_setup_entities(devices, async_add_entities, coordinator)
2022-04-01 15:02:13 +02:00
config_entry.async_on_unload(
async_dispatcher_connect(hass, VS_DISCOVERY.format(VS_HUMIDIFIERS), discover)
)
_setup_entities(
hass.data[DOMAIN][config_entry.entry_id][VS_HUMIDIFIERS],
async_add_entities,
coordinator,
2022-04-01 15:02:13 +02:00
)
@callback
def _setup_entities(devices, async_add_entities, coordinator):
2022-04-01 15:02:13 +02:00
"""Check if device is online and add entity."""
async_add_entities(
[VeSyncHumidifierHA(dev, coordinator) for dev in devices],
update_before_add=True,
)
2022-04-01 15:02:13 +02:00
def _get_ha_mode(vs_mode: str) -> str | None:
ha_mode = VS_TO_HA_MODE_MAP.get(vs_mode)
if ha_mode is None:
_LOGGER.warning("Unknown mode '%s'", vs_mode)
return ha_mode
def _get_vs_mode(ha_mode: str) -> str | None:
vs_mode = HA_TO_VS_MODE_MAP.get(ha_mode)
if vs_mode is None:
_LOGGER.warning("Unknown mode '%s'", ha_mode)
return vs_mode
2022-04-01 15:02:13 +02:00
class VeSyncHumidifierHA(VeSyncDevice, HumidifierEntity):
"""Representation of a VeSync humidifier."""
_attr_max_humidity = MAX_HUMIDITY
_attr_min_humidity = MIN_HUMIDITY
def __init__(self, humidifier: VeSyncHumid200300S, coordinator):
2022-04-01 15:02:13 +02:00
"""Initialize the VeSync humidifier device."""
super().__init__(humidifier, coordinator)
2022-04-01 15:02:13 +02:00
self.smarthumidifier = humidifier
@property
def available_modes(self) -> list[str]:
2022-04-01 15:02:13 +02:00
"""Return the available mist modes."""
modes = []
for vs_mode in self.smarthumidifier.mist_modes:
ha_mode = _get_ha_mode(vs_mode)
if ha_mode is None:
continue
modes.append(ha_mode)
return modes
2022-04-01 15:02:13 +02:00
@property
def supported_features(self):
"""Flag supported features."""
return SUPPORT_MODES
@property
def target_humidity(self) -> int:
2022-04-01 15:02:13 +02:00
"""Return the humidity we try to reach."""
return self.smarthumidifier.config["auto_target_humidity"]
@property
def mode(self) -> str | None:
2022-04-01 15:02:13 +02:00
"""Get the current preset mode."""
return _get_ha_mode(self.smarthumidifier.details["mode"])
2022-04-01 15:02:13 +02:00
@property
def is_on(self) -> bool:
2022-04-01 15:02:13 +02:00
"""Return True if humidifier is on."""
return self.smarthumidifier.enabled # device_status is always on
@property
def unique_info(self) -> str:
2022-04-01 15:02:13 +02:00
"""Return the ID of this humidifier."""
return self.smarthumidifier.uuid
@property
def extra_state_attributes(self) -> Mapping[str, Any]:
2022-04-01 15:02:13 +02:00
"""Return the state attributes of the humidifier."""
attr = {}
for k, v in self.smarthumidifier.details.items():
if k in VS_TO_HA_ATTRIBUTES:
attr[VS_TO_HA_ATTRIBUTES[k]] = v
elif k in self.state_attributes:
attr[f"vs_{k}"] = v
else:
attr[k] = v
2022-04-01 15:02:13 +02:00
return attr
def set_humidity(self, humidity: int) -> None:
2022-04-01 15:02:13 +02:00
"""Set the target humidity of the device."""
if humidity not in range(self.min_humidity, self.max_humidity + 1):
raise ValueError(
"{humidity} is not between {self.min_humidity} and {self.max_humidity} (inclusive)"
)
if self.smarthumidifier.set_humidity(humidity):
self.schedule_update_ha_state()
else:
raise ValueError("An error occurred while setting humidity.")
2022-04-01 15:02:13 +02:00
def set_mode(self, mode: str) -> None:
2022-04-01 15:02:13 +02:00
"""Set the mode of the device."""
if mode not in self.available_modes:
raise ValueError(
"{mode} is not one of the valid available modes: {self.available_modes}"
)
if self.smarthumidifier.set_humidity_mode(_get_vs_mode(mode)):
self.schedule_update_ha_state()
else:
raise ValueError("An error occurred while setting mode.")
2022-04-01 15:02:13 +02:00
def turn_on(
self,
**kwargs,
) -> None:
"""Turn the device on."""
success = self.smarthumidifier.turn_on()
if not success:
raise ValueError("An error occurred while turning on.")
def turn_off(self, **kwargs) -> None:
"""Turn the device off."""
success = self.smarthumidifier.turn_off()
if not success:
raise ValueError("An error occurred while turning off.")