2019-02-14 15:01:46 +00:00
|
|
|
"""Support for Apple HomeKit."""
|
2018-04-30 12:58:17 +00:00
|
|
|
import ipaddress
|
2018-02-19 22:46:22 +00:00
|
|
|
import logging
|
|
|
|
|
2020-04-21 22:38:43 +00:00
|
|
|
from aiohttp import web
|
2018-02-19 22:46:22 +00:00
|
|
|
import voluptuous as vol
|
2020-04-12 14:59:50 +00:00
|
|
|
from zeroconf import InterfaceChoice
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2020-04-20 15:00:52 +00:00
|
|
|
from homeassistant.components import cover, vacuum
|
2020-04-22 00:43:49 +00:00
|
|
|
from homeassistant.components.binary_sensor import DEVICE_CLASS_BATTERY_CHARGING
|
2020-04-10 12:42:34 +00:00
|
|
|
from homeassistant.components.cover import DEVICE_CLASS_GARAGE, DEVICE_CLASS_GATE
|
2020-04-21 22:38:43 +00:00
|
|
|
from homeassistant.components.http import HomeAssistantView
|
2019-05-05 15:51:48 +00:00
|
|
|
from homeassistant.components.media_player import DEVICE_CLASS_TV
|
2018-03-15 01:48:21 +00:00
|
|
|
from homeassistant.const import (
|
2020-04-22 00:43:49 +00:00
|
|
|
ATTR_BATTERY_CHARGING,
|
|
|
|
ATTR_BATTERY_LEVEL,
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_DEVICE_CLASS,
|
|
|
|
ATTR_ENTITY_ID,
|
2020-04-21 00:48:09 +00:00
|
|
|
ATTR_SERVICE,
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_SUPPORTED_FEATURES,
|
|
|
|
ATTR_UNIT_OF_MEASUREMENT,
|
|
|
|
CONF_IP_ADDRESS,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_PORT,
|
|
|
|
CONF_TYPE,
|
2020-04-22 00:43:49 +00:00
|
|
|
DEVICE_CLASS_BATTERY,
|
2019-07-31 19:25:30 +00:00
|
|
|
DEVICE_CLASS_HUMIDITY,
|
|
|
|
DEVICE_CLASS_ILLUMINANCE,
|
|
|
|
DEVICE_CLASS_TEMPERATURE,
|
|
|
|
EVENT_HOMEASSISTANT_START,
|
|
|
|
EVENT_HOMEASSISTANT_STOP,
|
|
|
|
TEMP_CELSIUS,
|
|
|
|
TEMP_FAHRENHEIT,
|
2020-02-28 19:46:48 +00:00
|
|
|
UNIT_PERCENTAGE,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2020-04-21 00:48:09 +00:00
|
|
|
from homeassistant.core import callback
|
2020-04-21 22:38:43 +00:00
|
|
|
from homeassistant.exceptions import Unauthorized
|
2020-04-22 00:43:49 +00:00
|
|
|
from homeassistant.helpers import entity_registry
|
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
|
2019-02-14 15:01:46 +00:00
|
|
|
|
2020-04-19 18:51:09 +00:00
|
|
|
from .aidmanager import AccessoryAidStorage
|
2018-03-15 01:48:21 +00:00
|
|
|
from .const import (
|
2020-04-19 18:51:09 +00:00
|
|
|
AID_STORAGE,
|
2020-04-21 00:48:09 +00:00
|
|
|
ATTR_DISPLAY_NAME,
|
|
|
|
ATTR_VALUE,
|
2019-07-31 19:25:30 +00:00
|
|
|
BRIDGE_NAME,
|
2019-10-23 05:06:21 +00:00
|
|
|
CONF_ADVERTISE_IP,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_AUTO_START,
|
|
|
|
CONF_ENTITY_CONFIG,
|
|
|
|
CONF_FEATURE_LIST,
|
|
|
|
CONF_FILTER,
|
2020-04-22 00:43:49 +00:00
|
|
|
CONF_LINKED_BATTERY_CHARGING_SENSOR,
|
|
|
|
CONF_LINKED_BATTERY_SENSOR,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_SAFE_MODE,
|
2020-04-12 14:59:50 +00:00
|
|
|
CONF_ZEROCONF_DEFAULT_INTERFACE,
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_AUTO_START,
|
|
|
|
DEFAULT_PORT,
|
|
|
|
DEFAULT_SAFE_MODE,
|
2020-04-12 14:59:50 +00:00
|
|
|
DEFAULT_ZEROCONF_DEFAULT_INTERFACE,
|
2019-07-31 19:25:30 +00:00
|
|
|
DEVICE_CLASS_CO,
|
|
|
|
DEVICE_CLASS_CO2,
|
|
|
|
DEVICE_CLASS_PM25,
|
|
|
|
DOMAIN,
|
2020-04-21 00:48:09 +00:00
|
|
|
EVENT_HOMEKIT_CHANGED,
|
2019-07-31 19:25:30 +00:00
|
|
|
HOMEKIT_FILE,
|
2020-04-21 22:38:43 +00:00
|
|
|
HOMEKIT_PAIRING_QR,
|
|
|
|
HOMEKIT_PAIRING_QR_SECRET,
|
2019-07-31 19:25:30 +00:00
|
|
|
SERVICE_HOMEKIT_RESET_ACCESSORY,
|
2019-12-08 17:16:49 +00:00
|
|
|
SERVICE_HOMEKIT_START,
|
2019-07-31 19:25:30 +00:00
|
|
|
TYPE_FAUCET,
|
|
|
|
TYPE_OUTLET,
|
|
|
|
TYPE_SHOWER,
|
|
|
|
TYPE_SPRINKLER,
|
|
|
|
TYPE_SWITCH,
|
|
|
|
TYPE_VALVE,
|
|
|
|
)
|
2018-05-25 09:37:20 +00:00
|
|
|
from .util import (
|
2019-07-31 19:25:30 +00:00
|
|
|
show_setup_message,
|
|
|
|
validate_entity_config,
|
|
|
|
validate_media_player_features,
|
|
|
|
)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2020-04-16 02:40:38 +00:00
|
|
|
MAX_DEVICES = 150
|
2018-10-05 10:32:26 +00:00
|
|
|
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 = {
|
2019-07-31 19:25:30 +00:00
|
|
|
TYPE_FAUCET: "Valve",
|
|
|
|
TYPE_OUTLET: "Outlet",
|
|
|
|
TYPE_SHOWER: "Valve",
|
|
|
|
TYPE_SPRINKLER: "Valve",
|
|
|
|
TYPE_SWITCH: "Switch",
|
|
|
|
TYPE_VALVE: "Valve",
|
|
|
|
}
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
DOMAIN: vol.All(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_NAME, default=BRIDGE_NAME): vol.All(
|
|
|
|
cv.string, vol.Length(min=3, max=25)
|
|
|
|
),
|
|
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
|
|
|
vol.Optional(CONF_IP_ADDRESS): vol.All(ipaddress.ip_address, cv.string),
|
2019-10-23 05:06:21 +00:00
|
|
|
vol.Optional(CONF_ADVERTISE_IP): vol.All(
|
|
|
|
ipaddress.ip_address, cv.string
|
|
|
|
),
|
2019-07-31 19:25:30 +00:00
|
|
|
vol.Optional(CONF_AUTO_START, default=DEFAULT_AUTO_START): cv.boolean,
|
|
|
|
vol.Optional(CONF_SAFE_MODE, default=DEFAULT_SAFE_MODE): cv.boolean,
|
|
|
|
vol.Optional(CONF_FILTER, default={}): FILTER_SCHEMA,
|
|
|
|
vol.Optional(CONF_ENTITY_CONFIG, default={}): validate_entity_config,
|
2020-04-12 14:59:50 +00:00
|
|
|
vol.Optional(
|
|
|
|
CONF_ZEROCONF_DEFAULT_INTERFACE,
|
|
|
|
default=DEFAULT_ZEROCONF_DEFAULT_INTERFACE,
|
|
|
|
): cv.boolean,
|
2019-07-31 19:25:30 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
|
|
|
|
|
|
|
RESET_ACCESSORY_SERVICE_SCHEMA = vol.Schema(
|
|
|
|
{vol.Required(ATTR_ENTITY_ID): cv.entity_ids}
|
|
|
|
)
|
2019-07-16 01:43:37 +00:00
|
|
|
|
2018-02-19 22:46:22 +00:00
|
|
|
|
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."""
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug("Begin setup HomeKit")
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2020-04-19 18:51:09 +00:00
|
|
|
aid_storage = hass.data[AID_STORAGE] = AccessoryAidStorage(hass)
|
|
|
|
await aid_storage.async_initialize()
|
|
|
|
|
2020-04-21 22:38:43 +00:00
|
|
|
hass.http.register_view(HomeKitPairingQRView)
|
|
|
|
|
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)
|
2019-10-23 05:06:21 +00:00
|
|
|
advertise_ip = conf.get(CONF_ADVERTISE_IP)
|
2018-03-15 01:48:21 +00:00
|
|
|
auto_start = conf[CONF_AUTO_START]
|
2018-11-16 10:08:39 +00:00
|
|
|
safe_mode = conf[CONF_SAFE_MODE]
|
2018-03-15 01:48:21 +00:00
|
|
|
entity_filter = conf[CONF_FILTER]
|
|
|
|
entity_config = conf[CONF_ENTITY_CONFIG]
|
2020-04-12 14:59:50 +00:00
|
|
|
interface_choice = (
|
2020-04-22 03:46:19 +00:00
|
|
|
InterfaceChoice.Default if conf.get(CONF_ZEROCONF_DEFAULT_INTERFACE) else None
|
2020-04-12 14:59:50 +00:00
|
|
|
)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
homekit = HomeKit(
|
2019-10-23 05:06:21 +00:00
|
|
|
hass,
|
|
|
|
name,
|
|
|
|
port,
|
|
|
|
ip_address,
|
|
|
|
entity_filter,
|
|
|
|
entity_config,
|
|
|
|
safe_mode,
|
|
|
|
advertise_ip,
|
2020-04-12 14:59:50 +00:00
|
|
|
interface_choice,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-10-19 22:14:05 +00:00
|
|
|
await hass.async_add_executor_job(homekit.setup)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2019-07-16 01:43:37 +00:00
|
|
|
def handle_homekit_reset_accessory(service):
|
|
|
|
"""Handle start HomeKit service call."""
|
|
|
|
if homekit.status != STATUS_RUNNING:
|
|
|
|
_LOGGER.warning(
|
2019-07-31 19:25:30 +00:00
|
|
|
"HomeKit is not running. Either it is waiting to be "
|
|
|
|
"started or has been stopped."
|
|
|
|
)
|
2019-07-16 01:43:37 +00:00
|
|
|
return
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
entity_ids = service.data.get("entity_id")
|
2019-07-16 01:43:37 +00:00
|
|
|
homekit.reset_accessories(entity_ids)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN,
|
|
|
|
SERVICE_HOMEKIT_RESET_ACCESSORY,
|
|
|
|
handle_homekit_reset_accessory,
|
|
|
|
schema=RESET_ACCESSORY_SERVICE_SCHEMA,
|
|
|
|
)
|
2019-07-16 01:43:37 +00:00
|
|
|
|
2020-04-21 00:48:09 +00:00
|
|
|
@callback
|
|
|
|
def async_describe_logbook_event(event):
|
|
|
|
"""Describe a logbook event."""
|
|
|
|
data = event.data
|
|
|
|
entity_id = data.get(ATTR_ENTITY_ID)
|
|
|
|
value = data.get(ATTR_VALUE)
|
|
|
|
|
|
|
|
value_msg = f" to {value}" if value else ""
|
|
|
|
message = f"send command {data[ATTR_SERVICE]}{value_msg} for {data[ATTR_DISPLAY_NAME]}"
|
|
|
|
|
|
|
|
return {
|
|
|
|
"name": "HomeKit",
|
|
|
|
"message": message,
|
|
|
|
"entity_id": entity_id,
|
|
|
|
}
|
|
|
|
|
|
|
|
hass.components.logbook.async_describe_event(
|
|
|
|
DOMAIN, EVENT_HOMEKIT_CHANGED, async_describe_logbook_event
|
|
|
|
)
|
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
if auto_start:
|
2020-04-22 00:43:49 +00:00
|
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_START, homekit.async_start)
|
2018-03-15 01:48:21 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-22 00:43:49 +00:00
|
|
|
async def async_handle_homekit_service_start(service):
|
2018-03-15 01:48:21 +00:00
|
|
|
"""Handle start HomeKit service call."""
|
2018-05-04 14:46:00 +00:00
|
|
|
if homekit.status != STATUS_READY:
|
|
|
|
_LOGGER.warning(
|
2019-07-31 19:25:30 +00:00
|
|
|
"HomeKit is not ready. Either it is already running or has "
|
|
|
|
"been stopped."
|
|
|
|
)
|
2018-03-15 01:48:21 +00:00
|
|
|
return
|
2020-04-22 00:43:49 +00:00
|
|
|
await homekit.async_start()
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.services.async_register(
|
2020-04-22 00:43:49 +00:00
|
|
|
DOMAIN, SERVICE_HOMEKIT_START, async_handle_homekit_service_start
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
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:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.warning(
|
|
|
|
'The entity "%s" is not supported, since it '
|
|
|
|
"generates an invalid aid, please change it.",
|
|
|
|
state.entity_id,
|
|
|
|
)
|
2018-03-15 01:48:21 +00:00
|
|
|
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
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if state.domain == "alarm_control_panel":
|
|
|
|
a_type = "SecuritySystem"
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
elif state.domain in ("binary_sensor", "device_tracker", "person"):
|
|
|
|
a_type = "BinarySensor"
|
2018-04-11 20:24:14 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
elif state.domain == "climate":
|
|
|
|
a_type = "Thermostat"
|
2018-04-09 13:32:29 +00:00
|
|
|
|
2019-07-31 19:25:30 +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
|
|
|
|
2020-04-10 12:42:34 +00:00
|
|
|
if device_class in (DEVICE_CLASS_GARAGE, DEVICE_CLASS_GATE) and features & (
|
2019-07-31 19:25:30 +00:00
|
|
|
cover.SUPPORT_OPEN | cover.SUPPORT_CLOSE
|
|
|
|
):
|
|
|
|
a_type = "GarageDoorOpener"
|
2018-05-28 14:26:33 +00:00
|
|
|
elif features & cover.SUPPORT_SET_POSITION:
|
2019-07-31 19:25:30 +00:00
|
|
|
a_type = "WindowCovering"
|
2018-05-28 14:26:33 +00:00
|
|
|
elif features & (cover.SUPPORT_OPEN | cover.SUPPORT_CLOSE):
|
2019-07-31 19:25:30 +00:00
|
|
|
a_type = "WindowCoveringBasic"
|
2018-03-07 12:17:52 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
elif state.domain == "fan":
|
|
|
|
a_type = "Fan"
|
2018-05-16 11:15:59 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
elif state.domain == "light":
|
|
|
|
a_type = "Light"
|
2018-03-16 00:05:28 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
elif state.domain == "lock":
|
|
|
|
a_type = "Lock"
|
2018-04-11 20:24:14 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
elif state.domain == "media_player":
|
2019-05-05 15:51:48 +00:00
|
|
|
device_class = state.attributes.get(ATTR_DEVICE_CLASS)
|
2018-05-28 14:26:33 +00:00
|
|
|
feature_list = config.get(CONF_FEATURE_LIST)
|
2019-05-05 15:51:48 +00:00
|
|
|
|
|
|
|
if device_class == DEVICE_CLASS_TV:
|
2019-07-31 19:25:30 +00:00
|
|
|
a_type = "TelevisionMediaPlayer"
|
2019-05-05 15:51:48 +00:00
|
|
|
else:
|
2019-07-31 19:25:30 +00:00
|
|
|
if feature_list and validate_media_player_features(state, feature_list):
|
|
|
|
a_type = "MediaPlayer"
|
2018-05-25 09:37:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +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
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if device_class == DEVICE_CLASS_TEMPERATURE or unit in (
|
|
|
|
TEMP_CELSIUS,
|
|
|
|
TEMP_FAHRENHEIT,
|
|
|
|
):
|
|
|
|
a_type = "TemperatureSensor"
|
2020-02-28 19:46:48 +00:00
|
|
|
elif device_class == DEVICE_CLASS_HUMIDITY and unit == UNIT_PERCENTAGE:
|
2019-07-31 19:25:30 +00:00
|
|
|
a_type = "HumiditySensor"
|
|
|
|
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:
|
2019-07-31 19:25:30 +00:00
|
|
|
a_type = "CarbonMonoxideSensor"
|
|
|
|
elif device_class == DEVICE_CLASS_CO2 or DEVICE_CLASS_CO2 in state.entity_id:
|
|
|
|
a_type = "CarbonDioxideSensor"
|
|
|
|
elif device_class == DEVICE_CLASS_ILLUMINANCE or unit in ("lm", "lx"):
|
|
|
|
a_type = "LightSensor"
|
|
|
|
|
|
|
|
elif state.domain == "switch":
|
2018-06-01 16:04:54 +00:00
|
|
|
switch_type = config.get(CONF_TYPE, TYPE_SWITCH)
|
|
|
|
a_type = SWITCH_TYPES[switch_type]
|
|
|
|
|
2020-04-20 15:00:52 +00:00
|
|
|
elif state.domain == "vacuum":
|
|
|
|
features = state.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
|
|
|
if features & (vacuum.SUPPORT_START | vacuum.SUPPORT_RETURN_HOME):
|
|
|
|
a_type = "DockVacuum"
|
|
|
|
else:
|
|
|
|
a_type = "Switch"
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
elif state.domain in ("automation", "input_boolean", "remote", "scene", "script"):
|
|
|
|
a_type = "Switch"
|
2018-04-11 20:24:14 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
elif state.domain == "water_heater":
|
|
|
|
a_type = "WaterHeater"
|
2018-10-19 19:04:05 +00:00
|
|
|
|
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
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
class HomeKit:
|
2018-02-25 09:58:13 +00:00
|
|
|
"""Class to handle all actions between HomeKit and Home Assistant."""
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
def __init__(
|
2019-10-23 05:06:21 +00:00
|
|
|
self,
|
|
|
|
hass,
|
|
|
|
name,
|
|
|
|
port,
|
|
|
|
ip_address,
|
|
|
|
entity_filter,
|
|
|
|
entity_config,
|
|
|
|
safe_mode,
|
|
|
|
advertise_ip=None,
|
2020-04-12 14:59:50 +00:00
|
|
|
interface_choice=None,
|
2019-07-31 19:25:30 +00:00
|
|
|
):
|
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-11-16 10:08:39 +00:00
|
|
|
self._safe_mode = safe_mode
|
2019-10-23 05:06:21 +00:00
|
|
|
self._advertise_ip = advertise_ip
|
2020-04-12 14:59:50 +00:00
|
|
|
self._interface_choice = interface_choice
|
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."""
|
2019-12-22 09:23:44 +00:00
|
|
|
# pylint: disable=import-outside-toplevel
|
|
|
|
from .accessories import HomeBridge, HomeDriver
|
2018-03-15 01:48:21 +00:00
|
|
|
|
2020-04-22 00:43:49 +00:00
|
|
|
self.hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, self.async_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)
|
2019-07-31 19:25:30 +00:00
|
|
|
self.driver = HomeDriver(
|
2019-10-23 05:06:21 +00:00
|
|
|
self.hass,
|
|
|
|
address=ip_addr,
|
|
|
|
port=self._port,
|
|
|
|
persist_file=path,
|
|
|
|
advertised_address=self._advertise_ip,
|
2020-04-12 14:59:50 +00:00
|
|
|
interface_choice=self._interface_choice,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-07-22 07:51:42 +00:00
|
|
|
self.bridge = HomeBridge(self.hass, self.driver, self._name)
|
2018-11-16 10:08:39 +00:00
|
|
|
if self._safe_mode:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug("Safe_mode selected")
|
2018-11-16 10:08:39 +00:00
|
|
|
self.driver.safe_mode = True
|
2018-03-15 01:48:21 +00:00
|
|
|
|
2019-07-16 01:43:37 +00:00
|
|
|
def reset_accessories(self, entity_ids):
|
|
|
|
"""Reset the accessory to load the latest configuration."""
|
2020-04-19 18:51:09 +00:00
|
|
|
aid_storage = self.hass.data[AID_STORAGE]
|
2019-07-16 01:43:37 +00:00
|
|
|
removed = []
|
|
|
|
for entity_id in entity_ids:
|
2020-04-19 18:51:09 +00:00
|
|
|
aid = aid_storage.get_or_allocate_aid_for_entity_id(entity_id)
|
2019-07-16 01:43:37 +00:00
|
|
|
if aid not in self.bridge.accessories:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.warning(
|
2020-01-02 19:17:10 +00:00
|
|
|
"Could not reset accessory. entity_id not found %s", entity_id
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-07-16 01:43:37 +00:00
|
|
|
continue
|
|
|
|
acc = self.remove_bridge_accessory(aid)
|
|
|
|
removed.append(acc)
|
|
|
|
self.driver.config_changed()
|
|
|
|
|
|
|
|
for acc in removed:
|
|
|
|
self.bridge.add_accessory(acc)
|
|
|
|
self.driver.config_changed()
|
|
|
|
|
2018-03-15 01:48:21 +00:00
|
|
|
def add_bridge_accessory(self, state):
|
|
|
|
"""Try adding accessory to bridge if configured beforehand."""
|
2020-04-22 00:43:49 +00:00
|
|
|
if not self._filter(state.entity_id):
|
2018-03-15 01:48:21 +00:00
|
|
|
return
|
2020-04-19 18:51:09 +00:00
|
|
|
|
|
|
|
# The bridge itself counts as an accessory
|
|
|
|
if len(self.bridge.accessories) + 1 >= MAX_DEVICES:
|
|
|
|
_LOGGER.warning(
|
|
|
|
"Cannot add %s as this would exceeded the %d device limit. Consider using the filter option.",
|
|
|
|
state.entity_id,
|
|
|
|
MAX_DEVICES,
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
aid = self.hass.data[AID_STORAGE].get_or_allocate_aid_for_entity_id(
|
|
|
|
state.entity_id
|
|
|
|
)
|
2018-03-15 01:48:21 +00:00
|
|
|
conf = self._config.pop(state.entity_id, {})
|
2020-04-16 02:40:38 +00:00
|
|
|
# If an accessory cannot be created or added due to an exception
|
|
|
|
# of any kind (usually in pyhap) it should not prevent
|
|
|
|
# the rest of the accessories from being created
|
|
|
|
try:
|
|
|
|
acc = get_accessory(self.hass, self.driver, state, aid, conf)
|
|
|
|
if acc is not None:
|
|
|
|
self.bridge.add_accessory(acc)
|
|
|
|
except Exception: # pylint: disable=broad-except
|
|
|
|
_LOGGER.exception(
|
|
|
|
"Failed to create a HomeKit accessory for %s", state.entity_id
|
|
|
|
)
|
2018-03-15 01:48:21 +00:00
|
|
|
|
2019-07-16 01:43:37 +00:00
|
|
|
def remove_bridge_accessory(self, aid):
|
|
|
|
"""Try adding accessory to bridge if configured beforehand."""
|
|
|
|
acc = None
|
|
|
|
if aid in self.bridge.accessories:
|
|
|
|
acc = self.bridge.accessories.pop(aid)
|
|
|
|
return acc
|
|
|
|
|
2020-04-22 00:43:49 +00:00
|
|
|
async def async_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
|
|
|
|
2020-04-22 00:43:49 +00:00
|
|
|
ent_reg = await entity_registry.async_get_registry(self.hass)
|
|
|
|
|
|
|
|
device_lookup = ent_reg.async_get_device_class_lookup(
|
|
|
|
{
|
|
|
|
("binary_sensor", DEVICE_CLASS_BATTERY_CHARGING),
|
|
|
|
("sensor", DEVICE_CLASS_BATTERY),
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
bridged_states = []
|
|
|
|
for state in self.hass.states.async_all():
|
|
|
|
if not self._filter(state.entity_id):
|
|
|
|
continue
|
|
|
|
|
|
|
|
self._async_configure_linked_battery_sensors(ent_reg, device_lookup, state)
|
|
|
|
bridged_states.append(state)
|
|
|
|
|
|
|
|
await self.hass.async_add_executor_job(self._start, bridged_states)
|
|
|
|
|
|
|
|
def _start(self, bridged_states):
|
2019-12-09 15:10:02 +00:00
|
|
|
from . import ( # noqa: F401 pylint: disable=unused-import, import-outside-toplevel
|
2019-07-31 19:25:30 +00:00
|
|
|
type_covers,
|
|
|
|
type_fans,
|
|
|
|
type_lights,
|
|
|
|
type_locks,
|
|
|
|
type_media_players,
|
|
|
|
type_security_systems,
|
|
|
|
type_sensors,
|
|
|
|
type_switches,
|
|
|
|
type_thermostats,
|
|
|
|
)
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2020-04-22 00:43:49 +00:00
|
|
|
for state in bridged_states:
|
2018-03-15 01:48:21 +00:00
|
|
|
self.add_bridge_accessory(state)
|
2020-04-19 18:51:09 +00:00
|
|
|
|
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:
|
2020-04-21 22:38:43 +00:00
|
|
|
show_setup_message(
|
|
|
|
self.hass, self.driver.state.pincode, self.bridge.xhm_uri()
|
|
|
|
)
|
2018-03-15 01:48:21 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug("Driver start")
|
2020-04-30 20:29:50 +00:00
|
|
|
self.hass.add_job(self.driver.start)
|
2018-05-04 14:46:00 +00:00
|
|
|
self.status = STATUS_RUNNING
|
2018-02-19 22:46:22 +00:00
|
|
|
|
2020-04-22 00:43:49 +00:00
|
|
|
async def async_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
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug("Driver stop")
|
2020-04-30 22:49:16 +00:00
|
|
|
self.hass.add_job(self.driver.stop)
|
2020-04-22 00:43:49 +00:00
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_configure_linked_battery_sensors(self, ent_reg, device_lookup, state):
|
|
|
|
entry = ent_reg.async_get(state.entity_id)
|
|
|
|
|
|
|
|
if (
|
|
|
|
entry is None
|
|
|
|
or entry.device_id is None
|
|
|
|
or entry.device_id not in device_lookup
|
|
|
|
or entry.device_class
|
|
|
|
in (DEVICE_CLASS_BATTERY_CHARGING, DEVICE_CLASS_BATTERY)
|
|
|
|
):
|
|
|
|
return
|
|
|
|
|
|
|
|
if ATTR_BATTERY_CHARGING not in state.attributes:
|
|
|
|
battery_charging_binary_sensor_entity_id = device_lookup[
|
|
|
|
entry.device_id
|
|
|
|
].get(("binary_sensor", DEVICE_CLASS_BATTERY_CHARGING))
|
|
|
|
if battery_charging_binary_sensor_entity_id:
|
|
|
|
self._config.setdefault(state.entity_id, {}).setdefault(
|
|
|
|
CONF_LINKED_BATTERY_CHARGING_SENSOR,
|
|
|
|
battery_charging_binary_sensor_entity_id,
|
|
|
|
)
|
|
|
|
|
|
|
|
if ATTR_BATTERY_LEVEL not in state.attributes:
|
|
|
|
battery_sensor_entity_id = device_lookup[entry.device_id].get(
|
|
|
|
("sensor", DEVICE_CLASS_BATTERY)
|
|
|
|
)
|
|
|
|
if battery_sensor_entity_id:
|
|
|
|
self._config.setdefault(state.entity_id, {}).setdefault(
|
|
|
|
CONF_LINKED_BATTERY_SENSOR, battery_sensor_entity_id
|
|
|
|
)
|
2020-04-21 22:38:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
class HomeKitPairingQRView(HomeAssistantView):
|
|
|
|
"""Display the homekit pairing code at a protected url."""
|
|
|
|
|
|
|
|
url = "/api/homekit/pairingqr"
|
|
|
|
name = "api:homekit:pairingqr"
|
|
|
|
requires_auth = False
|
|
|
|
|
|
|
|
# pylint: disable=no-self-use
|
|
|
|
async def get(self, request):
|
|
|
|
"""Retrieve the pairing QRCode image."""
|
|
|
|
if request.query_string != request.app["hass"].data[HOMEKIT_PAIRING_QR_SECRET]:
|
|
|
|
raise Unauthorized()
|
|
|
|
return web.Response(
|
|
|
|
body=request.app["hass"].data[HOMEKIT_PAIRING_QR],
|
|
|
|
content_type="image/svg+xml",
|
|
|
|
)
|