core/homeassistant/components/onvif/__init__.py

169 lines
4.9 KiB
Python
Raw Normal View History

2020-05-01 06:15:40 +00:00
"""The ONVIF integration."""
import asyncio
import requests
from requests.auth import HTTPDigestAuth
from urllib3.exceptions import ReadTimeoutError
2020-05-01 06:15:40 +00:00
import voluptuous as vol
from homeassistant.components.ffmpeg import CONF_EXTRA_ARGUMENTS
from homeassistant.config_entries import SOURCE_IMPORT, ConfigEntry
2020-05-01 18:35:30 +00:00
from homeassistant.const import (
CONF_HOST,
CONF_NAME,
CONF_PASSWORD,
CONF_PORT,
CONF_USERNAME,
EVENT_HOMEASSISTANT_STOP,
HTTP_BASIC_AUTHENTICATION,
HTTP_DIGEST_AUTHENTICATION,
HTTP_UNAUTHORIZED,
2020-05-01 18:35:30 +00:00
)
2020-05-01 06:15:40 +00:00
from homeassistant.core import HomeAssistant
2020-05-06 16:29:59 +00:00
from homeassistant.exceptions import ConfigEntryNotReady
2020-05-01 06:15:40 +00:00
from homeassistant.helpers import config_per_platform
from .const import (
CONF_RTSP_TRANSPORT,
CONF_SNAPSHOT_AUTH,
2020-05-01 06:15:40 +00:00
DEFAULT_ARGUMENTS,
2020-05-01 18:35:30 +00:00
DEFAULT_NAME,
DEFAULT_PASSWORD,
DEFAULT_PORT,
DEFAULT_USERNAME,
2020-05-01 06:15:40 +00:00
DOMAIN,
RTSP_TRANS_PROTOCOLS,
)
2020-05-06 16:29:59 +00:00
from .device import ONVIFDevice
2020-05-01 06:15:40 +00:00
CONFIG_SCHEMA = vol.Schema({DOMAIN: vol.Schema({})}, extra=vol.ALLOW_EXTRA)
async def async_setup(hass: HomeAssistant, config: dict):
"""Set up the ONVIF component."""
# Import from yaml
configs = {}
for p_type, p_config in config_per_platform(config, "camera"):
if p_type != DOMAIN:
continue
config = p_config.copy()
if config[CONF_HOST] not in configs.keys():
2020-05-01 18:35:30 +00:00
configs[config[CONF_HOST]] = {
CONF_HOST: config[CONF_HOST],
CONF_NAME: config.get(CONF_NAME, DEFAULT_NAME),
CONF_PASSWORD: config.get(CONF_PASSWORD, DEFAULT_PASSWORD),
CONF_PORT: config.get(CONF_PORT, DEFAULT_PORT),
CONF_USERNAME: config.get(CONF_USERNAME, DEFAULT_USERNAME),
}
2020-05-01 06:15:40 +00:00
for conf in configs.values():
hass.async_create_task(
hass.config_entries.flow.async_init(
DOMAIN, context={"source": SOURCE_IMPORT}, data=conf
)
)
return True
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry):
"""Set up ONVIF from a config entry."""
2020-05-06 16:29:59 +00:00
if DOMAIN not in hass.data:
hass.data[DOMAIN] = {}
2020-05-01 06:15:40 +00:00
if not entry.options:
await async_populate_options(hass, entry)
2020-05-06 16:29:59 +00:00
device = ONVIFDevice(hass, entry)
if not await device.async_setup():
return False
if not device.available:
raise ConfigEntryNotReady()
if not entry.data.get(CONF_SNAPSHOT_AUTH):
await async_populate_snapshot_auth(hass, device, entry)
2020-05-06 16:29:59 +00:00
hass.data[DOMAIN][entry.unique_id] = device
platforms = ["camera"]
if device.capabilities.events and await device.events.async_start():
platforms += ["binary_sensor", "sensor"]
for component in platforms:
2020-05-01 06:15:40 +00:00
hass.async_create_task(
hass.config_entries.async_forward_entry_setup(entry, component)
)
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, device.async_stop)
2020-05-01 06:15:40 +00:00
return True
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry):
"""Unload a config entry."""
device = hass.data[DOMAIN][entry.unique_id]
platforms = ["camera"]
if device.capabilities.events and device.events.started:
platforms += ["binary_sensor", "sensor"]
await device.events.async_stop()
2020-05-01 18:35:30 +00:00
return all(
2020-05-01 06:15:40 +00:00
await asyncio.gather(
*[
hass.config_entries.async_forward_entry_unload(entry, component)
for component in platforms
2020-05-01 06:15:40 +00:00
]
)
)
async def _get_snapshot_auth(hass, device, entry):
if not (device.username and device.password):
return HTTP_DIGEST_AUTHENTICATION
snapshot_uri = await device.async_get_snapshot_uri(device.profiles[0])
if not snapshot_uri:
return HTTP_DIGEST_AUTHENTICATION
auth = HTTPDigestAuth(device.username, device.password)
def _get():
# so we can handle keyword arguments
return requests.get(snapshot_uri, timeout=1, auth=auth)
try:
response = await hass.async_add_executor_job(_get)
if response.status_code == HTTP_UNAUTHORIZED:
return HTTP_BASIC_AUTHENTICATION
return HTTP_DIGEST_AUTHENTICATION
except requests.exceptions.Timeout:
return HTTP_BASIC_AUTHENTICATION
except requests.exceptions.ConnectionError as error:
if isinstance(error.args[0], ReadTimeoutError):
return HTTP_BASIC_AUTHENTICATION
return HTTP_DIGEST_AUTHENTICATION
async def async_populate_snapshot_auth(hass, device, entry):
"""Check if digest auth for snapshots is possible."""
auth = await _get_snapshot_auth(hass, device, entry)
new_data = {**entry.data, CONF_SNAPSHOT_AUTH: auth}
hass.config_entries.async_update_entry(entry, data=new_data)
2020-05-01 06:15:40 +00:00
async def async_populate_options(hass, entry):
"""Populate default options for device."""
options = {
CONF_EXTRA_ARGUMENTS: DEFAULT_ARGUMENTS,
CONF_RTSP_TRANSPORT: RTSP_TRANS_PROTOCOLS[0],
}
hass.config_entries.async_update_entry(entry, options=options)