181 lines
5.4 KiB
Python
181 lines
5.4 KiB
Python
"""Camera platform that receives images through HTTP POST."""
|
|
from __future__ import annotations
|
|
|
|
import asyncio
|
|
from collections import deque
|
|
from datetime import timedelta
|
|
import logging
|
|
|
|
import aiohttp
|
|
import async_timeout
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.camera import PLATFORM_SCHEMA, STATE_IDLE, Camera
|
|
from homeassistant.components.camera.const import DOMAIN
|
|
from homeassistant.const import CONF_NAME, CONF_TIMEOUT, CONF_WEBHOOK_ID
|
|
from homeassistant.core import callback
|
|
from homeassistant.helpers import config_validation as cv
|
|
from homeassistant.helpers.event import async_track_point_in_utc_time
|
|
import homeassistant.util.dt as dt_util
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
CONF_BUFFER_SIZE = "buffer"
|
|
CONF_IMAGE_FIELD = "field"
|
|
|
|
DEFAULT_NAME = "Push Camera"
|
|
|
|
ATTR_FILENAME = "filename"
|
|
ATTR_LAST_TRIP = "last_trip"
|
|
|
|
PUSH_CAMERA_DATA = "push_camera"
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
{
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
vol.Optional(CONF_BUFFER_SIZE, default=1): cv.positive_int,
|
|
vol.Optional(CONF_TIMEOUT, default=timedelta(seconds=5)): vol.All(
|
|
cv.time_period, cv.positive_timedelta
|
|
),
|
|
vol.Optional(CONF_IMAGE_FIELD, default="image"): cv.string,
|
|
vol.Required(CONF_WEBHOOK_ID): cv.string,
|
|
}
|
|
)
|
|
|
|
|
|
async def async_setup_platform(hass, config, async_add_entities, discovery_info=None):
|
|
"""Set up the Push Camera platform."""
|
|
if PUSH_CAMERA_DATA not in hass.data:
|
|
hass.data[PUSH_CAMERA_DATA] = {}
|
|
|
|
webhook_id = config.get(CONF_WEBHOOK_ID)
|
|
|
|
cameras = [
|
|
PushCamera(
|
|
hass,
|
|
config[CONF_NAME],
|
|
config[CONF_BUFFER_SIZE],
|
|
config[CONF_TIMEOUT],
|
|
config[CONF_IMAGE_FIELD],
|
|
webhook_id,
|
|
)
|
|
]
|
|
|
|
async_add_entities(cameras)
|
|
|
|
|
|
async def handle_webhook(hass, webhook_id, request):
|
|
"""Handle incoming webhook POST with image files."""
|
|
try:
|
|
async with async_timeout.timeout(5):
|
|
data = dict(await request.post())
|
|
except (asyncio.TimeoutError, aiohttp.web.HTTPException) as error:
|
|
_LOGGER.error("Could not get information from POST <%s>", error)
|
|
return
|
|
|
|
camera = hass.data[PUSH_CAMERA_DATA][webhook_id]
|
|
|
|
if camera.image_field not in data:
|
|
_LOGGER.warning("Webhook call without POST parameter <%s>", camera.image_field)
|
|
return
|
|
|
|
await camera.update_image(
|
|
data[camera.image_field].file.read(), data[camera.image_field].filename
|
|
)
|
|
|
|
|
|
class PushCamera(Camera):
|
|
"""The representation of a Push camera."""
|
|
|
|
def __init__(self, hass, name, buffer_size, timeout, image_field, webhook_id):
|
|
"""Initialize push camera component."""
|
|
super().__init__()
|
|
self._name = name
|
|
self._last_trip = None
|
|
self._filename = None
|
|
self._expired_listener = None
|
|
self._timeout = timeout
|
|
self.queue = deque([], buffer_size)
|
|
self._current_image = None
|
|
self._image_field = image_field
|
|
self.webhook_id = webhook_id
|
|
self.webhook_url = hass.components.webhook.async_generate_url(webhook_id)
|
|
|
|
async def async_added_to_hass(self):
|
|
"""Call when entity is added to hass."""
|
|
self.hass.data[PUSH_CAMERA_DATA][self.webhook_id] = self
|
|
|
|
try:
|
|
self.hass.components.webhook.async_register(
|
|
DOMAIN, self.name, self.webhook_id, handle_webhook
|
|
)
|
|
except ValueError:
|
|
_LOGGER.error(
|
|
"In <%s>, webhook_id <%s> already used", self.name, self.webhook_id
|
|
)
|
|
|
|
@property
|
|
def image_field(self):
|
|
"""HTTP field containing the image file."""
|
|
return self._image_field
|
|
|
|
async def update_image(self, image, filename):
|
|
"""Update the camera image."""
|
|
if self.state == STATE_IDLE:
|
|
self._attr_is_recording = True
|
|
self._last_trip = dt_util.utcnow()
|
|
self.queue.clear()
|
|
|
|
self._filename = filename
|
|
self.queue.appendleft(image)
|
|
|
|
@callback
|
|
def reset_state(now):
|
|
"""Set state to idle after no new images for a period of time."""
|
|
self._attr_is_recording = False
|
|
self._expired_listener = None
|
|
_LOGGER.debug("Reset state")
|
|
self.async_write_ha_state()
|
|
|
|
if self._expired_listener:
|
|
self._expired_listener()
|
|
|
|
self._expired_listener = async_track_point_in_utc_time(
|
|
self.hass, reset_state, dt_util.utcnow() + self._timeout
|
|
)
|
|
|
|
self.async_write_ha_state()
|
|
|
|
async def async_camera_image(
|
|
self, width: int | None = None, height: int | None = None
|
|
) -> bytes | None:
|
|
"""Return a still image response."""
|
|
if self.queue:
|
|
if self.state == STATE_IDLE:
|
|
self.queue.rotate(1)
|
|
self._current_image = self.queue[0]
|
|
|
|
return self._current_image
|
|
|
|
@property
|
|
def name(self):
|
|
"""Return the name of this camera."""
|
|
return self._name
|
|
|
|
@property
|
|
def motion_detection_enabled(self):
|
|
"""Camera Motion Detection Status."""
|
|
return False
|
|
|
|
@property
|
|
def extra_state_attributes(self):
|
|
"""Return the state attributes."""
|
|
return {
|
|
name: value
|
|
for name, value in (
|
|
(ATTR_LAST_TRIP, self._last_trip),
|
|
(ATTR_FILENAME, self._filename),
|
|
)
|
|
if value is not None
|
|
}
|