2016-02-06 01:24:44 +00:00
|
|
|
"""
|
2016-02-07 10:52:17 +00:00
|
|
|
Support for Ubiquiti's UVC cameras.
|
2016-02-06 01:24:44 +00:00
|
|
|
|
2016-02-07 10:52:17 +00:00
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/camera.uvc/
|
2016-02-06 01:24:44 +00:00
|
|
|
"""
|
|
|
|
import logging
|
|
|
|
import socket
|
|
|
|
|
|
|
|
import requests
|
2016-09-18 21:22:32 +00:00
|
|
|
import voluptuous as vol
|
2016-02-06 01:24:44 +00:00
|
|
|
|
2016-09-18 21:22:32 +00:00
|
|
|
from homeassistant.const import CONF_PORT
|
|
|
|
from homeassistant.components.camera import Camera, PLATFORM_SCHEMA
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-02-06 01:24:44 +00:00
|
|
|
|
2017-09-14 05:21:58 +00:00
|
|
|
REQUIREMENTS = ['uvcclient==0.10.1']
|
2016-02-06 01:24:44 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2016-09-18 21:22:32 +00:00
|
|
|
CONF_NVR = 'nvr'
|
|
|
|
CONF_KEY = 'key'
|
2017-05-10 04:54:38 +00:00
|
|
|
CONF_PASSWORD = 'password'
|
2016-09-18 21:22:32 +00:00
|
|
|
|
2017-05-10 04:54:38 +00:00
|
|
|
DEFAULT_PASSWORD = 'ubnt'
|
2016-09-18 21:22:32 +00:00
|
|
|
DEFAULT_PORT = 7080
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Required(CONF_NVR): cv.string,
|
|
|
|
vol.Required(CONF_KEY): cv.string,
|
2017-05-10 04:54:38 +00:00
|
|
|
vol.Optional(CONF_PASSWORD, default=DEFAULT_PASSWORD): cv.string,
|
2016-09-18 21:22:32 +00:00
|
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
|
|
|
})
|
|
|
|
|
2016-02-06 01:24:44 +00:00
|
|
|
|
|
|
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
2016-03-07 16:45:06 +00:00
|
|
|
"""Discover cameras on a Unifi NVR."""
|
2016-09-18 21:22:32 +00:00
|
|
|
addr = config[CONF_NVR]
|
|
|
|
key = config[CONF_KEY]
|
2017-05-10 04:54:38 +00:00
|
|
|
password = config[CONF_PASSWORD]
|
2016-09-18 21:22:32 +00:00
|
|
|
port = config[CONF_PORT]
|
2016-02-06 01:24:44 +00:00
|
|
|
|
|
|
|
from uvcclient import nvr
|
|
|
|
nvrconn = nvr.UVCRemote(addr, port, key)
|
|
|
|
try:
|
|
|
|
cameras = nvrconn.index()
|
|
|
|
except nvr.NotAuthorized:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.error("Authorization failure while connecting to NVR")
|
2016-02-06 01:24:44 +00:00
|
|
|
return False
|
|
|
|
except nvr.NvrError:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.error("NVR refuses to talk to me")
|
2016-02-06 01:24:44 +00:00
|
|
|
return False
|
|
|
|
except requests.exceptions.ConnectionError as ex:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.error("Unable to connect to NVR: %s", str(ex))
|
2016-02-06 01:24:44 +00:00
|
|
|
return False
|
|
|
|
|
2017-07-06 03:02:16 +00:00
|
|
|
identifier = 'id' if nvrconn.server_version >= (3, 2, 0) else 'uuid'
|
2016-02-23 20:01:51 +00:00
|
|
|
# Filter out airCam models, which are not supported in the latest
|
|
|
|
# version of UnifiVideo and which are EOL by Ubiquiti
|
2016-06-05 17:09:58 +00:00
|
|
|
cameras = [
|
|
|
|
camera for camera in cameras
|
|
|
|
if 'airCam' not in nvrconn.get_camera(camera[identifier])['model']]
|
2016-02-23 20:01:51 +00:00
|
|
|
|
2016-02-22 22:06:06 +00:00
|
|
|
add_devices([UnifiVideoCamera(nvrconn,
|
2016-06-05 17:09:58 +00:00
|
|
|
camera[identifier],
|
2017-05-10 04:54:38 +00:00
|
|
|
camera['name'],
|
|
|
|
password)
|
2016-02-22 22:06:06 +00:00
|
|
|
for camera in cameras])
|
|
|
|
return True
|
2016-02-06 01:24:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
class UnifiVideoCamera(Camera):
|
2016-03-07 16:45:06 +00:00
|
|
|
"""A Ubiquiti Unifi Video Camera."""
|
2016-03-07 19:29:54 +00:00
|
|
|
|
2017-05-10 04:54:38 +00:00
|
|
|
def __init__(self, nvr, uuid, name, password):
|
2016-03-07 19:29:54 +00:00
|
|
|
"""Initialize an Unifi camera."""
|
2016-02-06 01:24:44 +00:00
|
|
|
super(UnifiVideoCamera, self).__init__()
|
|
|
|
self._nvr = nvr
|
|
|
|
self._uuid = uuid
|
|
|
|
self._name = name
|
2017-05-10 04:54:38 +00:00
|
|
|
self._password = password
|
2016-02-06 01:24:44 +00:00
|
|
|
self.is_streaming = False
|
2016-02-14 17:06:46 +00:00
|
|
|
self._connect_addr = None
|
|
|
|
self._camera = None
|
2016-02-06 01:24:44 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
2016-03-07 16:45:06 +00:00
|
|
|
"""Return the name of this camera."""
|
2016-02-06 01:24:44 +00:00
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_recording(self):
|
2016-03-07 16:45:06 +00:00
|
|
|
"""Return true if the camera is recording."""
|
2016-02-06 01:24:44 +00:00
|
|
|
caminfo = self._nvr.get_camera(self._uuid)
|
|
|
|
return caminfo['recordingSettings']['fullTimeRecordEnabled']
|
|
|
|
|
2016-02-14 17:09:54 +00:00
|
|
|
@property
|
|
|
|
def brand(self):
|
2016-03-07 16:45:06 +00:00
|
|
|
"""Return the brand of this camera."""
|
2016-02-14 17:09:54 +00:00
|
|
|
return 'Ubiquiti'
|
|
|
|
|
|
|
|
@property
|
|
|
|
def model(self):
|
2016-03-07 16:45:06 +00:00
|
|
|
"""Return the model of this camera."""
|
2016-02-14 17:09:54 +00:00
|
|
|
caminfo = self._nvr.get_camera(self._uuid)
|
|
|
|
return caminfo['model']
|
|
|
|
|
2016-02-14 17:06:46 +00:00
|
|
|
def _login(self):
|
2016-03-07 16:45:06 +00:00
|
|
|
"""Login to the camera."""
|
2016-02-06 01:24:44 +00:00
|
|
|
from uvcclient import camera as uvc_camera
|
2016-02-22 21:17:53 +00:00
|
|
|
|
2016-02-06 01:24:44 +00:00
|
|
|
caminfo = self._nvr.get_camera(self._uuid)
|
2016-02-14 17:06:46 +00:00
|
|
|
if self._connect_addr:
|
|
|
|
addrs = [self._connect_addr]
|
|
|
|
else:
|
|
|
|
addrs = [caminfo['host'], caminfo['internalHost']]
|
|
|
|
|
2016-06-05 17:09:58 +00:00
|
|
|
if self._nvr.server_version >= (3, 2, 0):
|
|
|
|
client_cls = uvc_camera.UVCCameraClientV320
|
|
|
|
else:
|
|
|
|
client_cls = uvc_camera.UVCCameraClient
|
|
|
|
|
2016-02-06 01:24:44 +00:00
|
|
|
camera = None
|
2016-02-14 17:06:46 +00:00
|
|
|
for addr in addrs:
|
2016-02-06 01:24:44 +00:00
|
|
|
try:
|
2017-04-30 05:04:49 +00:00
|
|
|
camera = client_cls(
|
2017-05-10 04:54:38 +00:00
|
|
|
addr, caminfo['username'], self._password)
|
2016-02-14 17:06:46 +00:00
|
|
|
camera.login()
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.debug("Logged into UVC camera %(name)s via %(addr)s",
|
2016-02-06 01:24:44 +00:00
|
|
|
dict(name=self._name, addr=addr))
|
2016-02-14 17:06:46 +00:00
|
|
|
self._connect_addr = addr
|
2016-02-22 22:06:06 +00:00
|
|
|
break
|
2016-02-06 01:24:44 +00:00
|
|
|
except socket.error:
|
|
|
|
pass
|
2016-02-14 16:36:51 +00:00
|
|
|
except uvc_camera.CameraConnectError:
|
|
|
|
pass
|
|
|
|
except uvc_camera.CameraAuthError:
|
|
|
|
pass
|
2016-02-22 22:06:06 +00:00
|
|
|
if not self._connect_addr:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.error("Unable to login to camera")
|
2016-02-06 01:24:44 +00:00
|
|
|
return None
|
|
|
|
|
2016-02-14 17:06:46 +00:00
|
|
|
self._camera = camera
|
|
|
|
return True
|
|
|
|
|
|
|
|
def camera_image(self):
|
2016-03-07 16:45:06 +00:00
|
|
|
"""Return the image of this camera."""
|
2016-02-14 17:06:46 +00:00
|
|
|
from uvcclient import camera as uvc_camera
|
|
|
|
if not self._camera:
|
|
|
|
if not self._login():
|
|
|
|
return
|
|
|
|
|
|
|
|
def _get_image(retry=True):
|
|
|
|
try:
|
|
|
|
return self._camera.get_snapshot()
|
|
|
|
except uvc_camera.CameraConnectError:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.error("Unable to contact camera")
|
2016-02-14 17:06:46 +00:00
|
|
|
except uvc_camera.CameraAuthError:
|
|
|
|
if retry:
|
|
|
|
self._login()
|
|
|
|
return _get_image(retry=False)
|
|
|
|
else:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"Unable to log into camera, unable to get snapshot")
|
2016-02-14 17:06:46 +00:00
|
|
|
raise
|
|
|
|
|
|
|
|
return _get_image()
|