179 lines
5.4 KiB
Python
179 lines
5.4 KiB
Python
"""Support for Rflink sensors."""
|
|
from __future__ import annotations
|
|
|
|
from rflink.parser import PACKET_FIELDS, UNITS
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.sensor import PLATFORM_SCHEMA, SensorEntity
|
|
from homeassistant.const import (
|
|
ATTR_UNIT_OF_MEASUREMENT,
|
|
CONF_DEVICES,
|
|
CONF_NAME,
|
|
CONF_SENSOR_TYPE,
|
|
CONF_UNIT_OF_MEASUREMENT,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
import homeassistant.helpers.config_validation as cv
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
|
|
|
from . import (
|
|
CONF_ALIASES,
|
|
CONF_AUTOMATIC_ADD,
|
|
DATA_DEVICE_REGISTER,
|
|
DATA_ENTITY_LOOKUP,
|
|
EVENT_KEY_ID,
|
|
EVENT_KEY_SENSOR,
|
|
EVENT_KEY_UNIT,
|
|
SIGNAL_AVAILABILITY,
|
|
SIGNAL_HANDLE_EVENT,
|
|
TMP_ENTITY,
|
|
RflinkDevice,
|
|
)
|
|
|
|
SENSOR_ICONS = {
|
|
"humidity": "mdi:water-percent",
|
|
"battery": "mdi:battery",
|
|
"temperature": "mdi:thermometer",
|
|
}
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
{
|
|
vol.Optional(CONF_AUTOMATIC_ADD, default=True): cv.boolean,
|
|
vol.Optional(CONF_DEVICES, default={}): {
|
|
cv.string: vol.Schema(
|
|
{
|
|
vol.Optional(CONF_NAME): cv.string,
|
|
vol.Required(CONF_SENSOR_TYPE): cv.string,
|
|
vol.Optional(CONF_UNIT_OF_MEASUREMENT): cv.string,
|
|
vol.Optional(CONF_ALIASES, default=[]): vol.All(
|
|
cv.ensure_list, [cv.string]
|
|
),
|
|
}
|
|
)
|
|
},
|
|
},
|
|
extra=vol.ALLOW_EXTRA,
|
|
)
|
|
|
|
|
|
def lookup_unit_for_sensor_type(sensor_type):
|
|
"""Get unit for sensor type.
|
|
|
|
Async friendly.
|
|
"""
|
|
field_abbrev = {v: k for k, v in PACKET_FIELDS.items()}
|
|
|
|
return UNITS.get(field_abbrev.get(sensor_type))
|
|
|
|
|
|
def devices_from_config(domain_config):
|
|
"""Parse configuration and add Rflink sensor devices."""
|
|
devices = []
|
|
for device_id, config in domain_config[CONF_DEVICES].items():
|
|
if ATTR_UNIT_OF_MEASUREMENT not in config:
|
|
config[ATTR_UNIT_OF_MEASUREMENT] = lookup_unit_for_sensor_type(
|
|
config[CONF_SENSOR_TYPE]
|
|
)
|
|
device = RflinkSensor(device_id, **config)
|
|
devices.append(device)
|
|
|
|
return devices
|
|
|
|
|
|
async def async_setup_platform(
|
|
hass: HomeAssistant,
|
|
config: ConfigType,
|
|
async_add_entities: AddEntitiesCallback,
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
) -> None:
|
|
"""Set up the Rflink platform."""
|
|
async_add_entities(devices_from_config(config))
|
|
|
|
async def add_new_device(event):
|
|
"""Check if device is known, otherwise create device entity."""
|
|
device_id = event[EVENT_KEY_ID]
|
|
|
|
device = RflinkSensor(
|
|
device_id,
|
|
event[EVENT_KEY_SENSOR],
|
|
event[EVENT_KEY_UNIT],
|
|
initial_event=event,
|
|
)
|
|
# Add device entity
|
|
async_add_entities([device])
|
|
|
|
if config[CONF_AUTOMATIC_ADD]:
|
|
hass.data[DATA_DEVICE_REGISTER][EVENT_KEY_SENSOR] = add_new_device
|
|
|
|
|
|
class RflinkSensor(RflinkDevice, SensorEntity):
|
|
"""Representation of a Rflink sensor."""
|
|
|
|
def __init__(
|
|
self, device_id, sensor_type, unit_of_measurement, initial_event=None, **kwargs
|
|
):
|
|
"""Handle sensor specific args and super init."""
|
|
self._sensor_type = sensor_type
|
|
self._unit_of_measurement = unit_of_measurement
|
|
super().__init__(device_id, initial_event=initial_event, **kwargs)
|
|
|
|
def _handle_event(self, event):
|
|
"""Domain specific event handler."""
|
|
self._state = event["value"]
|
|
|
|
async def async_added_to_hass(self):
|
|
"""Register update callback."""
|
|
# Remove temporary bogus entity_id if added
|
|
tmp_entity = TMP_ENTITY.format(self._device_id)
|
|
if (
|
|
tmp_entity
|
|
in self.hass.data[DATA_ENTITY_LOOKUP][EVENT_KEY_SENSOR][self._device_id]
|
|
):
|
|
self.hass.data[DATA_ENTITY_LOOKUP][EVENT_KEY_SENSOR][
|
|
self._device_id
|
|
].remove(tmp_entity)
|
|
|
|
# Register id and aliases
|
|
self.hass.data[DATA_ENTITY_LOOKUP][EVENT_KEY_SENSOR][self._device_id].append(
|
|
self.entity_id
|
|
)
|
|
if self._aliases:
|
|
for _id in self._aliases:
|
|
self.hass.data[DATA_ENTITY_LOOKUP][EVENT_KEY_SENSOR][_id].append(
|
|
self.entity_id
|
|
)
|
|
self.async_on_remove(
|
|
async_dispatcher_connect(
|
|
self.hass, SIGNAL_AVAILABILITY, self._availability_callback
|
|
)
|
|
)
|
|
self.async_on_remove(
|
|
async_dispatcher_connect(
|
|
self.hass,
|
|
SIGNAL_HANDLE_EVENT.format(self.entity_id),
|
|
self.handle_event_callback,
|
|
)
|
|
)
|
|
|
|
# Process the initial event now that the entity is created
|
|
if self._initial_event:
|
|
self.handle_event_callback(self._initial_event)
|
|
|
|
@property
|
|
def native_unit_of_measurement(self):
|
|
"""Return measurement unit."""
|
|
return self._unit_of_measurement
|
|
|
|
@property
|
|
def native_value(self):
|
|
"""Return value."""
|
|
return self._state
|
|
|
|
@property
|
|
def icon(self):
|
|
"""Return possible sensor specific icon."""
|
|
if self._sensor_type in SENSOR_ICONS:
|
|
return SENSOR_ICONS[self._sensor_type]
|