core/homeassistant/components/zigbee.py

469 lines
14 KiB
Python
Raw Normal View History

2016-01-24 08:02:14 +00:00
"""
2018-10-24 16:59:52 +00:00
Support for Zigbee devices.
2016-01-24 08:02:14 +00:00
For more details about this component, please refer to the documentation at
https://home-assistant.io/components/zigbee/
"""
import logging
from binascii import hexlify, unhexlify
2016-01-24 08:02:14 +00:00
2016-09-07 01:28:55 +00:00
import voluptuous as vol
from homeassistant.const import (
EVENT_HOMEASSISTANT_STOP, CONF_DEVICE, CONF_NAME, CONF_PIN)
from homeassistant.helpers.entity import Entity
2016-09-07 01:28:55 +00:00
from homeassistant.helpers import config_validation as cv
from homeassistant.helpers.dispatcher import (
async_dispatcher_connect, dispatcher_send)
2016-09-07 01:28:55 +00:00
REQUIREMENTS = ['xbee-helper==0.0.7']
_LOGGER = logging.getLogger(__name__)
2016-01-24 08:02:14 +00:00
2016-09-07 01:28:55 +00:00
DOMAIN = 'zigbee'
SIGNAL_ZIGBEE_FRAME_RECEIVED = 'zigbee_frame_received'
2016-01-24 08:02:14 +00:00
2016-09-07 01:28:55 +00:00
CONF_ADDRESS = 'address'
CONF_BAUD = 'baud'
2016-01-24 08:02:14 +00:00
2016-09-07 01:28:55 +00:00
DEFAULT_DEVICE = '/dev/ttyUSB0'
2016-01-24 08:02:14 +00:00
DEFAULT_BAUD = 9600
DEFAULT_ADC_MAX_VOLTS = 1.2
# Copied from xbee_helper during setup()
GPIO_DIGITAL_OUTPUT_LOW = None
GPIO_DIGITAL_OUTPUT_HIGH = None
ADC_PERCENTAGE = None
DIGITAL_PINS = None
ANALOG_PINS = None
CONVERT_ADC = None
ZIGBEE_EXCEPTION = None
ZIGBEE_TX_FAILURE = None
2016-09-07 01:28:55 +00:00
ATTR_FRAME = 'frame'
2016-01-24 08:02:14 +00:00
DEVICE = None
2016-09-07 01:28:55 +00:00
CONFIG_SCHEMA = vol.Schema({
DOMAIN: vol.Schema({
vol.Optional(CONF_BAUD, default=DEFAULT_BAUD): cv.string,
vol.Optional(CONF_DEVICE, default=DEFAULT_DEVICE): cv.string,
}),
}, extra=vol.ALLOW_EXTRA)
PLATFORM_SCHEMA = vol.Schema({
vol.Required(CONF_NAME): cv.string,
vol.Optional(CONF_PIN): cv.positive_int,
2016-09-07 01:28:55 +00:00
vol.Optional(CONF_ADDRESS): cv.string,
}, extra=vol.ALLOW_EXTRA)
2016-01-24 08:02:14 +00:00
def setup(hass, config):
2018-10-24 16:59:52 +00:00
"""Set up the connection to the Zigbee device."""
2016-01-24 08:02:14 +00:00
global DEVICE
global GPIO_DIGITAL_OUTPUT_LOW
global GPIO_DIGITAL_OUTPUT_HIGH
global ADC_PERCENTAGE
global DIGITAL_PINS
global ANALOG_PINS
global CONVERT_ADC
global ZIGBEE_EXCEPTION
global ZIGBEE_TX_FAILURE
2016-01-24 08:02:14 +00:00
import xbee_helper.const as xb_const
from xbee_helper import ZigBee
from xbee_helper.device import convert_adc
from xbee_helper.exceptions import ZigBeeException, ZigBeeTxFailure
from serial import Serial, SerialException
2016-01-24 08:02:14 +00:00
GPIO_DIGITAL_OUTPUT_LOW = xb_const.GPIO_DIGITAL_OUTPUT_LOW
GPIO_DIGITAL_OUTPUT_HIGH = xb_const.GPIO_DIGITAL_OUTPUT_HIGH
ADC_PERCENTAGE = xb_const.ADC_PERCENTAGE
DIGITAL_PINS = xb_const.DIGITAL_PINS
ANALOG_PINS = xb_const.ANALOG_PINS
CONVERT_ADC = convert_adc
ZIGBEE_EXCEPTION = ZigBeeException
ZIGBEE_TX_FAILURE = ZigBeeTxFailure
2016-01-24 08:02:14 +00:00
usb_device = config[DOMAIN].get(CONF_DEVICE, DEFAULT_DEVICE)
baud = int(config[DOMAIN].get(CONF_BAUD, DEFAULT_BAUD))
try:
ser = Serial(usb_device, baud)
except SerialException as exc:
2018-10-24 16:59:52 +00:00
_LOGGER.exception("Unable to open serial port for Zigbee: %s", exc)
return False
2016-01-24 08:02:14 +00:00
DEVICE = ZigBee(ser)
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, close_serial_port)
def _frame_received(frame):
2018-10-24 16:59:52 +00:00
"""Run when a Zigbee frame is received.
Pickles the frame, then encodes it into base64 since it contains
non JSON serializable binary.
"""
dispatcher_send(hass, SIGNAL_ZIGBEE_FRAME_RECEIVED, frame)
DEVICE.add_frame_rx_handler(_frame_received)
2016-01-24 08:02:14 +00:00
return True
def close_serial_port(*args):
2018-10-24 16:59:52 +00:00
"""Close the serial port we're using to communicate with the Zigbee."""
2016-01-24 08:02:14 +00:00
DEVICE.zb.serial.close()
def frame_is_relevant(entity, frame):
"""Test whether the frame is relevant to the entity."""
2016-09-07 01:28:55 +00:00
if frame.get('source_addr_long') != entity.config.address:
return False
2016-09-07 01:28:55 +00:00
if 'samples' not in frame:
return False
return True
class ZigBeeConfig:
2016-03-08 16:55:57 +00:00
"""Handle the fetching of configuration from the config file."""
2016-01-24 08:02:14 +00:00
def __init__(self, config):
2016-03-08 16:55:57 +00:00
"""Initialize the configuration."""
2016-01-24 08:02:14 +00:00
self._config = config
self._should_poll = config.get("poll", True)
@property
def name(self):
"""Return the name given to the entity."""
2016-01-24 08:02:14 +00:00
return self._config["name"]
@property
def address(self):
"""Return the address of the device.
2016-03-08 16:55:57 +00:00
2016-01-24 08:02:14 +00:00
If an address has been provided, unhexlify it, otherwise return None
2018-10-24 16:59:52 +00:00
as we're talking to our local Zigbee device.
2016-01-24 08:02:14 +00:00
"""
address = self._config.get("address")
if address is not None:
address = unhexlify(address)
return address
@property
def should_poll(self):
"""Return the polling state."""
2016-01-24 08:02:14 +00:00
return self._should_poll
class ZigBeePinConfig(ZigBeeConfig):
"""Handle the fetching of configuration from the configuration file."""
2016-03-08 16:55:57 +00:00
2016-01-24 08:02:14 +00:00
@property
def pin(self):
"""Return the GPIO pin number."""
2016-01-24 08:02:14 +00:00
return self._config["pin"]
class ZigBeeDigitalInConfig(ZigBeePinConfig):
"""A subclass of ZigBeePinConfig."""
2016-03-08 16:55:57 +00:00
2016-01-24 08:02:14 +00:00
def __init__(self, config):
2018-10-24 16:59:52 +00:00
"""Initialise the Zigbee Digital input config."""
super(ZigBeeDigitalInConfig, self).__init__(config)
2016-01-24 08:02:14 +00:00
self._bool2state, self._state2bool = self.boolean_maps
@property
def boolean_maps(self):
"""Create mapping dictionaries for potential inversion of booleans.
2016-03-08 16:55:57 +00:00
Create dicts to map the pin state (true/false) to potentially inverted
values depending on the on_state config value which should be set to
"low" or "high".
2016-01-24 08:02:14 +00:00
"""
if self._config.get("on_state", "").lower() == "low":
bool2state = {
True: False,
False: True
2016-01-24 08:02:14 +00:00
}
else:
bool2state = {
True: True,
False: False
2016-01-24 08:02:14 +00:00
}
state2bool = {v: k for k, v in bool2state.items()}
return bool2state, state2bool
@property
def bool2state(self):
2018-10-24 16:59:52 +00:00
"""Return a dictionary mapping the internal value to the Zigbee value.
2016-03-08 16:55:57 +00:00
For the translation of on/off as being pin high or low.
2016-01-24 08:02:14 +00:00
"""
return self._bool2state
@property
def state2bool(self):
2018-10-24 16:59:52 +00:00
"""Return a dictionary mapping the Zigbee value to the internal value.
2016-03-08 16:55:57 +00:00
For the translation of pin high/low as being on or off.
2016-01-24 08:02:14 +00:00
"""
return self._state2bool
class ZigBeeDigitalOutConfig(ZigBeePinConfig):
"""A subclass of ZigBeePinConfig.
2016-03-08 16:55:57 +00:00
Set _should_poll to default as False instead of True. The value will
still be overridden by the presence of a 'poll' config entry.
2016-01-24 08:02:14 +00:00
"""
2016-03-08 16:55:57 +00:00
2016-01-24 08:02:14 +00:00
def __init__(self, config):
2018-10-24 16:59:52 +00:00
"""Initialize the Zigbee Digital out."""
2016-01-24 08:02:14 +00:00
super(ZigBeeDigitalOutConfig, self).__init__(config)
self._bool2state, self._state2bool = self.boolean_maps
2016-01-24 08:02:14 +00:00
self._should_poll = config.get("poll", False)
@property
def boolean_maps(self):
"""Create dicts to map booleans to pin high/low and vice versa.
Depends on the config item "on_state" which should be set to "low"
or "high".
"""
if self._config.get("on_state", "").lower() == "low":
bool2state = {
True: GPIO_DIGITAL_OUTPUT_LOW,
False: GPIO_DIGITAL_OUTPUT_HIGH
}
else:
bool2state = {
True: GPIO_DIGITAL_OUTPUT_HIGH,
False: GPIO_DIGITAL_OUTPUT_LOW
}
state2bool = {v: k for k, v in bool2state.items()}
return bool2state, state2bool
@property
def bool2state(self):
"""Return a dictionary mapping booleans to GPIOSetting objects.
For the translation of on/off as being pin high or low.
"""
return self._bool2state
@property
def state2bool(self):
"""Return a dictionary mapping GPIOSetting objects to booleans.
For the translation of pin high/low as being on or off.
"""
return self._state2bool
2016-01-24 08:02:14 +00:00
class ZigBeeAnalogInConfig(ZigBeePinConfig):
2018-10-24 16:59:52 +00:00
"""Representation of a Zigbee GPIO pin set to analog in."""
2016-03-08 16:55:57 +00:00
2016-01-24 08:02:14 +00:00
@property
def max_voltage(self):
"""Return the voltage for ADC to report its highest value."""
2016-01-24 08:02:14 +00:00
return float(self._config.get("max_volts", DEFAULT_ADC_MAX_VOLTS))
class ZigBeeDigitalIn(Entity):
2016-03-08 16:55:57 +00:00
"""Representation of a GPIO pin configured as a digital input."""
2016-01-24 08:02:14 +00:00
def __init__(self, hass, config):
2016-03-08 16:55:57 +00:00
"""Initialize the device."""
2016-01-24 08:02:14 +00:00
self._config = config
self._state = False
async def async_added_to_hass(self):
"""Register callbacks."""
def handle_frame(frame):
"""Handle an incoming frame.
Handle an incoming frame and update our status if it contains
information relating to this device.
"""
if not frame_is_relevant(self, frame):
return
sample = next(iter(frame['samples']))
pin_name = DIGITAL_PINS[self._config.pin]
if pin_name not in sample:
# Doesn't contain information about our pin
return
# Set state to the value of sample, respecting any inversion
# logic from the on_state config variable.
self._state = self._config.state2bool[
self._config.bool2state[sample[pin_name]]]
self.schedule_update_ha_state()
async_dispatcher_connect(
self.hass, SIGNAL_ZIGBEE_FRAME_RECEIVED, handle_frame)
2016-01-24 08:02:14 +00:00
@property
def name(self):
2016-03-08 16:55:57 +00:00
"""Return the name of the input."""
2016-01-24 08:02:14 +00:00
return self._config.name
@property
def config(self):
"""Return the entity's configuration."""
return self._config
2016-01-24 08:02:14 +00:00
@property
def should_poll(self):
2016-03-08 16:55:57 +00:00
"""Return the state of the polling, if needed."""
2016-01-24 08:02:14 +00:00
return self._config.should_poll
@property
def is_on(self):
2016-03-08 16:55:57 +00:00
"""Return True if the Entity is on, else False."""
2016-01-24 08:02:14 +00:00
return self._state
def update(self):
2018-10-24 16:59:52 +00:00
"""Ask the Zigbee device what state its input pin is in."""
try:
sample = DEVICE.get_sample(self._config.address)
except ZIGBEE_TX_FAILURE:
_LOGGER.warning(
"Transmission failure when attempting to get sample from "
"ZigBee device at address: %s", hexlify(self._config.address))
return
except ZIGBEE_EXCEPTION as exc:
_LOGGER.exception(
2018-10-24 16:59:52 +00:00
"Unable to get sample from Zigbee device: %s", exc)
return
pin_name = DIGITAL_PINS[self._config.pin]
if pin_name not in sample:
_LOGGER.warning(
2018-10-24 16:59:52 +00:00
"Pin %s (%s) was not in the sample provided by Zigbee device "
"%s.",
self._config.pin, pin_name, hexlify(self._config.address))
return
self._state = self._config.state2bool[sample[pin_name]]
2016-01-24 08:02:14 +00:00
class ZigBeeDigitalOut(ZigBeeDigitalIn):
2016-03-08 16:55:57 +00:00
"""Representation of a GPIO pin configured as a digital input."""
2016-01-24 08:02:14 +00:00
def _set_state(self, state):
2016-03-08 16:55:57 +00:00
"""Initialize the ZigBee digital out device."""
try:
DEVICE.set_gpio_pin(
self._config.pin,
self._config.bool2state[state],
self._config.address)
except ZIGBEE_TX_FAILURE:
_LOGGER.warning(
"Transmission failure when attempting to set output pin on "
"ZigBee device at address: %s", hexlify(self._config.address))
return
except ZIGBEE_EXCEPTION as exc:
_LOGGER.exception(
"Unable to set digital pin on ZigBee device: %s", exc)
return
2016-01-24 08:02:14 +00:00
self._state = state
if not self.should_poll:
self.schedule_update_ha_state()
2016-01-24 08:02:14 +00:00
def turn_on(self, **kwargs):
2016-03-08 16:55:57 +00:00
"""Set the digital output to its 'on' state."""
2016-01-24 08:02:14 +00:00
self._set_state(True)
def turn_off(self, **kwargs):
2016-03-08 16:55:57 +00:00
"""Set the digital output to its 'off' state."""
2016-01-24 08:02:14 +00:00
self._set_state(False)
def update(self):
"""Ask the ZigBee device what its output is set to."""
try:
pin_state = DEVICE.get_gpio_pin(
self._config.pin,
self._config.address)
except ZIGBEE_TX_FAILURE:
_LOGGER.warning(
"Transmission failure when attempting to get output pin status"
" from ZigBee device at address: %s",
hexlify(self._config.address))
return
except ZIGBEE_EXCEPTION as exc:
_LOGGER.exception(
"Unable to get output pin status from ZigBee device: %s", exc)
return
self._state = self._config.state2bool[pin_state]
2016-01-24 08:02:14 +00:00
class ZigBeeAnalogIn(Entity):
2016-03-08 16:55:57 +00:00
"""Representation of a GPIO pin configured as an analog input."""
2016-01-24 08:02:14 +00:00
def __init__(self, hass, config):
2016-03-08 16:55:57 +00:00
"""Initialize the ZigBee analog in device."""
2016-01-24 08:02:14 +00:00
self._config = config
self._value = None
async def async_added_to_hass(self):
"""Register callbacks."""
def handle_frame(frame):
"""Handle an incoming frame.
Handle an incoming frame and update our status if it contains
information relating to this device.
"""
if not frame_is_relevant(self, frame):
return
2016-09-07 01:28:55 +00:00
sample = frame['samples'].pop()
pin_name = ANALOG_PINS[self._config.pin]
if pin_name not in sample:
# Doesn't contain information about our pin
return
self._value = CONVERT_ADC(
sample[pin_name],
ADC_PERCENTAGE,
self._config.max_voltage
)
self.schedule_update_ha_state()
async_dispatcher_connect(
self.hass, SIGNAL_ZIGBEE_FRAME_RECEIVED, handle_frame)
2016-01-24 08:02:14 +00:00
@property
def name(self):
"""Return the name of the input."""
2016-01-24 08:02:14 +00:00
return self._config.name
@property
def config(self):
"""Return the entity's configuration."""
return self._config
2016-01-24 08:02:14 +00:00
@property
def should_poll(self):
"""Return the polling state, if needed."""
2016-01-24 08:02:14 +00:00
return self._config.should_poll
@property
def state(self):
2016-03-08 16:55:57 +00:00
"""Return the state of the entity."""
2016-01-24 08:02:14 +00:00
return self._value
@property
def unit_of_measurement(self):
2016-03-08 16:55:57 +00:00
"""Return the unit this state is expressed in."""
2016-01-24 08:02:14 +00:00
return "%"
def update(self):
2016-03-08 16:55:57 +00:00
"""Get the latest reading from the ADC."""
try:
self._value = DEVICE.read_analog_pin(
self._config.pin,
self._config.max_voltage,
self._config.address,
ADC_PERCENTAGE)
except ZIGBEE_TX_FAILURE:
_LOGGER.warning(
"Transmission failure when attempting to get sample from "
"ZigBee device at address: %s", hexlify(self._config.address))
except ZIGBEE_EXCEPTION as exc:
_LOGGER.exception(
"Unable to get sample from ZigBee device: %s", exc)