2019-03-17 06:36:31 +00:00
|
|
|
"""Support for ESPHome cameras."""
|
2021-03-17 22:49:01 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2019-03-17 06:36:31 +00:00
|
|
|
import asyncio
|
2023-07-08 10:36:40 +00:00
|
|
|
from collections.abc import Callable, Coroutine
|
|
|
|
from functools import partial
|
2021-07-12 20:56:10 +00:00
|
|
|
from typing import Any
|
2019-05-29 11:33:49 +00:00
|
|
|
|
|
|
|
from aioesphomeapi import CameraInfo, CameraState
|
2021-07-12 20:56:10 +00:00
|
|
|
from aiohttp import web
|
2019-03-17 06:36:31 +00:00
|
|
|
|
|
|
|
from homeassistant.components import camera
|
|
|
|
from homeassistant.components.camera import Camera
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2021-07-28 11:56:45 +00:00
|
|
|
from homeassistant.core import HomeAssistant, callback
|
2021-07-12 20:56:10 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2019-03-17 06:36:31 +00:00
|
|
|
|
2023-06-25 03:12:36 +00:00
|
|
|
from .entity import (
|
|
|
|
EsphomeEntity,
|
|
|
|
platform_async_setup_entry,
|
|
|
|
)
|
2019-03-17 06:36:31 +00:00
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
async def async_setup_entry(
|
2021-07-12 20:56:10 +00:00
|
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
2019-07-31 19:25:30 +00:00
|
|
|
) -> None:
|
2019-03-17 06:36:31 +00:00
|
|
|
"""Set up esphome cameras based on a config entry."""
|
|
|
|
await platform_async_setup_entry(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass,
|
|
|
|
entry,
|
|
|
|
async_add_entities,
|
|
|
|
info_type=CameraInfo,
|
|
|
|
entity_type=EsphomeCamera,
|
|
|
|
state_type=CameraState,
|
2019-03-17 06:36:31 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2021-07-28 11:56:45 +00:00
|
|
|
class EsphomeCamera(Camera, EsphomeEntity[CameraInfo, CameraState]):
|
2019-03-17 06:36:31 +00:00
|
|
|
"""A camera implementation for ESPHome."""
|
|
|
|
|
2021-07-12 20:56:10 +00:00
|
|
|
def __init__(self, *args: Any, **kwargs: Any) -> None:
|
2019-03-17 06:36:31 +00:00
|
|
|
"""Initialize."""
|
|
|
|
Camera.__init__(self)
|
2021-07-28 11:56:45 +00:00
|
|
|
EsphomeEntity.__init__(self, *args, **kwargs)
|
2023-07-08 10:36:40 +00:00
|
|
|
self._loop = asyncio.get_running_loop()
|
|
|
|
self._image_futures: list[asyncio.Future[bool | None]] = []
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _set_futures(self, result: bool) -> None:
|
|
|
|
"""Set futures to done."""
|
|
|
|
for future in self._image_futures:
|
|
|
|
if not future.done():
|
|
|
|
future.set_result(result)
|
|
|
|
self._image_futures.clear()
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _on_device_update(self) -> None:
|
|
|
|
"""Handle device going available or unavailable."""
|
|
|
|
super()._on_device_update()
|
|
|
|
if not self.available:
|
|
|
|
self._set_futures(False)
|
2019-03-17 06:36:31 +00:00
|
|
|
|
2021-07-28 11:56:45 +00:00
|
|
|
@callback
|
|
|
|
def _on_state_update(self) -> None:
|
2019-03-17 06:36:31 +00:00
|
|
|
"""Notify listeners of new image when update arrives."""
|
2021-07-28 11:56:45 +00:00
|
|
|
super()._on_state_update()
|
2023-07-08 10:36:40 +00:00
|
|
|
self._set_futures(True)
|
2019-03-17 06:36:31 +00:00
|
|
|
|
2021-08-11 00:33:06 +00:00
|
|
|
async def async_camera_image(
|
|
|
|
self, width: int | None = None, height: int | None = None
|
|
|
|
) -> bytes | None:
|
2019-03-17 06:36:31 +00:00
|
|
|
"""Return single camera image bytes."""
|
2023-07-08 10:36:40 +00:00
|
|
|
return await self._async_request_image(self._client.request_single_image)
|
2019-03-17 06:36:31 +00:00
|
|
|
|
2023-07-08 10:36:40 +00:00
|
|
|
async def _async_request_image(
|
|
|
|
self, request_method: Callable[[], Coroutine[Any, Any, None]]
|
|
|
|
) -> bytes | None:
|
|
|
|
"""Wait for an image to be available and return it."""
|
2019-03-17 06:36:31 +00:00
|
|
|
if not self.available:
|
|
|
|
return None
|
2023-07-08 10:36:40 +00:00
|
|
|
image_future = self._loop.create_future()
|
|
|
|
self._image_futures.append(image_future)
|
|
|
|
await request_method()
|
|
|
|
if not await image_future:
|
|
|
|
return None
|
|
|
|
return self._state.data
|
2019-03-17 06:36:31 +00:00
|
|
|
|
2021-07-12 20:56:10 +00:00
|
|
|
async def handle_async_mjpeg_stream(
|
|
|
|
self, request: web.Request
|
|
|
|
) -> web.StreamResponse:
|
2019-03-17 06:36:31 +00:00
|
|
|
"""Serve an HTTP MJPEG stream from the camera."""
|
2023-07-08 10:36:40 +00:00
|
|
|
stream_request = partial(
|
|
|
|
self._async_request_image, self._client.request_image_stream
|
|
|
|
)
|
2019-03-17 06:36:31 +00:00
|
|
|
return await camera.async_get_still_stream(
|
2023-07-08 10:36:40 +00:00
|
|
|
request, stream_request, camera.DEFAULT_CONTENT_TYPE, 0.0
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|