hass-core/homeassistant/components/renson/sensor.py
jimmyd-be 3d678f5b99
Renson integration (#56374)
* Implemented Renson integration

* -  renamed component to a better name
- Made cleaner code by splitting up files into different one
-  Fixed issues regarding getting data from library
- Added service.yaml file

* Added Renson services

* cleanup translations

* added config_flow tests

* changed config_flow, removed all services

* use SensorEntityDescription + introduced new binarySensor

* fixed config_flow test

* renamed renson_endura_delta to renson

* refactored sensors and implemented binary_sensor

* Changed some sensors to non measurement and added entity_registery_enabled_default for config sensors

* Enabled binary_sensor

* changed import to new renamed module

* Merge files into correct files + cleaned some code

* Change use of EntityDescription

* Update codeowners

* Fixed lint issues

* Fix sensor

* Create test.yml

* Update test.yml

* add github action tests

* Format json files

* Remove deprecated code

* Update homeassistant/components/renson/binary_sensor.py

Co-authored-by: Paulus Schoutsen <paulus@home-assistant.io>

* Use Coordinqte in Sensor

* Migrated binary sensor to use coordinate

* Removed firmwareSensor

* Add entity_catogory to binary_sensor

* Add services

* Revert "Add services"

This reverts commit 028760d8d8.

* update requirements of Renson integration

* Add services and fan

* Fixed some issue + fixed PR comments

* Cleanup code

* Go back 2 years ago to the bare minimum for PR approval

* Refactored code and added a lot of device classes to the entities

* Fix some bugs

* Add unique Id and some device class

* Show the level value for CURRENT_LEVEL_FIELD instead of the raw data

* Remove FILTER_PRESET_FIELD for now

* Make the _attr_unique_id unique

* Changed Renson tests

* Moved Renson hass data into @dataclass

* Changed test + added files to .coveragerc

* Add device_class=SensorDeviceClass.Duration

* Fix syntax

---------

Co-authored-by: Paulus Schoutsen <paulus@home-assistant.io>
Co-authored-by: Martin Hjelmare <marhje52@gmail.com>
2023-06-10 09:21:33 +02:00

317 lines
9.7 KiB
Python

"""Sensor data of the Renson ventilation unit."""
from __future__ import annotations
from dataclasses import dataclass
from renson_endura_delta.field_enum import (
AIR_QUALITY_FIELD,
BREEZE_LEVEL_FIELD,
BREEZE_TEMPERATURE_FIELD,
BYPASS_LEVEL_FIELD,
BYPASS_TEMPERATURE_FIELD,
CO2_FIELD,
CO2_HYSTERESIS_FIELD,
CO2_QUALITY_FIELD,
CO2_THRESHOLD_FIELD,
CURRENT_AIRFLOW_EXTRACT_FIELD,
CURRENT_AIRFLOW_INGOING_FIELD,
CURRENT_LEVEL_FIELD,
DAY_POLLUTION_FIELD,
DAYTIME_FIELD,
FILTER_REMAIN_FIELD,
HUMIDITY_FIELD,
INDOOR_TEMP_FIELD,
MANUAL_LEVEL_FIELD,
NIGHT_POLLUTION_FIELD,
NIGHTTIME_FIELD,
OUTDOOR_TEMP_FIELD,
FieldEnum,
)
from renson_endura_delta.renson import RensonVentilation
from homeassistant.components.sensor import (
SensorDeviceClass,
SensorEntity,
SensorEntityDescription,
SensorStateClass,
)
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import (
CONCENTRATION_PARTS_PER_MILLION,
PERCENTAGE,
UnitOfTemperature,
UnitOfTime,
UnitOfVolumeFlowRate,
)
from homeassistant.core import HomeAssistant, callback
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from . import RensonCoordinator
from .const import DOMAIN
from .entity import RensonEntity
@dataclass
class RensonSensorEntityDescriptionMixin:
"""Mixin for required keys."""
field: FieldEnum
raw_format: bool
@dataclass
class RensonSensorEntityDescription(
SensorEntityDescription, RensonSensorEntityDescriptionMixin
):
"""Description of sensor."""
SENSORS: tuple[RensonSensorEntityDescription, ...] = (
RensonSensorEntityDescription(
key="CO2_QUALITY_FIELD",
name="CO2 quality category",
field=CO2_QUALITY_FIELD,
raw_format=False,
device_class=SensorDeviceClass.ENUM,
options=["good", "bad", "poor"],
),
RensonSensorEntityDescription(
key="AIR_QUALITY_FIELD",
name="Air quality category",
field=AIR_QUALITY_FIELD,
raw_format=False,
device_class=SensorDeviceClass.ENUM,
options=["good", "bad", "poor"],
),
RensonSensorEntityDescription(
key="CO2_FIELD",
name="CO2 quality",
field=CO2_FIELD,
raw_format=True,
state_class=SensorStateClass.MEASUREMENT,
device_class=SensorDeviceClass.CO2,
native_unit_of_measurement=CONCENTRATION_PARTS_PER_MILLION,
),
RensonSensorEntityDescription(
key="AIR_FIELD",
name="Air quality",
field=AIR_QUALITY_FIELD,
state_class=SensorStateClass.MEASUREMENT,
raw_format=True,
native_unit_of_measurement=CONCENTRATION_PARTS_PER_MILLION,
),
RensonSensorEntityDescription(
key="CURRENT_LEVEL_FIELD",
name="Ventilation level",
field=CURRENT_LEVEL_FIELD,
raw_format=False,
device_class=SensorDeviceClass.ENUM,
options=["Off", "Level1", "Level2", "Level3", "Level4", "Breeze", "Holiday"],
),
RensonSensorEntityDescription(
key="CURRENT_AIRFLOW_EXTRACT_FIELD",
name="Total airflow out",
field=CURRENT_AIRFLOW_EXTRACT_FIELD,
raw_format=False,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=UnitOfVolumeFlowRate.CUBIC_METERS_PER_HOUR,
),
RensonSensorEntityDescription(
key="CURRENT_AIRFLOW_INGOING_FIELD",
name="Total airflow in",
field=CURRENT_AIRFLOW_INGOING_FIELD,
raw_format=False,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=UnitOfVolumeFlowRate.CUBIC_METERS_PER_HOUR,
),
RensonSensorEntityDescription(
key="OUTDOOR_TEMP_FIELD",
name="Outdoor air temperature",
field=OUTDOOR_TEMP_FIELD,
raw_format=False,
device_class=SensorDeviceClass.TEMPERATURE,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
),
RensonSensorEntityDescription(
key="INDOOR_TEMP_FIELD",
name="Extract air temperature",
field=INDOOR_TEMP_FIELD,
raw_format=False,
device_class=SensorDeviceClass.TEMPERATURE,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
),
RensonSensorEntityDescription(
key="FILTER_REMAIN_FIELD",
name="Filter change",
field=FILTER_REMAIN_FIELD,
raw_format=False,
device_class=SensorDeviceClass.DURATION,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=UnitOfTime.DAYS,
),
RensonSensorEntityDescription(
key="HUMIDITY_FIELD",
name="Relative humidity",
field=HUMIDITY_FIELD,
raw_format=False,
device_class=SensorDeviceClass.HUMIDITY,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=PERCENTAGE,
),
RensonSensorEntityDescription(
key="MANUAL_LEVEL_FIELD",
name="Manual level",
field=MANUAL_LEVEL_FIELD,
raw_format=False,
device_class=SensorDeviceClass.ENUM,
options=["Off", "Level1", "Level2", "Level3", "Level4", "Breeze", "Holiday"],
),
RensonSensorEntityDescription(
key="BREEZE_TEMPERATURE_FIELD",
name="Breeze temperature",
field=BREEZE_TEMPERATURE_FIELD,
raw_format=False,
device_class=SensorDeviceClass.TEMPERATURE,
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
entity_registry_enabled_default=False,
),
RensonSensorEntityDescription(
key="BREEZE_LEVEL_FIELD",
name="Breeze level",
field=BREEZE_LEVEL_FIELD,
raw_format=False,
entity_registry_enabled_default=False,
device_class=SensorDeviceClass.ENUM,
options=["Off", "Level1", "Level2", "Level3", "Level4", "Breeze"],
),
RensonSensorEntityDescription(
key="DAYTIME_FIELD",
name="Start day time",
field=DAYTIME_FIELD,
raw_format=False,
entity_registry_enabled_default=False,
),
RensonSensorEntityDescription(
key="NIGHTTIME_FIELD",
name="Start night time",
field=NIGHTTIME_FIELD,
raw_format=False,
entity_registry_enabled_default=False,
),
RensonSensorEntityDescription(
key="DAY_POLLUTION_FIELD",
name="Day pollution level",
field=DAY_POLLUTION_FIELD,
raw_format=False,
entity_registry_enabled_default=False,
device_class=SensorDeviceClass.ENUM,
options=[
"Level1",
"Level2",
"Level3",
"Level4",
],
),
RensonSensorEntityDescription(
key="NIGHT_POLLUTION_FIELD",
name="Night pollution level",
field=NIGHT_POLLUTION_FIELD,
raw_format=False,
entity_registry_enabled_default=False,
device_class=SensorDeviceClass.ENUM,
options=[
"Level1",
"Level2",
"Level3",
"Level4",
],
),
RensonSensorEntityDescription(
key="CO2_THRESHOLD_FIELD",
name="CO2 threshold",
field=CO2_THRESHOLD_FIELD,
raw_format=False,
native_unit_of_measurement=CONCENTRATION_PARTS_PER_MILLION,
entity_registry_enabled_default=False,
),
RensonSensorEntityDescription(
key="CO2_HYSTERESIS_FIELD",
name="CO2 hysteresis",
field=CO2_HYSTERESIS_FIELD,
raw_format=False,
native_unit_of_measurement=CONCENTRATION_PARTS_PER_MILLION,
entity_registry_enabled_default=False,
),
RensonSensorEntityDescription(
key="BYPASS_TEMPERATURE_FIELD",
name="Bypass activation temperature",
field=BYPASS_TEMPERATURE_FIELD,
raw_format=False,
device_class=SensorDeviceClass.TEMPERATURE,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
),
RensonSensorEntityDescription(
key="BYPASS_LEVEL_FIELD",
name="Bypass level",
field=BYPASS_LEVEL_FIELD,
raw_format=False,
device_class=SensorDeviceClass.POWER_FACTOR,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=PERCENTAGE,
),
)
class RensonSensor(RensonEntity, SensorEntity):
"""Get a sensor data from the Renson API and store it in the state of the class."""
def __init__(
self,
description: RensonSensorEntityDescription,
api: RensonVentilation,
coordinator: RensonCoordinator,
) -> None:
"""Initialize class."""
super().__init__(description.key, api, coordinator)
self.field = description.field
self.entity_description = description
self.data_type = description.field.field_type
self.raw_format = description.raw_format
@callback
def _handle_coordinator_update(self) -> None:
"""Handle updated data from the coordinator."""
all_data = self.coordinator.data
value = self.api.get_field_value(all_data, self.field.name)
if self.raw_format:
self._attr_native_value = value
else:
self._attr_native_value = self.api.parse_value(value, self.data_type)
self.async_write_ha_state()
async def async_setup_entry(
hass: HomeAssistant,
config_entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up the Renson sensor platform."""
api: RensonVentilation = hass.data[DOMAIN][config_entry.entry_id].api
coordinator: RensonCoordinator = hass.data[DOMAIN][
config_entry.entry_id
].coordinator
entities: list = []
for description in SENSORS:
entities.append(RensonSensor(description, api, coordinator))
async_add_entities(entities)