2019-02-13 20:21:14 +00:00
|
|
|
"""Support for deCONZ sensors."""
|
2020-01-03 14:15:32 +00:00
|
|
|
from pydeconz.sensor import (
|
2021-04-20 15:34:11 +00:00
|
|
|
AncillaryControl,
|
2020-01-03 14:15:32 +00:00
|
|
|
Battery,
|
|
|
|
Consumption,
|
|
|
|
Daylight,
|
2021-03-31 07:36:06 +00:00
|
|
|
DoorLock,
|
2020-06-02 14:17:21 +00:00
|
|
|
Humidity,
|
2020-01-03 14:15:32 +00:00
|
|
|
LightLevel,
|
|
|
|
Power,
|
2020-06-02 14:17:21 +00:00
|
|
|
Pressure,
|
2020-01-03 14:15:32 +00:00
|
|
|
Switch,
|
2020-06-02 14:17:21 +00:00
|
|
|
Temperature,
|
2020-01-03 14:15:32 +00:00
|
|
|
Thermostat,
|
|
|
|
)
|
2019-05-27 04:56:00 +00:00
|
|
|
|
2021-05-21 09:44:34 +00:00
|
|
|
from homeassistant.components.sensor import (
|
|
|
|
DOMAIN,
|
|
|
|
STATE_CLASS_MEASUREMENT,
|
|
|
|
SensorEntity,
|
|
|
|
)
|
2020-02-28 19:46:48 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_TEMPERATURE,
|
|
|
|
ATTR_VOLTAGE,
|
|
|
|
DEVICE_CLASS_BATTERY,
|
2020-06-02 14:17:21 +00:00
|
|
|
DEVICE_CLASS_HUMIDITY,
|
|
|
|
DEVICE_CLASS_ILLUMINANCE,
|
|
|
|
DEVICE_CLASS_POWER,
|
|
|
|
DEVICE_CLASS_PRESSURE,
|
|
|
|
DEVICE_CLASS_TEMPERATURE,
|
|
|
|
ENERGY_KILO_WATT_HOUR,
|
2020-09-23 18:48:01 +00:00
|
|
|
LIGHT_LUX,
|
2020-09-05 19:09:14 +00:00
|
|
|
PERCENTAGE,
|
2020-06-02 14:17:21 +00:00
|
|
|
POWER_WATT,
|
|
|
|
PRESSURE_HPA,
|
|
|
|
TEMP_CELSIUS,
|
2020-02-28 19:46:48 +00:00
|
|
|
)
|
2018-04-29 14:16:20 +00:00
|
|
|
from homeassistant.core import callback
|
2019-11-01 21:31:22 +00:00
|
|
|
from homeassistant.helpers.dispatcher import (
|
|
|
|
async_dispatcher_connect,
|
|
|
|
async_dispatcher_send,
|
|
|
|
)
|
2018-01-01 16:08:13 +00:00
|
|
|
|
2019-04-05 00:48:24 +00:00
|
|
|
from .const import ATTR_DARK, ATTR_ON, NEW_SENSOR
|
2019-01-16 07:33:04 +00:00
|
|
|
from .deconz_device import DeconzDevice
|
2020-02-05 00:37:01 +00:00
|
|
|
from .gateway import get_gateway_from_config_entry
|
2019-01-15 18:29:56 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_CURRENT = "current"
|
|
|
|
ATTR_POWER = "power"
|
|
|
|
ATTR_DAYLIGHT = "daylight"
|
|
|
|
ATTR_EVENT_ID = "event_id"
|
2018-01-01 16:08:13 +00:00
|
|
|
|
2020-06-02 14:17:21 +00:00
|
|
|
DEVICE_CLASS = {
|
|
|
|
Humidity: DEVICE_CLASS_HUMIDITY,
|
|
|
|
LightLevel: DEVICE_CLASS_ILLUMINANCE,
|
|
|
|
Power: DEVICE_CLASS_POWER,
|
|
|
|
Pressure: DEVICE_CLASS_PRESSURE,
|
|
|
|
Temperature: DEVICE_CLASS_TEMPERATURE,
|
|
|
|
}
|
|
|
|
|
|
|
|
ICON = {
|
|
|
|
Daylight: "mdi:white-balance-sunny",
|
|
|
|
Pressure: "mdi:gauge",
|
|
|
|
Temperature: "mdi:thermometer",
|
|
|
|
}
|
|
|
|
|
2021-05-21 09:44:34 +00:00
|
|
|
STATE_CLASS = {
|
2021-05-24 09:36:42 +00:00
|
|
|
Humidity: STATE_CLASS_MEASUREMENT,
|
|
|
|
Pressure: STATE_CLASS_MEASUREMENT,
|
2021-05-21 09:44:34 +00:00
|
|
|
Temperature: STATE_CLASS_MEASUREMENT,
|
|
|
|
}
|
|
|
|
|
2020-06-02 14:17:21 +00:00
|
|
|
UNIT_OF_MEASUREMENT = {
|
|
|
|
Consumption: ENERGY_KILO_WATT_HOUR,
|
2020-09-05 19:09:14 +00:00
|
|
|
Humidity: PERCENTAGE,
|
2020-09-23 18:48:01 +00:00
|
|
|
LightLevel: LIGHT_LUX,
|
2020-06-02 14:17:21 +00:00
|
|
|
Power: POWER_WATT,
|
|
|
|
Pressure: PRESSURE_HPA,
|
|
|
|
Temperature: TEMP_CELSIUS,
|
|
|
|
}
|
|
|
|
|
2018-01-01 16:08:13 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
2018-04-23 16:00:16 +00:00
|
|
|
"""Set up the deCONZ sensors."""
|
2019-04-05 00:48:24 +00:00
|
|
|
gateway = get_gateway_from_config_entry(hass, config_entry)
|
2020-09-25 20:49:28 +00:00
|
|
|
gateway.entities[DOMAIN] = set()
|
2018-11-05 15:21:44 +00:00
|
|
|
|
2019-11-01 21:31:22 +00:00
|
|
|
battery_handler = DeconzBatteryHandler(gateway)
|
2019-09-05 23:38:00 +00:00
|
|
|
|
2018-05-05 14:11:00 +00:00
|
|
|
@callback
|
2020-12-02 15:21:27 +00:00
|
|
|
def async_add_sensor(sensors=gateway.api.sensors.values()):
|
2019-09-14 17:15:18 +00:00
|
|
|
"""Add sensors from deCONZ.
|
|
|
|
|
|
|
|
Create DeconzBattery if sensor has a battery attribute.
|
2020-09-25 20:49:28 +00:00
|
|
|
Create DeconzSensor if not a battery, switch or thermostat and not a binary sensor.
|
2019-09-14 17:15:18 +00:00
|
|
|
"""
|
2018-05-05 14:11:00 +00:00
|
|
|
entities = []
|
2019-04-05 00:48:24 +00:00
|
|
|
|
2018-05-05 14:11:00 +00:00
|
|
|
for sensor in sensors:
|
2019-04-05 00:48:24 +00:00
|
|
|
|
2020-09-25 20:49:28 +00:00
|
|
|
if not gateway.option_allow_clip_sensor and sensor.type.startswith("CLIP"):
|
|
|
|
continue
|
2019-04-05 00:48:24 +00:00
|
|
|
|
2019-12-22 21:58:22 +00:00
|
|
|
if sensor.battery is not None:
|
2020-09-25 20:49:28 +00:00
|
|
|
battery_handler.remove_tracker(sensor)
|
|
|
|
|
2020-09-27 09:02:45 +00:00
|
|
|
known_batteries = set(gateway.entities[DOMAIN])
|
2019-09-14 17:15:18 +00:00
|
|
|
new_battery = DeconzBattery(sensor, gateway)
|
2020-09-25 20:49:28 +00:00
|
|
|
if new_battery.unique_id not in known_batteries:
|
2019-09-14 17:15:18 +00:00
|
|
|
entities.append(new_battery)
|
2020-09-25 20:49:28 +00:00
|
|
|
|
2019-11-01 21:31:22 +00:00
|
|
|
else:
|
|
|
|
battery_handler.create_tracker(sensor)
|
2019-04-05 00:48:24 +00:00
|
|
|
|
2020-09-25 20:49:28 +00:00
|
|
|
if (
|
|
|
|
not sensor.BINARY
|
|
|
|
and sensor.type
|
2021-04-20 15:34:11 +00:00
|
|
|
not in AncillaryControl.ZHATYPE
|
|
|
|
+ Battery.ZHATYPE
|
2021-03-31 07:36:06 +00:00
|
|
|
+ DoorLock.ZHATYPE
|
|
|
|
+ Switch.ZHATYPE
|
|
|
|
+ Thermostat.ZHATYPE
|
2020-09-25 20:49:28 +00:00
|
|
|
and sensor.uniqueid not in gateway.entities[DOMAIN]
|
|
|
|
):
|
|
|
|
entities.append(DeconzSensor(sensor, gateway))
|
|
|
|
|
2021-04-27 06:43:06 +00:00
|
|
|
if sensor.secondary_temperature:
|
|
|
|
known_temperature_sensors = set(gateway.entities[DOMAIN])
|
|
|
|
new_temperature_sensor = DeconzTemperature(sensor, gateway)
|
|
|
|
if new_temperature_sensor.unique_id not in known_temperature_sensors:
|
|
|
|
entities.append(new_temperature_sensor)
|
|
|
|
|
2020-10-02 09:20:33 +00:00
|
|
|
if entities:
|
2020-10-16 15:14:26 +00:00
|
|
|
async_add_entities(entities)
|
2018-05-29 14:09:53 +00:00
|
|
|
|
2021-04-20 18:20:57 +00:00
|
|
|
config_entry.async_on_unload(
|
2019-07-31 19:25:30 +00:00
|
|
|
async_dispatcher_connect(
|
2019-09-05 23:38:00 +00:00
|
|
|
hass, gateway.async_signal_new_device(NEW_SENSOR), async_add_sensor
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
)
|
2018-05-05 14:11:00 +00:00
|
|
|
|
2020-01-08 08:30:02 +00:00
|
|
|
async_add_sensor(
|
|
|
|
[gateway.api.sensors[key] for key in sorted(gateway.api.sensors, key=int)]
|
|
|
|
)
|
2018-01-01 16:08:13 +00:00
|
|
|
|
|
|
|
|
2021-03-22 19:05:13 +00:00
|
|
|
class DeconzSensor(DeconzDevice, SensorEntity):
|
2019-01-16 07:33:04 +00:00
|
|
|
"""Representation of a deCONZ sensor."""
|
2018-08-29 21:18:20 +00:00
|
|
|
|
2020-09-25 20:49:28 +00:00
|
|
|
TYPE = DOMAIN
|
|
|
|
|
2021-06-23 19:40:34 +00:00
|
|
|
def __init__(self, device, gateway):
|
|
|
|
"""Initialize deCONZ binary sensor."""
|
|
|
|
super().__init__(device, gateway)
|
|
|
|
|
|
|
|
self._attr_device_class = DEVICE_CLASS.get(type(self._device))
|
|
|
|
self._attr_icon = ICON.get(type(self._device))
|
|
|
|
self._attr_state_class = STATE_CLASS.get(type(self._device))
|
|
|
|
self._attr_unit_of_measurement = UNIT_OF_MEASUREMENT.get(type(self._device))
|
|
|
|
|
2018-01-01 16:08:13 +00:00
|
|
|
@callback
|
2020-09-30 15:24:30 +00:00
|
|
|
def async_update_callback(self, force_update=False):
|
2019-05-27 04:56:00 +00:00
|
|
|
"""Update the sensor's state."""
|
2019-09-14 17:15:18 +00:00
|
|
|
keys = {"on", "reachable", "state"}
|
2020-01-17 23:33:46 +00:00
|
|
|
if force_update or self._device.changed_keys.intersection(keys):
|
2020-09-30 15:24:30 +00:00
|
|
|
super().async_update_callback(force_update=force_update)
|
2018-01-01 16:08:13 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the sensor."""
|
2019-01-16 07:33:04 +00:00
|
|
|
return self._device.state
|
2018-01-30 22:42:24 +00:00
|
|
|
|
2018-01-01 16:08:13 +00:00
|
|
|
@property
|
2021-03-11 15:51:03 +00:00
|
|
|
def extra_state_attributes(self):
|
2018-01-01 16:08:13 +00:00
|
|
|
"""Return the state attributes of the sensor."""
|
2018-03-15 03:07:37 +00:00
|
|
|
attr = {}
|
2019-05-27 04:56:00 +00:00
|
|
|
|
2019-01-16 07:33:04 +00:00
|
|
|
if self._device.on is not None:
|
|
|
|
attr[ATTR_ON] = self._device.on
|
2019-05-27 04:56:00 +00:00
|
|
|
|
|
|
|
if self._device.secondary_temperature is not None:
|
|
|
|
attr[ATTR_TEMPERATURE] = self._device.secondary_temperature
|
|
|
|
|
2019-07-28 19:01:52 +00:00
|
|
|
if self._device.type in Consumption.ZHATYPE:
|
|
|
|
attr[ATTR_POWER] = self._device.power
|
|
|
|
|
|
|
|
elif self._device.type in Daylight.ZHATYPE:
|
|
|
|
attr[ATTR_DAYLIGHT] = self._device.daylight
|
|
|
|
|
2020-01-29 06:40:42 +00:00
|
|
|
elif self._device.type in LightLevel.ZHATYPE:
|
|
|
|
|
|
|
|
if self._device.dark is not None:
|
|
|
|
attr[ATTR_DARK] = self._device.dark
|
|
|
|
|
|
|
|
if self._device.daylight is not None:
|
|
|
|
attr[ATTR_DAYLIGHT] = self._device.daylight
|
2019-05-27 04:56:00 +00:00
|
|
|
|
2019-07-28 19:01:52 +00:00
|
|
|
elif self._device.type in Power.ZHATYPE:
|
2019-01-16 07:33:04 +00:00
|
|
|
attr[ATTR_CURRENT] = self._device.current
|
|
|
|
attr[ATTR_VOLTAGE] = self._device.voltage
|
2019-05-27 04:56:00 +00:00
|
|
|
|
2018-01-01 16:08:13 +00:00
|
|
|
return attr
|
|
|
|
|
|
|
|
|
2021-04-27 06:43:06 +00:00
|
|
|
class DeconzTemperature(DeconzDevice, SensorEntity):
|
|
|
|
"""Representation of a deCONZ temperature sensor.
|
|
|
|
|
|
|
|
Extra temperature sensor on certain Xiaomi devices.
|
|
|
|
"""
|
|
|
|
|
2021-06-23 19:40:34 +00:00
|
|
|
_attr_device_class = DEVICE_CLASS_TEMPERATURE
|
|
|
|
_attr_state_class = STATE_CLASS_MEASUREMENT
|
|
|
|
_attr_unit_of_measurement = TEMP_CELSIUS
|
|
|
|
|
2021-04-27 06:43:06 +00:00
|
|
|
TYPE = DOMAIN
|
|
|
|
|
2021-06-23 19:40:34 +00:00
|
|
|
def __init__(self, device, gateway):
|
|
|
|
"""Initialize deCONZ temperature sensor."""
|
|
|
|
super().__init__(device, gateway)
|
|
|
|
|
|
|
|
self._attr_name = f"{self._device.name} Temperature"
|
|
|
|
|
2021-04-27 06:43:06 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return a unique identifier for this device."""
|
|
|
|
return f"{self.serial}-temperature"
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_update_callback(self, force_update=False):
|
|
|
|
"""Update the sensor's state."""
|
|
|
|
keys = {"temperature", "reachable"}
|
|
|
|
if force_update or self._device.changed_keys.intersection(keys):
|
|
|
|
super().async_update_callback(force_update=force_update)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the sensor."""
|
|
|
|
return self._device.secondary_temperature
|
|
|
|
|
|
|
|
|
2021-03-22 19:05:13 +00:00
|
|
|
class DeconzBattery(DeconzDevice, SensorEntity):
|
2018-01-01 16:08:13 +00:00
|
|
|
"""Battery class for when a device is only represented as an event."""
|
|
|
|
|
2021-06-23 19:40:34 +00:00
|
|
|
_attr_device_class = DEVICE_CLASS_BATTERY
|
|
|
|
_attr_state_class = STATE_CLASS_MEASUREMENT
|
|
|
|
_attr_unit_of_measurement = PERCENTAGE
|
|
|
|
|
2020-09-25 20:49:28 +00:00
|
|
|
TYPE = DOMAIN
|
|
|
|
|
2021-06-23 19:40:34 +00:00
|
|
|
def __init__(self, device, gateway):
|
|
|
|
"""Initialize deCONZ battery level sensor."""
|
|
|
|
super().__init__(device, gateway)
|
|
|
|
|
|
|
|
self._attr_name = f"{self._device.name} Battery Level"
|
|
|
|
|
2018-01-01 16:08:13 +00:00
|
|
|
@callback
|
2020-09-30 15:24:30 +00:00
|
|
|
def async_update_callback(self, force_update=False):
|
2018-01-01 16:08:13 +00:00
|
|
|
"""Update the battery's state, if needed."""
|
2019-07-31 19:25:30 +00:00
|
|
|
keys = {"battery", "reachable"}
|
2020-01-17 23:33:46 +00:00
|
|
|
if force_update or self._device.changed_keys.intersection(keys):
|
2020-09-30 15:24:30 +00:00
|
|
|
super().async_update_callback(force_update=force_update)
|
2018-01-01 16:08:13 +00:00
|
|
|
|
2019-09-14 17:15:18 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self):
|
2021-02-25 08:54:46 +00:00
|
|
|
"""Return a unique identifier for this device.
|
|
|
|
|
|
|
|
Normally there should only be one battery sensor per device from deCONZ.
|
|
|
|
With specific Danfoss devices each endpoint can report its own battery state.
|
|
|
|
"""
|
|
|
|
if self._device.manufacturer == "Danfoss" and self._device.modelid in [
|
|
|
|
"0x8030",
|
|
|
|
"0x8031",
|
|
|
|
"0x8034",
|
|
|
|
"0x8035",
|
|
|
|
]:
|
|
|
|
return f"{super().unique_id}-battery"
|
2019-09-14 17:15:18 +00:00
|
|
|
return f"{self.serial}-battery"
|
|
|
|
|
2018-01-01 16:08:13 +00:00
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the battery."""
|
2019-01-16 07:33:04 +00:00
|
|
|
return self._device.battery
|
2018-01-01 16:08:13 +00:00
|
|
|
|
|
|
|
@property
|
2021-03-11 15:51:03 +00:00
|
|
|
def extra_state_attributes(self):
|
2018-01-01 16:08:13 +00:00
|
|
|
"""Return the state attributes of the battery."""
|
2019-09-14 17:15:18 +00:00
|
|
|
attr = {}
|
|
|
|
|
|
|
|
if self._device.type in Switch.ZHATYPE:
|
|
|
|
for event in self.gateway.events:
|
|
|
|
if self._device == event.device:
|
|
|
|
attr[ATTR_EVENT_ID] = event.event_id
|
|
|
|
|
2018-01-01 16:08:13 +00:00
|
|
|
return attr
|
2019-11-01 21:31:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
class DeconzSensorStateTracker:
|
|
|
|
"""Track sensors without a battery state and signal when battery state exist."""
|
|
|
|
|
|
|
|
def __init__(self, sensor, gateway):
|
|
|
|
"""Set up tracker."""
|
|
|
|
self.sensor = sensor
|
|
|
|
self.gateway = gateway
|
2020-02-02 18:07:20 +00:00
|
|
|
sensor.register_callback(self.async_update_callback)
|
2019-11-01 21:31:22 +00:00
|
|
|
|
|
|
|
@callback
|
|
|
|
def close(self):
|
|
|
|
"""Clean up tracker."""
|
|
|
|
self.sensor.remove_callback(self.async_update_callback)
|
|
|
|
self.gateway = None
|
|
|
|
self.sensor = None
|
|
|
|
|
|
|
|
@callback
|
2020-01-17 23:33:46 +00:00
|
|
|
def async_update_callback(self, ignore_update=False):
|
2019-11-01 21:31:22 +00:00
|
|
|
"""Sensor state updated."""
|
|
|
|
if "battery" in self.sensor.changed_keys:
|
|
|
|
async_dispatcher_send(
|
|
|
|
self.gateway.hass,
|
|
|
|
self.gateway.async_signal_new_device(NEW_SENSOR),
|
|
|
|
[self.sensor],
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class DeconzBatteryHandler:
|
|
|
|
"""Creates and stores trackers for sensors without a battery state."""
|
|
|
|
|
|
|
|
def __init__(self, gateway):
|
|
|
|
"""Set up battery handler."""
|
|
|
|
self.gateway = gateway
|
|
|
|
self._trackers = set()
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def create_tracker(self, sensor):
|
|
|
|
"""Create new tracker for battery state."""
|
2019-12-22 21:58:22 +00:00
|
|
|
for tracker in self._trackers:
|
|
|
|
if sensor == tracker.sensor:
|
|
|
|
return
|
2019-11-01 21:31:22 +00:00
|
|
|
self._trackers.add(DeconzSensorStateTracker(sensor, self.gateway))
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def remove_tracker(self, sensor):
|
|
|
|
"""Remove tracker of battery state."""
|
|
|
|
for tracker in self._trackers:
|
|
|
|
if sensor == tracker.sensor:
|
|
|
|
tracker.close()
|
|
|
|
self._trackers.remove(tracker)
|
|
|
|
break
|