mirror of https://github.com/home-assistant/core
196 lines
6.4 KiB
Python
196 lines
6.4 KiB
Python
"""Support for Fibaro sensors."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from contextlib import suppress
|
|
|
|
from pyfibaro.fibaro_device import DeviceModel
|
|
|
|
from homeassistant.components.sensor import (
|
|
ENTITY_ID_FORMAT,
|
|
SensorDeviceClass,
|
|
SensorEntity,
|
|
SensorEntityDescription,
|
|
SensorStateClass,
|
|
)
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
CONCENTRATION_PARTS_PER_MILLION,
|
|
LIGHT_LUX,
|
|
PERCENTAGE,
|
|
Platform,
|
|
UnitOfEnergy,
|
|
UnitOfPower,
|
|
UnitOfTemperature,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.util import convert
|
|
|
|
from . import FibaroController
|
|
from .const import DOMAIN
|
|
from .entity import FibaroEntity
|
|
|
|
# List of known sensors which represents a fibaro device
|
|
MAIN_SENSOR_TYPES: dict[str, SensorEntityDescription] = {
|
|
"com.fibaro.temperatureSensor": SensorEntityDescription(
|
|
key="com.fibaro.temperatureSensor",
|
|
name="Temperature",
|
|
device_class=SensorDeviceClass.TEMPERATURE,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
),
|
|
"com.fibaro.smokeSensor": SensorEntityDescription(
|
|
key="com.fibaro.smokeSensor",
|
|
name="Smoke",
|
|
native_unit_of_measurement=CONCENTRATION_PARTS_PER_MILLION,
|
|
icon="mdi:fire",
|
|
),
|
|
"CO2": SensorEntityDescription(
|
|
key="CO2",
|
|
name="CO2",
|
|
native_unit_of_measurement=CONCENTRATION_PARTS_PER_MILLION,
|
|
device_class=SensorDeviceClass.CO2,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
),
|
|
"com.fibaro.humiditySensor": SensorEntityDescription(
|
|
key="com.fibaro.humiditySensor",
|
|
name="Humidity",
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
device_class=SensorDeviceClass.HUMIDITY,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
),
|
|
"com.fibaro.lightSensor": SensorEntityDescription(
|
|
key="com.fibaro.lightSensor",
|
|
name="Light",
|
|
native_unit_of_measurement=LIGHT_LUX,
|
|
device_class=SensorDeviceClass.ILLUMINANCE,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
),
|
|
"com.fibaro.energyMeter": SensorEntityDescription(
|
|
key="com.fibaro.energyMeter",
|
|
name="Energy",
|
|
native_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
|
|
device_class=SensorDeviceClass.ENERGY,
|
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
|
),
|
|
}
|
|
|
|
# List of additional sensors which are created based on a property
|
|
# The key is the property name
|
|
ADDITIONAL_SENSOR_TYPES: tuple[SensorEntityDescription, ...] = (
|
|
SensorEntityDescription(
|
|
key="energy",
|
|
name="Energy",
|
|
native_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
|
|
device_class=SensorDeviceClass.ENERGY,
|
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
|
),
|
|
SensorEntityDescription(
|
|
key="power",
|
|
name="Power",
|
|
native_unit_of_measurement=UnitOfPower.WATT,
|
|
device_class=SensorDeviceClass.POWER,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
),
|
|
)
|
|
|
|
FIBARO_TO_HASS_UNIT: dict[str, str] = {
|
|
"lux": LIGHT_LUX,
|
|
"C": UnitOfTemperature.CELSIUS,
|
|
"F": UnitOfTemperature.FAHRENHEIT,
|
|
}
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant,
|
|
entry: ConfigEntry,
|
|
async_add_entities: AddEntitiesCallback,
|
|
) -> None:
|
|
"""Set up the Fibaro controller devices."""
|
|
|
|
controller: FibaroController = hass.data[DOMAIN][entry.entry_id]
|
|
entities: list[SensorEntity] = [
|
|
FibaroSensor(device, MAIN_SENSOR_TYPES.get(device.type))
|
|
for device in controller.fibaro_devices[Platform.SENSOR]
|
|
# Some sensor devices do not have a value but report power or energy.
|
|
# These sensors are added to the sensor list but need to be excluded
|
|
# here as the FibaroSensor expects a value. One example is the
|
|
# Qubino 3 phase power meter.
|
|
if device.value.has_value
|
|
]
|
|
|
|
entities.extend(
|
|
FibaroAdditionalSensor(device, entity_description)
|
|
for platform in (
|
|
Platform.BINARY_SENSOR,
|
|
Platform.CLIMATE,
|
|
Platform.COVER,
|
|
Platform.LIGHT,
|
|
Platform.LOCK,
|
|
Platform.SENSOR,
|
|
Platform.SWITCH,
|
|
)
|
|
for device in controller.fibaro_devices[platform]
|
|
for entity_description in ADDITIONAL_SENSOR_TYPES
|
|
if entity_description.key in device.properties
|
|
)
|
|
|
|
async_add_entities(entities, True)
|
|
|
|
|
|
class FibaroSensor(FibaroEntity, SensorEntity):
|
|
"""Representation of a Fibaro Sensor."""
|
|
|
|
def __init__(
|
|
self,
|
|
fibaro_device: DeviceModel,
|
|
entity_description: SensorEntityDescription | None,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
super().__init__(fibaro_device)
|
|
if entity_description is not None:
|
|
self.entity_description = entity_description
|
|
self.entity_id = ENTITY_ID_FORMAT.format(self.ha_id)
|
|
|
|
# Map unit if it was not defined in the entity description
|
|
# or there is no entity description at all
|
|
with suppress(KeyError, ValueError):
|
|
if not self.native_unit_of_measurement:
|
|
self._attr_native_unit_of_measurement = FIBARO_TO_HASS_UNIT.get(
|
|
fibaro_device.unit, fibaro_device.unit
|
|
)
|
|
|
|
def update(self) -> None:
|
|
"""Update the state."""
|
|
super().update()
|
|
with suppress(TypeError):
|
|
self._attr_native_value = self.fibaro_device.value.float_value()
|
|
|
|
|
|
class FibaroAdditionalSensor(FibaroEntity, SensorEntity):
|
|
"""Representation of a Fibaro Additional Sensor."""
|
|
|
|
def __init__(
|
|
self, fibaro_device: DeviceModel, entity_description: SensorEntityDescription
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
super().__init__(fibaro_device)
|
|
self.entity_description = entity_description
|
|
|
|
# To differentiate additional sensors from main sensors they need
|
|
# to get different names and ids
|
|
self.entity_id = ENTITY_ID_FORMAT.format(
|
|
f"{self.ha_id}_{entity_description.key}"
|
|
)
|
|
self._attr_name = f"{fibaro_device.friendly_name} {entity_description.name}"
|
|
self._attr_unique_id = f"{fibaro_device.unique_id_str}_{entity_description.key}"
|
|
|
|
def update(self) -> None:
|
|
"""Update the state."""
|
|
super().update()
|
|
with suppress(KeyError, ValueError):
|
|
self._attr_native_value = convert(
|
|
self.fibaro_device.properties[self.entity_description.key],
|
|
float,
|
|
)
|