mirror of
https://github.com/Andre0512/hon.git
synced 2024-01-19 02:50:19 +00:00
Add mypy check, add missing types and fix type issues
This commit is contained in:
parent
f0fb5742a4
commit
9d6b8297b2
7
.github/workflows/python_check.yml
vendored
7
.github/workflows/python_check.yml
vendored
|
@ -24,12 +24,17 @@ jobs:
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: |
|
run: |
|
||||||
python -m pip install --upgrade pip
|
python -m pip install --upgrade pip
|
||||||
python -m pip install flake8 pylint black
|
python -m pip install -r requirements.txt
|
||||||
|
python -m pip install -r requirements_dev.txt
|
||||||
- name: Lint with flake8
|
- name: Lint with flake8
|
||||||
run: |
|
run: |
|
||||||
# stop the build if there are Python syntax errors or undefined names
|
# stop the build if there are Python syntax errors or undefined names
|
||||||
flake8 . --count --select=E9,F63,F7,F82 --show-source --statistics
|
flake8 . --count --select=E9,F63,F7,F82 --show-source --statistics
|
||||||
flake8 . --count --exit-zero --max-complexity=10 --max-line-length=88 --statistics
|
flake8 . --count --exit-zero --max-complexity=10 --max-line-length=88 --statistics
|
||||||
|
- name: Type check with mypy
|
||||||
|
run: |
|
||||||
|
touch "$(python -c 'import inspect, homeassistant, os; print(os.path.dirname(inspect.getfile(homeassistant)))')"/py.typed
|
||||||
|
mypy -p custom_components.hon
|
||||||
# - name: Analysing the code with pylint
|
# - name: Analysing the code with pylint
|
||||||
# run: |
|
# run: |
|
||||||
# pylint --max-line-length 88 $(git ls-files '*.py')
|
# pylint --max-line-length 88 $(git ls-files '*.py')
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import logging
|
import logging
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
import voluptuous as vol
|
import voluptuous as vol # type: ignore[import]
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import CONF_EMAIL, CONF_PASSWORD
|
from homeassistant.const import CONF_EMAIL, CONF_PASSWORD
|
||||||
from homeassistant.helpers import config_validation as cv, aiohttp_client
|
from homeassistant.helpers import config_validation as cv, aiohttp_client
|
||||||
|
@ -25,13 +25,15 @@ CONFIG_SCHEMA = vol.Schema(
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass: HomeAssistantType, entry: ConfigEntry):
|
async def async_setup_entry(hass: HomeAssistantType, entry: ConfigEntry) -> None:
|
||||||
session = aiohttp_client.async_get_clientsession(hass)
|
session = aiohttp_client.async_get_clientsession(hass)
|
||||||
|
if (config_dir := hass.config.config_dir) is None:
|
||||||
|
raise ValueError("Missing Config Dir")
|
||||||
hon = await Hon(
|
hon = await Hon(
|
||||||
entry.data["email"],
|
entry.data["email"],
|
||||||
entry.data["password"],
|
entry.data["password"],
|
||||||
session=session,
|
session=session,
|
||||||
test_data_path=Path(hass.config.config_dir),
|
test_data_path=Path(config_dir),
|
||||||
).create()
|
).create()
|
||||||
hass.data.setdefault(DOMAIN, {})
|
hass.data.setdefault(DOMAIN, {})
|
||||||
hass.data[DOMAIN][entry.unique_id] = hon
|
hass.data[DOMAIN][entry.unique_id] = hon
|
||||||
|
@ -41,10 +43,10 @@ async def async_setup_entry(hass: HomeAssistantType, entry: ConfigEntry):
|
||||||
hass.async_create_task(
|
hass.async_create_task(
|
||||||
hass.config_entries.async_forward_entry_setup(entry, platform)
|
hass.config_entries.async_forward_entry_setup(entry, platform)
|
||||||
)
|
)
|
||||||
return True
|
return
|
||||||
|
|
||||||
|
|
||||||
async def async_unload_entry(hass, entry: ConfigEntry) -> bool:
|
async def async_unload_entry(hass: HomeAssistantType, entry: ConfigEntry) -> bool:
|
||||||
unload = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
unload = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
||||||
if unload:
|
if unload:
|
||||||
if not hass.data[DOMAIN]:
|
if not hass.data[DOMAIN]:
|
||||||
|
|
|
@ -8,6 +8,8 @@ from homeassistant.components.binary_sensor import (
|
||||||
)
|
)
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
|
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
from .hon import HonEntity, unique_entities
|
from .hon import HonEntity, unique_entities
|
||||||
|
@ -287,7 +289,9 @@ BINARY_SENSORS: dict[str, tuple[HonBinarySensorEntityDescription, ...]] = {
|
||||||
BINARY_SENSORS["WD"] = unique_entities(BINARY_SENSORS["WM"], BINARY_SENSORS["TD"])
|
BINARY_SENSORS["WD"] = unique_entities(BINARY_SENSORS["WM"], BINARY_SENSORS["TD"])
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
entities = []
|
entities = []
|
||||||
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
||||||
for description in BINARY_SENSORS.get(device.appliance_type, []):
|
for description in BINARY_SENSORS.get(device.appliance_type, []):
|
||||||
|
@ -304,13 +308,13 @@ class HonBinarySensorEntity(HonEntity, BinarySensorEntity):
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_on(self) -> bool:
|
def is_on(self) -> bool:
|
||||||
return (
|
return bool(
|
||||||
self._device.get(self.entity_description.key, "")
|
self._device.get(self.entity_description.key, "")
|
||||||
== self.entity_description.on_value
|
== self.entity_description.on_value
|
||||||
)
|
)
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
self._attr_native_value = (
|
self._attr_native_value = (
|
||||||
self._device.get(self.entity_description.key, "")
|
self._device.get(self.entity_description.key, "")
|
||||||
== self.entity_description.on_value
|
== self.entity_description.on_value
|
||||||
|
|
|
@ -5,10 +5,13 @@ from homeassistant.components import persistent_notification
|
||||||
from homeassistant.components.button import ButtonEntityDescription, ButtonEntity
|
from homeassistant.components.button import ButtonEntityDescription, ButtonEntity
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.helpers.entity import EntityCategory
|
from homeassistant.helpers.entity import EntityCategory
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
from pyhon.appliance import HonAppliance
|
from pyhon.appliance import HonAppliance
|
||||||
|
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
from .hon import HonEntity
|
from .hon import HonEntity
|
||||||
|
from .typedefs import HonButtonType
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@ -38,8 +41,10 @@ BUTTONS: dict[str, tuple[ButtonEntityDescription, ...]] = {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(
|
||||||
entities = []
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
|
entities: list[HonButtonType] = []
|
||||||
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
||||||
for description in BUTTONS.get(device.appliance_type, []):
|
for description in BUTTONS.get(device.appliance_type, []):
|
||||||
if not device.commands.get(description.key):
|
if not device.commands.get(description.key):
|
||||||
|
@ -70,7 +75,9 @@ class HonButtonEntity(HonEntity, ButtonEntity):
|
||||||
|
|
||||||
|
|
||||||
class HonDeviceInfo(HonEntity, ButtonEntity):
|
class HonDeviceInfo(HonEntity, ButtonEntity):
|
||||||
def __init__(self, hass, entry, device: HonAppliance) -> None:
|
def __init__(
|
||||||
|
self, hass: HomeAssistantType, entry: ConfigEntry, device: HonAppliance
|
||||||
|
) -> None:
|
||||||
super().__init__(hass, entry, device)
|
super().__init__(hass, entry, device)
|
||||||
|
|
||||||
self._attr_unique_id = f"{super().unique_id}_show_device_info"
|
self._attr_unique_id = f"{super().unique_id}_show_device_info"
|
||||||
|
@ -93,7 +100,9 @@ class HonDeviceInfo(HonEntity, ButtonEntity):
|
||||||
|
|
||||||
|
|
||||||
class HonDataArchive(HonEntity, ButtonEntity):
|
class HonDataArchive(HonEntity, ButtonEntity):
|
||||||
def __init__(self, hass, entry, device: HonAppliance) -> None:
|
def __init__(
|
||||||
|
self, hass: HomeAssistantType, entry: ConfigEntry, device: HonAppliance
|
||||||
|
) -> None:
|
||||||
super().__init__(hass, entry, device)
|
super().__init__(hass, entry, device)
|
||||||
|
|
||||||
self._attr_unique_id = f"{super().unique_id}_create_data_archive"
|
self._attr_unique_id = f"{super().unique_id}_create_data_archive"
|
||||||
|
@ -104,7 +113,9 @@ class HonDataArchive(HonEntity, ButtonEntity):
|
||||||
self._attr_entity_registry_enabled_default = False
|
self._attr_entity_registry_enabled_default = False
|
||||||
|
|
||||||
async def async_press(self) -> None:
|
async def async_press(self) -> None:
|
||||||
path = Path(self._hass.config.config_dir) / "www"
|
if (config_dir := self._hass.config.config_dir) is None:
|
||||||
|
raise ValueError("Missing Config Dir")
|
||||||
|
path = Path(config_dir) / "www"
|
||||||
data = await self._device.data_archive(path)
|
data = await self._device.data_archive(path)
|
||||||
title = f"{self._device.nick_name} Data Archive"
|
title = f"{self._device.nick_name} Data Archive"
|
||||||
text = (
|
text = (
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
import logging
|
import logging
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
from homeassistant.components.climate import (
|
from homeassistant.components.climate import (
|
||||||
ClimateEntity,
|
ClimateEntity,
|
||||||
|
@ -19,7 +20,10 @@ from homeassistant.const import (
|
||||||
TEMP_CELSIUS,
|
TEMP_CELSIUS,
|
||||||
)
|
)
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
from pyhon.appliance import HonAppliance
|
from pyhon.appliance import HonAppliance
|
||||||
|
from pyhon.parameter.range import HonParameterRange
|
||||||
|
|
||||||
from .const import HON_HVAC_MODE, HON_FAN, DOMAIN, HON_HVAC_PROGRAM
|
from .const import HON_HVAC_MODE, HON_FAN, DOMAIN, HON_HVAC_PROGRAM
|
||||||
from .hon import HonEntity
|
from .hon import HonEntity
|
||||||
|
@ -34,10 +38,12 @@ class HonACClimateEntityDescription(ClimateEntityDescription):
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class HonClimateEntityDescription(ClimateEntityDescription):
|
class HonClimateEntityDescription(ClimateEntityDescription):
|
||||||
mode: HVACMode = "auto"
|
mode: HVACMode = HVACMode.AUTO
|
||||||
|
|
||||||
|
|
||||||
CLIMATES = {
|
CLIMATES: dict[
|
||||||
|
str, tuple[HonACClimateEntityDescription | HonClimateEntityDescription, ...]
|
||||||
|
] = {
|
||||||
"AC": (
|
"AC": (
|
||||||
HonACClimateEntityDescription(
|
HonACClimateEntityDescription(
|
||||||
key="settings",
|
key="settings",
|
||||||
|
@ -90,8 +96,11 @@ CLIMATES = {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
entities = []
|
entities = []
|
||||||
|
entity: HonClimateEntity | HonACClimateEntity
|
||||||
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
||||||
for description in CLIMATES.get(device.appliance_type, []):
|
for description in CLIMATES.get(device.appliance_type, []):
|
||||||
if isinstance(description, HonACClimateEntityDescription):
|
if isinstance(description, HonACClimateEntityDescription):
|
||||||
|
@ -103,14 +112,22 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
||||||
continue
|
continue
|
||||||
entity = HonClimateEntity(hass, entry, device, description)
|
entity = HonClimateEntity(hass, entry, device, description)
|
||||||
else:
|
else:
|
||||||
continue
|
continue # type: ignore[unreachable]
|
||||||
await entity.coordinator.async_config_entry_first_refresh()
|
await entity.coordinator.async_config_entry_first_refresh()
|
||||||
entities.append(entity)
|
entities.append(entity)
|
||||||
async_add_entities(entities)
|
async_add_entities(entities)
|
||||||
|
|
||||||
|
|
||||||
class HonACClimateEntity(HonEntity, ClimateEntity):
|
class HonACClimateEntity(HonEntity, ClimateEntity):
|
||||||
def __init__(self, hass, entry, device: HonAppliance, description) -> None:
|
entity_description: HonACClimateEntityDescription
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
hass: HomeAssistantType,
|
||||||
|
entry: ConfigEntry,
|
||||||
|
device: HonAppliance,
|
||||||
|
description: HonACClimateEntityDescription,
|
||||||
|
) -> None:
|
||||||
super().__init__(hass, entry, device, description)
|
super().__init__(hass, entry, device, description)
|
||||||
|
|
||||||
self._attr_temperature_unit = TEMP_CELSIUS
|
self._attr_temperature_unit = TEMP_CELSIUS
|
||||||
|
@ -138,37 +155,38 @@ class HonACClimateEntity(HonEntity, ClimateEntity):
|
||||||
self._handle_coordinator_update(update=False)
|
self._handle_coordinator_update(update=False)
|
||||||
|
|
||||||
def _set_temperature_bound(self) -> None:
|
def _set_temperature_bound(self) -> None:
|
||||||
self._attr_target_temperature_step = self._device.settings[
|
temperature = self._device.settings[self.entity_description.key]
|
||||||
"settings.tempSel"
|
if not isinstance(temperature, HonParameterRange):
|
||||||
].step
|
raise ValueError
|
||||||
self._attr_max_temp = self._device.settings["settings.tempSel"].max
|
self._attr_max_temp = temperature.max
|
||||||
self._attr_min_temp = self._device.settings["settings.tempSel"].min
|
self._attr_target_temperature_step = temperature.step
|
||||||
|
self._attr_min_temp = temperature.min
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def target_temperature(self) -> int | None:
|
def target_temperature(self) -> float | None:
|
||||||
"""Return the temperature we try to reach."""
|
"""Return the temperature we try to reach."""
|
||||||
return self._device.get("tempSel")
|
return self._device.get("tempSel", 0.0)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def current_temperature(self) -> float | None:
|
def current_temperature(self) -> float | None:
|
||||||
"""Return the current temperature."""
|
"""Return the current temperature."""
|
||||||
return self._device.get("tempIndoor")
|
return self._device.get("tempIndoor", 0.0)
|
||||||
|
|
||||||
async def async_set_temperature(self, **kwargs):
|
async def async_set_temperature(self, **kwargs: Any) -> None:
|
||||||
if (temperature := kwargs.get(ATTR_TEMPERATURE)) is None:
|
if (temperature := kwargs.get(ATTR_TEMPERATURE)) is None:
|
||||||
return False
|
return
|
||||||
self._device.settings["settings.tempSel"].value = str(int(temperature))
|
self._device.settings["settings.tempSel"].value = str(int(temperature))
|
||||||
await self._device.commands["settings"].send()
|
await self._device.commands["settings"].send()
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def hvac_mode(self) -> HVACMode | str | None:
|
def hvac_mode(self) -> HVACMode:
|
||||||
if self._device.get("onOffStatus") == 0:
|
if self._device.get("onOffStatus") == 0:
|
||||||
return HVACMode.OFF
|
return HVACMode.OFF
|
||||||
else:
|
else:
|
||||||
return HON_HVAC_MODE[self._device.get("machMode")]
|
return HON_HVAC_MODE[self._device.get("machMode")]
|
||||||
|
|
||||||
async def async_set_hvac_mode(self, hvac_mode):
|
async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None:
|
||||||
self._attr_hvac_mode = hvac_mode
|
self._attr_hvac_mode = hvac_mode
|
||||||
if hvac_mode == HVACMode.OFF:
|
if hvac_mode == HVACMode.OFF:
|
||||||
await self._device.commands["stopProgram"].send()
|
await self._device.commands["stopProgram"].send()
|
||||||
|
@ -215,7 +233,7 @@ class HonACClimateEntity(HonEntity, ClimateEntity):
|
||||||
"""Return the fan setting."""
|
"""Return the fan setting."""
|
||||||
return HON_FAN[self._device.get("windSpeed")]
|
return HON_FAN[self._device.get("windSpeed")]
|
||||||
|
|
||||||
async def async_set_fan_mode(self, fan_mode):
|
async def async_set_fan_mode(self, fan_mode: str) -> None:
|
||||||
fan_modes = {}
|
fan_modes = {}
|
||||||
for mode in reversed(self._device.settings["settings.windSpeed"].values):
|
for mode in reversed(self._device.settings["settings.windSpeed"].values):
|
||||||
fan_modes[HON_FAN[int(mode)]] = mode
|
fan_modes[HON_FAN[int(mode)]] = mode
|
||||||
|
@ -231,14 +249,13 @@ class HonACClimateEntity(HonEntity, ClimateEntity):
|
||||||
vertical = self._device.get("windDirectionVertical")
|
vertical = self._device.get("windDirectionVertical")
|
||||||
if horizontal == 7 and vertical == 8:
|
if horizontal == 7 and vertical == 8:
|
||||||
return SWING_BOTH
|
return SWING_BOTH
|
||||||
elif horizontal == 7:
|
if horizontal == 7:
|
||||||
return SWING_HORIZONTAL
|
return SWING_HORIZONTAL
|
||||||
elif vertical == 8:
|
if vertical == 8:
|
||||||
return SWING_VERTICAL
|
return SWING_VERTICAL
|
||||||
else:
|
|
||||||
return SWING_OFF
|
return SWING_OFF
|
||||||
|
|
||||||
async def async_set_swing_mode(self, swing_mode):
|
async def async_set_swing_mode(self, swing_mode: str) -> None:
|
||||||
horizontal = self._device.settings["settings.windDirectionHorizontal"]
|
horizontal = self._device.settings["settings.windDirectionHorizontal"]
|
||||||
vertical = self._device.settings["settings.windDirectionVertical"]
|
vertical = self._device.settings["settings.windDirectionVertical"]
|
||||||
if swing_mode in [SWING_BOTH, SWING_HORIZONTAL]:
|
if swing_mode in [SWING_BOTH, SWING_HORIZONTAL]:
|
||||||
|
@ -254,13 +271,7 @@ class HonACClimateEntity(HonEntity, ClimateEntity):
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
self._attr_target_temperature = self.target_temperature
|
|
||||||
self._attr_current_temperature = self.current_temperature
|
|
||||||
self._attr_hvac_mode = self.hvac_mode
|
|
||||||
self._attr_fan_modes = self.fan_modes
|
|
||||||
self._attr_fan_mode = self.fan_mode
|
|
||||||
self._attr_swing_mode = self.swing_mode
|
|
||||||
if update:
|
if update:
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
|
@ -268,7 +279,13 @@ class HonACClimateEntity(HonEntity, ClimateEntity):
|
||||||
class HonClimateEntity(HonEntity, ClimateEntity):
|
class HonClimateEntity(HonEntity, ClimateEntity):
|
||||||
entity_description: HonClimateEntityDescription
|
entity_description: HonClimateEntityDescription
|
||||||
|
|
||||||
def __init__(self, hass, entry, device: HonAppliance, description) -> None:
|
def __init__(
|
||||||
|
self,
|
||||||
|
hass: HomeAssistantType,
|
||||||
|
entry: ConfigEntry,
|
||||||
|
device: HonAppliance,
|
||||||
|
description: HonClimateEntityDescription,
|
||||||
|
) -> None:
|
||||||
super().__init__(hass, entry, device, description)
|
super().__init__(hass, entry, device, description)
|
||||||
|
|
||||||
self._attr_temperature_unit = TEMP_CELSIUS
|
self._attr_temperature_unit = TEMP_CELSIUS
|
||||||
|
@ -288,7 +305,9 @@ class HonClimateEntity(HonEntity, ClimateEntity):
|
||||||
for mode, data in device.commands["startProgram"].categories.items():
|
for mode, data in device.commands["startProgram"].categories.items():
|
||||||
if mode not in data.parameters["program"].values:
|
if mode not in data.parameters["program"].values:
|
||||||
continue
|
continue
|
||||||
if zone := data.parameters.get("zone"):
|
if (zone := data.parameters.get("zone")) and isinstance(
|
||||||
|
self.entity_description.name, str
|
||||||
|
):
|
||||||
if self.entity_description.name.lower() in zone.values:
|
if self.entity_description.name.lower() in zone.values:
|
||||||
modes.append(mode)
|
modes.append(mode)
|
||||||
else:
|
else:
|
||||||
|
@ -300,29 +319,29 @@ class HonClimateEntity(HonEntity, ClimateEntity):
|
||||||
@property
|
@property
|
||||||
def target_temperature(self) -> float | None:
|
def target_temperature(self) -> float | None:
|
||||||
"""Return the temperature we try to reach."""
|
"""Return the temperature we try to reach."""
|
||||||
return self._device.get(self.entity_description.key)
|
return self._device.get(self.entity_description.key, 0.0)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def current_temperature(self) -> float | None:
|
def current_temperature(self) -> float | None:
|
||||||
"""Return the current temperature."""
|
"""Return the current temperature."""
|
||||||
temp_key = self.entity_description.key.split(".")[-1].replace("Sel", "")
|
temp_key = self.entity_description.key.split(".")[-1].replace("Sel", "")
|
||||||
return self._device.get(temp_key)
|
return self._device.get(temp_key, 0.0)
|
||||||
|
|
||||||
async def async_set_temperature(self, **kwargs):
|
async def async_set_temperature(self, **kwargs: Any) -> None:
|
||||||
if (temperature := kwargs.get(ATTR_TEMPERATURE)) is None:
|
if (temperature := kwargs.get(ATTR_TEMPERATURE)) is None:
|
||||||
return False
|
return
|
||||||
self._device.settings[self.entity_description.key].value = str(int(temperature))
|
self._device.settings[self.entity_description.key].value = str(int(temperature))
|
||||||
await self._device.commands["settings"].send()
|
await self._device.commands["settings"].send()
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def hvac_mode(self) -> HVACMode | str | None:
|
def hvac_mode(self) -> HVACMode:
|
||||||
if self._device.get("onOffStatus") == 0:
|
if self._device.get("onOffStatus") == 0:
|
||||||
return HVACMode.OFF
|
return HVACMode.OFF
|
||||||
else:
|
else:
|
||||||
return self.entity_description.mode
|
return self.entity_description.mode
|
||||||
|
|
||||||
async def async_set_hvac_mode(self, hvac_mode):
|
async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None:
|
||||||
if len(self.hvac_modes) <= 1:
|
if len(self.hvac_modes) <= 1:
|
||||||
return
|
return
|
||||||
if hvac_mode == HVACMode.OFF:
|
if hvac_mode == HVACMode.OFF:
|
||||||
|
@ -347,7 +366,8 @@ class HonClimateEntity(HonEntity, ClimateEntity):
|
||||||
command = "stopProgram" if preset_mode == "no_mode" else "startProgram"
|
command = "stopProgram" if preset_mode == "no_mode" else "startProgram"
|
||||||
if program := self._device.settings.get(f"{command}.program"):
|
if program := self._device.settings.get(f"{command}.program"):
|
||||||
program.value = preset_mode
|
program.value = preset_mode
|
||||||
if zone := self._device.settings.get(f"{command}.zone"):
|
zone = self._device.settings.get(f"{command}.zone")
|
||||||
|
if zone and isinstance(self.entity_description.name, str):
|
||||||
zone.value = self.entity_description.name.lower()
|
zone.value = self.entity_description.name.lower()
|
||||||
self._device.sync_command(command, "settings")
|
self._device.sync_command(command, "settings")
|
||||||
self._set_temperature_bound()
|
self._set_temperature_bound()
|
||||||
|
@ -356,18 +376,15 @@ class HonClimateEntity(HonEntity, ClimateEntity):
|
||||||
self._attr_preset_mode = preset_mode
|
self._attr_preset_mode = preset_mode
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
def _set_temperature_bound(self):
|
def _set_temperature_bound(self) -> None:
|
||||||
self._attr_target_temperature_step = self._device.settings[
|
temperature = self._device.settings[self.entity_description.key]
|
||||||
self.entity_description.key
|
if not isinstance(temperature, HonParameterRange):
|
||||||
].step
|
raise ValueError
|
||||||
self._attr_max_temp = self._device.settings[self.entity_description.key].max
|
self._attr_max_temp = temperature.max
|
||||||
self._attr_min_temp = self._device.settings[self.entity_description.key].min
|
self._attr_target_temperature_step = temperature.step
|
||||||
|
self._attr_min_temp = temperature.min
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
self._attr_target_temperature = self.target_temperature
|
|
||||||
self._attr_current_temperature = self.current_temperature
|
|
||||||
self._attr_hvac_mode = self.hvac_mode
|
|
||||||
self._attr_preset_mode = self.preset_mode
|
|
||||||
if update:
|
if update:
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
import logging
|
import logging
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
import voluptuous as vol
|
import voluptuous as vol # type: ignore[import]
|
||||||
from homeassistant import config_entries
|
from homeassistant import config_entries
|
||||||
from homeassistant.const import CONF_EMAIL, CONF_PASSWORD
|
from homeassistant.const import CONF_EMAIL, CONF_PASSWORD
|
||||||
|
from homeassistant.data_entry_flow import FlowResult
|
||||||
|
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
|
|
||||||
|
@ -13,11 +15,13 @@ class HonFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
|
||||||
VERSION = 1
|
VERSION = 1
|
||||||
CONNECTION_CLASS = config_entries.CONN_CLASS_LOCAL_POLL
|
CONNECTION_CLASS = config_entries.CONN_CLASS_LOCAL_POLL
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self) -> None:
|
||||||
self._email = None
|
self._email: str | None = None
|
||||||
self._password = None
|
self._password: str | None = None
|
||||||
|
|
||||||
async def async_step_user(self, user_input=None):
|
async def async_step_user(
|
||||||
|
self, user_input: dict[str, Any] | None = None
|
||||||
|
) -> FlowResult:
|
||||||
if user_input is None:
|
if user_input is None:
|
||||||
return self.async_show_form(
|
return self.async_show_form(
|
||||||
step_id="user",
|
step_id="user",
|
||||||
|
@ -29,6 +33,14 @@ class HonFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
|
||||||
self._email = user_input[CONF_EMAIL]
|
self._email = user_input[CONF_EMAIL]
|
||||||
self._password = user_input[CONF_PASSWORD]
|
self._password = user_input[CONF_PASSWORD]
|
||||||
|
|
||||||
|
if self._email is None or self._password is None:
|
||||||
|
return self.async_show_form(
|
||||||
|
step_id="user",
|
||||||
|
data_schema=vol.Schema(
|
||||||
|
{vol.Required(CONF_EMAIL): str, vol.Required(CONF_PASSWORD): str}
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
# Check if already configured
|
# Check if already configured
|
||||||
await self.async_set_unique_id(self._email)
|
await self.async_set_unique_id(self._email)
|
||||||
self._abort_if_unique_id_configured()
|
self._abort_if_unique_id_configured()
|
||||||
|
@ -41,5 +53,5 @@ class HonFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
async def async_step_import(self, user_input=None):
|
async def async_step_import(self, user_input: dict[str, str]) -> FlowResult:
|
||||||
return await self.async_step_user(user_input)
|
return await self.async_step_user(user_input)
|
||||||
|
|
|
@ -6,10 +6,10 @@ from homeassistant.components.climate import (
|
||||||
FAN_AUTO,
|
FAN_AUTO,
|
||||||
)
|
)
|
||||||
|
|
||||||
DOMAIN = "hon"
|
DOMAIN: str = "hon"
|
||||||
UPDATE_INTERVAL = 10
|
UPDATE_INTERVAL: int = 10
|
||||||
|
|
||||||
PLATFORMS = [
|
PLATFORMS: list[str] = [
|
||||||
"sensor",
|
"sensor",
|
||||||
"select",
|
"select",
|
||||||
"number",
|
"number",
|
||||||
|
@ -22,7 +22,7 @@ PLATFORMS = [
|
||||||
"lock",
|
"lock",
|
||||||
]
|
]
|
||||||
|
|
||||||
APPLIANCES = {
|
APPLIANCES: dict[str, str] = {
|
||||||
"AC": "Air Conditioner",
|
"AC": "Air Conditioner",
|
||||||
"AP": "Air Purifier",
|
"AP": "Air Purifier",
|
||||||
"AS": "Air Scanner",
|
"AS": "Air Scanner",
|
||||||
|
@ -40,7 +40,7 @@ APPLIANCES = {
|
||||||
"WM": "Washing Machine",
|
"WM": "Washing Machine",
|
||||||
}
|
}
|
||||||
|
|
||||||
HON_HVAC_MODE = {
|
HON_HVAC_MODE: dict[int, HVACMode] = {
|
||||||
0: HVACMode.AUTO,
|
0: HVACMode.AUTO,
|
||||||
1: HVACMode.COOL,
|
1: HVACMode.COOL,
|
||||||
2: HVACMode.DRY,
|
2: HVACMode.DRY,
|
||||||
|
@ -50,7 +50,7 @@ HON_HVAC_MODE = {
|
||||||
6: HVACMode.FAN_ONLY,
|
6: HVACMode.FAN_ONLY,
|
||||||
}
|
}
|
||||||
|
|
||||||
HON_HVAC_PROGRAM = {
|
HON_HVAC_PROGRAM: dict[str, str] = {
|
||||||
HVACMode.AUTO: "iot_auto",
|
HVACMode.AUTO: "iot_auto",
|
||||||
HVACMode.COOL: "iot_cool",
|
HVACMode.COOL: "iot_cool",
|
||||||
HVACMode.DRY: "iot_dry",
|
HVACMode.DRY: "iot_dry",
|
||||||
|
@ -58,7 +58,7 @@ HON_HVAC_PROGRAM = {
|
||||||
HVACMode.FAN_ONLY: "iot_fan",
|
HVACMode.FAN_ONLY: "iot_fan",
|
||||||
}
|
}
|
||||||
|
|
||||||
HON_FAN = {
|
HON_FAN: dict[int, str] = {
|
||||||
1: FAN_HIGH,
|
1: FAN_HIGH,
|
||||||
2: FAN_MEDIUM,
|
2: FAN_MEDIUM,
|
||||||
3: FAN_LOW,
|
3: FAN_LOW,
|
||||||
|
@ -67,7 +67,7 @@ HON_FAN = {
|
||||||
}
|
}
|
||||||
|
|
||||||
# These languages are official supported by hOn
|
# These languages are official supported by hOn
|
||||||
LANGUAGES = [
|
LANGUAGES: list[str] = [
|
||||||
"cs", # Czech
|
"cs", # Czech
|
||||||
"de", # German
|
"de", # German
|
||||||
"el", # Greek
|
"el", # Greek
|
||||||
|
@ -89,7 +89,7 @@ LANGUAGES = [
|
||||||
"zh", # Chinese
|
"zh", # Chinese
|
||||||
]
|
]
|
||||||
|
|
||||||
WASHING_PR_PHASE = {
|
WASHING_PR_PHASE: dict[int, str] = {
|
||||||
0: "ready",
|
0: "ready",
|
||||||
1: "washing",
|
1: "washing",
|
||||||
2: "washing",
|
2: "washing",
|
||||||
|
@ -116,7 +116,7 @@ WASHING_PR_PHASE = {
|
||||||
27: "washing",
|
27: "washing",
|
||||||
}
|
}
|
||||||
|
|
||||||
MACH_MODE = {
|
MACH_MODE: dict[int, str] = {
|
||||||
0: "ready", # NO_STATE
|
0: "ready", # NO_STATE
|
||||||
1: "ready", # SELECTION_MODE
|
1: "ready", # SELECTION_MODE
|
||||||
2: "running", # EXECUTION_MODE
|
2: "running", # EXECUTION_MODE
|
||||||
|
@ -129,7 +129,7 @@ MACH_MODE = {
|
||||||
9: "ending", # STOP_MODE
|
9: "ending", # STOP_MODE
|
||||||
}
|
}
|
||||||
|
|
||||||
TUMBLE_DRYER_PR_PHASE = {
|
TUMBLE_DRYER_PR_PHASE: dict[int, str] = {
|
||||||
0: "ready",
|
0: "ready",
|
||||||
1: "heat_stroke",
|
1: "heat_stroke",
|
||||||
2: "drying",
|
2: "drying",
|
||||||
|
@ -147,21 +147,21 @@ TUMBLE_DRYER_PR_PHASE = {
|
||||||
20: "drying",
|
20: "drying",
|
||||||
}
|
}
|
||||||
|
|
||||||
DIRTY_LEVEL = {
|
DIRTY_LEVEL: dict[int, str] = {
|
||||||
0: "unknown",
|
0: "unknown",
|
||||||
1: "little",
|
1: "little",
|
||||||
2: "normal",
|
2: "normal",
|
||||||
3: "very",
|
3: "very",
|
||||||
}
|
}
|
||||||
|
|
||||||
STEAM_LEVEL = {
|
STEAM_LEVEL: dict[int, str] = {
|
||||||
0: "no_steam",
|
0: "no_steam",
|
||||||
1: "cotton",
|
1: "cotton",
|
||||||
2: "delicate",
|
2: "delicate",
|
||||||
3: "synthetic",
|
3: "synthetic",
|
||||||
}
|
}
|
||||||
|
|
||||||
DISHWASHER_PR_PHASE = {
|
DISHWASHER_PR_PHASE: dict[int, str] = {
|
||||||
0: "ready",
|
0: "ready",
|
||||||
1: "prewash",
|
1: "prewash",
|
||||||
2: "washing",
|
2: "washing",
|
||||||
|
@ -171,7 +171,7 @@ DISHWASHER_PR_PHASE = {
|
||||||
6: "hot_rinse",
|
6: "hot_rinse",
|
||||||
}
|
}
|
||||||
|
|
||||||
TUMBLE_DRYER_DRY_LEVEL = {
|
TUMBLE_DRYER_DRY_LEVEL: dict[int, str] = {
|
||||||
0: "no_dry",
|
0: "no_dry",
|
||||||
1: "iron_dry",
|
1: "iron_dry",
|
||||||
2: "no_dry_iron",
|
2: "no_dry_iron",
|
||||||
|
@ -184,7 +184,7 @@ TUMBLE_DRYER_DRY_LEVEL = {
|
||||||
15: "extra_dry",
|
15: "extra_dry",
|
||||||
}
|
}
|
||||||
|
|
||||||
AC_MACH_MODE = {
|
AC_MACH_MODE: dict[int, str] = {
|
||||||
0: "auto",
|
0: "auto",
|
||||||
1: "cool",
|
1: "cool",
|
||||||
2: "cool",
|
2: "cool",
|
||||||
|
@ -194,7 +194,7 @@ AC_MACH_MODE = {
|
||||||
6: "fan",
|
6: "fan",
|
||||||
}
|
}
|
||||||
|
|
||||||
AC_FAN_MODE = {
|
AC_FAN_MODE: dict[int, str] = {
|
||||||
1: "high",
|
1: "high",
|
||||||
2: "mid",
|
2: "mid",
|
||||||
3: "low",
|
3: "low",
|
||||||
|
@ -202,14 +202,14 @@ AC_FAN_MODE = {
|
||||||
5: "auto",
|
5: "auto",
|
||||||
}
|
}
|
||||||
|
|
||||||
AC_HUMAN_SENSE = {
|
AC_HUMAN_SENSE: dict[int, str] = {
|
||||||
0: "touch_off",
|
0: "touch_off",
|
||||||
1: "avoid_touch",
|
1: "avoid_touch",
|
||||||
2: "follow_touch",
|
2: "follow_touch",
|
||||||
3: "unknown",
|
3: "unknown",
|
||||||
}
|
}
|
||||||
|
|
||||||
AP_MACH_MODE = {
|
AP_MACH_MODE: dict[int, str] = {
|
||||||
0: "standby",
|
0: "standby",
|
||||||
1: "sleep",
|
1: "sleep",
|
||||||
2: "auto",
|
2: "auto",
|
||||||
|
@ -217,7 +217,7 @@ AP_MACH_MODE = {
|
||||||
4: "max",
|
4: "max",
|
||||||
}
|
}
|
||||||
|
|
||||||
AP_DIFFUSER_LEVEL = {
|
AP_DIFFUSER_LEVEL: dict[int, str] = {
|
||||||
0: "off",
|
0: "off",
|
||||||
1: "soft",
|
1: "soft",
|
||||||
2: "mid",
|
2: "mid",
|
||||||
|
@ -225,4 +225,4 @@ AP_DIFFUSER_LEVEL = {
|
||||||
4: "custom",
|
4: "custom",
|
||||||
}
|
}
|
||||||
|
|
||||||
REF_HUMIDITY_LEVELS = {1: "low", 2: "mid", 3: "high"}
|
REF_HUMIDITY_LEVELS: dict[int, str] = {1: "low", 2: "mid", 3: "high"}
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import logging
|
import logging
|
||||||
import math
|
import math
|
||||||
from dataclasses import dataclass
|
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
from homeassistant.components.fan import (
|
from homeassistant.components.fan import (
|
||||||
|
@ -10,6 +9,8 @@ from homeassistant.components.fan import (
|
||||||
)
|
)
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
from homeassistant.util.percentage import (
|
from homeassistant.util.percentage import (
|
||||||
percentage_to_ranged_value,
|
percentage_to_ranged_value,
|
||||||
ranged_value_to_percentage,
|
ranged_value_to_percentage,
|
||||||
|
@ -19,18 +20,14 @@ from pyhon.parameter.range import HonParameterRange
|
||||||
|
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
from .hon import HonEntity
|
from .hon import HonEntity
|
||||||
|
from .typedefs import HonEntityDescription
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
FANS: dict[str, tuple[FanEntityDescription, ...]] = {
|
||||||
class HonFanEntityDescription(FanEntityDescription):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
FANS = {
|
|
||||||
"HO": (
|
"HO": (
|
||||||
HonFanEntityDescription(
|
FanEntityDescription(
|
||||||
key="settings.windSpeed",
|
key="settings.windSpeed",
|
||||||
name="Wind Speed",
|
name="Wind Speed",
|
||||||
translation_key="air_extraction",
|
translation_key="air_extraction",
|
||||||
|
@ -39,30 +36,36 @@ FANS = {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
entities = []
|
entities = []
|
||||||
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
||||||
for description in FANS.get(device.appliance_type, []):
|
for description in FANS.get(device.appliance_type, []):
|
||||||
if isinstance(description, HonFanEntityDescription):
|
|
||||||
if (
|
if (
|
||||||
description.key not in device.available_settings
|
description.key not in device.available_settings
|
||||||
or device.get(description.key.split(".")[-1]) is None
|
or device.get(description.key.split(".")[-1]) is None
|
||||||
):
|
):
|
||||||
continue
|
continue
|
||||||
entity = HonFanEntity(hass, entry, device, description)
|
entity = HonFanEntity(hass, entry, device, description)
|
||||||
else:
|
|
||||||
continue
|
|
||||||
await entity.coordinator.async_config_entry_first_refresh()
|
await entity.coordinator.async_config_entry_first_refresh()
|
||||||
entities.append(entity)
|
entities.append(entity)
|
||||||
async_add_entities(entities)
|
async_add_entities(entities)
|
||||||
|
|
||||||
|
|
||||||
class HonFanEntity(HonEntity, FanEntity):
|
class HonFanEntity(HonEntity, FanEntity):
|
||||||
entity_description: HonFanEntityDescription
|
entity_description: FanEntityDescription
|
||||||
|
|
||||||
def __init__(self, hass, entry, device: HonAppliance, description) -> None:
|
def __init__(
|
||||||
|
self,
|
||||||
|
hass: HomeAssistantType,
|
||||||
|
entry: ConfigEntry,
|
||||||
|
device: HonAppliance,
|
||||||
|
description: FanEntityDescription,
|
||||||
|
) -> None:
|
||||||
self._attr_supported_features = FanEntityFeature.SET_SPEED
|
self._attr_supported_features = FanEntityFeature.SET_SPEED
|
||||||
self._wind_speed: HonParameterRange = device.settings.get(description.key)
|
self._wind_speed: HonParameterRange
|
||||||
|
self._speed_range: tuple[int, int]
|
||||||
self._command, self._parameter = description.key.split(".")
|
self._command, self._parameter = description.key.split(".")
|
||||||
|
|
||||||
super().__init__(hass, entry, device, description)
|
super().__init__(hass, entry, device, description)
|
||||||
|
@ -89,8 +92,10 @@ class HonFanEntity(HonEntity, FanEntity):
|
||||||
@property
|
@property
|
||||||
def is_on(self) -> bool | None:
|
def is_on(self) -> bool | None:
|
||||||
"""Return true if device is on."""
|
"""Return true if device is on."""
|
||||||
|
if self.percentage is None:
|
||||||
|
return False
|
||||||
mode = math.ceil(percentage_to_ranged_value(self._speed_range, self.percentage))
|
mode = math.ceil(percentage_to_ranged_value(self._speed_range, self.percentage))
|
||||||
return mode > self._wind_speed.min
|
return bool(mode > self._wind_speed.min)
|
||||||
|
|
||||||
async def async_turn_on(
|
async def async_turn_on(
|
||||||
self,
|
self,
|
||||||
|
@ -112,9 +117,10 @@ class HonFanEntity(HonEntity, FanEntity):
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
self._wind_speed = self._device.settings.get(self.entity_description.key)
|
wind_speed = self._device.settings.get(self.entity_description.key)
|
||||||
if len(self._wind_speed.values) > 1:
|
if isinstance(wind_speed, HonParameterRange) and len(wind_speed.values) > 1:
|
||||||
|
self._wind_speed = wind_speed
|
||||||
self._speed_range = (
|
self._speed_range = (
|
||||||
int(self._wind_speed.values[1]),
|
int(self._wind_speed.values[1]),
|
||||||
int(self._wind_speed.values[-1]),
|
int(self._wind_speed.values[-1]),
|
||||||
|
|
|
@ -3,23 +3,79 @@ import logging
|
||||||
from contextlib import suppress
|
from contextlib import suppress
|
||||||
from datetime import timedelta
|
from datetime import timedelta
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
from typing import Optional, Any, TypeVar
|
||||||
|
|
||||||
import pkg_resources
|
import pkg_resources
|
||||||
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
from homeassistant.helpers.entity import DeviceInfo
|
from homeassistant.helpers.entity import DeviceInfo
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
||||||
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
||||||
from pyhon.appliance import HonAppliance
|
from pyhon.appliance import HonAppliance
|
||||||
|
|
||||||
from .const import DOMAIN, UPDATE_INTERVAL
|
from .const import DOMAIN, UPDATE_INTERVAL
|
||||||
|
from .typedefs import HonEntityDescription, HonOptionEntityDescription, T
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class HonEntity(CoordinatorEntity):
|
class HonInfo:
|
||||||
|
def __init__(self) -> None:
|
||||||
|
self._manifest: dict[str, Any] = self._get_manifest()
|
||||||
|
self._hon_version: str = self._manifest.get("version", "")
|
||||||
|
self._pyhon_version: str = pkg_resources.get_distribution("pyhon").version
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _get_manifest() -> dict[str, Any]:
|
||||||
|
manifest = Path(__file__).parent / "manifest.json"
|
||||||
|
with open(manifest, "r", encoding="utf-8") as file:
|
||||||
|
result: dict[str, Any] = json.loads(file.read())
|
||||||
|
return result
|
||||||
|
|
||||||
|
@property
|
||||||
|
def manifest(self) -> dict[str, Any]:
|
||||||
|
return self._manifest
|
||||||
|
|
||||||
|
@property
|
||||||
|
def hon_version(self) -> str:
|
||||||
|
return self._hon_version
|
||||||
|
|
||||||
|
@property
|
||||||
|
def pyhon_version(self) -> str:
|
||||||
|
return self._pyhon_version
|
||||||
|
|
||||||
|
|
||||||
|
class HonCoordinator(DataUpdateCoordinator[None]):
|
||||||
|
def __init__(self, hass: HomeAssistantType, device: HonAppliance):
|
||||||
|
"""Initialize my coordinator."""
|
||||||
|
super().__init__(
|
||||||
|
hass,
|
||||||
|
_LOGGER,
|
||||||
|
name=device.unique_id,
|
||||||
|
update_interval=timedelta(seconds=UPDATE_INTERVAL),
|
||||||
|
)
|
||||||
|
self._device = device
|
||||||
|
self._info = HonInfo()
|
||||||
|
|
||||||
|
async def _async_update_data(self) -> None:
|
||||||
|
return await self._device.update()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def info(self) -> HonInfo:
|
||||||
|
return self._info
|
||||||
|
|
||||||
|
|
||||||
|
class HonEntity(CoordinatorEntity[HonCoordinator]):
|
||||||
_attr_has_entity_name = True
|
_attr_has_entity_name = True
|
||||||
|
|
||||||
def __init__(self, hass, entry, device: HonAppliance, description=None) -> None:
|
def __init__(
|
||||||
|
self,
|
||||||
|
hass: HomeAssistantType,
|
||||||
|
entry: ConfigEntry,
|
||||||
|
device: HonAppliance,
|
||||||
|
description: Optional[HonEntityDescription] = None,
|
||||||
|
) -> None:
|
||||||
coordinator = get_coordinator(hass, device)
|
coordinator = get_coordinator(hass, device)
|
||||||
super().__init__(coordinator)
|
super().__init__(coordinator)
|
||||||
|
|
||||||
|
@ -36,7 +92,7 @@ class HonEntity(CoordinatorEntity):
|
||||||
self._handle_coordinator_update(update=False)
|
self._handle_coordinator_update(update=False)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def device_info(self):
|
def device_info(self) -> DeviceInfo:
|
||||||
return DeviceInfo(
|
return DeviceInfo(
|
||||||
identifiers={(DOMAIN, self._device.unique_id)},
|
identifiers={(DOMAIN, self._device.unique_id)},
|
||||||
manufacturer=self._device.get("brand", ""),
|
manufacturer=self._device.get("brand", ""),
|
||||||
|
@ -51,71 +107,34 @@ class HonEntity(CoordinatorEntity):
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
|
|
||||||
class HonInfo:
|
def unique_entities(
|
||||||
def __init__(self):
|
base_entities: tuple[T, ...],
|
||||||
self._manifest = self._get_manifest()
|
new_entities: tuple[T, ...],
|
||||||
self._hon_version = self._manifest.get("version", "")
|
) -> tuple[T, ...]:
|
||||||
self._pyhon_version = pkg_resources.get_distribution("pyhon").version
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _get_manifest():
|
|
||||||
manifest = Path(__file__).parent / "manifest.json"
|
|
||||||
with open(manifest, "r", encoding="utf-8") as file:
|
|
||||||
return json.loads(file.read())
|
|
||||||
|
|
||||||
@property
|
|
||||||
def manifest(self):
|
|
||||||
return self._manifest
|
|
||||||
|
|
||||||
@property
|
|
||||||
def hon_version(self):
|
|
||||||
return self._hon_version
|
|
||||||
|
|
||||||
@property
|
|
||||||
def pyhon_version(self):
|
|
||||||
return self._pyhon_version
|
|
||||||
|
|
||||||
|
|
||||||
class HonCoordinator(DataUpdateCoordinator):
|
|
||||||
def __init__(self, hass, device: HonAppliance):
|
|
||||||
"""Initialize my coordinator."""
|
|
||||||
super().__init__(
|
|
||||||
hass,
|
|
||||||
_LOGGER,
|
|
||||||
name=device.unique_id,
|
|
||||||
update_interval=timedelta(seconds=UPDATE_INTERVAL),
|
|
||||||
)
|
|
||||||
self._device = device
|
|
||||||
self._info = HonInfo()
|
|
||||||
|
|
||||||
async def _async_update_data(self):
|
|
||||||
await self._device.update()
|
|
||||||
|
|
||||||
@property
|
|
||||||
def info(self) -> HonInfo:
|
|
||||||
return self._info
|
|
||||||
|
|
||||||
|
|
||||||
def unique_entities(base_entities, new_entities):
|
|
||||||
result = list(base_entities)
|
result = list(base_entities)
|
||||||
existing_entities = [entity.key for entity in base_entities]
|
existing_entities = [entity.key for entity in base_entities]
|
||||||
|
entity: HonEntityDescription
|
||||||
for entity in new_entities:
|
for entity in new_entities:
|
||||||
if entity.key not in existing_entities:
|
if entity.key not in existing_entities:
|
||||||
result.append(entity)
|
result.append(entity)
|
||||||
return tuple(result)
|
return tuple(result)
|
||||||
|
|
||||||
|
|
||||||
def get_coordinator(hass, appliance):
|
def get_coordinator(hass: HomeAssistantType, appliance: HonAppliance) -> HonCoordinator:
|
||||||
coordinators = hass.data[DOMAIN]["coordinators"]
|
coordinators = hass.data[DOMAIN]["coordinators"]
|
||||||
if appliance.unique_id in coordinators:
|
if appliance.unique_id in coordinators:
|
||||||
coordinator = hass.data[DOMAIN]["coordinators"][appliance.unique_id]
|
coordinator: HonCoordinator = hass.data[DOMAIN]["coordinators"][
|
||||||
|
appliance.unique_id
|
||||||
|
]
|
||||||
else:
|
else:
|
||||||
coordinator = HonCoordinator(hass, appliance)
|
coordinator = HonCoordinator(hass, appliance)
|
||||||
hass.data[DOMAIN]["coordinators"][appliance.unique_id] = coordinator
|
hass.data[DOMAIN]["coordinators"][appliance.unique_id] = coordinator
|
||||||
return coordinator
|
return coordinator
|
||||||
|
|
||||||
|
|
||||||
def get_readable(description, value):
|
def get_readable(
|
||||||
|
description: HonOptionEntityDescription, value: float | str
|
||||||
|
) -> float | str:
|
||||||
if description.option_list is not None:
|
if description.option_list is not None:
|
||||||
with suppress(ValueError):
|
with suppress(ValueError):
|
||||||
return description.option_list.get(int(value), value)
|
return description.option_list.get(int(value), value)
|
||||||
|
|
|
@ -9,6 +9,8 @@ from homeassistant.components.light import (
|
||||||
)
|
)
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
from pyhon.appliance import HonAppliance
|
from pyhon.appliance import HonAppliance
|
||||||
from pyhon.parameter.range import HonParameterRange
|
from pyhon.parameter.range import HonParameterRange
|
||||||
|
|
||||||
|
@ -18,7 +20,7 @@ from .hon import HonEntity
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
LIGHTS = {
|
LIGHTS: dict[str, tuple[LightEntityDescription, ...]] = {
|
||||||
"WC": (
|
"WC": (
|
||||||
LightEntityDescription(
|
LightEntityDescription(
|
||||||
key="settings.lightStatus",
|
key="settings.lightStatus",
|
||||||
|
@ -43,7 +45,9 @@ LIGHTS = {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
entities = []
|
entities = []
|
||||||
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
||||||
for description in LIGHTS.get(device.appliance_type, []):
|
for description in LIGHTS.get(device.appliance_type, []):
|
||||||
|
@ -61,8 +65,16 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
||||||
class HonLightEntity(HonEntity, LightEntity):
|
class HonLightEntity(HonEntity, LightEntity):
|
||||||
entity_description: LightEntityDescription
|
entity_description: LightEntityDescription
|
||||||
|
|
||||||
def __init__(self, hass, entry, device: HonAppliance, description) -> None:
|
def __init__(
|
||||||
light: HonParameterRange = device.settings.get(description.key)
|
self,
|
||||||
|
hass: HomeAssistantType,
|
||||||
|
entry: ConfigEntry,
|
||||||
|
device: HonAppliance,
|
||||||
|
description: LightEntityDescription,
|
||||||
|
) -> None:
|
||||||
|
light = self._device.settings.get(self.entity_description.key)
|
||||||
|
if not isinstance(light, HonParameterRange):
|
||||||
|
raise ValueError()
|
||||||
self._light_range = (light.min, light.max)
|
self._light_range = (light.min, light.max)
|
||||||
self._attr_supported_color_modes: set[ColorMode] = set()
|
self._attr_supported_color_modes: set[ColorMode] = set()
|
||||||
if len(light.values) == 2:
|
if len(light.values) == 2:
|
||||||
|
@ -76,13 +88,13 @@ class HonLightEntity(HonEntity, LightEntity):
|
||||||
@property
|
@property
|
||||||
def is_on(self) -> bool:
|
def is_on(self) -> bool:
|
||||||
"""Return true if light is on."""
|
"""Return true if light is on."""
|
||||||
return self._device.get(self.entity_description.key.split(".")[-1]) > 0
|
return bool(self._device.get(self.entity_description.key.split(".")[-1]) > 0)
|
||||||
|
|
||||||
async def async_turn_on(self, **kwargs: Any) -> None:
|
async def async_turn_on(self, **kwargs: Any) -> None:
|
||||||
"""Turn on or control the light."""
|
"""Turn on or control the light."""
|
||||||
light: HonParameterRange = self._device.settings.get(
|
light = self._device.settings.get(self.entity_description.key)
|
||||||
self.entity_description.key
|
if not isinstance(light, HonParameterRange):
|
||||||
)
|
raise ValueError()
|
||||||
if ColorMode.BRIGHTNESS in self._attr_supported_color_modes:
|
if ColorMode.BRIGHTNESS in self._attr_supported_color_modes:
|
||||||
percent = int(100 / 255 * kwargs.get(ATTR_BRIGHTNESS, 128))
|
percent = int(100 / 255 * kwargs.get(ATTR_BRIGHTNESS, 128))
|
||||||
light.value = round(light.max / 100 * percent)
|
light.value = round(light.max / 100 * percent)
|
||||||
|
@ -96,9 +108,9 @@ class HonLightEntity(HonEntity, LightEntity):
|
||||||
|
|
||||||
async def async_turn_off(self, **kwargs: Any) -> None:
|
async def async_turn_off(self, **kwargs: Any) -> None:
|
||||||
"""Instruct the light to turn off."""
|
"""Instruct the light to turn off."""
|
||||||
light: HonParameterRange = self._device.settings.get(
|
light = self._device.settings.get(self.entity_description.key)
|
||||||
self.entity_description.key
|
if not isinstance(light, HonParameterRange):
|
||||||
)
|
raise ValueError()
|
||||||
light.value = light.min
|
light.value = light.min
|
||||||
await self._device.commands[self._command].send()
|
await self._device.commands[self._command].send()
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
@ -106,15 +118,15 @@ class HonLightEntity(HonEntity, LightEntity):
|
||||||
@property
|
@property
|
||||||
def brightness(self) -> int | None:
|
def brightness(self) -> int | None:
|
||||||
"""Return the brightness of the light."""
|
"""Return the brightness of the light."""
|
||||||
light: HonParameterRange = self._device.settings.get(
|
light = self._device.settings.get(self.entity_description.key)
|
||||||
self.entity_description.key
|
if not isinstance(light, HonParameterRange):
|
||||||
)
|
raise ValueError()
|
||||||
if light.value == light.min:
|
if light.value == light.min:
|
||||||
return None
|
return None
|
||||||
return int(255 / light.max * light.value)
|
return int(255 / light.max * float(light.value))
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
self._attr_is_on = self.is_on
|
self._attr_is_on = self.is_on
|
||||||
self._attr_brightness = self.brightness
|
self._attr_brightness = self.brightness
|
||||||
if update:
|
if update:
|
||||||
|
@ -122,7 +134,6 @@ class HonLightEntity(HonEntity, LightEntity):
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def available(self) -> bool:
|
def available(self) -> bool:
|
||||||
return (
|
if (entity := self._device.settings.get(self.entity_description.key)) is None:
|
||||||
super().available
|
return False
|
||||||
and len(self._device.settings.get(self.entity_description.key).values) > 1
|
return super().available and len(entity.values) > 1
|
||||||
)
|
|
||||||
|
|
|
@ -4,6 +4,8 @@ from typing import Any
|
||||||
from homeassistant.components.lock import LockEntity, LockEntityDescription
|
from homeassistant.components.lock import LockEntity, LockEntityDescription
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
from pyhon.parameter.base import HonParameter
|
from pyhon.parameter.base import HonParameter
|
||||||
from pyhon.parameter.range import HonParameterRange
|
from pyhon.parameter.range import HonParameterRange
|
||||||
|
|
||||||
|
@ -23,7 +25,9 @@ LOCKS: dict[str, tuple[LockEntityDescription, ...]] = {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
entities = []
|
entities = []
|
||||||
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
||||||
for description in LOCKS.get(device.appliance_type, []):
|
for description in LOCKS.get(device.appliance_type, []):
|
||||||
|
@ -45,13 +49,12 @@ class HonLockEntity(HonEntity, LockEntity):
|
||||||
@property
|
@property
|
||||||
def is_locked(self) -> bool | None:
|
def is_locked(self) -> bool | None:
|
||||||
"""Return a boolean for the state of the lock."""
|
"""Return a boolean for the state of the lock."""
|
||||||
"""Return True if entity is on."""
|
return bool(self._device.get(self.entity_description.key, 0) == 1)
|
||||||
return self._device.get(self.entity_description.key, 0) == 1
|
|
||||||
|
|
||||||
async def async_lock(self, **kwargs: Any) -> None:
|
async def async_lock(self, **kwargs: Any) -> None:
|
||||||
"""Lock method."""
|
"""Lock method."""
|
||||||
setting = self._device.settings[f"settings.{self.entity_description.key}"]
|
setting = self._device.settings.get(f"settings.{self.entity_description.key}")
|
||||||
if type(setting) == HonParameter:
|
if type(setting) == HonParameter or setting is None:
|
||||||
return
|
return
|
||||||
setting.value = setting.max if isinstance(setting, HonParameterRange) else 1
|
setting.value = setting.max if isinstance(setting, HonParameterRange) else 1
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
@ -78,8 +81,7 @@ class HonLockEntity(HonEntity, LockEntity):
|
||||||
)
|
)
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
value = self._device.get(self.entity_description.key, 0)
|
|
||||||
self._attr_is_locked = self.is_locked
|
self._attr_is_locked = self.is_locked
|
||||||
if update:
|
if update:
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
|
@ -10,6 +10,9 @@ from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import UnitOfTime, UnitOfTemperature
|
from homeassistant.const import UnitOfTime, UnitOfTemperature
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
from homeassistant.helpers.entity import EntityCategory
|
from homeassistant.helpers.entity import EntityCategory
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
|
from pyhon.appliance import HonAppliance
|
||||||
from pyhon.parameter.range import HonParameterRange
|
from pyhon.parameter.range import HonParameterRange
|
||||||
|
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
|
@ -183,8 +186,11 @@ NUMBERS: dict[str, tuple[NumberEntityDescription, ...]] = {
|
||||||
NUMBERS["WD"] = unique_entities(NUMBERS["WM"], NUMBERS["TD"])
|
NUMBERS["WD"] = unique_entities(NUMBERS["WM"], NUMBERS["TD"])
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
entities = []
|
entities = []
|
||||||
|
entity: HonNumberEntity | HonConfigNumberEntity
|
||||||
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
||||||
for description in NUMBERS.get(device.appliance_type, []):
|
for description in NUMBERS.get(device.appliance_type, []):
|
||||||
if description.key not in device.available_settings:
|
if description.key not in device.available_settings:
|
||||||
|
@ -203,7 +209,13 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
||||||
class HonNumberEntity(HonEntity, NumberEntity):
|
class HonNumberEntity(HonEntity, NumberEntity):
|
||||||
entity_description: HonNumberEntityDescription
|
entity_description: HonNumberEntityDescription
|
||||||
|
|
||||||
def __init__(self, hass, entry, device, description) -> None:
|
def __init__(
|
||||||
|
self,
|
||||||
|
hass: HomeAssistantType,
|
||||||
|
entry: ConfigEntry,
|
||||||
|
device: HonAppliance,
|
||||||
|
description: HonNumberEntityDescription,
|
||||||
|
) -> None:
|
||||||
super().__init__(hass, entry, device, description)
|
super().__init__(hass, entry, device, description)
|
||||||
|
|
||||||
self._data = device.settings[description.key]
|
self._data = device.settings[description.key]
|
||||||
|
@ -214,7 +226,9 @@ class HonNumberEntity(HonEntity, NumberEntity):
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def native_value(self) -> float | None:
|
def native_value(self) -> float | None:
|
||||||
return self._device.get(self.entity_description.key.split(".")[-1])
|
if value := self._device.get(self.entity_description.key.split(".")[-1]):
|
||||||
|
return float(value)
|
||||||
|
return None
|
||||||
|
|
||||||
async def async_set_native_value(self, value: float) -> None:
|
async def async_set_native_value(self, value: float) -> None:
|
||||||
setting = self._device.settings[self.entity_description.key]
|
setting = self._device.settings[self.entity_description.key]
|
||||||
|
@ -227,7 +241,7 @@ class HonNumberEntity(HonEntity, NumberEntity):
|
||||||
await self.coordinator.async_refresh()
|
await self.coordinator.async_refresh()
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
setting = self._device.settings[self.entity_description.key]
|
setting = self._device.settings[self.entity_description.key]
|
||||||
if isinstance(setting, HonParameterRange):
|
if isinstance(setting, HonParameterRange):
|
||||||
self._attr_native_max_value = setting.max
|
self._attr_native_max_value = setting.max
|
||||||
|
@ -247,14 +261,31 @@ class HonNumberEntity(HonEntity, NumberEntity):
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class HonConfigNumberEntity(HonNumberEntity):
|
class HonConfigNumberEntity(HonEntity, NumberEntity):
|
||||||
entity_description: HonConfigNumberEntityDescription
|
entity_description: HonConfigNumberEntityDescription
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
hass: HomeAssistantType,
|
||||||
|
entry: ConfigEntry,
|
||||||
|
device: HonAppliance,
|
||||||
|
description: HonConfigNumberEntityDescription,
|
||||||
|
) -> None:
|
||||||
|
super().__init__(hass, entry, device, description)
|
||||||
|
|
||||||
|
self._data = device.settings[description.key]
|
||||||
|
if isinstance(self._data, HonParameterRange):
|
||||||
|
self._attr_native_max_value = self._data.max
|
||||||
|
self._attr_native_min_value = self._data.min
|
||||||
|
self._attr_native_step = self._data.step
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def native_value(self) -> float | None:
|
def native_value(self) -> float | None:
|
||||||
return self._device.settings[self.entity_description.key].value
|
if value := self._device.settings[self.entity_description.key].value:
|
||||||
|
return float(value)
|
||||||
|
return None
|
||||||
|
|
||||||
async def async_set_native_value(self, value: str) -> None:
|
async def async_set_native_value(self, value: float) -> None:
|
||||||
setting = self._device.settings[self.entity_description.key]
|
setting = self._device.settings[self.entity_description.key]
|
||||||
if isinstance(setting, HonParameterRange):
|
if isinstance(setting, HonParameterRange):
|
||||||
setting.value = value
|
setting.value = value
|
||||||
|
@ -264,3 +295,14 @@ class HonConfigNumberEntity(HonNumberEntity):
|
||||||
def available(self) -> bool:
|
def available(self) -> bool:
|
||||||
"""Return True if entity is available."""
|
"""Return True if entity is available."""
|
||||||
return super(NumberEntity, self).available
|
return super(NumberEntity, self).available
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
|
setting = self._device.settings[self.entity_description.key]
|
||||||
|
if isinstance(setting, HonParameterRange):
|
||||||
|
self._attr_native_max_value = setting.max
|
||||||
|
self._attr_native_min_value = setting.min
|
||||||
|
self._attr_native_step = setting.step
|
||||||
|
self._attr_native_value = self.native_value
|
||||||
|
if update:
|
||||||
|
self.async_write_ha_state()
|
||||||
|
|
|
@ -2,13 +2,14 @@ from __future__ import annotations
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from typing import Dict, List
|
|
||||||
|
|
||||||
from homeassistant.components.select import SelectEntity, SelectEntityDescription
|
from homeassistant.components.select import SelectEntity, SelectEntityDescription
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import UnitOfTemperature, UnitOfTime, REVOLUTIONS_PER_MINUTE
|
from homeassistant.const import UnitOfTemperature, UnitOfTime, REVOLUTIONS_PER_MINUTE
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
from homeassistant.helpers.entity import EntityCategory
|
from homeassistant.helpers.entity import EntityCategory
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
|
|
||||||
from . import const
|
from . import const
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
|
@ -19,16 +20,16 @@ _LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class HonSelectEntityDescription(SelectEntityDescription):
|
class HonSelectEntityDescription(SelectEntityDescription):
|
||||||
option_list: Dict[int, str] = None
|
option_list: dict[int, str] | None = None
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class HonConfigSelectEntityDescription(SelectEntityDescription):
|
class HonConfigSelectEntityDescription(SelectEntityDescription):
|
||||||
entity_category: EntityCategory = EntityCategory.CONFIG
|
entity_category: EntityCategory = EntityCategory.CONFIG
|
||||||
option_list: Dict[int, str] = None
|
option_list: dict[int, str] | None = None
|
||||||
|
|
||||||
|
|
||||||
SELECTS = {
|
SELECTS: dict[str, tuple[SelectEntityDescription, ...]] = {
|
||||||
"WM": (
|
"WM": (
|
||||||
HonConfigSelectEntityDescription(
|
HonConfigSelectEntityDescription(
|
||||||
key="startProgram.spinSpeed",
|
key="startProgram.spinSpeed",
|
||||||
|
@ -168,8 +169,11 @@ SELECTS = {
|
||||||
SELECTS["WD"] = unique_entities(SELECTS["WM"], SELECTS["TD"])
|
SELECTS["WD"] = unique_entities(SELECTS["WM"], SELECTS["TD"])
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
entities = []
|
entities = []
|
||||||
|
entity: HonSelectEntity | HonConfigSelectEntity
|
||||||
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
||||||
for description in SELECTS.get(device.appliance_type, []):
|
for description in SELECTS.get(device.appliance_type, []):
|
||||||
if description.key not in device.available_settings:
|
if description.key not in device.available_settings:
|
||||||
|
@ -195,16 +199,18 @@ class HonConfigSelectEntity(HonEntity, SelectEntity):
|
||||||
value = get_readable(self.entity_description, setting.value)
|
value = get_readable(self.entity_description, setting.value)
|
||||||
if value not in self._attr_options:
|
if value not in self._attr_options:
|
||||||
return None
|
return None
|
||||||
return value
|
return str(value)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def options(self) -> list[str]:
|
def options(self) -> list[str]:
|
||||||
setting = self._device.settings.get(self.entity_description.key)
|
setting = self._device.settings.get(self.entity_description.key)
|
||||||
if setting is None:
|
if setting is None:
|
||||||
return []
|
return []
|
||||||
return [get_readable(self.entity_description, key) for key in setting.values]
|
return [
|
||||||
|
str(get_readable(self.entity_description, key)) for key in setting.values
|
||||||
|
]
|
||||||
|
|
||||||
def _option_to_number(self, option: str, values: List[str]):
|
def _option_to_number(self, option: str, values: list[str]) -> str:
|
||||||
if (options := self.entity_description.option_list) is not None:
|
if (options := self.entity_description.option_list) is not None:
|
||||||
return str(
|
return str(
|
||||||
next(
|
next(
|
||||||
|
@ -220,7 +226,7 @@ class HonConfigSelectEntity(HonEntity, SelectEntity):
|
||||||
await self.coordinator.async_refresh()
|
await self.coordinator.async_refresh()
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
self._attr_available = self.available
|
self._attr_available = self.available
|
||||||
self._attr_options = self.options
|
self._attr_options = self.options
|
||||||
self._attr_current_option = self.current_option
|
self._attr_current_option = self.current_option
|
||||||
|
@ -233,9 +239,37 @@ class HonConfigSelectEntity(HonEntity, SelectEntity):
|
||||||
return self._device.settings.get(self.entity_description.key) is not None
|
return self._device.settings.get(self.entity_description.key) is not None
|
||||||
|
|
||||||
|
|
||||||
class HonSelectEntity(HonConfigSelectEntity):
|
class HonSelectEntity(HonEntity, SelectEntity):
|
||||||
entity_description: HonSelectEntityDescription
|
entity_description: HonSelectEntityDescription
|
||||||
|
|
||||||
|
@property
|
||||||
|
def current_option(self) -> str | None:
|
||||||
|
if not (setting := self._device.settings.get(self.entity_description.key)):
|
||||||
|
return None
|
||||||
|
value = get_readable(self.entity_description, setting.value)
|
||||||
|
if value not in self._attr_options:
|
||||||
|
return None
|
||||||
|
return str(value)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def options(self) -> list[str]:
|
||||||
|
setting = self._device.settings.get(self.entity_description.key)
|
||||||
|
if setting is None:
|
||||||
|
return []
|
||||||
|
return [
|
||||||
|
str(get_readable(self.entity_description, key)) for key in setting.values
|
||||||
|
]
|
||||||
|
|
||||||
|
def _option_to_number(self, option: str, values: list[str]) -> str:
|
||||||
|
if (options := self.entity_description.option_list) is not None:
|
||||||
|
return str(
|
||||||
|
next(
|
||||||
|
(k for k, v in options.items() if str(k) in values and v == option),
|
||||||
|
option,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return option
|
||||||
|
|
||||||
async def async_select_option(self, option: str) -> None:
|
async def async_select_option(self, option: str) -> None:
|
||||||
setting = self._device.settings[self.entity_description.key]
|
setting = self._device.settings[self.entity_description.key]
|
||||||
setting.value = self._option_to_number(option, setting.values)
|
setting.value = self._option_to_number(option, setting.values)
|
||||||
|
@ -253,3 +287,11 @@ class HonSelectEntity(HonConfigSelectEntity):
|
||||||
and int(self._device.get("remoteCtrValid", 1)) == 1
|
and int(self._device.get("remoteCtrValid", 1)) == 1
|
||||||
and self._device.get("attributes.lastConnEvent.category") != "DISCONNECTED"
|
and self._device.get("attributes.lastConnEvent.category") != "DISCONNECTED"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
|
self._attr_available = self.available
|
||||||
|
self._attr_options = self.options
|
||||||
|
self._attr_current_option = self.current_option
|
||||||
|
if update:
|
||||||
|
self.async_write_ha_state()
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import logging
|
import logging
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from typing import Dict
|
|
||||||
|
|
||||||
from homeassistant.components.sensor import (
|
from homeassistant.components.sensor import (
|
||||||
SensorEntity,
|
SensorEntity,
|
||||||
|
@ -25,6 +24,8 @@ from homeassistant.const import (
|
||||||
)
|
)
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
from homeassistant.helpers.entity import EntityCategory
|
from homeassistant.helpers.entity import EntityCategory
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
|
|
||||||
from . import const
|
from . import const
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
|
@ -36,12 +37,12 @@ _LOGGER = logging.getLogger(__name__)
|
||||||
@dataclass
|
@dataclass
|
||||||
class HonConfigSensorEntityDescription(SensorEntityDescription):
|
class HonConfigSensorEntityDescription(SensorEntityDescription):
|
||||||
entity_category: EntityCategory = EntityCategory.CONFIG
|
entity_category: EntityCategory = EntityCategory.CONFIG
|
||||||
option_list: Dict[int, str] = None
|
option_list: dict[int, str] | None = None
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class HonSensorEntityDescription(SensorEntityDescription):
|
class HonSensorEntityDescription(SensorEntityDescription):
|
||||||
option_list: Dict[int, str] = None
|
option_list: dict[int, str] | None = None
|
||||||
|
|
||||||
|
|
||||||
SENSORS: dict[str, tuple[SensorEntityDescription, ...]] = {
|
SENSORS: dict[str, tuple[SensorEntityDescription, ...]] = {
|
||||||
|
@ -775,8 +776,11 @@ SENSORS: dict[str, tuple[SensorEntityDescription, ...]] = {
|
||||||
SENSORS["WD"] = unique_entities(SENSORS["WM"], SENSORS["TD"])
|
SENSORS["WD"] = unique_entities(SENSORS["WM"], SENSORS["TD"])
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
entities = []
|
entities = []
|
||||||
|
entity: HonSensorEntity | HonConfigSensorEntity
|
||||||
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
||||||
for description in SENSORS.get(device.appliance_type, []):
|
for description in SENSORS.get(device.appliance_type, []):
|
||||||
if isinstance(description, HonSensorEntityDescription):
|
if isinstance(description, HonSensorEntityDescription):
|
||||||
|
@ -799,15 +803,15 @@ class HonSensorEntity(HonEntity, SensorEntity):
|
||||||
entity_description: HonSensorEntityDescription
|
entity_description: HonSensorEntityDescription
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
value = self._device.get(self.entity_description.key, "")
|
value = self._device.get(self.entity_description.key, "")
|
||||||
if self.entity_description.key == "programName":
|
if self.entity_description.key == "programName":
|
||||||
self._attr_options = self._device.settings.get(
|
if not (options := self._device.settings.get("startProgram.program")):
|
||||||
"startProgram.program"
|
raise ValueError
|
||||||
).values + ["No Program"]
|
self._attr_options = options.values + ["No Program"]
|
||||||
elif self.entity_description.option_list is not None:
|
elif self.entity_description.option_list is not None:
|
||||||
self._attr_options = list(self.entity_description.option_list.values())
|
self._attr_options = list(self.entity_description.option_list.values())
|
||||||
value = get_readable(self.entity_description, value)
|
value = str(get_readable(self.entity_description, value))
|
||||||
if not value and self.entity_description.state_class is not None:
|
if not value and self.entity_description.state_class is not None:
|
||||||
self._attr_native_value = 0
|
self._attr_native_value = 0
|
||||||
self._attr_native_value = value
|
self._attr_native_value = value
|
||||||
|
@ -819,17 +823,22 @@ class HonConfigSensorEntity(HonEntity, SensorEntity):
|
||||||
entity_description: HonConfigSensorEntityDescription
|
entity_description: HonConfigSensorEntityDescription
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
value = self._device.settings.get(self.entity_description.key, None)
|
sensor = self._device.settings.get(self.entity_description.key, None)
|
||||||
|
value: float | str
|
||||||
if self.entity_description.state_class is not None:
|
if self.entity_description.state_class is not None:
|
||||||
if value and value.value:
|
if sensor and sensor.value:
|
||||||
value = (
|
value = (
|
||||||
float(value.value) if "." in str(value.value) else int(value.value)
|
float(sensor.value)
|
||||||
|
if "." in str(sensor.value)
|
||||||
|
else int(sensor.value)
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
value = 0
|
value = 0
|
||||||
|
elif sensor is not None:
|
||||||
|
value = sensor.value
|
||||||
else:
|
else:
|
||||||
value = value.value
|
value = 0
|
||||||
if self.entity_description.option_list is not None and not value == 0:
|
if self.entity_description.option_list is not None and not value == 0:
|
||||||
self._attr_options = list(self.entity_description.option_list.values())
|
self._attr_options = list(self.entity_description.option_list.values())
|
||||||
value = get_readable(self.entity_description, value)
|
value = get_readable(self.entity_description, value)
|
||||||
|
|
|
@ -7,6 +7,8 @@ from homeassistant.components.switch import SwitchEntityDescription, SwitchEntit
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
from homeassistant.helpers.entity import EntityCategory
|
from homeassistant.helpers.entity import EntityCategory
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
from homeassistant.helpers.typing import HomeAssistantType
|
||||||
from pyhon.parameter.base import HonParameter
|
from pyhon.parameter.base import HonParameter
|
||||||
from pyhon.parameter.range import HonParameterRange
|
from pyhon.parameter.range import HonParameterRange
|
||||||
|
|
||||||
|
@ -17,18 +19,11 @@ _LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class HonSwitchEntityDescriptionMixin:
|
class HonControlSwitchEntityDescription(SwitchEntityDescription):
|
||||||
turn_on_key: str = ""
|
turn_on_key: str = ""
|
||||||
turn_off_key: str = ""
|
turn_off_key: str = ""
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
|
||||||
class HonControlSwitchEntityDescription(
|
|
||||||
HonSwitchEntityDescriptionMixin, SwitchEntityDescription
|
|
||||||
):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class HonSwitchEntityDescription(SwitchEntityDescription):
|
class HonSwitchEntityDescription(SwitchEntityDescription):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@ -38,7 +33,7 @@ class HonConfigSwitchEntityDescription(SwitchEntityDescription):
|
||||||
entity_category: EntityCategory = EntityCategory.CONFIG
|
entity_category: EntityCategory = EntityCategory.CONFIG
|
||||||
|
|
||||||
|
|
||||||
SWITCHES: dict[str, tuple[HonSwitchEntityDescription, ...]] = {
|
SWITCHES: dict[str, tuple[SwitchEntityDescription, ...]] = {
|
||||||
"WM": (
|
"WM": (
|
||||||
HonControlSwitchEntityDescription(
|
HonControlSwitchEntityDescription(
|
||||||
key="active",
|
key="active",
|
||||||
|
@ -355,8 +350,11 @@ SWITCHES["WD"] = unique_entities(SWITCHES["WD"], SWITCHES["WM"])
|
||||||
SWITCHES["WD"] = unique_entities(SWITCHES["WD"], SWITCHES["TD"])
|
SWITCHES["WD"] = unique_entities(SWITCHES["WD"], SWITCHES["TD"])
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
entities = []
|
entities = []
|
||||||
|
entity: HonConfigSwitchEntity | HonControlSwitchEntity | HonSwitchEntity
|
||||||
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
for device in hass.data[DOMAIN][entry.unique_id].appliances:
|
||||||
for description in SWITCHES.get(device.appliance_type, []):
|
for description in SWITCHES.get(device.appliance_type, []):
|
||||||
if isinstance(description, HonConfigSwitchEntityDescription):
|
if isinstance(description, HonConfigSwitchEntityDescription):
|
||||||
|
@ -427,7 +425,7 @@ class HonSwitchEntity(HonEntity, SwitchEntity):
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
self._attr_is_on = self.is_on
|
self._attr_is_on = self.is_on
|
||||||
if update:
|
if update:
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
@ -507,7 +505,7 @@ class HonConfigSwitchEntity(HonEntity, SwitchEntity):
|
||||||
await self.coordinator.async_refresh()
|
await self.coordinator.async_refresh()
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self, update=True) -> None:
|
def _handle_coordinator_update(self, update: bool = True) -> None:
|
||||||
self._attr_is_on = self.is_on
|
self._attr_is_on = self.is_on
|
||||||
if update:
|
if update:
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
95
custom_components/hon/typedefs.py
Normal file
95
custom_components/hon/typedefs.py
Normal file
|
@ -0,0 +1,95 @@
|
||||||
|
from typing import Union, TypeVar
|
||||||
|
|
||||||
|
from homeassistant.components.button import ButtonEntityDescription
|
||||||
|
from homeassistant.components.fan import FanEntityDescription
|
||||||
|
from homeassistant.components.light import LightEntityDescription
|
||||||
|
from homeassistant.components.lock import LockEntityDescription
|
||||||
|
from homeassistant.components.number import NumberEntityDescription
|
||||||
|
from homeassistant.components.select import SelectEntityDescription
|
||||||
|
from homeassistant.components.sensor import SensorEntityDescription
|
||||||
|
from homeassistant.components.switch import SwitchEntityDescription
|
||||||
|
|
||||||
|
from .binary_sensor import HonBinarySensorEntityDescription
|
||||||
|
from .button import HonButtonEntity, HonDataArchive, HonDeviceInfo
|
||||||
|
from .climate import (
|
||||||
|
HonACClimateEntityDescription,
|
||||||
|
HonClimateEntityDescription,
|
||||||
|
)
|
||||||
|
from .number import (
|
||||||
|
HonConfigNumberEntityDescription,
|
||||||
|
HonNumberEntityDescription,
|
||||||
|
)
|
||||||
|
from .select import (
|
||||||
|
HonConfigSelectEntityDescription,
|
||||||
|
HonSelectEntityDescription,
|
||||||
|
)
|
||||||
|
from .sensor import (
|
||||||
|
HonSensorEntityDescription,
|
||||||
|
HonConfigSensorEntityDescription,
|
||||||
|
)
|
||||||
|
from .switch import (
|
||||||
|
HonControlSwitchEntityDescription,
|
||||||
|
HonSwitchEntityDescription,
|
||||||
|
HonConfigSwitchEntityDescription,
|
||||||
|
)
|
||||||
|
|
||||||
|
HonButtonType = Union[
|
||||||
|
HonButtonEntity,
|
||||||
|
HonDataArchive,
|
||||||
|
HonDeviceInfo,
|
||||||
|
]
|
||||||
|
|
||||||
|
HonEntityDescription = Union[
|
||||||
|
HonBinarySensorEntityDescription,
|
||||||
|
HonControlSwitchEntityDescription,
|
||||||
|
HonSwitchEntityDescription,
|
||||||
|
HonConfigSwitchEntityDescription,
|
||||||
|
HonSensorEntityDescription,
|
||||||
|
HonConfigSelectEntityDescription,
|
||||||
|
HonConfigNumberEntityDescription,
|
||||||
|
HonACClimateEntityDescription,
|
||||||
|
HonClimateEntityDescription,
|
||||||
|
HonNumberEntityDescription,
|
||||||
|
HonSelectEntityDescription,
|
||||||
|
HonConfigSensorEntityDescription,
|
||||||
|
FanEntityDescription,
|
||||||
|
LightEntityDescription,
|
||||||
|
LockEntityDescription,
|
||||||
|
ButtonEntityDescription,
|
||||||
|
SwitchEntityDescription,
|
||||||
|
SensorEntityDescription,
|
||||||
|
SelectEntityDescription,
|
||||||
|
NumberEntityDescription,
|
||||||
|
]
|
||||||
|
|
||||||
|
HonOptionEntityDescription = Union[
|
||||||
|
HonConfigSelectEntityDescription,
|
||||||
|
HonSelectEntityDescription,
|
||||||
|
HonConfigSensorEntityDescription,
|
||||||
|
HonSensorEntityDescription,
|
||||||
|
]
|
||||||
|
|
||||||
|
T = TypeVar(
|
||||||
|
"T",
|
||||||
|
HonBinarySensorEntityDescription,
|
||||||
|
HonControlSwitchEntityDescription,
|
||||||
|
HonSwitchEntityDescription,
|
||||||
|
HonConfigSwitchEntityDescription,
|
||||||
|
HonSensorEntityDescription,
|
||||||
|
HonConfigSelectEntityDescription,
|
||||||
|
HonConfigNumberEntityDescription,
|
||||||
|
HonACClimateEntityDescription,
|
||||||
|
HonClimateEntityDescription,
|
||||||
|
HonNumberEntityDescription,
|
||||||
|
HonSelectEntityDescription,
|
||||||
|
HonConfigSensorEntityDescription,
|
||||||
|
FanEntityDescription,
|
||||||
|
LightEntityDescription,
|
||||||
|
LockEntityDescription,
|
||||||
|
ButtonEntityDescription,
|
||||||
|
SwitchEntityDescription,
|
||||||
|
SensorEntityDescription,
|
||||||
|
SelectEntityDescription,
|
||||||
|
NumberEntityDescription,
|
||||||
|
)
|
||||||
|
|
25
mypy.ini
Normal file
25
mypy.ini
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
[mypy]
|
||||||
|
check_untyped_defs = true
|
||||||
|
disallow_any_generics = true
|
||||||
|
disallow_incomplete_defs = true
|
||||||
|
disallow_untyped_calls = true
|
||||||
|
disallow_untyped_decorators = true
|
||||||
|
disallow_untyped_defs = true
|
||||||
|
disable_error_code = annotation-unchecked
|
||||||
|
enable_error_code = ignore-without-code, redundant-self, truthy-iterable
|
||||||
|
follow_imports = silent
|
||||||
|
local_partial_types = true
|
||||||
|
no_implicit_optional = true
|
||||||
|
no_implicit_reexport = true
|
||||||
|
show_error_codes = true
|
||||||
|
strict_concatenate = false
|
||||||
|
strict_equality = true
|
||||||
|
warn_incomplete_stub = true
|
||||||
|
warn_redundant_casts = true
|
||||||
|
warn_return_any = true
|
||||||
|
warn_unreachable = true
|
||||||
|
warn_unused_configs = true
|
||||||
|
warn_unused_ignores = true
|
||||||
|
|
||||||
|
[mypy-homeassistant.*]
|
||||||
|
implicit_reexport = True
|
2
requirements.txt
Normal file
2
requirements.txt
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
pyhOn
|
||||||
|
homeassistant
|
|
@ -1,3 +1,4 @@
|
||||||
pyhOn
|
|
||||||
black
|
black
|
||||||
homeassistant
|
flake8
|
||||||
|
mypy
|
||||||
|
pylint
|
||||||
|
|
Loading…
Reference in a new issue