2019-04-03 15:40:03 +00:00
|
|
|
"""Support for ONVIF Cameras with FFmpeg as decoder."""
|
2017-06-16 05:28:17 +00:00
|
|
|
import asyncio
|
|
|
|
import logging
|
2018-03-16 03:30:41 +00:00
|
|
|
import os
|
2017-06-16 05:28:17 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.const import (
|
2018-02-18 16:08:56 +00:00
|
|
|
CONF_NAME, CONF_HOST, CONF_USERNAME, CONF_PASSWORD, CONF_PORT,
|
|
|
|
ATTR_ENTITY_ID)
|
2019-03-31 22:12:55 +00:00
|
|
|
from homeassistant.components.camera import (
|
|
|
|
Camera, PLATFORM_SCHEMA, SUPPORT_STREAM)
|
2019-03-26 12:31:29 +00:00
|
|
|
from homeassistant.components.camera.const import DOMAIN
|
2017-06-16 05:28:17 +00:00
|
|
|
from homeassistant.components.ffmpeg import (
|
2018-01-16 08:23:48 +00:00
|
|
|
DATA_FFMPEG, CONF_EXTRA_ARGUMENTS)
|
2017-06-16 05:28:17 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
from homeassistant.helpers.aiohttp_client import (
|
|
|
|
async_aiohttp_proxy_stream)
|
2018-02-18 16:08:56 +00:00
|
|
|
from homeassistant.helpers.service import extract_entity_ids
|
2017-06-16 05:28:17 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
DEFAULT_NAME = 'ONVIF Camera'
|
|
|
|
DEFAULT_PORT = 5000
|
|
|
|
DEFAULT_USERNAME = 'admin'
|
|
|
|
DEFAULT_PASSWORD = '888888'
|
2019-04-03 11:46:41 +00:00
|
|
|
DEFAULT_ARGUMENTS = '-pred 1'
|
2018-03-06 03:56:15 +00:00
|
|
|
DEFAULT_PROFILE = 0
|
|
|
|
|
|
|
|
CONF_PROFILE = "profile"
|
2017-06-16 05:28:17 +00:00
|
|
|
|
2018-02-18 16:08:56 +00:00
|
|
|
ATTR_PAN = "pan"
|
|
|
|
ATTR_TILT = "tilt"
|
|
|
|
ATTR_ZOOM = "zoom"
|
|
|
|
|
|
|
|
DIR_UP = "UP"
|
|
|
|
DIR_DOWN = "DOWN"
|
|
|
|
DIR_LEFT = "LEFT"
|
|
|
|
DIR_RIGHT = "RIGHT"
|
|
|
|
ZOOM_OUT = "ZOOM_OUT"
|
|
|
|
ZOOM_IN = "ZOOM_IN"
|
2018-10-02 12:36:28 +00:00
|
|
|
PTZ_NONE = "NONE"
|
2018-02-18 16:08:56 +00:00
|
|
|
|
|
|
|
SERVICE_PTZ = "onvif_ptz"
|
|
|
|
|
|
|
|
ONVIF_DATA = "onvif"
|
|
|
|
ENTITIES = "entities"
|
|
|
|
|
2017-06-16 05:28:17 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
|
|
vol.Optional(CONF_PASSWORD, default=DEFAULT_PASSWORD): cv.string,
|
|
|
|
vol.Optional(CONF_USERNAME, default=DEFAULT_USERNAME): cv.string,
|
|
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
2018-01-16 08:23:48 +00:00
|
|
|
vol.Optional(CONF_EXTRA_ARGUMENTS, default=DEFAULT_ARGUMENTS): cv.string,
|
2018-03-06 03:56:15 +00:00
|
|
|
vol.Optional(CONF_PROFILE, default=DEFAULT_PROFILE):
|
|
|
|
vol.All(vol.Coerce(int), vol.Range(min=0)),
|
2017-06-16 05:28:17 +00:00
|
|
|
})
|
|
|
|
|
2018-02-18 16:08:56 +00:00
|
|
|
SERVICE_PTZ_SCHEMA = vol.Schema({
|
|
|
|
ATTR_ENTITY_ID: cv.entity_ids,
|
2018-10-02 12:36:28 +00:00
|
|
|
ATTR_PAN: vol.In([DIR_LEFT, DIR_RIGHT, PTZ_NONE]),
|
|
|
|
ATTR_TILT: vol.In([DIR_UP, DIR_DOWN, PTZ_NONE]),
|
|
|
|
ATTR_ZOOM: vol.In([ZOOM_OUT, ZOOM_IN, PTZ_NONE])
|
2018-02-18 16:08:56 +00:00
|
|
|
})
|
|
|
|
|
2017-06-16 05:28:17 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2017-06-16 05:28:17 +00:00
|
|
|
"""Set up a ONVIF camera."""
|
2018-02-18 16:08:56 +00:00
|
|
|
def handle_ptz(service):
|
|
|
|
"""Handle PTZ service call."""
|
|
|
|
pan = service.data.get(ATTR_PAN, None)
|
|
|
|
tilt = service.data.get(ATTR_TILT, None)
|
|
|
|
zoom = service.data.get(ATTR_ZOOM, None)
|
|
|
|
all_cameras = hass.data[ONVIF_DATA][ENTITIES]
|
|
|
|
entity_ids = extract_entity_ids(hass, service)
|
|
|
|
target_cameras = []
|
|
|
|
if not entity_ids:
|
|
|
|
target_cameras = all_cameras
|
|
|
|
else:
|
|
|
|
target_cameras = [camera for camera in all_cameras
|
|
|
|
if camera.entity_id in entity_ids]
|
|
|
|
for camera in target_cameras:
|
|
|
|
camera.perform_ptz(pan, tilt, zoom)
|
|
|
|
|
2018-11-03 11:36:22 +00:00
|
|
|
hass.services.register(DOMAIN, SERVICE_PTZ, handle_ptz,
|
|
|
|
schema=SERVICE_PTZ_SCHEMA)
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities([ONVIFHassCamera(hass, config)])
|
2017-06-16 05:28:17 +00:00
|
|
|
|
|
|
|
|
2018-02-17 13:57:05 +00:00
|
|
|
class ONVIFHassCamera(Camera):
|
2017-06-16 05:28:17 +00:00
|
|
|
"""An implementation of an ONVIF camera."""
|
|
|
|
|
|
|
|
def __init__(self, hass, config):
|
|
|
|
"""Initialize a ONVIF camera."""
|
|
|
|
super().__init__()
|
2018-03-16 03:30:41 +00:00
|
|
|
import onvif
|
2017-06-16 05:28:17 +00:00
|
|
|
|
2018-03-16 03:30:41 +00:00
|
|
|
self._username = config.get(CONF_USERNAME)
|
|
|
|
self._password = config.get(CONF_PASSWORD)
|
|
|
|
self._host = config.get(CONF_HOST)
|
|
|
|
self._port = config.get(CONF_PORT)
|
2017-06-16 05:28:17 +00:00
|
|
|
self._name = config.get(CONF_NAME)
|
2018-01-16 08:23:48 +00:00
|
|
|
self._ffmpeg_arguments = config.get(CONF_EXTRA_ARGUMENTS)
|
2018-03-16 03:30:41 +00:00
|
|
|
self._profile_index = config.get(CONF_PROFILE)
|
2018-02-17 13:57:05 +00:00
|
|
|
self._input = None
|
2018-03-16 03:30:41 +00:00
|
|
|
self._media_service = \
|
|
|
|
onvif.ONVIFService('http://{}:{}/onvif/device_service'.format(
|
|
|
|
self._host, self._port),
|
|
|
|
self._username, self._password,
|
|
|
|
'{}/wsdl/media.wsdl'.format(os.path.dirname(
|
|
|
|
onvif.__file__)))
|
|
|
|
|
|
|
|
self._ptz_service = \
|
|
|
|
onvif.ONVIFService('http://{}:{}/onvif/device_service'.format(
|
|
|
|
self._host, self._port),
|
|
|
|
self._username, self._password,
|
|
|
|
'{}/wsdl/ptz.wsdl'.format(os.path.dirname(
|
|
|
|
onvif.__file__)))
|
|
|
|
|
|
|
|
def obtain_input_uri(self):
|
|
|
|
"""Set the input uri for the camera."""
|
|
|
|
from onvif import exceptions
|
|
|
|
_LOGGER.debug("Connecting with ONVIF Camera: %s on port %s",
|
|
|
|
self._host, self._port)
|
|
|
|
|
2018-02-17 13:57:05 +00:00
|
|
|
try:
|
2018-03-16 03:30:41 +00:00
|
|
|
profiles = self._media_service.GetProfiles()
|
|
|
|
|
|
|
|
if self._profile_index >= len(profiles):
|
2018-03-06 03:56:15 +00:00
|
|
|
_LOGGER.warning("ONVIF Camera '%s' doesn't provide profile %d."
|
|
|
|
" Using the last profile.",
|
|
|
|
self._name, self._profile_index)
|
|
|
|
self._profile_index = -1
|
2018-03-16 03:30:41 +00:00
|
|
|
|
|
|
|
req = self._media_service.create_type('GetStreamUri')
|
|
|
|
|
2018-03-06 03:56:15 +00:00
|
|
|
# pylint: disable=protected-access
|
2018-03-16 03:30:41 +00:00
|
|
|
req.ProfileToken = profiles[self._profile_index]._token
|
|
|
|
uri_no_auth = self._media_service.GetStreamUri(req).Uri
|
|
|
|
uri_for_log = uri_no_auth.replace(
|
|
|
|
'rtsp://', 'rtsp://<user>:<password>@', 1)
|
|
|
|
self._input = uri_no_auth.replace(
|
|
|
|
'rtsp://', 'rtsp://{}:{}@'.format(self._username,
|
|
|
|
self._password), 1)
|
2018-02-17 13:57:05 +00:00
|
|
|
_LOGGER.debug(
|
|
|
|
"ONVIF Camera Using the following URL for %s: %s",
|
2018-03-16 03:30:41 +00:00
|
|
|
self._name, uri_for_log)
|
|
|
|
# we won't need the media service anymore
|
|
|
|
self._media_service = None
|
2018-02-18 16:08:56 +00:00
|
|
|
except exceptions.ONVIFError as err:
|
2018-03-16 03:30:41 +00:00
|
|
|
_LOGGER.debug("Couldn't setup camera '%s'. Error: %s",
|
|
|
|
self._name, err)
|
|
|
|
return
|
2018-02-18 16:08:56 +00:00
|
|
|
|
|
|
|
def perform_ptz(self, pan, tilt, zoom):
|
|
|
|
"""Perform a PTZ action on the camera."""
|
2018-03-16 03:30:41 +00:00
|
|
|
from onvif import exceptions
|
|
|
|
if self._ptz_service:
|
2018-02-18 16:08:56 +00:00
|
|
|
pan_val = 1 if pan == DIR_RIGHT else -1 if pan == DIR_LEFT else 0
|
|
|
|
tilt_val = 1 if tilt == DIR_UP else -1 if tilt == DIR_DOWN else 0
|
|
|
|
zoom_val = 1 if zoom == ZOOM_IN else -1 if zoom == ZOOM_OUT else 0
|
|
|
|
req = {"Velocity": {
|
|
|
|
"PanTilt": {"_x": pan_val, "_y": tilt_val},
|
|
|
|
"Zoom": {"_x": zoom_val}}}
|
2018-03-16 03:30:41 +00:00
|
|
|
try:
|
|
|
|
self._ptz_service.ContinuousMove(req)
|
|
|
|
except exceptions.ONVIFError as err:
|
|
|
|
if "Bad Request" in err.reason:
|
|
|
|
self._ptz_service = None
|
|
|
|
_LOGGER.debug("Camera '%s' doesn't support PTZ.",
|
|
|
|
self._name)
|
|
|
|
else:
|
|
|
|
_LOGGER.debug("Camera '%s' doesn't support PTZ.", self._name)
|
2018-02-18 16:08:56 +00:00
|
|
|
|
2018-03-16 03:30:41 +00:00
|
|
|
async def async_added_to_hass(self):
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Handle entity addition to hass."""
|
2018-02-18 16:08:56 +00:00
|
|
|
if ONVIF_DATA not in self.hass.data:
|
|
|
|
self.hass.data[ONVIF_DATA] = {}
|
|
|
|
self.hass.data[ONVIF_DATA][ENTITIES] = []
|
|
|
|
self.hass.data[ONVIF_DATA][ENTITIES].append(self)
|
2019-03-31 22:12:55 +00:00
|
|
|
await self.hass.async_add_executor_job(self.obtain_input_uri)
|
2017-06-16 05:28:17 +00:00
|
|
|
|
2018-03-16 03:30:41 +00:00
|
|
|
async def async_camera_image(self):
|
2017-06-16 05:28:17 +00:00
|
|
|
"""Return a still image response from the camera."""
|
2019-03-27 06:55:05 +00:00
|
|
|
from haffmpeg.tools import ImageFrame, IMAGE_JPEG
|
2018-03-16 03:30:41 +00:00
|
|
|
|
|
|
|
if not self._input:
|
2019-03-31 22:12:55 +00:00
|
|
|
await self.hass.async_add_executor_job(self.obtain_input_uri)
|
2018-03-16 03:30:41 +00:00
|
|
|
if not self._input:
|
|
|
|
return None
|
|
|
|
|
2017-06-16 05:28:17 +00:00
|
|
|
ffmpeg = ImageFrame(
|
|
|
|
self.hass.data[DATA_FFMPEG].binary, loop=self.hass.loop)
|
|
|
|
|
2018-03-16 03:30:41 +00:00
|
|
|
image = await asyncio.shield(ffmpeg.get_image(
|
2017-06-16 05:28:17 +00:00
|
|
|
self._input, output_format=IMAGE_JPEG,
|
2017-10-18 15:11:22 +00:00
|
|
|
extra_cmd=self._ffmpeg_arguments), loop=self.hass.loop)
|
2017-06-16 05:28:17 +00:00
|
|
|
return image
|
|
|
|
|
2018-03-16 03:30:41 +00:00
|
|
|
async def handle_async_mjpeg_stream(self, request):
|
2017-06-16 05:28:17 +00:00
|
|
|
"""Generate an HTTP MJPEG stream from the camera."""
|
2019-03-27 06:55:05 +00:00
|
|
|
from haffmpeg.camera import CameraMjpeg
|
2017-06-16 05:28:17 +00:00
|
|
|
|
2018-03-16 03:30:41 +00:00
|
|
|
if not self._input:
|
2019-03-31 22:12:55 +00:00
|
|
|
await self.hass.async_add_executor_job(self.obtain_input_uri)
|
2018-03-16 03:30:41 +00:00
|
|
|
if not self._input:
|
|
|
|
return None
|
|
|
|
|
2019-02-04 17:57:22 +00:00
|
|
|
ffmpeg_manager = self.hass.data[DATA_FFMPEG]
|
|
|
|
stream = CameraMjpeg(ffmpeg_manager.binary,
|
2017-06-16 05:28:17 +00:00
|
|
|
loop=self.hass.loop)
|
2018-03-16 03:30:41 +00:00
|
|
|
await stream.open_camera(
|
2017-06-16 05:28:17 +00:00
|
|
|
self._input, extra_cmd=self._ffmpeg_arguments)
|
|
|
|
|
2018-11-01 08:28:23 +00:00
|
|
|
try:
|
2019-03-27 06:55:05 +00:00
|
|
|
stream_reader = await stream.get_reader()
|
2018-11-01 08:28:23 +00:00
|
|
|
return await async_aiohttp_proxy_stream(
|
2019-03-27 06:55:05 +00:00
|
|
|
self.hass, request, stream_reader,
|
2019-02-04 17:57:22 +00:00
|
|
|
ffmpeg_manager.ffmpeg_stream_content_type)
|
2018-11-01 08:28:23 +00:00
|
|
|
finally:
|
|
|
|
await stream.close()
|
2017-06-16 05:28:17 +00:00
|
|
|
|
2019-03-31 22:12:55 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Return supported features."""
|
|
|
|
if self._input:
|
|
|
|
return SUPPORT_STREAM
|
|
|
|
return 0
|
|
|
|
|
|
|
|
@property
|
|
|
|
def stream_source(self):
|
|
|
|
"""Return the stream source."""
|
|
|
|
return self._input
|
|
|
|
|
2017-06-16 05:28:17 +00:00
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of this camera."""
|
|
|
|
return self._name
|