2016-07-10 17:36:54 +00:00
|
|
|
"""
|
2017-09-07 07:11:55 +00:00
|
|
|
Connects to KNX platform.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
2016-07-10 17:36:54 +00:00
|
|
|
https://home-assistant.io/components/knx/
|
|
|
|
"""
|
2018-02-24 18:24:33 +00:00
|
|
|
|
2018-01-21 06:35:38 +00:00
|
|
|
import logging
|
2016-07-10 17:36:54 +00:00
|
|
|
|
2016-09-14 06:03:30 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2018-02-26 07:44:09 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_ENTITY_ID, CONF_HOST, CONF_PORT, EVENT_HOMEASSISTANT_STOP)
|
|
|
|
from homeassistant.core import callback
|
2017-09-07 07:11:55 +00:00
|
|
|
from homeassistant.helpers import discovery
|
2017-04-30 05:04:49 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2018-02-26 07:44:09 +00:00
|
|
|
from homeassistant.helpers.event import async_track_state_change
|
2017-09-07 07:11:55 +00:00
|
|
|
from homeassistant.helpers.script import Script
|
|
|
|
|
2018-03-11 04:26:21 +00:00
|
|
|
REQUIREMENTS = ['xknx==0.8.5']
|
2018-01-21 06:35:38 +00:00
|
|
|
|
2017-09-07 07:11:55 +00:00
|
|
|
DOMAIN = "knx"
|
|
|
|
DATA_KNX = "data_knx"
|
|
|
|
CONF_KNX_CONFIG = "config_file"
|
2016-07-10 17:36:54 +00:00
|
|
|
|
2017-09-07 07:11:55 +00:00
|
|
|
CONF_KNX_ROUTING = "routing"
|
|
|
|
CONF_KNX_TUNNELING = "tunneling"
|
|
|
|
CONF_KNX_LOCAL_IP = "local_ip"
|
|
|
|
CONF_KNX_FIRE_EVENT = "fire_event"
|
|
|
|
CONF_KNX_FIRE_EVENT_FILTER = "fire_event_filter"
|
2017-10-15 21:46:55 +00:00
|
|
|
CONF_KNX_STATE_UPDATER = "state_updater"
|
2018-02-26 07:44:09 +00:00
|
|
|
CONF_KNX_EXPOSE = "expose"
|
|
|
|
CONF_KNX_EXPOSE_TYPE = "type"
|
|
|
|
CONF_KNX_EXPOSE_ADDRESS = "address"
|
2017-09-07 07:11:55 +00:00
|
|
|
|
|
|
|
SERVICE_KNX_SEND = "send"
|
|
|
|
SERVICE_KNX_ATTR_ADDRESS = "address"
|
|
|
|
SERVICE_KNX_ATTR_PAYLOAD = "payload"
|
|
|
|
|
|
|
|
ATTR_DISCOVER_DEVICES = 'devices'
|
2016-07-10 17:36:54 +00:00
|
|
|
|
2016-09-14 06:03:30 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2016-07-10 17:36:54 +00:00
|
|
|
|
2017-09-07 07:11:55 +00:00
|
|
|
TUNNELING_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Required(CONF_KNX_LOCAL_IP): cv.string,
|
2018-01-21 06:35:38 +00:00
|
|
|
vol.Optional(CONF_PORT): cv.port,
|
2017-09-07 07:11:55 +00:00
|
|
|
})
|
2016-07-10 17:36:54 +00:00
|
|
|
|
2017-09-07 07:11:55 +00:00
|
|
|
ROUTING_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_KNX_LOCAL_IP): cv.string,
|
|
|
|
})
|
2016-07-10 17:36:54 +00:00
|
|
|
|
2018-02-26 07:44:09 +00:00
|
|
|
EXPOSE_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_KNX_EXPOSE_TYPE): cv.string,
|
|
|
|
vol.Optional(CONF_ENTITY_ID): cv.entity_id,
|
|
|
|
vol.Required(CONF_KNX_EXPOSE_ADDRESS): cv.string,
|
|
|
|
})
|
|
|
|
|
2016-09-14 06:03:30 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
2017-09-07 07:11:55 +00:00
|
|
|
vol.Optional(CONF_KNX_CONFIG): cv.string,
|
|
|
|
vol.Exclusive(CONF_KNX_ROUTING, 'connection_type'): ROUTING_SCHEMA,
|
|
|
|
vol.Exclusive(CONF_KNX_TUNNELING, 'connection_type'):
|
|
|
|
TUNNELING_SCHEMA,
|
|
|
|
vol.Inclusive(CONF_KNX_FIRE_EVENT, 'fire_ev'):
|
|
|
|
cv.boolean,
|
|
|
|
vol.Inclusive(CONF_KNX_FIRE_EVENT_FILTER, 'fire_ev'):
|
2018-01-21 06:35:38 +00:00
|
|
|
vol.All(cv.ensure_list, [cv.string]),
|
2017-10-15 21:46:55 +00:00
|
|
|
vol.Optional(CONF_KNX_STATE_UPDATER, default=True): cv.boolean,
|
2018-02-26 07:44:09 +00:00
|
|
|
vol.Optional(CONF_KNX_EXPOSE):
|
|
|
|
vol.All(
|
|
|
|
cv.ensure_list,
|
|
|
|
[EXPOSE_SCHEMA]),
|
2017-09-07 07:11:55 +00:00
|
|
|
})
|
2016-09-14 06:03:30 +00:00
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
2016-07-10 17:36:54 +00:00
|
|
|
|
2017-09-07 07:11:55 +00:00
|
|
|
SERVICE_KNX_SEND_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(SERVICE_KNX_ATTR_ADDRESS): cv.string,
|
|
|
|
vol.Required(SERVICE_KNX_ATTR_PAYLOAD): vol.Any(
|
|
|
|
cv.positive_int, [cv.positive_int]),
|
2017-07-20 05:01:05 +00:00
|
|
|
})
|
|
|
|
|
2016-07-10 17:36:54 +00:00
|
|
|
|
2018-02-24 18:24:33 +00:00
|
|
|
async def async_setup(hass, config):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Set up the KNX component."""
|
2017-09-07 07:11:55 +00:00
|
|
|
from xknx.exceptions import XKNXException
|
2016-07-10 17:36:54 +00:00
|
|
|
try:
|
2017-09-07 07:11:55 +00:00
|
|
|
hass.data[DATA_KNX] = KNXModule(hass, config)
|
2018-02-26 07:44:09 +00:00
|
|
|
hass.data[DATA_KNX].async_create_exposures()
|
2018-02-24 18:24:33 +00:00
|
|
|
await hass.data[DATA_KNX].start()
|
2017-07-20 05:01:05 +00:00
|
|
|
|
2017-09-07 07:11:55 +00:00
|
|
|
except XKNXException as ex:
|
2018-01-07 21:39:14 +00:00
|
|
|
_LOGGER.warning("Can't connect to KNX interface: %s", ex)
|
|
|
|
hass.components.persistent_notification.async_create(
|
|
|
|
"Can't connect to KNX interface: <br>"
|
|
|
|
"<b>{0}</b>".format(ex),
|
|
|
|
title="KNX")
|
2017-07-07 05:21:40 +00:00
|
|
|
|
2017-09-07 07:11:55 +00:00
|
|
|
for component, discovery_type in (
|
|
|
|
('switch', 'Switch'),
|
|
|
|
('climate', 'Climate'),
|
|
|
|
('cover', 'Cover'),
|
|
|
|
('light', 'Light'),
|
|
|
|
('sensor', 'Sensor'),
|
|
|
|
('binary_sensor', 'BinarySensor'),
|
2018-02-26 07:44:09 +00:00
|
|
|
('scene', 'Scene'),
|
2017-09-07 07:11:55 +00:00
|
|
|
('notify', 'Notification')):
|
|
|
|
found_devices = _get_devices(hass, discovery_type)
|
2018-07-23 12:05:38 +00:00
|
|
|
hass.async_create_task(
|
2017-09-07 07:11:55 +00:00
|
|
|
discovery.async_load_platform(hass, component, DOMAIN, {
|
|
|
|
ATTR_DISCOVER_DEVICES: found_devices
|
|
|
|
}, config))
|
|
|
|
|
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN, SERVICE_KNX_SEND,
|
|
|
|
hass.data[DATA_KNX].service_send_to_knx_bus,
|
|
|
|
schema=SERVICE_KNX_SEND_SCHEMA)
|
2017-07-12 10:21:15 +00:00
|
|
|
|
2016-07-10 17:36:54 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2017-09-07 07:11:55 +00:00
|
|
|
def _get_devices(hass, discovery_type):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Get the KNX devices."""
|
2017-09-07 07:11:55 +00:00
|
|
|
return list(
|
|
|
|
map(lambda device: device.name,
|
|
|
|
filter(
|
|
|
|
lambda device: type(device).__name__ == discovery_type,
|
|
|
|
hass.data[DATA_KNX].xknx.devices)))
|
2016-07-10 17:36:54 +00:00
|
|
|
|
|
|
|
|
2018-07-20 08:45:20 +00:00
|
|
|
class KNXModule:
|
2017-09-07 07:11:55 +00:00
|
|
|
"""Representation of KNX Object."""
|
2016-07-10 17:36:54 +00:00
|
|
|
|
|
|
|
def __init__(self, hass, config):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Initialize of KNX module."""
|
2017-09-07 07:11:55 +00:00
|
|
|
self.hass = hass
|
|
|
|
self.config = config
|
2018-01-07 21:39:14 +00:00
|
|
|
self.connected = False
|
2017-09-07 07:11:55 +00:00
|
|
|
self.init_xknx()
|
|
|
|
self.register_callbacks()
|
2018-02-26 07:44:09 +00:00
|
|
|
self.exposures = []
|
2017-09-07 07:11:55 +00:00
|
|
|
|
|
|
|
def init_xknx(self):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Initialize of KNX object."""
|
2017-09-07 07:11:55 +00:00
|
|
|
from xknx import XKNX
|
2018-01-21 06:35:38 +00:00
|
|
|
self.xknx = XKNX(config=self.config_file(), loop=self.hass.loop)
|
2017-09-07 07:11:55 +00:00
|
|
|
|
2018-02-24 18:24:33 +00:00
|
|
|
async def start(self):
|
2017-09-07 07:11:55 +00:00
|
|
|
"""Start KNX object. Connect to tunneling or Routing device."""
|
|
|
|
connection_config = self.connection_config()
|
2018-02-24 18:24:33 +00:00
|
|
|
await self.xknx.start(
|
2017-10-15 21:46:55 +00:00
|
|
|
state_updater=self.config[DOMAIN][CONF_KNX_STATE_UPDATER],
|
2017-09-07 07:11:55 +00:00
|
|
|
connection_config=connection_config)
|
|
|
|
self.hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, self.stop)
|
2018-01-07 21:39:14 +00:00
|
|
|
self.connected = True
|
2017-09-07 07:11:55 +00:00
|
|
|
|
2018-02-24 18:24:33 +00:00
|
|
|
async def stop(self, event):
|
2017-09-07 07:11:55 +00:00
|
|
|
"""Stop KNX object. Disconnect from tunneling or Routing device."""
|
2018-02-24 18:24:33 +00:00
|
|
|
await self.xknx.stop()
|
2017-09-07 07:11:55 +00:00
|
|
|
|
|
|
|
def config_file(self):
|
|
|
|
"""Resolve and return the full path of xknx.yaml if configured."""
|
|
|
|
config_file = self.config[DOMAIN].get(CONF_KNX_CONFIG)
|
|
|
|
if not config_file:
|
|
|
|
return None
|
|
|
|
if not config_file.startswith("/"):
|
|
|
|
return self.hass.config.path(config_file)
|
|
|
|
return config_file
|
|
|
|
|
|
|
|
def connection_config(self):
|
|
|
|
"""Return the connection_config."""
|
|
|
|
if CONF_KNX_TUNNELING in self.config[DOMAIN]:
|
|
|
|
return self.connection_config_tunneling()
|
2018-07-23 08:16:05 +00:00
|
|
|
if CONF_KNX_ROUTING in self.config[DOMAIN]:
|
2017-09-07 07:11:55 +00:00
|
|
|
return self.connection_config_routing()
|
|
|
|
return self.connection_config_auto()
|
|
|
|
|
|
|
|
def connection_config_routing(self):
|
|
|
|
"""Return the connection_config if routing is configured."""
|
|
|
|
from xknx.io import ConnectionConfig, ConnectionType
|
|
|
|
local_ip = \
|
|
|
|
self.config[DOMAIN][CONF_KNX_ROUTING].get(CONF_KNX_LOCAL_IP)
|
|
|
|
return ConnectionConfig(
|
|
|
|
connection_type=ConnectionType.ROUTING,
|
|
|
|
local_ip=local_ip)
|
|
|
|
|
|
|
|
def connection_config_tunneling(self):
|
|
|
|
"""Return the connection_config if tunneling is configured."""
|
|
|
|
from xknx.io import ConnectionConfig, ConnectionType, \
|
|
|
|
DEFAULT_MCAST_PORT
|
|
|
|
gateway_ip = \
|
|
|
|
self.config[DOMAIN][CONF_KNX_TUNNELING].get(CONF_HOST)
|
|
|
|
gateway_port = \
|
|
|
|
self.config[DOMAIN][CONF_KNX_TUNNELING].get(CONF_PORT)
|
|
|
|
local_ip = \
|
|
|
|
self.config[DOMAIN][CONF_KNX_TUNNELING].get(CONF_KNX_LOCAL_IP)
|
|
|
|
if gateway_port is None:
|
|
|
|
gateway_port = DEFAULT_MCAST_PORT
|
|
|
|
return ConnectionConfig(
|
2018-01-21 06:35:38 +00:00
|
|
|
connection_type=ConnectionType.TUNNELING, gateway_ip=gateway_ip,
|
|
|
|
gateway_port=gateway_port, local_ip=local_ip)
|
2017-09-07 07:11:55 +00:00
|
|
|
|
|
|
|
def connection_config_auto(self):
|
|
|
|
"""Return the connection_config if auto is configured."""
|
|
|
|
# pylint: disable=no-self-use
|
|
|
|
from xknx.io import ConnectionConfig
|
|
|
|
return ConnectionConfig()
|
|
|
|
|
|
|
|
def register_callbacks(self):
|
|
|
|
"""Register callbacks within XKNX object."""
|
|
|
|
if CONF_KNX_FIRE_EVENT in self.config[DOMAIN] and \
|
|
|
|
self.config[DOMAIN][CONF_KNX_FIRE_EVENT]:
|
|
|
|
from xknx.knx import AddressFilter
|
|
|
|
address_filters = list(map(
|
|
|
|
AddressFilter,
|
|
|
|
self.config[DOMAIN][CONF_KNX_FIRE_EVENT_FILTER]))
|
|
|
|
self.xknx.telegram_queue.register_telegram_received_cb(
|
|
|
|
self.telegram_received_cb, address_filters)
|
|
|
|
|
2018-02-26 07:44:09 +00:00
|
|
|
@callback
|
|
|
|
def async_create_exposures(self):
|
|
|
|
"""Create exposures."""
|
|
|
|
if CONF_KNX_EXPOSE not in self.config[DOMAIN]:
|
|
|
|
return
|
|
|
|
for to_expose in self.config[DOMAIN][CONF_KNX_EXPOSE]:
|
|
|
|
expose_type = to_expose.get(CONF_KNX_EXPOSE_TYPE)
|
|
|
|
entity_id = to_expose.get(CONF_ENTITY_ID)
|
|
|
|
address = to_expose.get(CONF_KNX_EXPOSE_ADDRESS)
|
|
|
|
if expose_type in ['time', 'date', 'datetime']:
|
|
|
|
exposure = KNXExposeTime(
|
|
|
|
self.xknx, expose_type, address)
|
|
|
|
exposure.async_register()
|
|
|
|
self.exposures.append(exposure)
|
|
|
|
else:
|
|
|
|
exposure = KNXExposeSensor(
|
|
|
|
self.hass, self.xknx, expose_type, entity_id, address)
|
|
|
|
exposure.async_register()
|
|
|
|
self.exposures.append(exposure)
|
|
|
|
|
2018-02-24 18:24:33 +00:00
|
|
|
async def telegram_received_cb(self, telegram):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Call invoked after a KNX telegram was received."""
|
2017-09-07 07:11:55 +00:00
|
|
|
self.hass.bus.fire('knx_event', {
|
2018-03-11 04:26:21 +00:00
|
|
|
'address': str(telegram.group_address),
|
2017-09-07 07:11:55 +00:00
|
|
|
'data': telegram.payload.value
|
|
|
|
})
|
|
|
|
# False signals XKNX to proceed with processing telegrams.
|
2016-07-10 17:36:54 +00:00
|
|
|
return False
|
|
|
|
|
2018-02-24 18:24:33 +00:00
|
|
|
async def service_send_to_knx_bus(self, call):
|
2017-09-23 15:15:46 +00:00
|
|
|
"""Service for sending an arbitrary KNX message to the KNX bus."""
|
2018-02-15 06:06:36 +00:00
|
|
|
from xknx.knx import Telegram, GroupAddress, DPTBinary, DPTArray
|
2017-09-07 07:11:55 +00:00
|
|
|
attr_payload = call.data.get(SERVICE_KNX_ATTR_PAYLOAD)
|
|
|
|
attr_address = call.data.get(SERVICE_KNX_ATTR_ADDRESS)
|
|
|
|
|
|
|
|
def calculate_payload(attr_payload):
|
|
|
|
"""Calculate payload depending on type of attribute."""
|
|
|
|
if isinstance(attr_payload, int):
|
|
|
|
return DPTBinary(attr_payload)
|
|
|
|
return DPTArray(attr_payload)
|
|
|
|
payload = calculate_payload(attr_payload)
|
2018-02-15 06:06:36 +00:00
|
|
|
address = GroupAddress(attr_address)
|
2017-09-07 07:11:55 +00:00
|
|
|
|
|
|
|
telegram = Telegram()
|
|
|
|
telegram.payload = payload
|
|
|
|
telegram.group_address = address
|
2018-02-24 18:24:33 +00:00
|
|
|
await self.xknx.telegrams.put(telegram)
|
2017-09-07 07:11:55 +00:00
|
|
|
|
|
|
|
|
|
|
|
class KNXAutomation():
|
|
|
|
"""Wrapper around xknx.devices.ActionCallback object.."""
|
|
|
|
|
|
|
|
def __init__(self, hass, device, hook, action, counter=1):
|
|
|
|
"""Initialize Automation class."""
|
|
|
|
self.hass = hass
|
|
|
|
self.device = device
|
|
|
|
script_name = "{} turn ON script".format(device.get_name())
|
|
|
|
self.script = Script(hass, action, script_name)
|
|
|
|
|
|
|
|
import xknx
|
|
|
|
self.action = xknx.devices.ActionCallback(
|
2018-01-21 06:35:38 +00:00
|
|
|
hass.data[DATA_KNX].xknx, self.script.async_run,
|
|
|
|
hook=hook, counter=counter)
|
2017-09-07 07:11:55 +00:00
|
|
|
device.actions.append(self.action)
|
2018-02-26 07:44:09 +00:00
|
|
|
|
|
|
|
|
2018-07-20 08:45:20 +00:00
|
|
|
class KNXExposeTime:
|
2018-02-26 07:44:09 +00:00
|
|
|
"""Object to Expose Time/Date object to KNX bus."""
|
|
|
|
|
|
|
|
def __init__(self, xknx, expose_type, address):
|
|
|
|
"""Initialize of Expose class."""
|
|
|
|
self.xknx = xknx
|
|
|
|
self.type = expose_type
|
|
|
|
self.address = address
|
|
|
|
self.device = None
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_register(self):
|
|
|
|
"""Register listener."""
|
|
|
|
from xknx.devices import DateTime, DateTimeBroadcastType
|
|
|
|
broadcast_type_string = self.type.upper()
|
|
|
|
broadcast_type = DateTimeBroadcastType[broadcast_type_string]
|
|
|
|
self.device = DateTime(
|
|
|
|
self.xknx,
|
|
|
|
'Time',
|
|
|
|
broadcast_type=broadcast_type,
|
|
|
|
group_address=self.address)
|
|
|
|
self.xknx.devices.add(self.device)
|
|
|
|
|
|
|
|
|
2018-07-20 08:45:20 +00:00
|
|
|
class KNXExposeSensor:
|
2018-02-26 07:44:09 +00:00
|
|
|
"""Object to Expose HASS entity to KNX bus."""
|
|
|
|
|
|
|
|
def __init__(self, hass, xknx, expose_type, entity_id, address):
|
|
|
|
"""Initialize of Expose class."""
|
|
|
|
self.hass = hass
|
|
|
|
self.xknx = xknx
|
|
|
|
self.type = expose_type
|
|
|
|
self.entity_id = entity_id
|
|
|
|
self.address = address
|
|
|
|
self.device = None
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_register(self):
|
|
|
|
"""Register listener."""
|
|
|
|
from xknx.devices import ExposeSensor
|
|
|
|
self.device = ExposeSensor(
|
|
|
|
self.xknx,
|
|
|
|
name=self.entity_id,
|
|
|
|
group_address=self.address,
|
|
|
|
value_type=self.type)
|
|
|
|
self.xknx.devices.add(self.device)
|
|
|
|
async_track_state_change(
|
|
|
|
self.hass, self.entity_id, self._async_entity_changed)
|
|
|
|
|
|
|
|
async def _async_entity_changed(self, entity_id, old_state, new_state):
|
|
|
|
"""Callback after entity changed."""
|
|
|
|
if new_state is None:
|
|
|
|
return
|
|
|
|
await self.device.set(float(new_state.state))
|