192 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Python
		
	
	
			
		
		
	
	
			192 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Python
		
	
	
"""The motionEye integration."""
 | 
						|
from __future__ import annotations
 | 
						|
 | 
						|
import logging
 | 
						|
from types import MappingProxyType
 | 
						|
from typing import Any
 | 
						|
 | 
						|
import aiohttp
 | 
						|
from motioneye_client.client import MotionEyeClient
 | 
						|
from motioneye_client.const import (
 | 
						|
    DEFAULT_SURVEILLANCE_USERNAME,
 | 
						|
    KEY_MOTION_DETECTION,
 | 
						|
    KEY_NAME,
 | 
						|
    KEY_STREAMING_AUTH_MODE,
 | 
						|
)
 | 
						|
 | 
						|
from homeassistant.components.mjpeg.camera import (
 | 
						|
    CONF_MJPEG_URL,
 | 
						|
    CONF_STILL_IMAGE_URL,
 | 
						|
    CONF_VERIFY_SSL,
 | 
						|
    MjpegCamera,
 | 
						|
)
 | 
						|
from homeassistant.config_entries import ConfigEntry
 | 
						|
from homeassistant.const import (
 | 
						|
    CONF_AUTHENTICATION,
 | 
						|
    CONF_NAME,
 | 
						|
    CONF_PASSWORD,
 | 
						|
    CONF_USERNAME,
 | 
						|
    HTTP_BASIC_AUTHENTICATION,
 | 
						|
    HTTP_DIGEST_AUTHENTICATION,
 | 
						|
)
 | 
						|
from homeassistant.core import HomeAssistant, callback
 | 
						|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
 | 
						|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
 | 
						|
 | 
						|
from . import (
 | 
						|
    MotionEyeEntity,
 | 
						|
    get_camera_from_cameras,
 | 
						|
    is_acceptable_camera,
 | 
						|
    listen_for_new_cameras,
 | 
						|
)
 | 
						|
from .const import (
 | 
						|
    CONF_CLIENT,
 | 
						|
    CONF_COORDINATOR,
 | 
						|
    CONF_SURVEILLANCE_PASSWORD,
 | 
						|
    CONF_SURVEILLANCE_USERNAME,
 | 
						|
    DOMAIN,
 | 
						|
    MOTIONEYE_MANUFACTURER,
 | 
						|
    TYPE_MOTIONEYE_MJPEG_CAMERA,
 | 
						|
)
 | 
						|
 | 
						|
_LOGGER = logging.getLogger(__name__)
 | 
						|
 | 
						|
PLATFORMS = ["camera"]
 | 
						|
 | 
						|
 | 
						|
async def async_setup_entry(
 | 
						|
    hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
 | 
						|
) -> None:
 | 
						|
    """Set up motionEye from a config entry."""
 | 
						|
    entry_data = hass.data[DOMAIN][entry.entry_id]
 | 
						|
 | 
						|
    @callback
 | 
						|
    def camera_add(camera: dict[str, Any]) -> None:
 | 
						|
        """Add a new motionEye camera."""
 | 
						|
        async_add_entities(
 | 
						|
            [
 | 
						|
                MotionEyeMjpegCamera(
 | 
						|
                    entry.entry_id,
 | 
						|
                    entry.data.get(
 | 
						|
                        CONF_SURVEILLANCE_USERNAME, DEFAULT_SURVEILLANCE_USERNAME
 | 
						|
                    ),
 | 
						|
                    entry.data.get(CONF_SURVEILLANCE_PASSWORD, ""),
 | 
						|
                    camera,
 | 
						|
                    entry_data[CONF_CLIENT],
 | 
						|
                    entry_data[CONF_COORDINATOR],
 | 
						|
                    entry.options,
 | 
						|
                )
 | 
						|
            ]
 | 
						|
        )
 | 
						|
 | 
						|
    listen_for_new_cameras(hass, entry, camera_add)
 | 
						|
 | 
						|
 | 
						|
class MotionEyeMjpegCamera(MotionEyeEntity, MjpegCamera):
 | 
						|
    """motionEye mjpeg camera."""
 | 
						|
 | 
						|
    def __init__(
 | 
						|
        self,
 | 
						|
        config_entry_id: str,
 | 
						|
        username: str,
 | 
						|
        password: str,
 | 
						|
        camera: dict[str, Any],
 | 
						|
        client: MotionEyeClient,
 | 
						|
        coordinator: DataUpdateCoordinator,
 | 
						|
        options: MappingProxyType[str, str],
 | 
						|
    ) -> None:
 | 
						|
        """Initialize a MJPEG camera."""
 | 
						|
        self._surveillance_username = username
 | 
						|
        self._surveillance_password = password
 | 
						|
        self._motion_detection_enabled: bool = camera.get(KEY_MOTION_DETECTION, False)
 | 
						|
 | 
						|
        # motionEye cameras are always streaming or unavailable.
 | 
						|
        self.is_streaming = True
 | 
						|
 | 
						|
        MotionEyeEntity.__init__(
 | 
						|
            self,
 | 
						|
            config_entry_id,
 | 
						|
            TYPE_MOTIONEYE_MJPEG_CAMERA,
 | 
						|
            camera,
 | 
						|
            client,
 | 
						|
            coordinator,
 | 
						|
            options,
 | 
						|
        )
 | 
						|
        MjpegCamera.__init__(
 | 
						|
            self,
 | 
						|
            {
 | 
						|
                CONF_VERIFY_SSL: False,
 | 
						|
                **self._get_mjpeg_camera_properties_for_camera(camera),
 | 
						|
            },
 | 
						|
        )
 | 
						|
 | 
						|
    @callback
 | 
						|
    def _get_mjpeg_camera_properties_for_camera(
 | 
						|
        self, camera: dict[str, Any]
 | 
						|
    ) -> dict[str, Any]:
 | 
						|
        """Convert a motionEye camera to MjpegCamera internal properties."""
 | 
						|
        auth = None
 | 
						|
        if camera.get(KEY_STREAMING_AUTH_MODE) in (
 | 
						|
            HTTP_BASIC_AUTHENTICATION,
 | 
						|
            HTTP_DIGEST_AUTHENTICATION,
 | 
						|
        ):
 | 
						|
            auth = camera[KEY_STREAMING_AUTH_MODE]
 | 
						|
 | 
						|
        return {
 | 
						|
            CONF_NAME: camera[KEY_NAME],
 | 
						|
            CONF_USERNAME: self._surveillance_username if auth is not None else None,
 | 
						|
            CONF_PASSWORD: self._surveillance_password if auth is not None else None,
 | 
						|
            CONF_MJPEG_URL: self._client.get_camera_stream_url(camera) or "",
 | 
						|
            CONF_STILL_IMAGE_URL: self._client.get_camera_snapshot_url(camera),
 | 
						|
            CONF_AUTHENTICATION: auth,
 | 
						|
        }
 | 
						|
 | 
						|
    @callback
 | 
						|
    def _set_mjpeg_camera_state_for_camera(self, camera: dict[str, Any]) -> None:
 | 
						|
        """Set the internal state to match the given camera."""
 | 
						|
 | 
						|
        # Sets the state of the underlying (inherited) MjpegCamera based on the updated
 | 
						|
        # MotionEye camera dictionary.
 | 
						|
        properties = self._get_mjpeg_camera_properties_for_camera(camera)
 | 
						|
        self._name = properties[CONF_NAME]
 | 
						|
        self._username = properties[CONF_USERNAME]
 | 
						|
        self._password = properties[CONF_PASSWORD]
 | 
						|
        self._mjpeg_url = properties[CONF_MJPEG_URL]
 | 
						|
        self._still_image_url = properties[CONF_STILL_IMAGE_URL]
 | 
						|
        self._authentication = properties[CONF_AUTHENTICATION]
 | 
						|
 | 
						|
        if self._authentication == HTTP_BASIC_AUTHENTICATION:
 | 
						|
            self._auth = aiohttp.BasicAuth(self._username, password=self._password)
 | 
						|
 | 
						|
    def _is_acceptable_streaming_camera(self) -> bool:
 | 
						|
        """Determine if a camera is streaming/usable."""
 | 
						|
        return is_acceptable_camera(
 | 
						|
            self._camera
 | 
						|
        ) and MotionEyeClient.is_camera_streaming(self._camera)
 | 
						|
 | 
						|
    @property
 | 
						|
    def available(self) -> bool:
 | 
						|
        """Return if entity is available."""
 | 
						|
        return super().available and self._is_acceptable_streaming_camera()
 | 
						|
 | 
						|
    @callback
 | 
						|
    def _handle_coordinator_update(self) -> None:
 | 
						|
        """Handle updated data from the coordinator."""
 | 
						|
        self._camera = get_camera_from_cameras(self._camera_id, self.coordinator.data)
 | 
						|
        if self._camera and self._is_acceptable_streaming_camera():
 | 
						|
            self._set_mjpeg_camera_state_for_camera(self._camera)
 | 
						|
            self._motion_detection_enabled = self._camera.get(
 | 
						|
                KEY_MOTION_DETECTION, False
 | 
						|
            )
 | 
						|
        super()._handle_coordinator_update()
 | 
						|
 | 
						|
    @property
 | 
						|
    def brand(self) -> str:
 | 
						|
        """Return the camera brand."""
 | 
						|
        return MOTIONEYE_MANUFACTURER
 | 
						|
 | 
						|
    @property
 | 
						|
    def motion_detection_enabled(self) -> bool:
 | 
						|
        """Return the camera motion detection status."""
 | 
						|
        return self._motion_detection_enabled
 |