Add Somfy climate platform (#43895)
* Add climate platform * Upgrade pymfy to 0.9.3
This commit is contained in:
parent
266f82ac76
commit
03bfc3bcf0
6 changed files with 232 additions and 18 deletions
|
@ -47,7 +47,7 @@ CONFIG_SCHEMA = vol.Schema(
|
|||
extra=vol.ALLOW_EXTRA,
|
||||
)
|
||||
|
||||
SOMFY_COMPONENTS = ["cover", "switch"]
|
||||
SOMFY_COMPONENTS = ["cover", "switch", "climate"]
|
||||
|
||||
|
||||
async def async_setup(hass, config):
|
||||
|
|
214
homeassistant/components/somfy/climate.py
Normal file
214
homeassistant/components/somfy/climate.py
Normal file
|
@ -0,0 +1,214 @@
|
|||
"""Support for Somfy Thermostat."""
|
||||
|
||||
import logging
|
||||
from typing import Any, Dict, List, Optional
|
||||
|
||||
from pymfy.api.devices.category import Category
|
||||
from pymfy.api.devices.thermostat import (
|
||||
DurationType,
|
||||
HvacState,
|
||||
RegulationState,
|
||||
TargetMode,
|
||||
Thermostat,
|
||||
)
|
||||
|
||||
from homeassistant.components.climate import ClimateEntity
|
||||
from homeassistant.components.climate.const import (
|
||||
CURRENT_HVAC_COOL,
|
||||
CURRENT_HVAC_HEAT,
|
||||
CURRENT_HVAC_IDLE,
|
||||
HVAC_MODE_AUTO,
|
||||
HVAC_MODE_COOL,
|
||||
HVAC_MODE_HEAT,
|
||||
PRESET_AWAY,
|
||||
PRESET_HOME,
|
||||
PRESET_SLEEP,
|
||||
SUPPORT_PRESET_MODE,
|
||||
SUPPORT_TARGET_TEMPERATURE,
|
||||
)
|
||||
from homeassistant.const import ATTR_BATTERY_LEVEL, ATTR_TEMPERATURE, TEMP_CELSIUS
|
||||
|
||||
from . import SomfyEntity
|
||||
from .const import API, COORDINATOR, DOMAIN
|
||||
|
||||
_LOGGER = logging.getLogger(__name__)
|
||||
|
||||
SUPPORTED_CATEGORIES = {Category.HVAC.value}
|
||||
|
||||
PRESET_FROST_GUARD = "Frost Guard"
|
||||
PRESET_GEOFENCING = "Geofencing"
|
||||
PRESET_MANUAL = "Manual"
|
||||
|
||||
PRESETS_MAPPING = {
|
||||
TargetMode.AT_HOME: PRESET_HOME,
|
||||
TargetMode.AWAY: PRESET_AWAY,
|
||||
TargetMode.SLEEP: PRESET_SLEEP,
|
||||
TargetMode.MANUAL: PRESET_MANUAL,
|
||||
TargetMode.GEOFENCING: PRESET_GEOFENCING,
|
||||
TargetMode.FROST_PROTECTION: PRESET_FROST_GUARD,
|
||||
}
|
||||
REVERSE_PRESET_MAPPING = {v: k for k, v in PRESETS_MAPPING.items()}
|
||||
|
||||
HVAC_MODES_MAPPING = {HvacState.COOL: HVAC_MODE_COOL, HvacState.HEAT: HVAC_MODE_HEAT}
|
||||
|
||||
|
||||
async def async_setup_entry(hass, config_entry, async_add_entities):
|
||||
"""Set up the Somfy climate platform."""
|
||||
|
||||
def get_thermostats():
|
||||
"""Retrieve thermostats."""
|
||||
domain_data = hass.data[DOMAIN]
|
||||
coordinator = domain_data[COORDINATOR]
|
||||
api = domain_data[API]
|
||||
|
||||
return [
|
||||
SomfyClimate(coordinator, device_id, api)
|
||||
for device_id, device in coordinator.data.items()
|
||||
if SUPPORTED_CATEGORIES & set(device.categories)
|
||||
]
|
||||
|
||||
async_add_entities(await hass.async_add_executor_job(get_thermostats))
|
||||
|
||||
|
||||
class SomfyClimate(SomfyEntity, ClimateEntity):
|
||||
"""Representation of a Somfy thermostat device."""
|
||||
|
||||
def __init__(self, coordinator, device_id, api):
|
||||
"""Initialize the Somfy device."""
|
||||
super().__init__(coordinator, device_id, api)
|
||||
self._climate = None
|
||||
self._create_device()
|
||||
|
||||
def _create_device(self):
|
||||
"""Update the device with the latest data."""
|
||||
self._climate = Thermostat(self.device, self.api)
|
||||
|
||||
@property
|
||||
def supported_features(self) -> int:
|
||||
"""Return the list of supported features."""
|
||||
return SUPPORT_TARGET_TEMPERATURE | SUPPORT_PRESET_MODE
|
||||
|
||||
@property
|
||||
def device_state_attributes(self) -> Dict[str, Any]:
|
||||
"""Return the state attributes of the device."""
|
||||
return {ATTR_BATTERY_LEVEL: self._climate.get_battery()}
|
||||
|
||||
@property
|
||||
def temperature_unit(self):
|
||||
"""Return the unit of measurement used by the platform."""
|
||||
return TEMP_CELSIUS
|
||||
|
||||
@property
|
||||
def current_temperature(self):
|
||||
"""Return the current temperature."""
|
||||
return self._climate.get_ambient_temperature()
|
||||
|
||||
@property
|
||||
def target_temperature(self):
|
||||
"""Return the temperature we try to reach."""
|
||||
return self._climate.get_target_temperature()
|
||||
|
||||
def set_temperature(self, **kwargs) -> None:
|
||||
"""Set new target temperature."""
|
||||
temperature = kwargs.get(ATTR_TEMPERATURE)
|
||||
if temperature is None:
|
||||
return
|
||||
|
||||
self._climate.set_target(TargetMode.MANUAL, temperature, DurationType.NEXT_MODE)
|
||||
|
||||
@property
|
||||
def max_temp(self) -> float:
|
||||
"""Return the maximum temperature."""
|
||||
return 26.0
|
||||
|
||||
@property
|
||||
def min_temp(self) -> float:
|
||||
"""Return the minimum temperature."""
|
||||
return 15.0
|
||||
|
||||
@property
|
||||
def current_humidity(self):
|
||||
"""Return the current humidity."""
|
||||
return self._climate.get_humidity()
|
||||
|
||||
@property
|
||||
def hvac_mode(self) -> str:
|
||||
"""Return hvac operation ie. heat, cool mode."""
|
||||
if self._climate.get_regulation_state() == RegulationState.TIMETABLE:
|
||||
return HVAC_MODE_AUTO
|
||||
return HVAC_MODES_MAPPING.get(self._climate.get_hvac_state())
|
||||
|
||||
@property
|
||||
def hvac_modes(self) -> List[str]:
|
||||
"""Return the list of available hvac operation modes.
|
||||
|
||||
HEAT and COOL mode are exclusive. End user has to enable a mode manually within the Somfy application.
|
||||
So only one mode can be displayed. Auto mode is a scheduler.
|
||||
"""
|
||||
hvac_state = HVAC_MODES_MAPPING.get(self._climate.get_hvac_state())
|
||||
return [HVAC_MODE_AUTO, hvac_state]
|
||||
|
||||
def set_hvac_mode(self, hvac_mode: str) -> None:
|
||||
"""Set new target hvac mode."""
|
||||
if hvac_mode == self.hvac_mode:
|
||||
return
|
||||
if hvac_mode == HVAC_MODE_AUTO:
|
||||
self._climate.cancel_target()
|
||||
else:
|
||||
self._climate.set_target(
|
||||
TargetMode.MANUAL, self.target_temperature, DurationType.FURTHER_NOTICE
|
||||
)
|
||||
|
||||
@property
|
||||
def hvac_action(self) -> str:
|
||||
"""Return the current running hvac operation if supported."""
|
||||
if not self.current_temperature or not self.target_temperature:
|
||||
return CURRENT_HVAC_IDLE
|
||||
|
||||
if (
|
||||
self.hvac_mode == HVAC_MODE_HEAT
|
||||
and self.current_temperature < self.target_temperature
|
||||
):
|
||||
return CURRENT_HVAC_HEAT
|
||||
|
||||
if (
|
||||
self.hvac_mode == HVAC_MODE_COOL
|
||||
and self.current_temperature > self.target_temperature
|
||||
):
|
||||
return CURRENT_HVAC_COOL
|
||||
|
||||
return CURRENT_HVAC_IDLE
|
||||
|
||||
@property
|
||||
def preset_mode(self) -> Optional[str]:
|
||||
"""Return the current preset mode."""
|
||||
mode = self._climate.get_target_mode()
|
||||
return PRESETS_MAPPING.get(mode)
|
||||
|
||||
@property
|
||||
def preset_modes(self) -> Optional[List[str]]:
|
||||
"""Return a list of available preset modes."""
|
||||
return list(PRESETS_MAPPING.values())
|
||||
|
||||
def set_preset_mode(self, preset_mode: str) -> None:
|
||||
"""Set new preset mode."""
|
||||
if self.preset_mode == preset_mode:
|
||||
return
|
||||
|
||||
if preset_mode == PRESET_HOME:
|
||||
temperature = self._climate.get_at_home_temperature()
|
||||
elif preset_mode == PRESET_AWAY:
|
||||
temperature = self._climate.get_away_temperature()
|
||||
elif preset_mode == PRESET_SLEEP:
|
||||
temperature = self._climate.get_night_temperature()
|
||||
elif preset_mode == PRESET_FROST_GUARD:
|
||||
temperature = self._climate.get_frost_protection_temperature()
|
||||
elif preset_mode in [PRESET_MANUAL, PRESET_GEOFENCING]:
|
||||
temperature = self.target_temperature
|
||||
else:
|
||||
_LOGGER.error("Preset mode not supported: %s", preset_mode)
|
||||
return
|
||||
|
||||
self._climate.set_target(
|
||||
REVERSE_PRESET_MAPPING[preset_mode], temperature, DurationType.NEXT_MODE
|
||||
)
|
|
@ -62,12 +62,12 @@ class SomfyCover(SomfyEntity, RestoreEntity, CoverEntity):
|
|||
self._closed = None
|
||||
self._is_opening = None
|
||||
self._is_closing = None
|
||||
self.cover = None
|
||||
self._cover = None
|
||||
self._create_device()
|
||||
|
||||
def _create_device(self) -> Blind:
|
||||
"""Update the device with the latest data."""
|
||||
self.cover = Blind(self.device, self.api)
|
||||
self._cover = Blind(self.device, self.api)
|
||||
|
||||
@property
|
||||
def supported_features(self) -> int:
|
||||
|
@ -97,7 +97,7 @@ class SomfyCover(SomfyEntity, RestoreEntity, CoverEntity):
|
|||
self.async_write_ha_state()
|
||||
try:
|
||||
# Blocks until the close command is sent
|
||||
await self.hass.async_add_executor_job(self.cover.close)
|
||||
await self.hass.async_add_executor_job(self._cover.close)
|
||||
self._closed = True
|
||||
finally:
|
||||
self._is_closing = None
|
||||
|
@ -109,7 +109,7 @@ class SomfyCover(SomfyEntity, RestoreEntity, CoverEntity):
|
|||
self.async_write_ha_state()
|
||||
try:
|
||||
# Blocks until the open command is sent
|
||||
await self.hass.async_add_executor_job(self.cover.open)
|
||||
await self.hass.async_add_executor_job(self._cover.open)
|
||||
self._closed = False
|
||||
finally:
|
||||
self._is_opening = None
|
||||
|
@ -117,11 +117,11 @@ class SomfyCover(SomfyEntity, RestoreEntity, CoverEntity):
|
|||
|
||||
def stop_cover(self, **kwargs):
|
||||
"""Stop the cover."""
|
||||
self.cover.stop()
|
||||
self._cover.stop()
|
||||
|
||||
def set_cover_position(self, **kwargs):
|
||||
"""Move the cover shutter to a specific position."""
|
||||
self.cover.set_position(100 - kwargs[ATTR_POSITION])
|
||||
self._cover.set_position(100 - kwargs[ATTR_POSITION])
|
||||
|
||||
@property
|
||||
def device_class(self):
|
||||
|
@ -137,7 +137,7 @@ class SomfyCover(SomfyEntity, RestoreEntity, CoverEntity):
|
|||
"""Return the current position of cover shutter."""
|
||||
if not self.has_state("position"):
|
||||
return None
|
||||
return 100 - self.cover.get_position()
|
||||
return 100 - self._cover.get_position()
|
||||
|
||||
@property
|
||||
def is_opening(self):
|
||||
|
@ -158,7 +158,7 @@ class SomfyCover(SomfyEntity, RestoreEntity, CoverEntity):
|
|||
"""Return if the cover is closed."""
|
||||
is_closed = None
|
||||
if self.has_state("position"):
|
||||
is_closed = self.cover.is_closed()
|
||||
is_closed = self._cover.is_closed()
|
||||
elif self.optimistic:
|
||||
is_closed = self._closed
|
||||
return is_closed
|
||||
|
@ -171,23 +171,23 @@ class SomfyCover(SomfyEntity, RestoreEntity, CoverEntity):
|
|||
"""
|
||||
if not self.has_state("orientation"):
|
||||
return None
|
||||
return 100 - self.cover.orientation
|
||||
return 100 - self._cover.orientation
|
||||
|
||||
def set_cover_tilt_position(self, **kwargs):
|
||||
"""Move the cover tilt to a specific position."""
|
||||
self.cover.orientation = 100 - kwargs[ATTR_TILT_POSITION]
|
||||
self._cover.orientation = 100 - kwargs[ATTR_TILT_POSITION]
|
||||
|
||||
def open_cover_tilt(self, **kwargs):
|
||||
"""Open the cover tilt."""
|
||||
self.cover.orientation = 0
|
||||
self._cover.orientation = 0
|
||||
|
||||
def close_cover_tilt(self, **kwargs):
|
||||
"""Close the cover tilt."""
|
||||
self.cover.orientation = 100
|
||||
self._cover.orientation = 100
|
||||
|
||||
def stop_cover_tilt(self, **kwargs):
|
||||
"""Stop the cover."""
|
||||
self.cover.stop()
|
||||
self._cover.stop()
|
||||
|
||||
async def async_added_to_hass(self):
|
||||
"""Complete the initialization."""
|
||||
|
|
|
@ -5,5 +5,5 @@
|
|||
"documentation": "https://www.home-assistant.io/integrations/somfy",
|
||||
"dependencies": ["http"],
|
||||
"codeowners": ["@tetienne"],
|
||||
"requirements": ["pymfy==0.9.1"]
|
||||
"requirements": ["pymfy==0.9.3"]
|
||||
}
|
|
@ -1521,7 +1521,7 @@ pymediaroom==0.6.4.1
|
|||
pymelcloud==2.5.2
|
||||
|
||||
# homeassistant.components.somfy
|
||||
pymfy==0.9.1
|
||||
pymfy==0.9.3
|
||||
|
||||
# homeassistant.components.xiaomi_tv
|
||||
pymitv==1.4.3
|
||||
|
|
|
@ -770,7 +770,7 @@ pymata-express==1.19
|
|||
pymelcloud==2.5.2
|
||||
|
||||
# homeassistant.components.somfy
|
||||
pymfy==0.9.1
|
||||
pymfy==0.9.3
|
||||
|
||||
# homeassistant.components.mochad
|
||||
pymochad==0.2.0
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue