core/homeassistant/components/zeroconf/__init__.py

166 lines
4.4 KiB
Python
Raw Normal View History

"""Support for exposing Home Assistant via Zeroconf."""
# PyLint bug confuses absolute/relative imports
# https://github.com/PyCQA/pylint/issues/1931
# pylint: disable=no-name-in-module
2016-04-10 22:34:04 +00:00
import logging
import socket
2016-04-10 22:34:04 +00:00
import ipaddress
import voluptuous as vol
from zeroconf import ServiceBrowser, ServiceInfo, ServiceStateChange, Zeroconf
from homeassistant import util
from homeassistant.const import (
EVENT_HOMEASSISTANT_STOP,
EVENT_HOMEASSISTANT_START,
__version__,
)
from homeassistant.generated.zeroconf import ZEROCONF, HOMEKIT
2016-04-10 22:34:04 +00:00
_LOGGER = logging.getLogger(__name__)
2016-04-10 22:34:04 +00:00
2019-07-31 19:25:30 +00:00
DOMAIN = "zeroconf"
2019-07-31 19:25:30 +00:00
ATTR_HOST = "host"
ATTR_PORT = "port"
ATTR_HOSTNAME = "hostname"
ATTR_TYPE = "type"
ATTR_NAME = "name"
ATTR_PROPERTIES = "properties"
2016-04-10 22:34:04 +00:00
2019-07-31 19:25:30 +00:00
ZEROCONF_TYPE = "_home-assistant._tcp.local."
HOMEKIT_TYPE = "_hap._tcp.local."
2016-04-10 22:34:04 +00:00
2019-07-31 19:25:30 +00:00
CONFIG_SCHEMA = vol.Schema({DOMAIN: vol.Schema({})}, extra=vol.ALLOW_EXTRA)
2016-04-10 22:34:04 +00:00
def setup(hass, config):
"""Set up Zeroconf and make Home Assistant discoverable."""
zeroconf = Zeroconf()
zeroconf_name = f"{hass.config.location_name}.{ZEROCONF_TYPE}"
2016-04-10 22:34:04 +00:00
params = {
2019-07-31 19:25:30 +00:00
"version": __version__,
"base_url": hass.config.api.base_url,
# always needs authentication
2019-07-31 19:25:30 +00:00
"requires_api_password": True,
}
2016-04-10 22:34:04 +00:00
host_ip = util.get_local_ip()
try:
host_ip_pton = socket.inet_pton(socket.AF_INET, host_ip)
except socket.error:
host_ip_pton = socket.inet_pton(socket.AF_INET6, host_ip)
2019-07-31 19:25:30 +00:00
info = ServiceInfo(
ZEROCONF_TYPE,
zeroconf_name,
None,
addresses=[host_ip_pton],
port=hass.http.server_port,
properties=params,
)
def zeroconf_hass_start(_event):
"""Expose Home Assistant on zeroconf when it starts.
Wait till started or otherwise HTTP is not up and running.
"""
_LOGGER.info("Starting Zeroconf broadcast")
zeroconf.register_service(info)
hass.bus.listen_once(EVENT_HOMEASSISTANT_START, zeroconf_hass_start)
2016-04-10 22:34:04 +00:00
def service_update(zeroconf, service_type, name, state_change):
"""Service state changed."""
if state_change != ServiceStateChange.Added:
return
service_info = zeroconf.get_service_info(service_type, name)
info = info_from_service(service_info)
_LOGGER.debug("Discovered new device %s %s", name, info)
# If we can handle it as a HomeKit discovery, we do that here.
if service_type == HOMEKIT_TYPE and handle_homekit(hass, info):
return
for domain in ZEROCONF[service_type]:
hass.add_job(
hass.config_entries.flow.async_init(
2019-07-31 19:25:30 +00:00
domain, context={"source": DOMAIN}, data=info
)
)
for service in ZEROCONF:
ServiceBrowser(zeroconf, service, handlers=[service_update])
if HOMEKIT_TYPE not in ZEROCONF:
ServiceBrowser(zeroconf, HOMEKIT_TYPE, handlers=[service_update])
def stop_zeroconf(_):
"""Stop Zeroconf."""
zeroconf.unregister_service(info)
zeroconf.close()
2016-04-10 22:34:04 +00:00
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, stop_zeroconf)
2016-04-10 22:34:04 +00:00
return True
def handle_homekit(hass, info) -> bool:
"""Handle a HomeKit discovery.
Return if discovery was forwarded.
"""
model = None
2019-07-31 19:25:30 +00:00
props = info.get("properties", {})
for key in props:
2019-07-31 19:25:30 +00:00
if key.lower() == "md":
model = props[key]
break
if model is None:
return False
for test_model in HOMEKIT:
if model != test_model and not model.startswith(test_model + " "):
continue
hass.add_job(
hass.config_entries.flow.async_init(
2019-07-31 19:25:30 +00:00
HOMEKIT[test_model], context={"source": "homekit"}, data=info
)
)
return True
return False
def info_from_service(service):
"""Return prepared info from mDNS entries."""
properties = {}
for key, value in service.properties.items():
try:
if isinstance(value, bytes):
2019-07-31 19:25:30 +00:00
value = value.decode("utf-8")
properties[key.decode("utf-8")] = value
except UnicodeDecodeError:
_LOGGER.warning("Unicode decode error on %s: %s", key, value)
address = service.addresses[0]
info = {
ATTR_HOST: str(ipaddress.ip_address(address)),
ATTR_PORT: service.port,
ATTR_HOSTNAME: service.server,
ATTR_TYPE: service.type,
ATTR_NAME: service.name,
ATTR_PROPERTIES: properties,
}
return info