2021-12-26 06:12:57 +00:00
|
|
|
"""Support for Ubiquiti's UniFi Protect NVR."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-12-27 11:39:24 +00:00
|
|
|
from collections.abc import Generator
|
2021-12-26 06:12:57 +00:00
|
|
|
import logging
|
|
|
|
|
|
|
|
from pyunifiprotect.api import ProtectApiClient
|
2022-01-08 16:49:55 +00:00
|
|
|
from pyunifiprotect.data import Camera as UFPCamera, StateType
|
2021-12-26 06:12:57 +00:00
|
|
|
from pyunifiprotect.data.devices import CameraChannel
|
|
|
|
|
2022-04-07 07:35:15 +00:00
|
|
|
from homeassistant.components.camera import Camera, CameraEntityFeature
|
2021-12-26 06:12:57 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
2021-12-27 11:39:24 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2021-12-26 06:12:57 +00:00
|
|
|
|
|
|
|
from .const import (
|
|
|
|
ATTR_BITRATE,
|
|
|
|
ATTR_CHANNEL_ID,
|
|
|
|
ATTR_FPS,
|
|
|
|
ATTR_HEIGHT,
|
|
|
|
ATTR_WIDTH,
|
|
|
|
DOMAIN,
|
|
|
|
)
|
|
|
|
from .data import ProtectData
|
|
|
|
from .entity import ProtectDeviceEntity
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
def get_camera_channels(
|
|
|
|
protect: ProtectApiClient,
|
|
|
|
) -> Generator[tuple[UFPCamera, CameraChannel, bool], None, None]:
|
|
|
|
"""Get all the camera channels."""
|
|
|
|
for camera in protect.bootstrap.cameras.values():
|
2021-12-27 11:39:24 +00:00
|
|
|
if not camera.channels:
|
2021-12-26 06:12:57 +00:00
|
|
|
_LOGGER.warning(
|
|
|
|
"Camera does not have any channels: %s (id: %s)", camera.name, camera.id
|
|
|
|
)
|
|
|
|
continue
|
|
|
|
|
|
|
|
is_default = True
|
|
|
|
for channel in camera.channels:
|
2022-01-12 21:27:41 +00:00
|
|
|
if channel.is_package:
|
|
|
|
yield camera, channel, True
|
|
|
|
elif channel.is_rtsp_enabled:
|
2021-12-26 06:12:57 +00:00
|
|
|
yield camera, channel, is_default
|
|
|
|
is_default = False
|
|
|
|
|
|
|
|
# no RTSP enabled use first channel with no stream
|
|
|
|
if is_default:
|
|
|
|
yield camera, camera.channels[0], True
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
entry: ConfigEntry,
|
2021-12-27 11:39:24 +00:00
|
|
|
async_add_entities: AddEntitiesCallback,
|
2021-12-26 06:12:57 +00:00
|
|
|
) -> None:
|
|
|
|
"""Discover cameras on a UniFi Protect NVR."""
|
|
|
|
data: ProtectData = hass.data[DOMAIN][entry.entry_id]
|
|
|
|
disable_stream = data.disable_stream
|
|
|
|
|
|
|
|
entities = []
|
|
|
|
for camera, channel, is_default in get_camera_channels(data.api):
|
2022-01-12 21:27:41 +00:00
|
|
|
# do not enable streaming for package camera
|
|
|
|
# 2 FPS causes a lot of buferring
|
2021-12-26 06:12:57 +00:00
|
|
|
entities.append(
|
|
|
|
ProtectCamera(
|
|
|
|
data,
|
|
|
|
camera,
|
|
|
|
channel,
|
|
|
|
is_default,
|
|
|
|
True,
|
2022-01-12 21:27:41 +00:00
|
|
|
disable_stream or channel.is_package,
|
2021-12-26 06:12:57 +00:00
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2022-01-12 21:27:41 +00:00
|
|
|
if channel.is_rtsp_enabled and not channel.is_package:
|
2021-12-26 06:12:57 +00:00
|
|
|
entities.append(
|
|
|
|
ProtectCamera(
|
|
|
|
data,
|
|
|
|
camera,
|
|
|
|
channel,
|
|
|
|
is_default,
|
|
|
|
False,
|
|
|
|
disable_stream,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
async_add_entities(entities)
|
|
|
|
|
|
|
|
|
|
|
|
class ProtectCamera(ProtectDeviceEntity, Camera):
|
|
|
|
"""A Ubiquiti UniFi Protect Camera."""
|
|
|
|
|
2022-01-10 04:37:24 +00:00
|
|
|
device: UFPCamera
|
|
|
|
|
2021-12-26 06:12:57 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
data: ProtectData,
|
|
|
|
camera: UFPCamera,
|
|
|
|
channel: CameraChannel,
|
|
|
|
is_default: bool,
|
|
|
|
secure: bool,
|
|
|
|
disable_stream: bool,
|
|
|
|
) -> None:
|
|
|
|
"""Initialize an UniFi camera."""
|
|
|
|
self.channel = channel
|
|
|
|
self._secure = secure
|
|
|
|
self._disable_stream = disable_stream
|
|
|
|
self._last_image: bytes | None = None
|
2022-01-10 04:37:24 +00:00
|
|
|
super().__init__(data, camera)
|
2021-12-26 06:12:57 +00:00
|
|
|
|
|
|
|
if self._secure:
|
|
|
|
self._attr_unique_id = f"{self.device.id}_{self.channel.id}"
|
|
|
|
self._attr_name = f"{self.device.name} {self.channel.name}"
|
|
|
|
else:
|
|
|
|
self._attr_unique_id = f"{self.device.id}_{self.channel.id}_insecure"
|
|
|
|
self._attr_name = f"{self.device.name} {self.channel.name} Insecure"
|
|
|
|
# only the default (first) channel is enabled by default
|
|
|
|
self._attr_entity_registry_enabled_default = is_default and secure
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_set_stream_source(self) -> None:
|
|
|
|
disable_stream = self._disable_stream
|
|
|
|
if not self.channel.is_rtsp_enabled:
|
|
|
|
disable_stream = False
|
|
|
|
|
|
|
|
rtsp_url = self.channel.rtsp_url
|
|
|
|
if self._secure:
|
|
|
|
rtsp_url = self.channel.rtsps_url
|
|
|
|
|
|
|
|
# _async_set_stream_source called by __init__
|
|
|
|
self._stream_source = ( # pylint: disable=attribute-defined-outside-init
|
|
|
|
None if disable_stream else rtsp_url
|
|
|
|
)
|
|
|
|
self._attr_supported_features: int = (
|
2022-04-07 07:35:15 +00:00
|
|
|
CameraEntityFeature.STREAM if self._stream_source else 0
|
2021-12-26 06:12:57 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_update_device_from_protect(self) -> None:
|
|
|
|
super()._async_update_device_from_protect()
|
|
|
|
self.channel = self.device.channels[self.channel.id]
|
|
|
|
self._attr_motion_detection_enabled = (
|
2022-01-08 16:49:55 +00:00
|
|
|
self.device.state == StateType.CONNECTED
|
|
|
|
and self.device.feature_flags.has_motion_zones
|
|
|
|
)
|
|
|
|
self._attr_is_recording = (
|
|
|
|
self.device.state == StateType.CONNECTED and self.device.is_recording
|
2021-12-26 06:12:57 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
self._async_set_stream_source()
|
2021-12-27 11:39:24 +00:00
|
|
|
self._attr_extra_state_attributes = {
|
2021-12-26 06:12:57 +00:00
|
|
|
ATTR_WIDTH: self.channel.width,
|
|
|
|
ATTR_HEIGHT: self.channel.height,
|
|
|
|
ATTR_FPS: self.channel.fps,
|
|
|
|
ATTR_BITRATE: self.channel.bitrate,
|
|
|
|
ATTR_CHANNEL_ID: self.channel.id,
|
|
|
|
}
|
|
|
|
|
|
|
|
async def async_camera_image(
|
|
|
|
self, width: int | None = None, height: int | None = None
|
|
|
|
) -> bytes | None:
|
|
|
|
"""Return the Camera Image."""
|
2022-01-14 23:38:01 +00:00
|
|
|
if self.channel.is_package:
|
|
|
|
last_image = await self.device.get_package_snapshot(width, height)
|
|
|
|
else:
|
|
|
|
last_image = await self.device.get_snapshot(width, height)
|
2021-12-26 06:12:57 +00:00
|
|
|
self._last_image = last_image
|
|
|
|
return self._last_image
|
|
|
|
|
|
|
|
async def stream_source(self) -> str | None:
|
|
|
|
"""Return the Stream Source."""
|
|
|
|
return self._stream_source
|