core/homeassistant/components/http/data_validator.py

64 lines
2.0 KiB
Python
Raw Normal View History

"""Decorator for view methods to help with data validation."""
from functools import wraps
import logging
from typing import Any, Awaitable, Callable
from aiohttp import web
import voluptuous as vol
2020-04-09 19:43:42 +00:00
from homeassistant.const import HTTP_BAD_REQUEST
from .view import HomeAssistantView
_LOGGER = logging.getLogger(__name__)
class RequestDataValidator:
"""Decorator that will validate the incoming data.
Takes in a voluptuous schema and adds 'data' as
keyword argument to the function call.
Will return a 400 if no JSON provided or doesn't match schema.
"""
def __init__(self, schema: vol.Schema, allow_empty: bool = False) -> None:
"""Initialize the decorator."""
2019-12-21 21:45:06 +00:00
if isinstance(schema, dict):
schema = vol.Schema(schema)
self._schema = schema
self._allow_empty = allow_empty
def __call__(
self, method: Callable[..., Awaitable[web.StreamResponse]]
) -> Callable:
"""Decorate a function."""
2019-07-31 19:25:30 +00:00
@wraps(method)
async def wrapper(
view: HomeAssistantView, request: web.Request, *args: Any, **kwargs: Any
) -> web.StreamResponse:
"""Wrap a request handler with data validation."""
data = None
try:
data = await request.json()
except ValueError:
2019-07-31 19:25:30 +00:00
if not self._allow_empty or (await request.content.read()) != b"":
_LOGGER.error("Invalid JSON received")
2020-04-09 19:43:42 +00:00
return view.json_message("Invalid JSON.", HTTP_BAD_REQUEST)
data = {}
try:
2019-07-31 19:25:30 +00:00
kwargs["data"] = self._schema(data)
except vol.Invalid as err:
2019-07-31 19:25:30 +00:00
_LOGGER.error("Data does not match schema: %s", err)
2020-04-09 19:43:42 +00:00
return view.json_message(
f"Message format incorrect: {err}", HTTP_BAD_REQUEST
)
result = await method(view, request, *args, **kwargs)
return result
return wrapper