mirror of
https://github.com/home-assistant/core.git
synced 2025-04-25 09:47:52 +00:00
Add sensors to venstar integration (#58851)
This commit is contained in:
parent
766c889e70
commit
3639481027
@ -1178,6 +1178,7 @@ omit =
|
||||
homeassistant/components/venstar/__init__.py
|
||||
homeassistant/components/venstar/binary_sensor.py
|
||||
homeassistant/components/venstar/climate.py
|
||||
homeassistant/components/venstar/sensor.py
|
||||
homeassistant/components/verisure/__init__.py
|
||||
homeassistant/components/verisure/alarm_control_panel.py
|
||||
homeassistant/components/verisure/binary_sensor.py
|
||||
|
@ -19,7 +19,7 @@ from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
||||
|
||||
from .const import _LOGGER, DOMAIN, VENSTAR_TIMEOUT
|
||||
|
||||
PLATFORMS = ["binary_sensor", "climate"]
|
||||
PLATFORMS = ["binary_sensor", "climate", "sensor"]
|
||||
|
||||
|
||||
async def async_setup_entry(hass, config):
|
||||
@ -88,7 +88,7 @@ class VenstarDataUpdateCoordinator(update_coordinator.DataUpdateCoordinator):
|
||||
) from ex
|
||||
|
||||
# older venstars sometimes cannot handle rapid sequential connections
|
||||
await asyncio.sleep(3)
|
||||
await asyncio.sleep(1)
|
||||
|
||||
try:
|
||||
await self.hass.async_add_executor_job(self.client.update_sensors)
|
||||
@ -98,7 +98,7 @@ class VenstarDataUpdateCoordinator(update_coordinator.DataUpdateCoordinator):
|
||||
) from ex
|
||||
|
||||
# older venstars sometimes cannot handle rapid sequential connections
|
||||
await asyncio.sleep(3)
|
||||
await asyncio.sleep(1)
|
||||
|
||||
try:
|
||||
await self.hass.async_add_executor_job(self.client.update_alerts)
|
||||
@ -129,16 +129,6 @@ class VenstarEntity(CoordinatorEntity):
|
||||
"""Handle updated data from the coordinator."""
|
||||
self.async_write_ha_state()
|
||||
|
||||
@property
|
||||
def name(self):
|
||||
"""Return the name of the thermostat."""
|
||||
return self._client.name
|
||||
|
||||
@property
|
||||
def unique_id(self):
|
||||
"""Set unique_id for this entity."""
|
||||
return f"{self._config.entry_id}"
|
||||
|
||||
@property
|
||||
def device_info(self):
|
||||
"""Return the device information for this entity."""
|
||||
|
@ -120,6 +120,8 @@ class VenstarThermostat(VenstarEntity, ClimateEntity):
|
||||
HVAC_MODE_COOL: self._client.MODE_COOL,
|
||||
HVAC_MODE_AUTO: self._client.MODE_AUTO,
|
||||
}
|
||||
self._attr_unique_id = config.entry_id
|
||||
self._attr_name = self._client.name
|
||||
|
||||
@property
|
||||
def supported_features(self):
|
||||
|
135
homeassistant/components/venstar/sensor.py
Normal file
135
homeassistant/components/venstar/sensor.py
Normal file
@ -0,0 +1,135 @@
|
||||
"""Representation of Venstar sensors."""
|
||||
from __future__ import annotations
|
||||
|
||||
from dataclasses import dataclass
|
||||
|
||||
from homeassistant.components.sensor import (
|
||||
DEVICE_CLASS_HUMIDITY,
|
||||
DEVICE_CLASS_TEMPERATURE,
|
||||
STATE_CLASS_MEASUREMENT,
|
||||
SensorEntity,
|
||||
SensorEntityDescription,
|
||||
)
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import PERCENTAGE, TEMP_CELSIUS, TEMP_FAHRENHEIT
|
||||
from homeassistant.helpers.entity import Entity
|
||||
|
||||
from . import VenstarDataUpdateCoordinator, VenstarEntity
|
||||
from .const import DOMAIN
|
||||
|
||||
|
||||
@dataclass
|
||||
class VenstarSensorTypeMixin:
|
||||
"""Mixin for sensor required keys."""
|
||||
|
||||
cls: type[VenstarSensor]
|
||||
stype: str
|
||||
|
||||
|
||||
@dataclass
|
||||
class VenstarSensorEntityDescription(SensorEntityDescription, VenstarSensorTypeMixin):
|
||||
"""Base description of a Sensor entity."""
|
||||
|
||||
|
||||
async def async_setup_entry(hass, config_entry, async_add_entities) -> None:
|
||||
"""Set up Vensar device binary_sensors based on a config entry."""
|
||||
coordinator = hass.data[DOMAIN][config_entry.entry_id]
|
||||
entities: list[Entity] = []
|
||||
|
||||
sensors = coordinator.client.get_sensor_list()
|
||||
if not sensors:
|
||||
return
|
||||
|
||||
entities = []
|
||||
|
||||
for sensor_name in sensors:
|
||||
entities.extend(
|
||||
[
|
||||
description.cls(coordinator, config_entry, description, sensor_name)
|
||||
for description in SENSOR_ENTITIES
|
||||
if coordinator.client.get_sensor(sensor_name, description.stype)
|
||||
is not None
|
||||
]
|
||||
)
|
||||
|
||||
async_add_entities(entities)
|
||||
|
||||
|
||||
class VenstarSensor(VenstarEntity, SensorEntity):
|
||||
"""Base class for a Venstar sensor."""
|
||||
|
||||
entity_description: VenstarSensorEntityDescription
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
coordinator: VenstarDataUpdateCoordinator,
|
||||
config: ConfigEntry,
|
||||
entity_description: VenstarSensorEntityDescription,
|
||||
sensor_name: str,
|
||||
) -> None:
|
||||
"""Initialize the sensor."""
|
||||
super().__init__(coordinator, config)
|
||||
self.entity_description = entity_description
|
||||
self.sensor_name = sensor_name
|
||||
self._config = config
|
||||
|
||||
@property
|
||||
def unique_id(self):
|
||||
"""Return the unique id."""
|
||||
return f"{self._config.entry_id}_{self.sensor_name.replace(' ', '_')}_{self.entity_description.key}"
|
||||
|
||||
|
||||
class VenstarHumiditySensor(VenstarSensor):
|
||||
"""Represent a Venstar humidity sensor."""
|
||||
|
||||
@property
|
||||
def name(self):
|
||||
"""Return the name of the device."""
|
||||
return f"{self._client.name} {self.sensor_name} Humidity"
|
||||
|
||||
@property
|
||||
def native_value(self) -> int:
|
||||
"""Return state of the sensor."""
|
||||
return self._client.get_sensor(self.sensor_name, "hum")
|
||||
|
||||
|
||||
class VenstarTemperatureSensor(VenstarSensor):
|
||||
"""Represent a Venstar temperature sensor."""
|
||||
|
||||
@property
|
||||
def name(self):
|
||||
"""Return the name of the device."""
|
||||
return (
|
||||
f"{self._client.name} {self.sensor_name.replace(' Temp', '')} Temperature"
|
||||
)
|
||||
|
||||
@property
|
||||
def native_unit_of_measurement(self) -> str:
|
||||
"""Return unit of measurement the value is expressed in."""
|
||||
if self._client.tempunits == self._client.TEMPUNITS_F:
|
||||
return TEMP_FAHRENHEIT
|
||||
return TEMP_CELSIUS
|
||||
|
||||
@property
|
||||
def native_value(self) -> float:
|
||||
"""Return state of the sensor."""
|
||||
return round(float(self._client.get_sensor(self.sensor_name, "temp")), 1)
|
||||
|
||||
|
||||
SENSOR_ENTITIES: tuple[VenstarSensorEntityDescription, ...] = (
|
||||
VenstarSensorEntityDescription(
|
||||
key="humidity",
|
||||
device_class=DEVICE_CLASS_HUMIDITY,
|
||||
state_class=STATE_CLASS_MEASUREMENT,
|
||||
native_unit_of_measurement=PERCENTAGE,
|
||||
cls=VenstarHumiditySensor,
|
||||
stype="hum",
|
||||
),
|
||||
VenstarSensorEntityDescription(
|
||||
key="temperature",
|
||||
device_class=DEVICE_CLASS_TEMPERATURE,
|
||||
state_class=STATE_CLASS_MEASUREMENT,
|
||||
cls=VenstarTemperatureSensor,
|
||||
stype="temp",
|
||||
),
|
||||
)
|
1
tests/components/venstar/fixtures/colortouch_alerts.json
Normal file
1
tests/components/venstar/fixtures/colortouch_alerts.json
Normal file
@ -0,0 +1 @@
|
||||
{"alerts":[{"name":"Air Filter","active":true},{"name":"UV Lamp","active":false},{"name":"Service","active":false}]}
|
1
tests/components/venstar/fixtures/t2k_alerts.json
Normal file
1
tests/components/venstar/fixtures/t2k_alerts.json
Normal file
@ -0,0 +1 @@
|
||||
{"alerts":[{"name":"Air Filter","active":true},{"name":"UV Lamp","active":false},{"name":"Service","active":false}]}
|
@ -1,5 +1,7 @@
|
||||
"""The climate tests for the venstar integration."""
|
||||
|
||||
from unittest.mock import patch
|
||||
|
||||
from homeassistant.components.climate.const import (
|
||||
SUPPORT_FAN_MODE,
|
||||
SUPPORT_PRESET_MODE,
|
||||
@ -18,7 +20,8 @@ EXPECTED_BASE_SUPPORTED_FEATURES = (
|
||||
async def test_colortouch(hass):
|
||||
"""Test interfacing with a venstar colortouch with attached humidifier."""
|
||||
|
||||
await async_init_integration(hass)
|
||||
with patch("homeassistant.components.onewire.sensor.asyncio.sleep"):
|
||||
await async_init_integration(hass)
|
||||
|
||||
state = hass.states.get("climate.colortouch")
|
||||
assert state.state == "heat"
|
||||
@ -53,7 +56,8 @@ async def test_colortouch(hass):
|
||||
async def test_t2000(hass):
|
||||
"""Test interfacing with a venstar T2000 presently turned off."""
|
||||
|
||||
await async_init_integration(hass)
|
||||
with patch("homeassistant.components.onewire.sensor.asyncio.sleep"):
|
||||
await async_init_integration(hass)
|
||||
|
||||
state = hass.states.get("climate.t2000")
|
||||
assert state.state == "off"
|
||||
|
@ -36,6 +36,8 @@ async def test_setup_entry(hass: HomeAssistant):
|
||||
), patch(
|
||||
"homeassistant.components.venstar.VenstarColorTouch.update_alerts",
|
||||
new=VenstarColorTouchMock.update_alerts,
|
||||
), patch(
|
||||
"homeassistant.components.onewire.sensor.asyncio.sleep"
|
||||
):
|
||||
await hass.config_entries.async_setup(config_entry.entry_id)
|
||||
await hass.async_block_till_done()
|
||||
|
@ -33,6 +33,10 @@ def mock_venstar_devices(f):
|
||||
f"http://venstar-{model}.localdomain/query/sensors",
|
||||
text=load_fixture(f"venstar/{model}_sensors.json"),
|
||||
)
|
||||
m.get(
|
||||
f"http://venstar-{model}.localdomain/query/alerts",
|
||||
text=load_fixture(f"venstar/{model}_alerts.json"),
|
||||
)
|
||||
return await f(hass)
|
||||
|
||||
return wrapper
|
||||
|
Loading…
x
Reference in New Issue
Block a user