core/homeassistant/components/camera/__init__.py

287 lines
8.1 KiB
Python
Raw Normal View History

2015-06-05 12:51:29 +00:00
# pylint: disable=too-many-lines
"""
2016-02-17 07:52:05 +00:00
Component to interface with cameras.
2015-06-05 12:51:29 +00:00
2015-11-09 12:12:18 +00:00
For more details about this component, please refer to the documentation at
https://home-assistant.io/components/camera/
2015-06-05 12:51:29 +00:00
"""
import asyncio
import collections
from contextlib import suppress
from datetime import timedelta
2015-06-05 12:51:29 +00:00
import logging
import hashlib
from random import SystemRandom
import aiohttp
from aiohttp import web
import async_timeout
2015-11-29 21:49:05 +00:00
from homeassistant.core import callback
from homeassistant.const import ATTR_ENTITY_PICTURE
from homeassistant.exceptions import HomeAssistantError
from homeassistant.helpers.aiohttp_client import async_get_clientsession
2015-06-05 12:51:29 +00:00
from homeassistant.helpers.entity import Entity
2015-11-29 21:49:05 +00:00
from homeassistant.helpers.entity_component import EntityComponent
2016-03-28 01:48:51 +00:00
from homeassistant.helpers.config_validation import PLATFORM_SCHEMA # noqa
from homeassistant.components.http import HomeAssistantView, KEY_AUTHENTICATED
from homeassistant.helpers.event import async_track_time_interval
2015-06-05 12:51:29 +00:00
_LOGGER = logging.getLogger(__name__)
2015-06-05 12:51:29 +00:00
DOMAIN = 'camera'
DEPENDENCIES = ['http']
SCAN_INTERVAL = timedelta(seconds=30)
2015-06-05 12:51:29 +00:00
ENTITY_ID_FORMAT = DOMAIN + '.{}'
STATE_RECORDING = 'recording'
2015-06-05 12:51:29 +00:00
STATE_STREAMING = 'streaming'
STATE_IDLE = 'idle'
2016-05-27 08:29:48 +00:00
ENTITY_IMAGE_URL = '/api/camera_proxy/{0}?token={1}'
2015-07-10 08:03:46 +00:00
TOKEN_CHANGE_INTERVAL = timedelta(minutes=5)
_RND = SystemRandom()
2015-06-05 12:51:29 +00:00
@asyncio.coroutine
def async_get_image(hass, entity_id, timeout=10):
"""Fetch a image from a camera entity."""
websession = async_get_clientsession(hass)
state = hass.states.get(entity_id)
if state is None:
raise HomeAssistantError(
"No entity '{0}' for grab a image".format(entity_id))
url = "{0}{1}".format(
hass.config.api.base_url,
state.attributes.get(ATTR_ENTITY_PICTURE)
)
try:
with async_timeout.timeout(timeout, loop=hass.loop):
response = yield from websession.get(url)
if response.status != 200:
raise HomeAssistantError("Error {0} on {1}".format(
response.status, url))
image = yield from response.read()
return image
except (asyncio.TimeoutError, aiohttp.ClientError):
raise HomeAssistantError("Can't connect to {0}".format(url))
@asyncio.coroutine
def async_setup(hass, config):
"""Set up the camera component."""
component = EntityComponent(_LOGGER, DOMAIN, hass, SCAN_INTERVAL)
2015-06-05 12:51:29 +00:00
hass.http.register_view(CameraImageView(component.entities))
hass.http.register_view(CameraMjpegStream(component.entities))
2015-06-05 12:51:29 +00:00
yield from component.async_setup(config)
@callback
def update_tokens(time):
"""Update tokens of the entities."""
for entity in component.entities.values():
entity.async_update_token()
hass.async_add_job(entity.async_update_ha_state())
async_track_time_interval(hass, update_tokens, TOKEN_CHANGE_INTERVAL)
2015-07-10 09:42:22 +00:00
return True
2015-06-05 12:51:29 +00:00
class Camera(Entity):
2016-02-17 07:52:05 +00:00
"""The base class for camera entities."""
2016-03-07 19:29:54 +00:00
2015-07-11 06:17:12 +00:00
def __init__(self):
2016-02-17 07:52:05 +00:00
"""Initialize a camera."""
2015-07-11 06:17:12 +00:00
self.is_streaming = False
self.access_tokens = collections.deque([], 2)
self.async_update_token()
2016-02-17 07:52:05 +00:00
@property
def should_poll(self):
"""No need to poll cameras."""
return False
2016-02-24 06:41:24 +00:00
@property
def entity_picture(self):
"""Return a link to the camera feed as entity picture."""
return ENTITY_IMAGE_URL.format(self.entity_id, self.access_tokens[-1])
2016-02-24 06:41:24 +00:00
2015-06-05 12:51:29 +00:00
@property
def is_recording(self):
2016-02-17 07:52:05 +00:00
"""Return true if the device is recording."""
2015-06-05 12:51:29 +00:00
return False
@property
def brand(self):
"""Return the camera brand."""
2015-06-05 12:51:29 +00:00
return None
@property
def model(self):
"""Return the camera model."""
2015-06-05 12:51:29 +00:00
return None
2015-07-11 06:17:12 +00:00
def camera_image(self):
2016-02-17 07:52:05 +00:00
"""Return bytes of camera image."""
2015-06-05 12:51:29 +00:00
raise NotImplementedError()
def async_camera_image(self):
"""Return bytes of camera image.
This method must be run in the event loop and returns a coroutine.
"""
return self.hass.async_add_job(self.camera_image)
@asyncio.coroutine
def handle_async_mjpeg_stream(self, request):
"""Generate an HTTP MJPEG stream from camera images.
This method must be run in the event loop.
"""
response = web.StreamResponse()
response.content_type = ('multipart/x-mixed-replace; '
'boundary=--jpegboundary')
yield from response.prepare(request)
def write(img_bytes):
"""Write image to stream."""
response.write(bytes(
'--jpegboundary\r\n'
'Content-Type: image/jpeg\r\n'
'Content-Length: {}\r\n\r\n'.format(
len(img_bytes)), 'utf-8') + img_bytes + b'\r\n')
last_image = None
try:
while True:
img_bytes = yield from self.async_camera_image()
if not img_bytes:
break
if img_bytes and img_bytes != last_image:
write(img_bytes)
# Chrome seems to always ignore first picture,
# print it twice.
if last_image is None:
write(img_bytes)
last_image = img_bytes
yield from response.drain()
yield from asyncio.sleep(.5)
except asyncio.CancelledError:
_LOGGER.debug("Stream closed by frontend.")
response = None
finally:
if response is not None:
yield from response.write_eof()
2015-06-05 12:51:29 +00:00
@property
def state(self):
"""Return the camera state."""
2015-06-05 12:51:29 +00:00
if self.is_recording:
return STATE_RECORDING
elif self.is_streaming:
return STATE_STREAMING
else:
return STATE_IDLE
@property
def state_attributes(self):
"""Return the camera state attributes."""
attr = {
'access_token': self.access_tokens[-1],
}
2015-07-11 18:55:25 +00:00
if self.model:
attr['model_name'] = self.model
if self.brand:
attr['brand'] = self.brand
return attr
2016-05-14 07:58:36 +00:00
@callback
def async_update_token(self):
"""Update the used token."""
self.access_tokens.append(
hashlib.sha256(
_RND.getrandbits(256).to_bytes(32, 'little')).hexdigest())
2016-05-14 07:58:36 +00:00
class CameraView(HomeAssistantView):
"""Base CameraView."""
requires_auth = False
def __init__(self, entities):
2016-05-14 07:58:36 +00:00
"""Initialize a basic camera view."""
self.entities = entities
@asyncio.coroutine
def get(self, request, entity_id):
"""Start a GET request."""
camera = self.entities.get(entity_id)
if camera is None:
status = 404 if request[KEY_AUTHENTICATED] else 401
return web.Response(status=status)
authenticated = (request[KEY_AUTHENTICATED] or
request.query.get('token') in camera.access_tokens)
if not authenticated:
return web.Response(status=401)
response = yield from self.handle(request, camera)
return response
@asyncio.coroutine
def handle(self, request, camera):
"""Handle the camera request."""
raise NotImplementedError()
2016-05-14 07:58:36 +00:00
class CameraImageView(CameraView):
"""Camera view to serve an image."""
url = '/api/camera_proxy/{entity_id}'
name = 'api:camera:image'
2016-05-14 07:58:36 +00:00
@asyncio.coroutine
def handle(self, request, camera):
2016-05-14 07:58:36 +00:00
"""Serve camera image."""
with suppress(asyncio.CancelledError, asyncio.TimeoutError):
with async_timeout.timeout(10, loop=request.app['hass'].loop):
image = yield from camera.async_camera_image()
if image:
return web.Response(body=image, content_type='image/jpeg')
2016-05-14 07:58:36 +00:00
return web.Response(status=500)
2016-05-14 07:58:36 +00:00
class CameraMjpegStream(CameraView):
"""Camera View to serve an MJPEG stream."""
url = '/api/camera_proxy_stream/{entity_id}'
name = 'api:camera:stream'
2016-05-14 07:58:36 +00:00
@asyncio.coroutine
def handle(self, request, camera):
2016-05-14 07:58:36 +00:00
"""Serve camera image."""
yield from camera.handle_async_mjpeg_stream(request)