2022-09-05 11:52:50 +00:00
|
|
|
"""Support for BTHome sensors."""
|
2022-08-27 13:25:29 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
from typing import Optional, Union
|
|
|
|
|
2022-09-14 10:09:03 +00:00
|
|
|
from bthome_ble import SensorDeviceClass as BTHomeSensorDeviceClass, SensorUpdate, Units
|
2022-08-27 13:25:29 +00:00
|
|
|
|
|
|
|
from homeassistant import config_entries
|
|
|
|
from homeassistant.components.bluetooth.passive_update_processor import (
|
|
|
|
PassiveBluetoothDataProcessor,
|
|
|
|
PassiveBluetoothDataUpdate,
|
|
|
|
PassiveBluetoothProcessorCoordinator,
|
|
|
|
PassiveBluetoothProcessorEntity,
|
|
|
|
)
|
|
|
|
from homeassistant.components.sensor import (
|
|
|
|
SensorDeviceClass,
|
|
|
|
SensorEntity,
|
|
|
|
SensorEntityDescription,
|
|
|
|
SensorStateClass,
|
|
|
|
)
|
|
|
|
from homeassistant.const import (
|
|
|
|
CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
CONCENTRATION_PARTS_PER_MILLION,
|
|
|
|
ELECTRIC_POTENTIAL_VOLT,
|
|
|
|
ENERGY_KILO_WATT_HOUR,
|
|
|
|
LIGHT_LUX,
|
|
|
|
MASS_KILOGRAMS,
|
2022-08-30 21:03:41 +00:00
|
|
|
MASS_POUNDS,
|
2022-08-27 13:25:29 +00:00
|
|
|
PERCENTAGE,
|
|
|
|
POWER_WATT,
|
|
|
|
PRESSURE_MBAR,
|
|
|
|
SIGNAL_STRENGTH_DECIBELS_MILLIWATT,
|
|
|
|
TEMP_CELSIUS,
|
|
|
|
)
|
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from homeassistant.helpers.entity import EntityCategory
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
|
|
|
|
from .const import DOMAIN
|
|
|
|
from .device import device_key_to_bluetooth_entity_key, sensor_device_info_to_hass
|
|
|
|
|
|
|
|
SENSOR_DESCRIPTIONS = {
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.TEMPERATURE, Units.TEMP_CELSIUS): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.TEMPERATURE}_{Units.TEMP_CELSIUS}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.TEMPERATURE,
|
|
|
|
native_unit_of_measurement=TEMP_CELSIUS,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.HUMIDITY, Units.PERCENTAGE): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.HUMIDITY}_{Units.PERCENTAGE}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.HUMIDITY,
|
|
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.ILLUMINANCE, Units.LIGHT_LUX): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.ILLUMINANCE}_{Units.LIGHT_LUX}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.ILLUMINANCE,
|
|
|
|
native_unit_of_measurement=LIGHT_LUX,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.PRESSURE, Units.PRESSURE_MBAR): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.PRESSURE}_{Units.PRESSURE_MBAR}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.PRESSURE,
|
|
|
|
native_unit_of_measurement=PRESSURE_MBAR,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.BATTERY, Units.PERCENTAGE): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.BATTERY}_{Units.PERCENTAGE}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.BATTERY,
|
|
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
|
|
),
|
2022-09-14 10:09:03 +00:00
|
|
|
(
|
|
|
|
BTHomeSensorDeviceClass.VOLTAGE,
|
|
|
|
Units.ELECTRIC_POTENTIAL_VOLT,
|
|
|
|
): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.VOLTAGE}_{Units.ELECTRIC_POTENTIAL_VOLT}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.VOLTAGE,
|
|
|
|
native_unit_of_measurement=ELECTRIC_POTENTIAL_VOLT,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
2022-09-14 10:09:03 +00:00
|
|
|
(
|
|
|
|
BTHomeSensorDeviceClass.ENERGY,
|
|
|
|
Units.ENERGY_KILO_WATT_HOUR,
|
|
|
|
): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.ENERGY}_{Units.ENERGY_KILO_WATT_HOUR}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.ENERGY,
|
|
|
|
native_unit_of_measurement=ENERGY_KILO_WATT_HOUR,
|
|
|
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
|
|
|
),
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.POWER, Units.POWER_WATT): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.POWER}_{Units.POWER_WATT}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.POWER,
|
|
|
|
native_unit_of_measurement=POWER_WATT,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
|
|
|
(
|
2022-09-14 10:09:03 +00:00
|
|
|
BTHomeSensorDeviceClass.PM10,
|
2022-08-27 13:25:29 +00:00
|
|
|
Units.CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
): SensorEntityDescription(
|
2022-09-14 10:09:03 +00:00
|
|
|
key=f"{BTHomeSensorDeviceClass.PM10}_{Units.CONCENTRATION_MICROGRAMS_PER_CUBIC_METER}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.PM10,
|
|
|
|
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
|
|
|
(
|
2022-09-14 10:09:03 +00:00
|
|
|
BTHomeSensorDeviceClass.PM25,
|
2022-08-27 13:25:29 +00:00
|
|
|
Units.CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
): SensorEntityDescription(
|
2022-09-14 10:09:03 +00:00
|
|
|
key=f"{BTHomeSensorDeviceClass.PM25}_{Units.CONCENTRATION_MICROGRAMS_PER_CUBIC_METER}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.PM25,
|
|
|
|
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
2022-09-14 10:09:03 +00:00
|
|
|
(
|
|
|
|
BTHomeSensorDeviceClass.CO2,
|
|
|
|
Units.CONCENTRATION_PARTS_PER_MILLION,
|
|
|
|
): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.CO2}_{Units.CONCENTRATION_PARTS_PER_MILLION}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.CO2,
|
|
|
|
native_unit_of_measurement=CONCENTRATION_PARTS_PER_MILLION,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
|
|
|
(
|
2022-09-14 10:09:03 +00:00
|
|
|
BTHomeSensorDeviceClass.VOLATILE_ORGANIC_COMPOUNDS,
|
2022-08-27 13:25:29 +00:00
|
|
|
Units.CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
): SensorEntityDescription(
|
2022-09-14 10:09:03 +00:00
|
|
|
key=f"{BTHomeSensorDeviceClass.VOLATILE_ORGANIC_COMPOUNDS}_{Units.CONCENTRATION_MICROGRAMS_PER_CUBIC_METER}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.VOLATILE_ORGANIC_COMPOUNDS,
|
|
|
|
native_unit_of_measurement=CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
|
|
|
(
|
2022-09-14 10:09:03 +00:00
|
|
|
BTHomeSensorDeviceClass.SIGNAL_STRENGTH,
|
2022-08-27 13:25:29 +00:00
|
|
|
Units.SIGNAL_STRENGTH_DECIBELS_MILLIWATT,
|
|
|
|
): SensorEntityDescription(
|
2022-09-14 10:09:03 +00:00
|
|
|
key=f"{BTHomeSensorDeviceClass.SIGNAL_STRENGTH}_{Units.SIGNAL_STRENGTH_DECIBELS_MILLIWATT}",
|
2022-08-27 13:25:29 +00:00
|
|
|
device_class=SensorDeviceClass.SIGNAL_STRENGTH,
|
|
|
|
native_unit_of_measurement=SIGNAL_STRENGTH_DECIBELS_MILLIWATT,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
|
|
entity_registry_enabled_default=False,
|
|
|
|
),
|
2022-08-30 21:03:41 +00:00
|
|
|
# Used for mass sensor with kg unit
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.MASS, Units.MASS_KILOGRAMS): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.MASS}_{Units.MASS_KILOGRAMS}",
|
2022-09-29 09:28:59 +00:00
|
|
|
device_class=SensorDeviceClass.WEIGHT,
|
2022-08-27 13:25:29 +00:00
|
|
|
native_unit_of_measurement=MASS_KILOGRAMS,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
2022-08-30 21:03:41 +00:00
|
|
|
# Used for mass sensor with lb unit
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.MASS, Units.MASS_POUNDS): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.MASS}_{Units.MASS_POUNDS}",
|
2022-09-29 09:28:59 +00:00
|
|
|
device_class=SensorDeviceClass.WEIGHT,
|
2022-08-30 21:03:41 +00:00
|
|
|
native_unit_of_measurement=MASS_POUNDS,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
|
|
|
# Used for moisture sensor
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.MOISTURE, Units.PERCENTAGE): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.MOISTURE}_{Units.PERCENTAGE}",
|
2022-09-10 02:43:25 +00:00
|
|
|
device_class=SensorDeviceClass.MOISTURE,
|
2022-08-30 21:03:41 +00:00
|
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
|
|
|
# Used for dew point sensor
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.DEW_POINT, Units.TEMP_CELSIUS): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.DEW_POINT}_{Units.TEMP_CELSIUS}",
|
2022-08-30 21:03:41 +00:00
|
|
|
device_class=SensorDeviceClass.TEMPERATURE,
|
|
|
|
native_unit_of_measurement=TEMP_CELSIUS,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
|
|
|
# Used for count sensor
|
2022-09-14 10:09:03 +00:00
|
|
|
(BTHomeSensorDeviceClass.COUNT, None): SensorEntityDescription(
|
|
|
|
key=f"{BTHomeSensorDeviceClass.COUNT}",
|
2022-08-30 21:03:41 +00:00
|
|
|
device_class=None,
|
|
|
|
native_unit_of_measurement=None,
|
|
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
|
|
),
|
2022-08-27 13:25:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def sensor_update_to_bluetooth_data_update(
|
|
|
|
sensor_update: SensorUpdate,
|
|
|
|
) -> PassiveBluetoothDataUpdate:
|
|
|
|
"""Convert a sensor update to a bluetooth data update."""
|
|
|
|
return PassiveBluetoothDataUpdate(
|
|
|
|
devices={
|
|
|
|
device_id: sensor_device_info_to_hass(device_info)
|
|
|
|
for device_id, device_info in sensor_update.devices.items()
|
|
|
|
},
|
|
|
|
entity_descriptions={
|
|
|
|
device_key_to_bluetooth_entity_key(device_key): SENSOR_DESCRIPTIONS[
|
|
|
|
(description.device_class, description.native_unit_of_measurement)
|
|
|
|
]
|
|
|
|
for device_key, description in sensor_update.entity_descriptions.items()
|
2022-09-14 10:09:03 +00:00
|
|
|
if description.device_class
|
2022-08-27 13:25:29 +00:00
|
|
|
},
|
|
|
|
entity_data={
|
|
|
|
device_key_to_bluetooth_entity_key(device_key): sensor_values.native_value
|
|
|
|
for device_key, sensor_values in sensor_update.entity_values.items()
|
|
|
|
},
|
|
|
|
entity_names={
|
|
|
|
device_key_to_bluetooth_entity_key(device_key): sensor_values.name
|
|
|
|
for device_key, sensor_values in sensor_update.entity_values.items()
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
entry: config_entries.ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
2022-09-05 11:52:50 +00:00
|
|
|
"""Set up the BTHome BLE sensors."""
|
2022-08-27 13:25:29 +00:00
|
|
|
coordinator: PassiveBluetoothProcessorCoordinator = hass.data[DOMAIN][
|
|
|
|
entry.entry_id
|
|
|
|
]
|
|
|
|
processor = PassiveBluetoothDataProcessor(sensor_update_to_bluetooth_data_update)
|
|
|
|
entry.async_on_unload(
|
|
|
|
processor.async_add_entities_listener(
|
2022-09-05 11:52:50 +00:00
|
|
|
BTHomeBluetoothSensorEntity, async_add_entities
|
2022-08-27 13:25:29 +00:00
|
|
|
)
|
|
|
|
)
|
|
|
|
entry.async_on_unload(coordinator.async_register_processor(processor))
|
|
|
|
|
|
|
|
|
2022-09-05 11:52:50 +00:00
|
|
|
class BTHomeBluetoothSensorEntity(
|
2022-08-27 13:25:29 +00:00
|
|
|
PassiveBluetoothProcessorEntity[
|
|
|
|
PassiveBluetoothDataProcessor[Optional[Union[float, int]]]
|
|
|
|
],
|
|
|
|
SensorEntity,
|
|
|
|
):
|
2022-09-05 11:52:50 +00:00
|
|
|
"""Representation of a BTHome BLE sensor."""
|
2022-08-27 13:25:29 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def native_value(self) -> int | float | None:
|
|
|
|
"""Return the native value."""
|
|
|
|
return self.processor.entity_data.get(self.entity_key)
|