Skip to content

Commit

Permalink
Added default values handling for json schema validators
Browse files Browse the repository at this point in the history
  • Loading branch information
trezorg committed Mar 3, 2018
1 parent beb3cef commit 7d72e74
Show file tree
Hide file tree
Showing 4 changed files with 319 additions and 30 deletions.
85 changes: 55 additions & 30 deletions aiohttp_swagger/helpers/validation.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,13 +21,13 @@
Response,
json_response,
)
from collections import defaultdict
from collections import defaultdict, MutableMapping
from jsonschema import (
validate,
ValidationError,
FormatChecker,
Draft4Validator,
validators,
)
from jsonschema.validators import validator_for


__all__ = (
Expand All @@ -54,12 +54,35 @@ def multi_dict_to_dict(mld: Mapping) -> Mapping:
}


def validate_schema(obj: Mapping, schema: Mapping):
validate(obj, schema, format_checker=FormatChecker())
def extend_with_default(validator_class):

validate_properties = validator_class.VALIDATORS["properties"]

def validate_multi_dict(obj, schema):
validate(multi_dict_to_dict(obj), schema, format_checker=FormatChecker())
def set_defaults(validator, properties, instance, schema):
if isinstance(instance, MutableMapping):
for prop, sub_schema in properties.items():
if "default" in sub_schema:
instance.setdefault(prop, sub_schema["default"])
for error in validate_properties(
validator, properties, instance, schema):
yield error

return validators.extend(validator_class, {"properties": set_defaults})


json_schema_validator = extend_with_default(Draft4Validator)


def validate_schema(obj: Mapping, schema: Mapping) -> Mapping:
json_schema_validator(schema, format_checker=FormatChecker()).validate(obj)
return obj


def validate_multi_dict(obj, schema) -> Mapping:
_obj = multi_dict_to_dict(obj)
json_schema_validator(
schema, format_checker=FormatChecker()).validate(_obj)
return _obj


def validate_content_type(swagger: Mapping, content_type: str):
Expand All @@ -73,34 +96,34 @@ async def validate_request(
request: Request,
parameter_groups: Mapping,
swagger: Mapping):
res = {}
validate_content_type(swagger, request.content_type)
for group_name, group_schemas in parameter_groups.items():
for group_name, group_schema in parameter_groups.items():
if group_name == 'header':
headers = request.headers
for schema in group_schemas:
validate_multi_dict(headers, schema)
res['headers'] = validate_multi_dict(request.headers, group_schema)
if group_name == 'query':
query = request.query
for schema in group_schemas:
validate_multi_dict(query, schema)
res['query'] = validate_multi_dict(request.query, group_schema)
if group_name == 'formData':
try:
data = await request.post()
except ValueError:
data = None
for schema in group_schemas:
validate_multi_dict(data, schema)
res['formData'] = validate_multi_dict(data, group_schema)
if group_name == 'body':
try:
content = await request.json()
except json.JSONDecodeError:
content = None
for schema in group_schemas:
validate_schema(content, schema)
if request.content_type == 'application/json':
try:
content = await request.json()
except json.JSONDecodeError:
content = None
elif request.content_type.startswith('text'):
content = await request.text()
else:
content = await request.read()
res['body'] = validate_schema(content, group_schema)
if group_name == 'path':
params = dict(request.match_info)
for schema in group_schemas:
validate_schema(params, schema)
res['path'] = validate_schema(params, group_schema)
return res


def adjust_swagger_item_to_json_schemes(*schemes: Mapping) -> Mapping:
Expand All @@ -124,7 +147,7 @@ def adjust_swagger_item_to_json_schemes(*schemes: Mapping) -> Mapping:
required_fields.append(name)
if required_fields:
new_schema['required'] = required_fields
validator_for(new_schema).check_schema(new_schema)
validators.validator_for(new_schema).check_schema(new_schema)
return new_schema


Expand All @@ -139,21 +162,21 @@ def adjust_swagger_body_item_to_json_schema(schema: Mapping) -> Mapping:
new_schema,
]
}
validator_for(new_schema).check_schema(new_schema)
validators.validator_for(new_schema).check_schema(new_schema)
return new_schema


def adjust_swagger_to_json_schema(parameter_groups: Iterable) -> Mapping:
res = defaultdict(list)
res = {}
for group_name, group_schemas in parameter_groups:
if group_name in ('query', 'header', 'path', 'formData'):
json_schema = adjust_swagger_item_to_json_schemes(*group_schemas)
res[group_name].append(json_schema)
res[group_name] = json_schema
else:
# only one possible schema for in: body
schema = list(group_schemas)[0]
json_schema = adjust_swagger_body_item_to_json_schema(schema)
res[group_name].append(json_schema)
res[group_name] = json_schema
return res


Expand Down Expand Up @@ -216,7 +239,9 @@ async def _wrapper(*args, **kwargs) -> Response:
request = args[0].request \
if isinstance(args[0], web.View) else args[0]
try:
await validate_request(request, parameter_groups, schema)
validation = \
await validate_request(request, parameter_groups, schema)
request.validation = validation
except ValidationError as exc:
logger.exception(exc)
exc_dict = validation_exc_to_dict(exc)
Expand Down
11 changes: 11 additions & 0 deletions doc/source/customizing.rst
Original file line number Diff line number Diff line change
Expand Up @@ -212,6 +212,17 @@ Global Swagger YAML
:samp:`aiohttp-swagger` also allow to validate swagger schema against json schema:
Validated object would be added as **request.validation**. Default values also will be filled into object.

.. code-block:: javascript
{
'query': {}, // validated request.query
'path': {}, // validated request.path
'body': {}, // validated request.json()
'formData': {}, // validated post request.data()
'headers': {}, // validated post request.headers
}
.. code-block:: python
Expand Down
170 changes: 170 additions & 0 deletions tests/test_validation_body.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,170 @@
import asyncio
import json

import pytest
from aiohttp import web
from aiohttp_swagger import *


@asyncio.coroutine
@swagger_validation
def post1(request, *args, **kwargs):
"""
---
description: Post resources
tags:
- Function View
produces:
- application/json
consumes:
- application/json
parameters:
- in: body
name: body
required: true
schema:
type: object
properties:
test:
type: string
default: default
minLength: 2
test1:
type: string
default: default1
minLength: 2
responses:
"200":
description: successful operation.
"405":
description: invalid HTTP Method
"""
return web.json_response(data=request.validation['body'])


@asyncio.coroutine
@swagger_validation
def post2(request, *args, **kwargs):
"""
---
description: Post resources
tags:
- Function View
produces:
- text/plain
consumes:
- text/plain
parameters:
- in: body
name: body
required: true
schema:
type: string
default: default
minLength: 2
responses:
"200":
description: successful operation.
"405":
description: invalid HTTP Method
"""
return web.Response(text=request.validation['body'])


POST1_METHOD_PARAMETERS = [
# success
(
'post',
'/example12',
{'test': 'default'},
{'Content-Type': 'application/json'},
200
),
# success
(
'post',
'/example12',
{},
{'Content-Type': 'application/json'},
200
),
# error
(
'post',
'/example12',
None,
{'Content-Type': 'application/json'},
400
),
]

POST2_METHOD_PARAMETERS = [
# success
(
'post',
'/example12',
'1234',
{'Content-Type': 'text/plain'},
200
),
(
'post',
'/example12',
None,
{'Content-Type': 'text/plain'},
400
),
]


@pytest.mark.parametrize("method,url,body,headers,response",
POST1_METHOD_PARAMETERS)
@asyncio.coroutine
def test_function_post1_method_body_validation(
test_client, loop, swagger_file, method, url, body, headers, response):
app = web.Application(loop=loop)
app.router.add_post("/example12", post1)
setup_swagger(
app,
swagger_merge_with_file=True,
swagger_validate_schema=True,
swagger_from_file=swagger_file,
)
client = yield from test_client(app)
data = json.dumps(body) \
if headers['Content-Type'] == 'application/json' else body
resp = yield from getattr(client, method)(url, data=data, headers=headers)
text = yield from resp.json()
assert resp.status == response, text
if response != 200:
assert 'error' in text
else:
assert 'error' not in text
assert 'test' in text
assert text['test'] == 'default'
assert text['test1'] == 'default1'


@pytest.mark.parametrize("method,url,body,headers,response",
POST2_METHOD_PARAMETERS)
@asyncio.coroutine
def test_function_post2_method_body_validation(
test_client, loop, swagger_file, method, url, body, headers, response):
app = web.Application(loop=loop)
app.router.add_post("/example12", post2)
setup_swagger(
app,
swagger_merge_with_file=True,
swagger_validate_schema=True,
swagger_from_file=swagger_file,
)
client = yield from test_client(app)
data = json.dumps(body) \
if headers['Content-Type'] == 'application/json' else body
resp = yield from getattr(client, method)(url, data=data, headers=headers)
text = yield from resp.text()
assert resp.status == response, text
if response != 200:
assert 'error' in text
else:
assert isinstance(text, str)
Loading

0 comments on commit 7d72e74

Please sign in to comment.