hass-core/homeassistant/components/sensor/rfxtrx.py

157 lines
5 KiB
Python
Raw Normal View History

2015-07-23 19:36:05 +02:00
"""
2016-02-23 06:21:49 +01:00
Support for RFXtrx sensors.
2015-07-23 19:36:05 +02:00
2015-10-08 11:08:47 +02:00
For more details about this platform, please refer to the documentation at
2015-11-09 13:12:18 +01:00
https://home-assistant.io/components/sensor.rfxtrx/
2015-07-23 19:36:05 +02:00
"""
import logging
from collections import OrderedDict
import voluptuous as vol
2015-07-23 19:36:05 +02:00
2015-11-01 12:51:09 +01:00
import homeassistant.components.rfxtrx as rfxtrx
2016-04-19 20:30:44 -07:00
from homeassistant.const import TEMP_CELSIUS
import homeassistant.helpers.config_validation as cv
2016-02-18 21:27:50 -08:00
from homeassistant.helpers.entity import Entity
2015-09-27 11:13:49 +02:00
from homeassistant.util import slugify
2016-02-23 18:01:53 +01:00
from homeassistant.components.rfxtrx import (
ATTR_AUTOMATIC_ADD, ATTR_PACKETID, ATTR_NAME,
CONF_DEVICES, ATTR_DATA_TYPE)
2015-07-23 19:36:05 +02:00
DEPENDENCIES = ['rfxtrx']
2015-07-23 19:36:05 +02:00
DATA_TYPES = OrderedDict([
2016-04-19 20:30:44 -07:00
('Temperature', TEMP_CELSIUS),
2015-07-23 19:36:05 +02:00
('Humidity', '%'),
('Barometer', ''),
('Wind direction', ''),
('Rain rate', ''),
('Energy usage', 'W'),
('Total usage', 'W')])
_LOGGER = logging.getLogger(__name__)
2015-07-23 19:36:05 +02:00
DEVICE_SCHEMA = vol.Schema({
vol.Optional(ATTR_NAME, default=None): cv.string,
vol.Required(ATTR_PACKETID): rfxtrx.validate_packetid,
vol.Optional(ATTR_DATA_TYPE, default=None):
vol.In(list(DATA_TYPES.keys())),
})
def _valid_device(value):
"""Validate a dictionary of devices definitions."""
config = OrderedDict()
for key, device in value.items():
try:
key = rfxtrx.VALID_SENSOR_DEVICE_ID(key)
config[key] = DEVICE_SCHEMA(device)
if not config[key][ATTR_NAME]:
config[key][ATTR_NAME] = key
except vol.MultipleInvalid as ex:
raise vol.Invalid('Rfxtrx sensor {} is invalid: {}'
.format(key, ex))
return config
PLATFORM_SCHEMA = vol.Schema({
vol.Required("platform"): rfxtrx.DOMAIN,
vol.Required(CONF_DEVICES): vol.All(dict, _valid_device),
vol.Optional(ATTR_AUTOMATIC_ADD, default=False): cv.boolean,
}, extra=vol.ALLOW_EXTRA)
2015-07-23 19:36:05 +02:00
2015-09-27 11:13:49 +02:00
def setup_platform(hass, config, add_devices_callback, discovery_info=None):
2016-02-23 06:21:49 +01:00
"""Setup the RFXtrx platform."""
2015-11-17 00:18:42 -08:00
from RFXtrx import SensorEvent
2015-07-23 19:36:05 +02:00
2016-02-23 18:01:53 +01:00
sensors = []
for device_id, entity_info in config['devices'].items():
2016-02-25 17:40:24 +01:00
if device_id in rfxtrx.RFX_DEVICES:
continue
_LOGGER.info("Add %s rfxtrx.sensor", entity_info[ATTR_NAME])
event = rfxtrx.get_rfx_object(entity_info[ATTR_PACKETID])
new_sensor = RfxtrxSensor(event, entity_info[ATTR_NAME],
entity_info[ATTR_DATA_TYPE])
2016-03-30 10:52:29 +02:00
rfxtrx.RFX_DEVICES[slugify(device_id)] = new_sensor
2016-02-25 17:40:24 +01:00
sensors.append(new_sensor)
2016-02-23 18:01:53 +01:00
add_devices_callback(sensors)
2015-07-23 19:36:05 +02:00
def sensor_update(event):
2016-02-23 06:21:49 +01:00
"""Callback for sensor updates from the RFXtrx gateway."""
2016-02-23 18:01:53 +01:00
if not isinstance(event, SensorEvent):
return
device_id = "sensor_" + slugify(event.device.id_string.lower())
if device_id in rfxtrx.RFX_DEVICES:
rfxtrx.RFX_DEVICES[device_id].event = event
2016-03-30 10:52:29 +02:00
k = 2
_device_id = device_id + "_" + str(k)
while _device_id in rfxtrx.RFX_DEVICES:
rfxtrx.RFX_DEVICES[_device_id].event = event
k = k + 1
_device_id = device_id + "_" + str(k)
2016-02-23 18:01:53 +01:00
return
# Add entity if not exist and the automatic_add is True
if config[ATTR_AUTOMATIC_ADD]:
2016-02-23 18:01:53 +01:00
pkt_id = "".join("{0:02x}".format(x) for x in event.data)
entity_name = "%s : %s" % (device_id, pkt_id)
_LOGGER.info(
"Automatic add rfxtrx.sensor: (%s : %s)",
device_id,
pkt_id)
new_sensor = RfxtrxSensor(event, entity_name)
rfxtrx.RFX_DEVICES[device_id] = new_sensor
add_devices_callback([new_sensor])
2015-09-27 11:13:49 +02:00
if sensor_update not in rfxtrx.RECEIVED_EVT_SUBSCRIBERS:
rfxtrx.RECEIVED_EVT_SUBSCRIBERS.append(sensor_update)
2015-07-23 19:36:05 +02:00
2015-07-23 19:36:05 +02:00
class RfxtrxSensor(Entity):
2016-03-08 16:46:34 +01:00
"""Representation of a RFXtrx sensor."""
2015-07-23 19:36:05 +02:00
2016-02-23 18:01:53 +01:00
def __init__(self, event, name, data_type=None):
2016-03-08 16:46:34 +01:00
"""Initialize the sensor."""
2015-07-23 19:36:05 +02:00
self.event = event
self._unit_of_measurement = None
self._data_type = None
2016-02-23 18:01:53 +01:00
self._name = name
if data_type:
self._data_type = data_type
self._unit_of_measurement = DATA_TYPES[data_type]
return
2015-07-23 19:36:05 +02:00
for data_type in DATA_TYPES:
if data_type in self.event.values:
self._unit_of_measurement = DATA_TYPES[data_type]
self._data_type = data_type
break
def __str__(self):
2016-03-08 16:46:34 +01:00
"""Return the name of the sensor."""
2015-07-23 19:36:05 +02:00
return self._name
@property
def state(self):
2016-03-08 16:46:34 +01:00
"""Return the state of the sensor."""
2015-07-23 19:36:05 +02:00
if self._data_type:
return self.event.values[self._data_type]
return None
@property
def name(self):
2016-02-23 06:21:49 +01:00
"""Get the name of the sensor."""
2015-07-23 19:36:05 +02:00
return self._name
@property
def device_state_attributes(self):
2016-03-08 16:46:34 +01:00
"""Return the state attributes."""
2015-07-24 12:35:03 +02:00
return self.event.values
2015-07-23 19:36:05 +02:00
@property
def unit_of_measurement(self):
2016-03-08 16:46:34 +01:00
"""Return the unit this state is expressed in."""
2015-07-23 19:36:05 +02:00
return self._unit_of_measurement