2019-02-14 15:01:46 +00:00
|
|
|
"""Support for views."""
|
2018-03-09 01:51:49 +00:00
|
|
|
import asyncio
|
|
|
|
import json
|
|
|
|
import logging
|
|
|
|
|
|
|
|
from aiohttp import web
|
2018-11-30 20:28:35 +00:00
|
|
|
from aiohttp.web_exceptions import (
|
2019-02-14 15:01:46 +00:00
|
|
|
HTTPBadRequest, HTTPInternalServerError, HTTPUnauthorized)
|
2018-11-30 20:28:35 +00:00
|
|
|
import voluptuous as vol
|
2018-03-09 01:51:49 +00:00
|
|
|
|
2019-02-14 15:01:46 +00:00
|
|
|
from homeassistant import exceptions
|
2018-03-09 01:51:49 +00:00
|
|
|
from homeassistant.const import CONTENT_TYPE_JSON
|
2019-02-14 15:01:46 +00:00
|
|
|
from homeassistant.core import Context, is_callback
|
2018-08-21 13:49:58 +00:00
|
|
|
from homeassistant.helpers.json import JSONEncoder
|
2018-03-09 01:51:49 +00:00
|
|
|
|
2019-03-21 05:56:46 +00:00
|
|
|
from .ban import process_success_login
|
|
|
|
from .const import KEY_AUTHENTICATED, KEY_HASS, KEY_REAL_IP
|
2018-03-09 01:51:49 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2018-07-20 08:45:20 +00:00
|
|
|
class HomeAssistantView:
|
2018-03-09 01:51:49 +00:00
|
|
|
"""Base view for all views."""
|
|
|
|
|
|
|
|
url = None
|
|
|
|
extra_urls = []
|
2018-07-19 06:37:00 +00:00
|
|
|
# Views inheriting from this class can override this
|
|
|
|
requires_auth = True
|
|
|
|
cors_allowed = False
|
2018-03-09 01:51:49 +00:00
|
|
|
|
|
|
|
# pylint: disable=no-self-use
|
2018-07-29 00:53:37 +00:00
|
|
|
def context(self, request):
|
|
|
|
"""Generate a context from a request."""
|
|
|
|
user = request.get('hass_user')
|
|
|
|
if user is None:
|
|
|
|
return Context()
|
|
|
|
|
|
|
|
return Context(user_id=user.id)
|
|
|
|
|
2018-03-09 01:51:49 +00:00
|
|
|
def json(self, result, status_code=200, headers=None):
|
|
|
|
"""Return a JSON response."""
|
2018-03-12 22:22:08 +00:00
|
|
|
try:
|
|
|
|
msg = json.dumps(
|
2018-11-28 12:25:23 +00:00
|
|
|
result, sort_keys=True, cls=JSONEncoder, allow_nan=False
|
|
|
|
).encode('UTF-8')
|
|
|
|
except (ValueError, TypeError) as err:
|
2018-03-12 22:22:08 +00:00
|
|
|
_LOGGER.error('Unable to serialize to JSON: %s\n%s', err, result)
|
|
|
|
raise HTTPInternalServerError
|
2018-03-09 01:51:49 +00:00
|
|
|
response = web.Response(
|
|
|
|
body=msg, content_type=CONTENT_TYPE_JSON, status=status_code,
|
|
|
|
headers=headers)
|
|
|
|
response.enable_compression()
|
|
|
|
return response
|
|
|
|
|
|
|
|
def json_message(self, message, status_code=200, message_code=None,
|
|
|
|
headers=None):
|
|
|
|
"""Return a JSON message response."""
|
|
|
|
data = {'message': message}
|
|
|
|
if message_code is not None:
|
|
|
|
data['code'] = message_code
|
|
|
|
return self.json(data, status_code, headers=headers)
|
|
|
|
|
2018-07-19 06:37:00 +00:00
|
|
|
def register(self, app, router):
|
2018-03-09 01:51:49 +00:00
|
|
|
"""Register the view with a router."""
|
|
|
|
assert self.url is not None, 'No url set for view'
|
|
|
|
urls = [self.url] + self.extra_urls
|
2018-07-19 06:37:00 +00:00
|
|
|
routes = []
|
2018-03-09 01:51:49 +00:00
|
|
|
|
2019-04-15 22:27:13 +00:00
|
|
|
for method in ('get', 'post', 'delete', 'put', 'patch', 'head',
|
|
|
|
'options'):
|
2018-03-09 01:51:49 +00:00
|
|
|
handler = getattr(self, method, None)
|
|
|
|
|
|
|
|
if not handler:
|
|
|
|
continue
|
|
|
|
|
|
|
|
handler = request_handler_factory(self, handler)
|
|
|
|
|
|
|
|
for url in urls:
|
2018-07-25 09:36:44 +00:00
|
|
|
routes.append(router.add_route(method, url, handler))
|
2018-03-09 01:51:49 +00:00
|
|
|
|
2018-07-19 06:37:00 +00:00
|
|
|
if not self.cors_allowed:
|
|
|
|
return
|
2018-03-09 01:51:49 +00:00
|
|
|
|
2018-07-25 09:36:44 +00:00
|
|
|
for route in routes:
|
|
|
|
app['allow_cors'](route)
|
2018-03-09 01:51:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
def request_handler_factory(view, handler):
|
|
|
|
"""Wrap the handler classes."""
|
|
|
|
assert asyncio.iscoroutinefunction(handler) or is_callback(handler), \
|
|
|
|
"Handler should be a coroutine or a callback."
|
|
|
|
|
|
|
|
async def handle(request):
|
|
|
|
"""Handle incoming request."""
|
2019-03-11 02:55:36 +00:00
|
|
|
if not request.app[KEY_HASS].is_running:
|
2018-03-09 01:51:49 +00:00
|
|
|
return web.Response(status=503)
|
|
|
|
|
|
|
|
authenticated = request.get(KEY_AUTHENTICATED, False)
|
|
|
|
|
2018-07-20 10:09:48 +00:00
|
|
|
if view.requires_auth:
|
|
|
|
if authenticated:
|
2019-04-03 04:23:59 +00:00
|
|
|
if 'deprecate_warning_message' in request:
|
|
|
|
_LOGGER.warning(request['deprecate_warning_message'])
|
2018-07-20 10:09:48 +00:00
|
|
|
await process_success_login(request)
|
|
|
|
else:
|
|
|
|
raise HTTPUnauthorized()
|
2018-03-09 01:51:49 +00:00
|
|
|
|
2019-03-26 23:18:32 +00:00
|
|
|
_LOGGER.debug('Serving %s to %s (auth: %s)',
|
|
|
|
request.path, request.get(KEY_REAL_IP), authenticated)
|
2018-03-09 01:51:49 +00:00
|
|
|
|
2018-11-25 17:04:48 +00:00
|
|
|
try:
|
|
|
|
result = handler(request, **request.match_info)
|
2018-03-09 01:51:49 +00:00
|
|
|
|
2018-11-25 17:04:48 +00:00
|
|
|
if asyncio.iscoroutine(result):
|
|
|
|
result = await result
|
2018-11-30 20:28:35 +00:00
|
|
|
except vol.Invalid:
|
|
|
|
raise HTTPBadRequest()
|
|
|
|
except exceptions.ServiceNotFound:
|
|
|
|
raise HTTPInternalServerError()
|
2018-11-25 17:04:48 +00:00
|
|
|
except exceptions.Unauthorized:
|
|
|
|
raise HTTPUnauthorized()
|
2018-03-09 01:51:49 +00:00
|
|
|
|
|
|
|
if isinstance(result, web.StreamResponse):
|
|
|
|
# The method handler returned a ready-made Response, how nice of it
|
|
|
|
return result
|
|
|
|
|
|
|
|
status_code = 200
|
|
|
|
|
|
|
|
if isinstance(result, tuple):
|
|
|
|
result, status_code = result
|
|
|
|
|
|
|
|
if isinstance(result, str):
|
|
|
|
result = result.encode('utf-8')
|
|
|
|
elif result is None:
|
|
|
|
result = b''
|
|
|
|
elif not isinstance(result, bytes):
|
|
|
|
assert False, ('Result should be None, string, bytes or Response. '
|
|
|
|
'Got: {}').format(result)
|
|
|
|
|
|
|
|
return web.Response(body=result, status=status_code)
|
|
|
|
|
|
|
|
return handle
|