2018-02-25 09:58:13 +00:00
|
|
|
"""Support for Apple HomeKit.
|
2018-02-19 22:46:22 +00:00
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/homekit/
|
|
|
|
"""
|
2018-04-30 12:58:17 +00:00
|
|
|
import ipaddress
|
2018-02-19 22:46:22 +00:00
|
|
|
import logging
|
2018-03-15 01:48:21 +00:00
|
|
|
from zlib import adler32
|
2018-02-19 22:46:22 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2018-07-18 09:54:27 +00:00
|
|
|
from homeassistant.components import cover
|
2018-03-15 01:48:21 +00:00
|
|
|
from homeassistant.const import (
|
2018-05-11 12:22:45 +00:00
|
|
|
ATTR_DEVICE_CLASS, ATTR_SUPPORTED_FEATURES, ATTR_UNIT_OF_MEASUREMENT,
|
2018-06-01 16:04:54 +00:00
|
|
|
CONF_IP_ADDRESS, CONF_NAME, CONF_PORT, CONF_TYPE, DEVICE_CLASS_HUMIDITY,
|
2018-05-28 14:26:33 +00:00
|
|
|
DEVICE_CLASS_ILLUMINANCE, DEVICE_CLASS_TEMPERATURE,
|
2018-05-11 12:22:45 +00:00
|
|
|
EVENT_HOMEASSISTANT_START, EVENT_HOMEASSISTANT_STOP,
|
2018-05-21 02:25:53 +00:00
|
|
|
TEMP_CELSIUS, TEMP_FAHRENHEIT)
|
2018-03-15 01:48:21 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
from homeassistant.helpers.entityfilter import FILTER_SCHEMA
|
2018-02-19 22:46:22 +00:00
|
|
|
from homeassistant.util import get_local_ip
|
|
|
|
from homeassistant.util.decorator import Registry
|
2018-03-15 01:48:21 +00:00
|
|
|
from .const import (
|
2018-07-22 07:51:42 +00:00
|
|
|
BRIDGE_NAME, CONF_AUTO_START, CONF_ENTITY_CONFIG, CONF_FEATURE_LIST,
|
2018-09-21 10:51:02 +00:00
|
|
|
CONF_FILTER, DEFAULT_AUTO_START, DEFAULT_PORT, DEVICE_CLASS_CO,
|
|
|
|
DEVICE_CLASS_CO2, DEVICE_CLASS_PM25, DOMAIN, HOMEKIT_FILE,
|
2018-10-05 10:43:50 +00:00
|
|
|
SERVICE_HOMEKIT_START, TYPE_FAUCET, TYPE_OUTLET, TYPE_SHOWER,
|
|
|
|
TYPE_SPRINKLER, TYPE_SWITCH, TYPE_VALVE)
|
2018-05-25 09:37:20 +00:00
|
|
|
from .util import (
|
2018-05-28 14:26:33 +00:00
|
|
|
show_setup_message, validate_entity_config, validate_media_player_features)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-11-01 18:35:02 +00:00
|
|
|
REQUIREMENTS = ['HAP-python==2.2.2']
|
2018-10-05 10:32:26 +00:00
|
|
|
|
2018-02-19 22:46:22 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2018-10-05 10:32:26 +00:00
|
|
|
MAX_DEVICES = 100
|
|
|
|
TYPES = Registry()
|
2018-05-04 14:46:00 +00:00
|
|
|
|
|
|
|
# #### Driver Status ####
|
|
|
|
STATUS_READY = 0
|
|
|
|
STATUS_RUNNING = 1
|
|
|
|
STATUS_STOPPED = 2
|
|
|
|
STATUS_WAIT = 3
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-10-05 10:43:50 +00:00
|
|
|
SWITCH_TYPES = {
|
|
|
|
TYPE_FAUCET: 'Valve',
|
|
|
|
TYPE_OUTLET: 'Outlet',
|
|
|
|
TYPE_SHOWER: 'Valve',
|
|
|
|
TYPE_SPRINKLER: 'Valve',
|
|
|
|
TYPE_SWITCH: 'Switch',
|
|
|
|
TYPE_VALVE: 'Valve'}
|
2018-02-19 22:46:22 +00:00
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.All({
|
2018-07-22 07:51:42 +00:00
|
|
|
vol.Optional(CONF_NAME, default=BRIDGE_NAME):
|
|
|
|
vol.All(cv.string, vol.Length(min=3, max=25)),
|
2018-03-15 01:48:21 +00:00
|
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
2018-04-30 12:58:17 +00:00
|
|
|
vol.Optional(CONF_IP_ADDRESS):
|
|
|
|
vol.All(ipaddress.ip_address, cv.string),
|
2018-03-15 01:48:21 +00:00
|
|
|
vol.Optional(CONF_AUTO_START, default=DEFAULT_AUTO_START): cv.boolean,
|
|
|
|
vol.Optional(CONF_FILTER, default={}): FILTER_SCHEMA,
|
|
|
|
vol.Optional(CONF_ENTITY_CONFIG, default={}): validate_entity_config,
|
2018-02-19 22:46:22 +00:00
|
|
|
})
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
async def async_setup(hass, config):
|
2018-08-19 20:29:08 +00:00
|
|
|
"""Set up the HomeKit component."""
|
2018-03-15 01:48:21 +00:00
|
|
|
_LOGGER.debug('Begin setup HomeKit')
|
2018-02-19 22:46:22 +00:00
|
|
|
|
|
|
|
conf = config[DOMAIN]
|
2018-07-22 07:51:42 +00:00
|
|
|
name = conf[CONF_NAME]
|
2018-03-15 01:48:21 +00:00
|
|
|
port = conf[CONF_PORT]
|
2018-04-30 12:58:17 +00:00
|
|
|
ip_address = conf.get(CONF_IP_ADDRESS)
|
2018-03-15 01:48:21 +00:00
|
|
|
auto_start = conf[CONF_AUTO_START]
|
|
|
|
entity_filter = conf[CONF_FILTER]
|
|
|
|
entity_config = conf[CONF_ENTITY_CONFIG]
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-07-22 07:51:42 +00:00
|
|
|
homekit = HomeKit(hass, name, port, ip_address, entity_filter,
|
|
|
|
entity_config)
|
2018-10-19 22:14:05 +00:00
|
|
|
await hass.async_add_executor_job(homekit.setup)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
if auto_start:
|
|
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_START, homekit.start)
|
|
|
|
return True
|
|
|
|
|
|
|
|
def handle_homekit_service_start(service):
|
|
|
|
"""Handle start HomeKit service call."""
|
2018-05-04 14:46:00 +00:00
|
|
|
if homekit.status != STATUS_READY:
|
|
|
|
_LOGGER.warning(
|
|
|
|
'HomeKit is not ready. Either it is already running or has '
|
|
|
|
'been stopped.')
|
2018-03-15 01:48:21 +00:00
|
|
|
return
|
|
|
|
homekit.start()
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
hass.services.async_register(DOMAIN, SERVICE_HOMEKIT_START,
|
|
|
|
handle_homekit_service_start)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
return True
|
2018-02-19 22:46:22 +00:00
|
|
|
|
|
|
|
|
2018-05-29 20:43:26 +00:00
|
|
|
def get_accessory(hass, driver, state, aid, config):
|
2018-02-19 22:46:22 +00:00
|
|
|
"""Take state and return an accessory object if supported."""
|
2018-03-15 01:48:21 +00:00
|
|
|
if not aid:
|
|
|
|
_LOGGER.warning('The entitiy "%s" is not supported, since it '
|
|
|
|
'generates an invalid aid, please change it.',
|
|
|
|
state.entity_id)
|
|
|
|
return None
|
|
|
|
|
2018-04-11 20:24:14 +00:00
|
|
|
a_type = None
|
2018-05-11 12:22:45 +00:00
|
|
|
name = config.get(CONF_NAME, state.name)
|
2018-04-11 20:24:14 +00:00
|
|
|
|
|
|
|
if state.domain == 'alarm_control_panel':
|
|
|
|
a_type = 'SecuritySystem'
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-04-09 13:32:29 +00:00
|
|
|
elif state.domain == 'binary_sensor' or state.domain == 'device_tracker':
|
2018-04-11 20:24:14 +00:00
|
|
|
a_type = 'BinarySensor'
|
|
|
|
|
|
|
|
elif state.domain == 'climate':
|
|
|
|
a_type = 'Thermostat'
|
2018-04-09 13:32:29 +00:00
|
|
|
|
2018-02-19 22:46:22 +00:00
|
|
|
elif state.domain == 'cover':
|
2018-04-12 16:08:48 +00:00
|
|
|
device_class = state.attributes.get(ATTR_DEVICE_CLASS)
|
2018-06-17 11:37:44 +00:00
|
|
|
features = state.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
2018-04-12 16:08:48 +00:00
|
|
|
|
|
|
|
if device_class == 'garage' and \
|
2018-05-28 14:26:33 +00:00
|
|
|
features & (cover.SUPPORT_OPEN | cover.SUPPORT_CLOSE):
|
2018-04-12 16:08:48 +00:00
|
|
|
a_type = 'GarageDoorOpener'
|
2018-05-28 14:26:33 +00:00
|
|
|
elif features & cover.SUPPORT_SET_POSITION:
|
2018-04-11 20:24:14 +00:00
|
|
|
a_type = 'WindowCovering'
|
2018-05-28 14:26:33 +00:00
|
|
|
elif features & (cover.SUPPORT_OPEN | cover.SUPPORT_CLOSE):
|
2018-04-18 12:39:58 +00:00
|
|
|
a_type = 'WindowCoveringBasic'
|
2018-03-07 12:17:52 +00:00
|
|
|
|
2018-05-16 11:15:59 +00:00
|
|
|
elif state.domain == 'fan':
|
|
|
|
a_type = 'Fan'
|
|
|
|
|
2018-03-16 00:05:28 +00:00
|
|
|
elif state.domain == 'light':
|
2018-04-11 20:24:14 +00:00
|
|
|
a_type = 'Light'
|
2018-03-16 00:05:28 +00:00
|
|
|
|
2018-04-09 14:23:49 +00:00
|
|
|
elif state.domain == 'lock':
|
2018-04-11 20:24:14 +00:00
|
|
|
a_type = 'Lock'
|
|
|
|
|
2018-05-25 09:37:20 +00:00
|
|
|
elif state.domain == 'media_player':
|
2018-05-28 14:26:33 +00:00
|
|
|
feature_list = config.get(CONF_FEATURE_LIST)
|
|
|
|
if feature_list and \
|
|
|
|
validate_media_player_features(state, feature_list):
|
2018-05-25 09:37:20 +00:00
|
|
|
a_type = 'MediaPlayer'
|
|
|
|
|
2018-04-11 20:24:14 +00:00
|
|
|
elif state.domain == 'sensor':
|
2018-04-12 13:01:41 +00:00
|
|
|
device_class = state.attributes.get(ATTR_DEVICE_CLASS)
|
2018-06-17 11:37:44 +00:00
|
|
|
unit = state.attributes.get(ATTR_UNIT_OF_MEASUREMENT)
|
2018-04-12 13:01:41 +00:00
|
|
|
|
2018-05-04 20:48:38 +00:00
|
|
|
if device_class == DEVICE_CLASS_TEMPERATURE or \
|
|
|
|
unit in (TEMP_CELSIUS, TEMP_FAHRENHEIT):
|
2018-04-11 20:24:14 +00:00
|
|
|
a_type = 'TemperatureSensor'
|
2018-05-04 20:48:38 +00:00
|
|
|
elif device_class == DEVICE_CLASS_HUMIDITY and unit == '%':
|
2018-04-11 20:24:14 +00:00
|
|
|
a_type = 'HumiditySensor'
|
2018-04-12 13:01:41 +00:00
|
|
|
elif device_class == DEVICE_CLASS_PM25 \
|
|
|
|
or DEVICE_CLASS_PM25 in state.entity_id:
|
|
|
|
a_type = 'AirQualitySensor'
|
2018-09-21 10:51:02 +00:00
|
|
|
elif device_class == DEVICE_CLASS_CO:
|
|
|
|
a_type = 'CarbonMonoxideSensor'
|
2018-04-12 13:01:41 +00:00
|
|
|
elif device_class == DEVICE_CLASS_CO2 \
|
|
|
|
or DEVICE_CLASS_CO2 in state.entity_id:
|
|
|
|
a_type = 'CarbonDioxideSensor'
|
2018-05-05 13:37:40 +00:00
|
|
|
elif device_class == DEVICE_CLASS_ILLUMINANCE or unit in ('lm', 'lx'):
|
2018-04-12 13:01:41 +00:00
|
|
|
a_type = 'LightSensor'
|
2018-04-09 14:23:49 +00:00
|
|
|
|
2018-06-01 16:04:54 +00:00
|
|
|
elif state.domain == 'switch':
|
|
|
|
switch_type = config.get(CONF_TYPE, TYPE_SWITCH)
|
|
|
|
a_type = SWITCH_TYPES[switch_type]
|
|
|
|
|
2018-11-05 20:36:30 +00:00
|
|
|
elif state.domain in ('automation', 'input_boolean', 'remote', 'scene',
|
|
|
|
'script'):
|
2018-04-11 20:24:14 +00:00
|
|
|
a_type = 'Switch'
|
|
|
|
|
2018-10-19 19:04:05 +00:00
|
|
|
elif state.domain == 'water_heater':
|
|
|
|
a_type = 'WaterHeater'
|
|
|
|
|
2018-04-11 20:24:14 +00:00
|
|
|
if a_type is None:
|
|
|
|
return None
|
2018-03-07 12:17:52 +00:00
|
|
|
|
2018-04-11 20:24:14 +00:00
|
|
|
_LOGGER.debug('Add "%s" as "%s"', state.entity_id, a_type)
|
2018-05-29 20:43:26 +00:00
|
|
|
return TYPES[a_type](hass, driver, name, state.entity_id, aid, config)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
def generate_aid(entity_id):
|
|
|
|
"""Generate accessory aid with zlib adler32."""
|
|
|
|
aid = adler32(entity_id.encode('utf-8'))
|
2018-07-17 17:34:29 +00:00
|
|
|
if aid in (0, 1):
|
2018-03-15 01:48:21 +00:00
|
|
|
return None
|
|
|
|
return aid
|
|
|
|
|
|
|
|
|
2018-02-25 09:58:13 +00:00
|
|
|
class HomeKit():
|
|
|
|
"""Class to handle all actions between HomeKit and Home Assistant."""
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-07-22 07:51:42 +00:00
|
|
|
def __init__(self, hass, name, port, ip_address, entity_filter,
|
|
|
|
entity_config):
|
2018-02-25 09:58:13 +00:00
|
|
|
"""Initialize a HomeKit object."""
|
2018-04-11 20:24:14 +00:00
|
|
|
self.hass = hass
|
2018-07-22 07:51:42 +00:00
|
|
|
self._name = name
|
2018-02-19 22:46:22 +00:00
|
|
|
self._port = port
|
2018-04-30 12:58:17 +00:00
|
|
|
self._ip_address = ip_address
|
2018-03-15 01:48:21 +00:00
|
|
|
self._filter = entity_filter
|
|
|
|
self._config = entity_config
|
2018-05-04 14:46:00 +00:00
|
|
|
self.status = STATUS_READY
|
2018-03-15 01:48:21 +00:00
|
|
|
|
2018-02-19 22:46:22 +00:00
|
|
|
self.bridge = None
|
|
|
|
self.driver = None
|
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
def setup(self):
|
2018-08-19 20:29:08 +00:00
|
|
|
"""Set up bridge and accessory driver."""
|
2018-03-15 01:48:21 +00:00
|
|
|
from .accessories import HomeBridge, HomeDriver
|
|
|
|
|
2018-04-11 20:24:14 +00:00
|
|
|
self.hass.bus.async_listen_once(
|
2018-03-15 01:48:21 +00:00
|
|
|
EVENT_HOMEASSISTANT_STOP, self.stop)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-04-30 12:58:17 +00:00
|
|
|
ip_addr = self._ip_address or get_local_ip()
|
2018-04-11 20:24:14 +00:00
|
|
|
path = self.hass.config.path(HOMEKIT_FILE)
|
2018-05-29 20:43:26 +00:00
|
|
|
self.driver = HomeDriver(self.hass, address=ip_addr,
|
|
|
|
port=self._port, persist_file=path)
|
2018-07-22 07:51:42 +00:00
|
|
|
self.bridge = HomeBridge(self.hass, self.driver, self._name)
|
2018-03-15 01:48:21 +00:00
|
|
|
|
|
|
|
def add_bridge_accessory(self, state):
|
|
|
|
"""Try adding accessory to bridge if configured beforehand."""
|
|
|
|
if not state or not self._filter(state.entity_id):
|
|
|
|
return
|
|
|
|
aid = generate_aid(state.entity_id)
|
|
|
|
conf = self._config.pop(state.entity_id, {})
|
2018-05-29 20:43:26 +00:00
|
|
|
acc = get_accessory(self.hass, self.driver, state, aid, conf)
|
2018-03-15 01:48:21 +00:00
|
|
|
if acc is not None:
|
|
|
|
self.bridge.add_accessory(acc)
|
|
|
|
|
|
|
|
def start(self, *args):
|
2018-02-19 22:46:22 +00:00
|
|
|
"""Start the accessory driver."""
|
2018-05-04 14:46:00 +00:00
|
|
|
if self.status != STATUS_READY:
|
2018-03-15 01:48:21 +00:00
|
|
|
return
|
2018-05-04 14:46:00 +00:00
|
|
|
self.status = STATUS_WAIT
|
2018-03-15 01:48:21 +00:00
|
|
|
|
|
|
|
# pylint: disable=unused-variable
|
|
|
|
from . import ( # noqa F401
|
2018-05-16 11:15:59 +00:00
|
|
|
type_covers, type_fans, type_lights, type_locks,
|
2018-05-25 09:37:20 +00:00
|
|
|
type_media_players, type_security_systems, type_sensors,
|
|
|
|
type_switches, type_thermostats)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-04-11 20:24:14 +00:00
|
|
|
for state in self.hass.states.all():
|
2018-03-15 01:48:21 +00:00
|
|
|
self.add_bridge_accessory(state)
|
2018-05-29 20:43:26 +00:00
|
|
|
self.driver.add_accessory(self.bridge)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-05-18 14:32:57 +00:00
|
|
|
if not self.driver.state.paired:
|
|
|
|
show_setup_message(self.hass, self.driver.state.pincode)
|
2018-03-15 01:48:21 +00:00
|
|
|
|
2018-10-05 10:32:26 +00:00
|
|
|
if len(self.bridge.accessories) > MAX_DEVICES:
|
|
|
|
_LOGGER.warning('You have exceeded the device limit, which might '
|
|
|
|
'cause issues. Consider using the filter option.')
|
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
_LOGGER.debug('Driver start')
|
2018-05-04 14:46:00 +00:00
|
|
|
self.hass.add_job(self.driver.start)
|
|
|
|
self.status = STATUS_RUNNING
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
def stop(self, *args):
|
2018-02-19 22:46:22 +00:00
|
|
|
"""Stop the accessory driver."""
|
2018-05-04 14:46:00 +00:00
|
|
|
if self.status != STATUS_RUNNING:
|
2018-03-15 01:48:21 +00:00
|
|
|
return
|
2018-05-04 14:46:00 +00:00
|
|
|
self.status = STATUS_STOPPED
|
2018-03-15 01:48:21 +00:00
|
|
|
|
|
|
|
_LOGGER.debug('Driver stop')
|
2018-05-04 14:46:00 +00:00
|
|
|
self.hass.add_job(self.driver.stop)
|