Compare commits
	
		
			23 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 25fcac18ec | ||
|  | f2b16a46b5 | ||
|  | c4c18ee4a1 | ||
|  | 93ec0f6c80 | ||
|  | a6d96d711b | ||
|  | 8aee135f95 | ||
|  | 462d8d8b98 | ||
|  | 0d3a33c964 | ||
|  | 22979b7e59 | ||
|  | b9519bb868 | ||
|  | 913f50298c | ||
|  | 03854cf939 | ||
|  | 2db23d3328 | ||
|  | d866ce5358 | ||
|  | 13c19105d8 | ||
|  | e4b23398b8 | ||
|  | 57b50725ea | ||
|  | cda4fba4c2 | ||
|  | 236374240e | ||
|  | 635f38e33a | ||
|  | 62d871fb5c | ||
|  | 77954cdd69 | ||
|  | d6b5fc26f3 | 
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,9 @@ | ||||
| .coverage | ||||
| .idea/ | ||||
| .pytest_cache | ||||
| __pycache__ | ||||
| aiohttp_pydantic.egg-info/ | ||||
| build/ | ||||
| coverage.xml | ||||
| dist/ | ||||
|  | ||||
|   | ||||
| @@ -2,9 +2,14 @@ language: python | ||||
| python: | ||||
| - '3.8' | ||||
| script: | ||||
| - pytest tests/ | ||||
| - pytest --cov-report=xml --cov=aiohttp_pydantic tests/ | ||||
| install: | ||||
| - pip install -r test_requirements.txt | ||||
| - pip install -U setuptools wheel pip | ||||
| - pip install -r requirements/test.txt | ||||
| - pip install -r requirements/ci.txt | ||||
| - pip install . | ||||
| after_success: | ||||
|   - codecov | ||||
| deploy: | ||||
|   provider: pypi | ||||
|   username: __token__ | ||||
|   | ||||
							
								
								
									
										139
									
								
								README.rst
									
									
									
									
									
								
							
							
						
						
									
										139
									
								
								README.rst
									
									
									
									
									
								
							| @@ -1,6 +1,28 @@ | ||||
| Aiohttp pydantic - Aiohttp View to validate and parse request | ||||
| ============================================================= | ||||
|  | ||||
| .. image:: https://travis-ci.org/Maillol/aiohttp-pydantic.svg?branch=main | ||||
|   :target: https://travis-ci.org/Maillol/aiohttp-pydantic | ||||
|  | ||||
| .. image:: https://img.shields.io/pypi/v/aiohttp-pydantic | ||||
|   :target: https://img.shields.io/pypi/v/aiohttp-pydantic | ||||
|   :alt: Latest PyPI package version | ||||
|  | ||||
| .. image:: https://codecov.io/gh/Maillol/aiohttp-pydantic/branch/main/graph/badge.svg | ||||
|   :target: https://codecov.io/gh/Maillol/aiohttp-pydantic | ||||
|   :alt: codecov.io status for master branch | ||||
|  | ||||
| Aiohttp pydantic is an `aiohttp view`_ to easily parse and validate request. | ||||
| You define using the function annotations what your methods for handling HTTP verbs expects and Aiohttp pydantic parses the HTTP request | ||||
| for you, validates the data, and injects that you want as parameters. | ||||
|  | ||||
|  | ||||
| Features: | ||||
|  | ||||
| - Query string, request body, URL path and HTTP headers validation. | ||||
| - Open API Specification generation. | ||||
|  | ||||
|  | ||||
| How to install | ||||
| -------------- | ||||
|  | ||||
| @@ -46,6 +68,7 @@ Example: | ||||
|     $ curl -X GET http://127.0.0.1:8080/article?with_comments=a | ||||
|     [ | ||||
|       { | ||||
|         "in": "query string", | ||||
|         "loc": [ | ||||
|           "with_comments" | ||||
|         ], | ||||
| @@ -60,6 +83,7 @@ Example: | ||||
|     $ curl -H "Content-Type: application/json" -X post http://127.0.0.1:8080/article --data '{}' | ||||
|     [ | ||||
|       { | ||||
|         "in": "body", | ||||
|         "loc": [ | ||||
|           "name" | ||||
|         ], | ||||
| @@ -94,7 +118,7 @@ Example: | ||||
| Inject Query String Parameters | ||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
| To declare a query parameters, you must declare your argument as simple argument: | ||||
| To declare a query parameters, you must declare your argument as a simple argument: | ||||
|  | ||||
|  | ||||
| .. code-block:: python3 | ||||
| @@ -109,7 +133,7 @@ To declare a query parameters, you must declare your argument as simple argument | ||||
| Inject Request Body | ||||
| ~~~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
| To declare a body parameters, you must declare your argument as a simple argument annotated with `pydantic Model`_. | ||||
| To declare a body parameter, you must declare your argument as a simple argument annotated with `pydantic Model`_. | ||||
|  | ||||
|  | ||||
| .. code-block:: python3 | ||||
| @@ -144,3 +168,114 @@ To declare a HTTP headers parameters, you must declare your argument as a `keywo | ||||
| .. _positional-only parameters: https://www.python.org/dev/peps/pep-0570/ | ||||
| .. _pydantic Model: https://pydantic-docs.helpmanual.io/usage/models/ | ||||
| .. _keyword-only argument: https://www.python.org/dev/peps/pep-3102/ | ||||
|  | ||||
| Add route to generate Open Api Specification (OAS) | ||||
| -------------------------------------------------- | ||||
|  | ||||
| aiohttp_pydantic provides a sub-application to serve a route to generate Open Api Specification | ||||
| reading annotation in your PydanticView. Use *aiohttp_pydantic.oas.setup()* to add the sub-application | ||||
|  | ||||
| .. code-block:: python3 | ||||
|  | ||||
|     from aiohttp import web | ||||
|     from aiohttp_pydantic import oas | ||||
|  | ||||
|  | ||||
|     app = web.Application() | ||||
|     oas.setup(app) | ||||
|  | ||||
| By default, the route to display the Open Api Specification is /oas but you can change it using | ||||
| *url_prefix* parameter | ||||
|  | ||||
|  | ||||
| .. code-block:: python3 | ||||
|  | ||||
|     oas.setup(app, url_prefix='/spec-api') | ||||
|  | ||||
| If you want generate the Open Api Specification from specific aiohttp sub-applications. | ||||
| on the same route, you must use *apps_to_expose* parameter. | ||||
|  | ||||
|  | ||||
| .. code-block:: python3 | ||||
|  | ||||
|     from aiohttp import web | ||||
|     from aiohttp_pydantic import oas | ||||
|  | ||||
|     app = web.Application() | ||||
|     sub_app_1 = web.Application() | ||||
|     sub_app_2 = web.Application() | ||||
|  | ||||
|     oas.setup(app, apps_to_expose=[sub_app_1, sub_app_2]) | ||||
|  | ||||
| Add annotation to define response content | ||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
| The module aiohttp_pydantic.oas.typing provides class to annotate a | ||||
| response content. | ||||
|  | ||||
| For example *r200[List[Pet]]* means the server responses with | ||||
| the status code 200 and the response content is a List of Pet where Pet will be | ||||
| defined using a pydantic.BaseModel | ||||
|  | ||||
|  | ||||
| .. code-block:: python3 | ||||
|  | ||||
|     from aiohttp_pydantic import PydanticView | ||||
|     from aiohttp_pydantic.oas.typing import r200, r201, r204, r404 | ||||
|  | ||||
|  | ||||
|     class Pet(BaseModel): | ||||
|         id: int | ||||
|         name: str | ||||
|  | ||||
|  | ||||
|     class Error(BaseModel): | ||||
|         error: str | ||||
|  | ||||
|  | ||||
|     class PetCollectionView(PydanticView): | ||||
|         async def get(self) -> r200[List[Pet]]: | ||||
|             pets = self.request.app["model"].list_pets() | ||||
|             return web.json_response([pet.dict() for pet in pets]) | ||||
|  | ||||
|         async def post(self, pet: Pet) -> r201[Pet]: | ||||
|             self.request.app["model"].add_pet(pet) | ||||
|             return web.json_response(pet.dict()) | ||||
|  | ||||
|  | ||||
|     class PetItemView(PydanticView): | ||||
|         async def get(self, id: int, /) -> Union[r200[Pet], r404[Error]]: | ||||
|             pet = self.request.app["model"].find_pet(id) | ||||
|             return web.json_response(pet.dict()) | ||||
|  | ||||
|         async def put(self, id: int, /, pet: Pet) -> r200[Pet]: | ||||
|             self.request.app["model"].update_pet(id, pet) | ||||
|             return web.json_response(pet.dict()) | ||||
|  | ||||
|         async def delete(self, id: int, /) -> r204: | ||||
|             self.request.app["model"].remove_pet(id) | ||||
|             return web.Response(status=204) | ||||
|  | ||||
| Demo | ||||
| ---- | ||||
|  | ||||
| Have a look at `demo`_ for a complete example | ||||
|  | ||||
| .. code-block:: bash | ||||
|  | ||||
|     git clone https://github.com/Maillol/aiohttp-pydantic.git | ||||
|     cd aiohttp-pydantic | ||||
|     pip install . | ||||
|     python -m demo | ||||
|  | ||||
| Go to http://127.0.0.1:8080/oas | ||||
|  | ||||
| You can generate the OAS in a json file using the command: | ||||
|  | ||||
| .. code-block:: bash | ||||
|  | ||||
|     python -m aiohttp_pydantic.oas demo.main | ||||
|  | ||||
|  | ||||
| .. _demo: https://github.com/Maillol/aiohttp-pydantic/tree/main/demo | ||||
| .. _aiohttp view: https://docs.aiohttp.org/en/stable/web_quickstart.html#class-based-views | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| from .view import PydanticView | ||||
|  | ||||
| __all__ = ("PydanticView",) | ||||
| __version__ = "1.6.1" | ||||
|  | ||||
| __all__ = ("PydanticView", "__version__") | ||||
|   | ||||
| @@ -1,11 +1,13 @@ | ||||
| import abc | ||||
| from inspect import signature | ||||
| from json.decoder import JSONDecodeError | ||||
| from typing import Callable, Tuple | ||||
|  | ||||
| from aiohttp.web_exceptions import HTTPBadRequest | ||||
| from aiohttp.web_request import BaseRequest | ||||
| from pydantic import BaseModel | ||||
| from inspect import signature | ||||
|  | ||||
|  | ||||
| import abc | ||||
| from .utils import is_pydantic_base_model | ||||
|  | ||||
|  | ||||
| class AbstractInjector(metaclass=abc.ABCMeta): | ||||
| @@ -13,8 +15,16 @@ class AbstractInjector(metaclass=abc.ABCMeta): | ||||
|     An injector parse HTTP request and inject params to the view. | ||||
|     """ | ||||
|  | ||||
|     @property | ||||
|     @abc.abstractmethod | ||||
|     def __init__(self, args_spec: dict): | ||||
|     def context(self) -> str: | ||||
|         """ | ||||
|         The name of part of parsed request | ||||
|         i.e "HTTP header", "URL path", ... | ||||
|         """ | ||||
|  | ||||
|     @abc.abstractmethod | ||||
|     def __init__(self, args_spec: dict, default_values: dict): | ||||
|         """ | ||||
|         args_spec - ordered mapping: arg_name -> type | ||||
|         """ | ||||
| @@ -31,8 +41,12 @@ class MatchInfoGetter(AbstractInjector): | ||||
|     Validates and injects the part of URL path inside the view positional args. | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, args_spec: dict): | ||||
|         self.model = type("PathModel", (BaseModel,), {"__annotations__": args_spec}) | ||||
|     context = "path" | ||||
|  | ||||
|     def __init__(self, args_spec: dict, default_values: dict): | ||||
|         attrs = {"__annotations__": args_spec} | ||||
|         attrs.update(default_values) | ||||
|         self.model = type("PathModel", (BaseModel,), attrs) | ||||
|  | ||||
|     def inject(self, request: BaseRequest, args_view: list, kwargs_view: dict): | ||||
|         args_view.extend(self.model(**request.match_info).dict().values()) | ||||
| @@ -43,11 +57,19 @@ class BodyGetter(AbstractInjector): | ||||
|     Validates and injects the content of request body inside the view kwargs. | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, args_spec: dict): | ||||
|     context = "body" | ||||
|  | ||||
|     def __init__(self, args_spec: dict, default_values: dict): | ||||
|         self.arg_name, self.model = next(iter(args_spec.items())) | ||||
|  | ||||
|     async def inject(self, request: BaseRequest, args_view: list, kwargs_view: dict): | ||||
|         try: | ||||
|             body = await request.json() | ||||
|         except JSONDecodeError: | ||||
|             raise HTTPBadRequest( | ||||
|                 text='{"error": "Malformed JSON"}', content_type="application/json" | ||||
|             ) from None | ||||
|  | ||||
|         kwargs_view[self.arg_name] = self.model(**body) | ||||
|  | ||||
|  | ||||
| @@ -56,8 +78,12 @@ class QueryGetter(AbstractInjector): | ||||
|     Validates and injects the query string inside the view kwargs. | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, args_spec: dict): | ||||
|         self.model = type("QueryModel", (BaseModel,), {"__annotations__": args_spec}) | ||||
|     context = "query string" | ||||
|  | ||||
|     def __init__(self, args_spec: dict, default_values: dict): | ||||
|         attrs = {"__annotations__": args_spec} | ||||
|         attrs.update(default_values) | ||||
|         self.model = type("QueryModel", (BaseModel,), attrs) | ||||
|  | ||||
|     def inject(self, request: BaseRequest, args_view: list, kwargs_view: dict): | ||||
|         kwargs_view.update(self.model(**request.query).dict()) | ||||
| @@ -68,36 +94,49 @@ class HeadersGetter(AbstractInjector): | ||||
|     Validates and injects the HTTP headers inside the view kwargs. | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, args_spec: dict): | ||||
|         self.model = type("HeaderModel", (BaseModel,), {"__annotations__": args_spec}) | ||||
|     context = "headers" | ||||
|  | ||||
|     def __init__(self, args_spec: dict, default_values: dict): | ||||
|         attrs = {"__annotations__": args_spec} | ||||
|         attrs.update(default_values) | ||||
|         self.model = type("HeaderModel", (BaseModel,), attrs) | ||||
|  | ||||
|     def inject(self, request: BaseRequest, args_view: list, kwargs_view: dict): | ||||
|         header = {k.lower().replace("-", "_"): v for k, v in request.headers.items()} | ||||
|         kwargs_view.update(self.model(**header).dict()) | ||||
|  | ||||
|  | ||||
| def _parse_func_signature(func: Callable) -> Tuple[dict, dict, dict, dict]: | ||||
| def _parse_func_signature(func: Callable) -> Tuple[dict, dict, dict, dict, dict]: | ||||
|     """ | ||||
|     Analyse function signature and returns 4-tuple: | ||||
|         0 - arguments will be set from the url path | ||||
|         1 - argument will be set from the request body. | ||||
|         2 - argument will be set from the query string. | ||||
|         3 - argument will be set from the HTTP headers. | ||||
|         4 - Default value for each parameters | ||||
|     """ | ||||
|  | ||||
|     path_args = {} | ||||
|     body_args = {} | ||||
|     qs_args = {} | ||||
|     header_args = {} | ||||
|     defaults = {} | ||||
|  | ||||
|     for param_name, param_spec in signature(func).parameters.items(): | ||||
|         if param_name == "self": | ||||
|             continue | ||||
|  | ||||
|         if param_spec.annotation == param_spec.empty: | ||||
|             raise RuntimeError(f"The parameter {param_name} must have an annotation") | ||||
|  | ||||
|         if param_spec.default is not param_spec.empty: | ||||
|             defaults[param_name] = param_spec.default | ||||
|  | ||||
|         if param_spec.kind is param_spec.POSITIONAL_ONLY: | ||||
|             path_args[param_name] = param_spec.annotation | ||||
|  | ||||
|         elif param_spec.kind is param_spec.POSITIONAL_OR_KEYWORD: | ||||
|             if issubclass(param_spec.annotation, BaseModel): | ||||
|             if is_pydantic_base_model(param_spec.annotation): | ||||
|                 body_args[param_name] = param_spec.annotation | ||||
|             else: | ||||
|                 qs_args[param_name] = param_spec.annotation | ||||
| @@ -106,4 +145,4 @@ def _parse_func_signature(func: Callable) -> Tuple[dict, dict, dict, dict]: | ||||
|         else: | ||||
|             raise RuntimeError(f"You cannot use {param_spec.VAR_POSITIONAL} parameters") | ||||
|  | ||||
|     return path_args, body_args, qs_args, header_args | ||||
|     return path_args, body_args, qs_args, header_args, defaults | ||||
|   | ||||
							
								
								
									
										27
									
								
								aiohttp_pydantic/oas/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								aiohttp_pydantic/oas/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| from importlib import resources | ||||
| from typing import Iterable | ||||
|  | ||||
| import jinja2 | ||||
| from aiohttp import web | ||||
| from swagger_ui_bundle import swagger_ui_path | ||||
|  | ||||
| from .view import get_oas, oas_ui | ||||
|  | ||||
|  | ||||
| def setup( | ||||
|     app: web.Application, | ||||
|     apps_to_expose: Iterable[web.Application] = (), | ||||
|     url_prefix: str = "/oas", | ||||
|     enable: bool = True, | ||||
| ): | ||||
|     if enable: | ||||
|         oas_app = web.Application() | ||||
|         oas_app["apps to expose"] = tuple(apps_to_expose) or (app,) | ||||
|         oas_app["index template"] = jinja2.Template( | ||||
|             resources.read_text("aiohttp_pydantic.oas", "index.j2") | ||||
|         ) | ||||
|         oas_app.router.add_get("/spec", get_oas, name="spec") | ||||
|         oas_app.router.add_static("/static", swagger_ui_path, name="static") | ||||
|         oas_app.router.add_get("", oas_ui, name="index") | ||||
|  | ||||
|         app.add_subapp(url_prefix, oas_app) | ||||
							
								
								
									
										8
									
								
								aiohttp_pydantic/oas/__main__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								aiohttp_pydantic/oas/__main__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| import argparse | ||||
|  | ||||
| from .cmd import setup | ||||
|  | ||||
| parser = argparse.ArgumentParser(description="Generate Open API Specification") | ||||
| setup(parser) | ||||
| args = parser.parse_args() | ||||
| args.func(args) | ||||
							
								
								
									
										45
									
								
								aiohttp_pydantic/oas/cmd.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								aiohttp_pydantic/oas/cmd.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| import argparse | ||||
| import importlib | ||||
| import json | ||||
|  | ||||
| from .view import generate_oas | ||||
|  | ||||
|  | ||||
| def application_type(value): | ||||
|     """ | ||||
|     Return aiohttp application defined in the value. | ||||
|     """ | ||||
|     try: | ||||
|         module_name, app_name = value.split(":") | ||||
|     except ValueError: | ||||
|         module_name, app_name = value, "app" | ||||
|  | ||||
|     module = importlib.import_module(module_name) | ||||
|     try: | ||||
|         if app_name.endswith("()"): | ||||
|             app_name = app_name.strip("()") | ||||
|             factory_app = getattr(module, app_name) | ||||
|             return factory_app() | ||||
|         return getattr(module, app_name) | ||||
|  | ||||
|     except AttributeError as error: | ||||
|         raise argparse.ArgumentTypeError(error) from error | ||||
|  | ||||
|  | ||||
| def setup(parser: argparse.ArgumentParser): | ||||
|     parser.add_argument( | ||||
|         "apps", | ||||
|         metavar="APP", | ||||
|         type=application_type, | ||||
|         nargs="*", | ||||
|         help="The name of the module containing the asyncio.web.Application." | ||||
|         " By default the variable named 'app' is loaded but you can define" | ||||
|         " an other variable name ending the name of module with : characters" | ||||
|         " and the name of variable. Example: my_package.my_module:my_app", | ||||
|     ) | ||||
|  | ||||
|     parser.set_defaults(func=show_oas) | ||||
|  | ||||
|  | ||||
| def show_oas(args: argparse.Namespace): | ||||
|     print(json.dumps(generate_oas(args.apps), sort_keys=True, indent=4)) | ||||
							
								
								
									
										69
									
								
								aiohttp_pydantic/oas/index.j2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								aiohttp_pydantic/oas/index.j2
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| {# This updated file is part of swagger_ui_bundle (https://github.com/dtkav/swagger_ui_bundle) #} | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|   <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <title>{{ title | default('Swagger UI') }}</title> | ||||
|     <link rel="stylesheet" type="text/css" href="{{ static_url | trim('/') }}/swagger-ui.css" > | ||||
|     <link rel="icon" type="image/png" href="{{ static_url | trim('/') }}/favicon-32x32.png" sizes="32x32" /> | ||||
|     <link rel="icon" type="image/png" href="{{ static_url | trim('/') }}/favicon-16x16.png" sizes="16x16" /> | ||||
|     <style> | ||||
|       html | ||||
|       { | ||||
|         box-sizing: border-box; | ||||
|         overflow: -moz-scrollbars-vertical; | ||||
|         overflow-y: scroll; | ||||
|       } | ||||
|  | ||||
|       *, | ||||
|       *:before, | ||||
|       *:after | ||||
|       { | ||||
|         box-sizing: inherit; | ||||
|       } | ||||
|  | ||||
|       body | ||||
|       { | ||||
|         margin:0; | ||||
|         background: #fafafa; | ||||
|       } | ||||
|     </style> | ||||
|   </head> | ||||
|  | ||||
|   <body> | ||||
|     <div id="swagger-ui"></div> | ||||
|  | ||||
|     <script src="{{ static_url | trim('/') }}/swagger-ui-bundle.js"> </script> | ||||
|     <script src="{{ static_url | trim('/') }}/swagger-ui-standalone-preset.js"> </script> | ||||
|     <script> | ||||
|     window.onload = function() { | ||||
|       // Begin Swagger UI call region | ||||
|       const ui = SwaggerUIBundle({ | ||||
|         url: "{{ openapi_spec_url }}", | ||||
|         validatorUrl: {{ validatorUrl | default('null') }}, | ||||
|         {% if configUrl is defined %} | ||||
|         configUrl: "{{ configUrl }}", | ||||
|         {% endif %} | ||||
|         dom_id: '#swagger-ui', | ||||
|         deepLinking: true, | ||||
|         presets: [ | ||||
|           SwaggerUIBundle.presets.apis, | ||||
|           SwaggerUIStandalonePreset | ||||
|         ], | ||||
|         plugins: [ | ||||
|           SwaggerUIBundle.plugins.DownloadUrl | ||||
|         ], | ||||
|         layout: "StandaloneLayout" | ||||
|       }) | ||||
|       {% if initOAuth is defined %} | ||||
|       ui.initOAuth( | ||||
|         {{ initOAuth|tojson|safe }} | ||||
|       ) | ||||
|       {% endif %} | ||||
|       // End Swagger UI call region | ||||
|  | ||||
|       window.ui = ui | ||||
|     } | ||||
|   </script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										327
									
								
								aiohttp_pydantic/oas/struct.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										327
									
								
								aiohttp_pydantic/oas/struct.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,327 @@ | ||||
| """ | ||||
| Utility to write Open Api Specifications using the Python language. | ||||
| """ | ||||
|  | ||||
| from typing import Union | ||||
|  | ||||
|  | ||||
| class Info: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec.setdefault("info", {}) | ||||
|  | ||||
|     @property | ||||
|     def title(self): | ||||
|         return self._spec.get("title") | ||||
|  | ||||
|     @title.setter | ||||
|     def title(self, title): | ||||
|         self._spec["title"] = title | ||||
|  | ||||
|     @property | ||||
|     def description(self): | ||||
|         return self._spec.get("description") | ||||
|  | ||||
|     @description.setter | ||||
|     def description(self, description): | ||||
|         self._spec["description"] = description | ||||
|  | ||||
|     @property | ||||
|     def version(self): | ||||
|         return self._spec.get("version") | ||||
|  | ||||
|     @version.setter | ||||
|     def version(self, version): | ||||
|         self._spec["version"] = version | ||||
|  | ||||
|     @property | ||||
|     def terms_of_service(self): | ||||
|         return self._spec.get("termsOfService") | ||||
|  | ||||
|     @terms_of_service.setter | ||||
|     def terms_of_service(self, terms_of_service): | ||||
|         self._spec["termsOfService"] = terms_of_service | ||||
|  | ||||
|  | ||||
| class RequestBody: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec.setdefault("requestBody", {}) | ||||
|  | ||||
|     @property | ||||
|     def description(self): | ||||
|         return self._spec["description"] | ||||
|  | ||||
|     @description.setter | ||||
|     def description(self, description: str): | ||||
|         self._spec["description"] = description | ||||
|  | ||||
|     @property | ||||
|     def required(self) -> bool: | ||||
|         return self._spec.get("required", False) | ||||
|  | ||||
|     @required.setter | ||||
|     def required(self, required: bool): | ||||
|         self._spec["required"] = required | ||||
|  | ||||
|     @property | ||||
|     def content(self): | ||||
|         return self._spec["content"] | ||||
|  | ||||
|     @content.setter | ||||
|     def content(self, content: dict): | ||||
|         self._spec["content"] = content | ||||
|  | ||||
|  | ||||
| class Parameter: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec | ||||
|  | ||||
|     @property | ||||
|     def name(self) -> str: | ||||
|         return self._spec["name"] | ||||
|  | ||||
|     @name.setter | ||||
|     def name(self, name: str): | ||||
|         self._spec["name"] = name | ||||
|  | ||||
|     @property | ||||
|     def in_(self) -> str: | ||||
|         return self._spec["in"] | ||||
|  | ||||
|     @in_.setter | ||||
|     def in_(self, in_: str): | ||||
|         self._spec["in"] = in_ | ||||
|  | ||||
|     @property | ||||
|     def description(self) -> str: | ||||
|         return self._spec["description"] | ||||
|  | ||||
|     @description.setter | ||||
|     def description(self, description: str): | ||||
|         self._spec["description"] = description | ||||
|  | ||||
|     @property | ||||
|     def required(self) -> bool: | ||||
|         return self._spec["required"] | ||||
|  | ||||
|     @required.setter | ||||
|     def required(self, required: bool): | ||||
|         self._spec["required"] = required | ||||
|  | ||||
|     @property | ||||
|     def schema(self) -> dict: | ||||
|         return self._spec["schema"] | ||||
|  | ||||
|     @schema.setter | ||||
|     def schema(self, schema: dict): | ||||
|         self._spec["schema"] = schema | ||||
|  | ||||
|  | ||||
| class Parameters: | ||||
|     def __init__(self, spec): | ||||
|         self._spec = spec | ||||
|         self._spec.setdefault("parameters", []) | ||||
|  | ||||
|     def __getitem__(self, item: int) -> Parameter: | ||||
|         if item == len(self._spec["parameters"]): | ||||
|             spec = {} | ||||
|             self._spec["parameters"].append(spec) | ||||
|         else: | ||||
|             spec = self._spec["parameters"][item] | ||||
|         return Parameter(spec) | ||||
|  | ||||
|  | ||||
| class Response: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec | ||||
|  | ||||
|     @property | ||||
|     def description(self) -> str: | ||||
|         return self._spec["description"] | ||||
|  | ||||
|     @description.setter | ||||
|     def description(self, description: str): | ||||
|         self._spec["description"] = description | ||||
|  | ||||
|     @property | ||||
|     def content(self): | ||||
|         return self._spec["content"] | ||||
|  | ||||
|     @content.setter | ||||
|     def content(self, content: dict): | ||||
|         self._spec["content"] = content | ||||
|  | ||||
|  | ||||
| class Responses: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec.setdefault("responses", {}) | ||||
|  | ||||
|     def __getitem__(self, status_code: Union[int, str]) -> Response: | ||||
|         if not (100 <= int(status_code) < 600): | ||||
|             raise ValueError("status_code must be between 100 and 599") | ||||
|  | ||||
|         spec = self._spec.setdefault(str(status_code), {}) | ||||
|         return Response(spec) | ||||
|  | ||||
|  | ||||
| class OperationObject: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec | ||||
|  | ||||
|     @property | ||||
|     def summary(self) -> str: | ||||
|         return self._spec["summary"] | ||||
|  | ||||
|     @summary.setter | ||||
|     def summary(self, summary: str): | ||||
|         self._spec["summary"] = summary | ||||
|  | ||||
|     @property | ||||
|     def description(self) -> str: | ||||
|         return self._spec["description"] | ||||
|  | ||||
|     @description.setter | ||||
|     def description(self, description: str): | ||||
|         self._spec["description"] = description | ||||
|  | ||||
|     @property | ||||
|     def request_body(self) -> RequestBody: | ||||
|         return RequestBody(self._spec) | ||||
|  | ||||
|     @property | ||||
|     def parameters(self) -> Parameters: | ||||
|         return Parameters(self._spec) | ||||
|  | ||||
|     @property | ||||
|     def responses(self) -> Responses: | ||||
|         return Responses(self._spec) | ||||
|  | ||||
|  | ||||
| class PathItem: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec | ||||
|  | ||||
|     @property | ||||
|     def get(self) -> OperationObject: | ||||
|         return OperationObject(self._spec.setdefault("get", {})) | ||||
|  | ||||
|     @property | ||||
|     def put(self) -> OperationObject: | ||||
|         return OperationObject(self._spec.setdefault("put", {})) | ||||
|  | ||||
|     @property | ||||
|     def post(self) -> OperationObject: | ||||
|         return OperationObject(self._spec.setdefault("post", {})) | ||||
|  | ||||
|     @property | ||||
|     def delete(self) -> OperationObject: | ||||
|         return OperationObject(self._spec.setdefault("delete", {})) | ||||
|  | ||||
|     @property | ||||
|     def options(self) -> OperationObject: | ||||
|         return OperationObject(self._spec.setdefault("options", {})) | ||||
|  | ||||
|     @property | ||||
|     def head(self) -> OperationObject: | ||||
|         return OperationObject(self._spec.setdefault("head", {})) | ||||
|  | ||||
|     @property | ||||
|     def patch(self) -> OperationObject: | ||||
|         return OperationObject(self._spec.setdefault("patch", {})) | ||||
|  | ||||
|     @property | ||||
|     def trace(self) -> OperationObject: | ||||
|         return OperationObject(self._spec.setdefault("trace", {})) | ||||
|  | ||||
|     @property | ||||
|     def description(self) -> str: | ||||
|         return self._spec["description"] | ||||
|  | ||||
|     @description.setter | ||||
|     def description(self, description: str): | ||||
|         self._spec["description"] = description | ||||
|  | ||||
|     @property | ||||
|     def summary(self) -> str: | ||||
|         return self._spec["summary"] | ||||
|  | ||||
|     @summary.setter | ||||
|     def summary(self, summary: str): | ||||
|         self._spec["summary"] = summary | ||||
|  | ||||
|  | ||||
| class Paths: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec.setdefault("paths", {}) | ||||
|  | ||||
|     def __getitem__(self, path: str) -> PathItem: | ||||
|         spec = self._spec.setdefault(path, {}) | ||||
|         return PathItem(spec) | ||||
|  | ||||
|  | ||||
| class Server: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec | ||||
|  | ||||
|     @property | ||||
|     def url(self) -> str: | ||||
|         return self._spec["url"] | ||||
|  | ||||
|     @url.setter | ||||
|     def url(self, url: str): | ||||
|         self._spec["url"] = url | ||||
|  | ||||
|     @property | ||||
|     def description(self) -> str: | ||||
|         return self._spec["description"] | ||||
|  | ||||
|     @description.setter | ||||
|     def description(self, description: str): | ||||
|         self._spec["description"] = description | ||||
|  | ||||
|  | ||||
| class Servers: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec | ||||
|         self._spec.setdefault("servers", []) | ||||
|  | ||||
|     def __getitem__(self, item: int) -> Server: | ||||
|         if item == len(self._spec["servers"]): | ||||
|             spec = {} | ||||
|             self._spec["servers"].append(spec) | ||||
|         else: | ||||
|             spec = self._spec["servers"][item] | ||||
|         return Server(spec) | ||||
|  | ||||
|  | ||||
| class Components: | ||||
|     def __init__(self, spec: dict): | ||||
|         self._spec = spec.setdefault("components", {}) | ||||
|  | ||||
|     @property | ||||
|     def schemas(self) -> dict: | ||||
|         return self._spec.setdefault("schemas", {}) | ||||
|  | ||||
|  | ||||
| class OpenApiSpec3: | ||||
|     def __init__(self): | ||||
|         self._spec = {"openapi": "3.0.0"} | ||||
|  | ||||
|     @property | ||||
|     def info(self) -> Info: | ||||
|         return Info(self._spec) | ||||
|  | ||||
|     @property | ||||
|     def servers(self) -> Servers: | ||||
|         return Servers(self._spec) | ||||
|  | ||||
|     @property | ||||
|     def paths(self) -> Paths: | ||||
|         return Paths(self._spec) | ||||
|  | ||||
|     @property | ||||
|     def components(self) -> Components: | ||||
|         return Components(self._spec) | ||||
|  | ||||
|     @property | ||||
|     def spec(self): | ||||
|         return self._spec | ||||
							
								
								
									
										48
									
								
								aiohttp_pydantic/oas/typing.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								aiohttp_pydantic/oas/typing.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| """ | ||||
| This module provides type to annotate the content of web.Response returned by | ||||
| the HTTP handlers. | ||||
|  | ||||
| The type are: r100, r101, ..., r599 | ||||
|  | ||||
| Example: | ||||
|  | ||||
|     class PetCollectionView(PydanticView): | ||||
|         async def get(self) -> Union[r200[List[Pet]], r404]: | ||||
|             ... | ||||
| """ | ||||
|  | ||||
| from functools import lru_cache | ||||
| from types import new_class | ||||
| from typing import Protocol, TypeVar | ||||
|  | ||||
| RespContents = TypeVar("RespContents", covariant=True) | ||||
|  | ||||
| _status_code = frozenset(f"r{code}" for code in range(100, 600)) | ||||
|  | ||||
|  | ||||
| @lru_cache(maxsize=len(_status_code)) | ||||
| def _make_status_code_type(status_code): | ||||
|     if status_code in _status_code: | ||||
|         return new_class(status_code, (Protocol[RespContents],)) | ||||
|     return None | ||||
|  | ||||
|  | ||||
| def is_status_code_type(obj) -> bool: | ||||
|     """ | ||||
|     Return True if obj is a status code type such as _200 or _404. | ||||
|     """ | ||||
|     name = getattr(obj, "__name__", None) | ||||
|     if name not in _status_code: | ||||
|         return False | ||||
|  | ||||
|     return obj is _make_status_code_type(name) | ||||
|  | ||||
|  | ||||
| def __getattr__(name): | ||||
|     if (status_code_type := _make_status_code_type(name)) is None: | ||||
|         raise AttributeError(f"module {__name__!r} has no attribute {name!r}") | ||||
|     return status_code_type | ||||
|  | ||||
|  | ||||
| __all__ = list(_status_code) | ||||
| __all__.append("is_status_code_type") | ||||
							
								
								
									
										183
									
								
								aiohttp_pydantic/oas/view.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										183
									
								
								aiohttp_pydantic/oas/view.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,183 @@ | ||||
| import typing | ||||
| from inspect import getdoc | ||||
| from itertools import count | ||||
| from typing import List, Type | ||||
|  | ||||
| from aiohttp.web import Response, json_response | ||||
| from aiohttp.web_app import Application | ||||
| from pydantic import BaseModel | ||||
|  | ||||
| from aiohttp_pydantic.oas.struct import OpenApiSpec3, OperationObject, PathItem | ||||
|  | ||||
| from ..injectors import _parse_func_signature | ||||
| from ..utils import is_pydantic_base_model | ||||
| from ..view import PydanticView, is_pydantic_view | ||||
| from .typing import is_status_code_type | ||||
|  | ||||
|  | ||||
| def _handle_optional(type_): | ||||
|     """ | ||||
|     Returns the type wrapped in Optional or None. | ||||
|  | ||||
|     >>>  _handle_optional(int) | ||||
|     >>>  _handle_optional(Optional[str]) | ||||
|     <class 'str'> | ||||
|     """ | ||||
|     if typing.get_origin(type_) is typing.Union: | ||||
|         args = typing.get_args(type_) | ||||
|         if len(args) == 2 and type(None) in args: | ||||
|             return next(iter(set(args) - {type(None)})) | ||||
|     return None | ||||
|  | ||||
|  | ||||
| class _OASResponseBuilder: | ||||
|     """ | ||||
|     Parse the type annotated as returned by a function and | ||||
|     generate the OAS operation response. | ||||
|     """ | ||||
|  | ||||
|     def __init__(self, oas: OpenApiSpec3, oas_operation): | ||||
|         self._oas_operation = oas_operation | ||||
|         self._oas = oas | ||||
|  | ||||
|     def _handle_pydantic_base_model(self, obj): | ||||
|         if is_pydantic_base_model(obj): | ||||
|             response_schema = obj.schema(ref_template="#/components/schemas/{model}") | ||||
|             if def_sub_schemas := response_schema.get("definitions", None): | ||||
|                 self._oas.components.schemas.update(def_sub_schemas) | ||||
|             return response_schema | ||||
|         return {} | ||||
|  | ||||
|     def _handle_list(self, obj): | ||||
|         if typing.get_origin(obj) is list: | ||||
|             return { | ||||
|                 "type": "array", | ||||
|                 "items": self._handle_pydantic_base_model(typing.get_args(obj)[0]), | ||||
|             } | ||||
|         return self._handle_pydantic_base_model(obj) | ||||
|  | ||||
|     def _handle_status_code_type(self, obj): | ||||
|         if is_status_code_type(typing.get_origin(obj)): | ||||
|             status_code = typing.get_origin(obj).__name__[1:] | ||||
|             self._oas_operation.responses[status_code].content = { | ||||
|                 "application/json": { | ||||
|                     "schema": self._handle_list(typing.get_args(obj)[0]) | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|         elif is_status_code_type(obj): | ||||
|             status_code = obj.__name__[1:] | ||||
|             self._oas_operation.responses[status_code].content = {} | ||||
|  | ||||
|     def _handle_union(self, obj): | ||||
|         if typing.get_origin(obj) is typing.Union: | ||||
|             for arg in typing.get_args(obj): | ||||
|                 self._handle_status_code_type(arg) | ||||
|         self._handle_status_code_type(obj) | ||||
|  | ||||
|     def build(self, obj): | ||||
|         self._handle_union(obj) | ||||
|  | ||||
|  | ||||
| def _add_http_method_to_oas( | ||||
|     oas: OpenApiSpec3, oas_path: PathItem, http_method: str, view: Type[PydanticView] | ||||
| ): | ||||
|     http_method = http_method.lower() | ||||
|     oas_operation: OperationObject = getattr(oas_path, http_method) | ||||
|     handler = getattr(view, http_method) | ||||
|     path_args, body_args, qs_args, header_args, defaults = _parse_func_signature( | ||||
|         handler | ||||
|     ) | ||||
|     description = getdoc(handler) | ||||
|     if description: | ||||
|         oas_operation.description = description | ||||
|  | ||||
|     if body_args: | ||||
|         body_schema = next(iter(body_args.values())).schema( | ||||
|             ref_template="#/components/schemas/{model}" | ||||
|         ) | ||||
|         if def_sub_schemas := body_schema.get("definitions", None): | ||||
|             oas.components.schemas.update(def_sub_schemas) | ||||
|  | ||||
|         oas_operation.request_body.content = { | ||||
|             "application/json": {"schema": body_schema} | ||||
|         } | ||||
|  | ||||
|     indexes = count() | ||||
|     for args_location, args in ( | ||||
|         ("path", path_args.items()), | ||||
|         ("query", qs_args.items()), | ||||
|         ("header", header_args.items()), | ||||
|     ): | ||||
|         for name, type_ in args: | ||||
|             i = next(indexes) | ||||
|             oas_operation.parameters[i].in_ = args_location | ||||
|             oas_operation.parameters[i].name = name | ||||
|             optional_type = _handle_optional(type_) | ||||
|  | ||||
|             attrs = {"__annotations__": {"__root__": type_}} | ||||
|             if name in defaults: | ||||
|                 attrs["__root__"] = defaults[name] | ||||
|  | ||||
|             oas_operation.parameters[i].schema = type(name, (BaseModel,), attrs).schema( | ||||
|                 ref_template="#/components/schemas/{model}" | ||||
|             ) | ||||
|  | ||||
|             oas_operation.parameters[i].required = optional_type is None | ||||
|  | ||||
|     return_type = handler.__annotations__.get("return") | ||||
|     if return_type is not None: | ||||
|         _OASResponseBuilder(oas, oas_operation).build(return_type) | ||||
|  | ||||
|  | ||||
| def generate_oas(apps: List[Application]) -> dict: | ||||
|     """ | ||||
|     Generate and return Open Api Specification from PydanticView in application. | ||||
|     """ | ||||
|     oas = OpenApiSpec3() | ||||
|     for app in apps: | ||||
|         for resources in app.router.resources(): | ||||
|             for resource_route in resources: | ||||
|                 if not is_pydantic_view(resource_route.handler): | ||||
|                     continue | ||||
|  | ||||
|                 view: Type[PydanticView] = resource_route.handler | ||||
|                 info = resource_route.get_info() | ||||
|                 path = oas.paths[info.get("path", info.get("formatter"))] | ||||
|                 if resource_route.method == "*": | ||||
|                     for method_name in view.allowed_methods: | ||||
|                         _add_http_method_to_oas(oas, path, method_name, view) | ||||
|                 else: | ||||
|                     _add_http_method_to_oas(oas, path, resource_route.method, view) | ||||
|  | ||||
|     return oas.spec | ||||
|  | ||||
|  | ||||
| async def get_oas(request): | ||||
|     """ | ||||
|     View to generate the Open Api Specification from PydanticView in application. | ||||
|     """ | ||||
|     apps = request.app["apps to expose"] | ||||
|     return json_response(generate_oas(apps)) | ||||
|  | ||||
|  | ||||
| async def oas_ui(request): | ||||
|     """ | ||||
|     View to serve the swagger-ui to read open api specification of application. | ||||
|     """ | ||||
|     template = request.app["index template"] | ||||
|  | ||||
|     static_url = request.app.router["static"].url_for(filename="") | ||||
|     spec_url = request.app.router["spec"].url_for() | ||||
|     host = request.url.origin() | ||||
|  | ||||
|     return Response( | ||||
|         text=template.render( | ||||
|             { | ||||
|                 "openapi_spec_url": host.with_path(str(spec_url)), | ||||
|                 "static_url": host.with_path(str(static_url)), | ||||
|             } | ||||
|         ), | ||||
|         content_type="text/html", | ||||
|         charset="utf-8", | ||||
|     ) | ||||
							
								
								
									
										11
									
								
								aiohttp_pydantic/utils.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								aiohttp_pydantic/utils.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| from pydantic import BaseModel | ||||
|  | ||||
|  | ||||
| def is_pydantic_base_model(obj): | ||||
|     """ | ||||
|     Return true is obj is a pydantic.BaseModel subclass. | ||||
|     """ | ||||
|     try: | ||||
|         return issubclass(obj, BaseModel) | ||||
|     except TypeError: | ||||
|         return False | ||||
| @@ -1,23 +1,16 @@ | ||||
| from functools import update_wrapper | ||||
| from inspect import iscoroutinefunction | ||||
| from typing import Any, Callable, Generator, Iterable | ||||
|  | ||||
| from aiohttp.abc import AbstractView | ||||
| from aiohttp.hdrs import METH_ALL | ||||
| from aiohttp.web import json_response | ||||
| from aiohttp.web_exceptions import HTTPMethodNotAllowed | ||||
| from aiohttp.web_response import StreamResponse | ||||
| from pydantic import ValidationError | ||||
| from typing import Generator, Any, Callable, List, Iterable | ||||
| from aiohttp.web import json_response | ||||
| from functools import update_wrapper | ||||
|  | ||||
|  | ||||
| from .injectors import ( | ||||
|     MatchInfoGetter, | ||||
|     HeadersGetter, | ||||
|     QueryGetter, | ||||
|     BodyGetter, | ||||
|     AbstractInjector, | ||||
|     _parse_func_signature, | ||||
| ) | ||||
| from .injectors import (AbstractInjector, BodyGetter, HeadersGetter, | ||||
|                         MatchInfoGetter, QueryGetter, _parse_func_signature) | ||||
|  | ||||
|  | ||||
| class PydanticView(AbstractView): | ||||
| @@ -34,33 +27,42 @@ class PydanticView(AbstractView): | ||||
|         return self._iter().__await__() | ||||
|  | ||||
|     def __init_subclass__(cls, **kwargs): | ||||
|         allowed_methods = { | ||||
|         cls.allowed_methods = { | ||||
|             meth_name for meth_name in METH_ALL if hasattr(cls, meth_name.lower()) | ||||
|         } | ||||
|  | ||||
|         async def raise_not_allowed(self): | ||||
|             raise HTTPMethodNotAllowed(self.request.method, allowed_methods) | ||||
|  | ||||
|         for meth_name in METH_ALL: | ||||
|             if meth_name not in allowed_methods: | ||||
|                 setattr(cls, meth_name.lower(), raise_not_allowed) | ||||
|             if meth_name not in cls.allowed_methods: | ||||
|                 setattr(cls, meth_name.lower(), cls.raise_not_allowed) | ||||
|             else: | ||||
|                 handler = getattr(cls, meth_name.lower()) | ||||
|                 decorated_handler = inject_params(handler, cls.parse_func_signature) | ||||
|                 setattr(cls, meth_name.lower(), decorated_handler) | ||||
|  | ||||
|     async def raise_not_allowed(self): | ||||
|         raise HTTPMethodNotAllowed(self.request.method, self.allowed_methods) | ||||
|  | ||||
|     @staticmethod | ||||
|     def parse_func_signature(func: Callable) -> Iterable[AbstractInjector]: | ||||
|         path_args, body_args, qs_args, header_args = _parse_func_signature(func) | ||||
|         path_args, body_args, qs_args, header_args, defaults = _parse_func_signature( | ||||
|             func | ||||
|         ) | ||||
|         injectors = [] | ||||
|  | ||||
|         def default_value(args: dict) -> dict: | ||||
|             """ | ||||
|             Returns the default values of args. | ||||
|             """ | ||||
|             return {name: defaults[name] for name in args if name in defaults} | ||||
|  | ||||
|         if path_args: | ||||
|             injectors.append(MatchInfoGetter(path_args)) | ||||
|             injectors.append(MatchInfoGetter(path_args, default_value(path_args))) | ||||
|         if body_args: | ||||
|             injectors.append(BodyGetter(body_args)) | ||||
|             injectors.append(BodyGetter(body_args, default_value(body_args))) | ||||
|         if qs_args: | ||||
|             injectors.append(QueryGetter(qs_args)) | ||||
|             injectors.append(QueryGetter(qs_args, default_value(qs_args))) | ||||
|         if header_args: | ||||
|             injectors.append(HeadersGetter(header_args)) | ||||
|             injectors.append(HeadersGetter(header_args, default_value(header_args))) | ||||
|         return injectors | ||||
|  | ||||
|  | ||||
| @@ -84,9 +86,23 @@ def inject_params( | ||||
|                 else: | ||||
|                     injector.inject(self.request, args, kwargs) | ||||
|             except ValidationError as error: | ||||
|                 return json_response(text=error.json(), status=400) | ||||
|                 errors = error.errors() | ||||
|                 for error in errors: | ||||
|                     error["in"] = injector.context | ||||
|  | ||||
|                 return json_response(data=errors, status=400) | ||||
|  | ||||
|         return await handler(self, *args, **kwargs) | ||||
|  | ||||
|     update_wrapper(wrapped_handler, handler) | ||||
|     return wrapped_handler | ||||
|  | ||||
|  | ||||
| def is_pydantic_view(obj) -> bool: | ||||
|     """ | ||||
|     Return True if obj is a PydanticView subclass else False. | ||||
|     """ | ||||
|     try: | ||||
|         return issubclass(obj, PydanticView) | ||||
|     except TypeError: | ||||
|         return False | ||||
|   | ||||
							
								
								
									
										0
									
								
								demo/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								demo/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										5
									
								
								demo/__main__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								demo/__main__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| from aiohttp import web | ||||
|  | ||||
| from .main import app | ||||
|  | ||||
| web.run_app(app) | ||||
							
								
								
									
										22
									
								
								demo/main.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								demo/main.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| from aiohttp.web import Application, json_response, middleware | ||||
|  | ||||
| from aiohttp_pydantic import oas | ||||
|  | ||||
| from .model import Model | ||||
| from .view import PetCollectionView, PetItemView | ||||
|  | ||||
|  | ||||
| @middleware | ||||
| async def pet_not_found_to_404(request, handler): | ||||
|     try: | ||||
|         return await handler(request) | ||||
|     except Model.NotFound as key: | ||||
|         return json_response({"error": f"Pet {key} does not exist"}, status=404) | ||||
|  | ||||
|  | ||||
| app = Application(middlewares=[pet_not_found_to_404]) | ||||
| oas.setup(app) | ||||
|  | ||||
| app["model"] = Model() | ||||
| app.router.add_view("/pets", PetCollectionView) | ||||
| app.router.add_view("/pets/{id}", PetItemView) | ||||
							
								
								
									
										55
									
								
								demo/model.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								demo/model.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| from pydantic import BaseModel | ||||
| from typing import List | ||||
|  | ||||
|  | ||||
| class Friend(BaseModel): | ||||
|     name: str | ||||
|     age: str | ||||
|  | ||||
|  | ||||
| class Pet(BaseModel): | ||||
|     id: int | ||||
|     name: str | ||||
|     age: int | ||||
|     friends: Friend | ||||
|  | ||||
|  | ||||
| class Error(BaseModel): | ||||
|     error: str | ||||
|  | ||||
|  | ||||
| class Model: | ||||
|     """ | ||||
|     To keep simple this demo, we use a simple dict as database to | ||||
|     store the models. | ||||
|     """ | ||||
|  | ||||
|     class NotFound(KeyError): | ||||
|         """ | ||||
|         Raised when a pet is not found. | ||||
|         """ | ||||
|  | ||||
|     def __init__(self): | ||||
|         self.storage = {} | ||||
|  | ||||
|     def add_pet(self, pet: Pet): | ||||
|         self.storage[pet.id] = pet | ||||
|  | ||||
|     def remove_pet(self, id: int): | ||||
|         try: | ||||
|             del self.storage[id] | ||||
|         except KeyError as error: | ||||
|             raise self.NotFound(str(error)) | ||||
|  | ||||
|     def update_pet(self, id: int, pet: Pet): | ||||
|         self.remove_pet(id) | ||||
|         self.add_pet(pet) | ||||
|  | ||||
|     def find_pet(self, id: int): | ||||
|         try: | ||||
|             return self.storage[id] | ||||
|         except KeyError as error: | ||||
|             raise self.NotFound(str(error)) | ||||
|  | ||||
|     def list_pets(self): | ||||
|         return list(self.storage.values()) | ||||
							
								
								
									
										34
									
								
								demo/view.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								demo/view.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| from typing import List, Optional, Union | ||||
|  | ||||
| from aiohttp import web | ||||
|  | ||||
| from aiohttp_pydantic import PydanticView | ||||
| from aiohttp_pydantic.oas.typing import r200, r201, r204, r404 | ||||
|  | ||||
| from .model import Error, Pet | ||||
|  | ||||
|  | ||||
| class PetCollectionView(PydanticView): | ||||
|     async def get(self, age: Optional[int] = None) -> r200[List[Pet]]: | ||||
|         pets = self.request.app["model"].list_pets() | ||||
|         return web.json_response( | ||||
|             [pet.dict() for pet in pets if age is None or age == pet.age] | ||||
|         ) | ||||
|  | ||||
|     async def post(self, pet: Pet) -> r201[Pet]: | ||||
|         self.request.app["model"].add_pet(pet) | ||||
|         return web.json_response(pet.dict()) | ||||
|  | ||||
|  | ||||
| class PetItemView(PydanticView): | ||||
|     async def get(self, id: int, /) -> Union[r200[Pet], r404[Error]]: | ||||
|         pet = self.request.app["model"].find_pet(id) | ||||
|         return web.json_response(pet.dict()) | ||||
|  | ||||
|     async def put(self, id: int, /, pet: Pet) -> r200[Pet]: | ||||
|         self.request.app["model"].update_pet(id, pet) | ||||
|         return web.json_response(pet.dict()) | ||||
|  | ||||
|     async def delete(self, id: int, /) -> r204: | ||||
|         self.request.app["model"].remove_pet(id) | ||||
|         return web.Response(status=204) | ||||
							
								
								
									
										6
									
								
								pyproject.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								pyproject.toml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| [build-system] | ||||
| requires = [ | ||||
|   "setuptools >= 46.4.0", | ||||
|   "wheel", | ||||
| ] | ||||
| build-backend = "setuptools.build_meta" | ||||
							
								
								
									
										7
									
								
								requirements/ci.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								requirements/ci.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| certifi==2020.11.8 | ||||
| chardet==3.0.4 | ||||
| codecov==2.1.10 | ||||
| coverage==5.3 | ||||
| idna==2.10 | ||||
| requests==2.25.0 | ||||
| urllib3==1.26.2 | ||||
							
								
								
									
										13
									
								
								requirements/test.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								requirements/test.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| attrs==20.3.0 | ||||
| coverage==5.3 | ||||
| iniconfig==1.1.1 | ||||
| packaging==20.4 | ||||
| pluggy==0.13.1 | ||||
| py==1.9.0 | ||||
| pyparsing==2.4.7 | ||||
| pytest==6.1.2 | ||||
| pytest-aiohttp==0.3.0 | ||||
| pytest-cov==2.10.1 | ||||
| six==1.15.0 | ||||
| toml==0.10.2 | ||||
| typing-extensions==3.7.4.3 | ||||
							
								
								
									
										47
									
								
								setup.cfg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								setup.cfg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| [metadata] | ||||
| name = aiohttp_pydantic | ||||
| version = attr: aiohttp_pydantic.__version__ | ||||
| url = https://github.com/Maillol/aiohttp-pydantic | ||||
| author = Vincent Maillol | ||||
| author_email = vincent.maillol@gmail.com | ||||
| description = Aiohttp View using pydantic to validate request body and query sting regarding method annotations. | ||||
| long_description = file: README.rst | ||||
| keywords = | ||||
|     aiohttp | ||||
|     pydantic | ||||
|     annotations | ||||
|     validation | ||||
| license = MIT | ||||
| classifiers = | ||||
|     Intended Audience :: Developers | ||||
|     Intended Audience :: Information Technology | ||||
|     Programming Language :: Python | ||||
|     Programming Language :: Python :: 3 | ||||
|     Programming Language :: Python :: 3 :: Only | ||||
|     Programming Language :: Python :: 3.8 | ||||
|     Programming Language :: Python :: 3.9 | ||||
|     Topic :: Software Development :: Libraries :: Application Frameworks | ||||
|     Framework :: AsyncIO | ||||
|     License :: OSI Approved :: MIT License | ||||
|  | ||||
| [options] | ||||
| zip_safe = False | ||||
| include_package_data = True | ||||
| packages = find: | ||||
| python_requires = >=3.8 | ||||
| install_requires = | ||||
|     aiohttp | ||||
|     pydantic>=1.7 | ||||
|     swagger-ui-bundle | ||||
|  | ||||
| [options.extras_require] | ||||
| test = pytest==6.1.2; pytest-aiohttp==0.3.0; pytest-cov==2.10.1 | ||||
| ci = pytest==6.1.2; pytest-aiohttp==0.3.0; pytest-cov==2.10.1; codecov==2.1.10 | ||||
|  | ||||
| [options.packages.find] | ||||
| exclude = | ||||
|     tests | ||||
|     demo | ||||
|  | ||||
| [options.package_data] | ||||
| aiohttp_pydantic.oas = index.j2 | ||||
							
								
								
									
										27
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								setup.py
									
									
									
									
									
								
							| @@ -1,28 +1,3 @@ | ||||
| from setuptools import setup | ||||
|  | ||||
|  | ||||
| setup( | ||||
|     name='aiohttp_pydantic', | ||||
|     version='1.0.0', | ||||
|     description='Aiohttp View using pydantic to validate request body and query sting regarding method annotation', | ||||
|     keywords='aiohttp pydantic annotation unpack inject validate', | ||||
|     author='Vincent Maillol', | ||||
|     author_email='vincent.maillol@gmail.com', | ||||
|     url='https://github.com/Maillol/aiohttp-pydantic', | ||||
|     license='MIT', | ||||
|     packages=['aiohttp_pydantic'], | ||||
|     classifiers=[ | ||||
|         'Intended Audience :: Developers', | ||||
|         'Intended Audience :: Information Technology', | ||||
|         'Programming Language :: Python', | ||||
|         'Programming Language :: Python :: 3', | ||||
|         'Programming Language :: Python :: 3 :: Only', | ||||
|         'Programming Language :: Python :: 3.8', | ||||
|         'Programming Language :: Python :: 3.9', | ||||
|         'Topic :: Software Development :: Libraries :: Application Frameworks', | ||||
|         'Framework :: AsyncIO', | ||||
|         'License :: OSI Approved :: MIT License' | ||||
|     ], | ||||
|     python_requires='>=3.6', | ||||
|     install_requires=['aiohttp', 'pydantic'] | ||||
| ) | ||||
| setup() | ||||
|   | ||||
| @@ -1,3 +0,0 @@ | ||||
| . | ||||
| pytest==6.1.1 | ||||
| pytest-aiohttp==0.3.0 | ||||
							
								
								
									
										0
									
								
								tests/test_oas/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								tests/test_oas/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								tests/test_oas/test_cmd/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								tests/test_oas/test_cmd/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										27
									
								
								tests/test_oas/test_cmd/sample.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								tests/test_oas/test_cmd/sample.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| from aiohttp import web | ||||
|  | ||||
| from aiohttp_pydantic import PydanticView | ||||
|  | ||||
|  | ||||
| class View1(PydanticView): | ||||
|     async def get(self, a: int, /): | ||||
|         return web.json_response() | ||||
|  | ||||
|  | ||||
| class View2(PydanticView): | ||||
|     async def post(self, b: int, /): | ||||
|         return web.json_response() | ||||
|  | ||||
|  | ||||
| sub_app = web.Application() | ||||
| sub_app.router.add_view("/route-2/{b}", View2) | ||||
|  | ||||
| app = web.Application() | ||||
| app.router.add_view("/route-1/{a}", View1) | ||||
| app.add_subapp("/sub-app", sub_app) | ||||
|  | ||||
|  | ||||
| def make_app(): | ||||
|     app = web.Application() | ||||
|     app.router.add_view("/route-3/{a}", View1) | ||||
|     return app | ||||
							
								
								
									
										124
									
								
								tests/test_oas/test_cmd/test_cmd.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								tests/test_oas/test_cmd/test_cmd.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,124 @@ | ||||
| import argparse | ||||
| from textwrap import dedent | ||||
|  | ||||
| import pytest | ||||
|  | ||||
| from aiohttp_pydantic.oas import cmd | ||||
|  | ||||
|  | ||||
| @pytest.fixture | ||||
| def cmd_line(): | ||||
|     parser = argparse.ArgumentParser() | ||||
|     cmd.setup(parser) | ||||
|     return parser | ||||
|  | ||||
|  | ||||
| def test_show_oad_of_app(cmd_line, capfd): | ||||
|     args = cmd_line.parse_args(["tests.test_oas.test_cmd.sample"]) | ||||
|     args.func(args) | ||||
|     captured = capfd.readouterr() | ||||
|     expected = dedent( | ||||
|         """ | ||||
|         { | ||||
|         "openapi": "3.0.0", | ||||
|         "paths": { | ||||
|             "/route-1/{a}": { | ||||
|                 "get": { | ||||
|                     "parameters": [ | ||||
|                         { | ||||
|                             "in": "path", | ||||
|                             "name": "a", | ||||
|                             "required": true, | ||||
|                             "schema": { | ||||
|                                 "title": "a", | ||||
|                                 "type": "integer" | ||||
|                             } | ||||
|                         } | ||||
|                     ] | ||||
|                 } | ||||
|             }, | ||||
|             "/sub-app/route-2/{b}": { | ||||
|                 "post": { | ||||
|                     "parameters": [ | ||||
|                         { | ||||
|                             "in": "path", | ||||
|                             "name": "b", | ||||
|                             "required": true, | ||||
|                             "schema": { | ||||
|                                 "title": "b", | ||||
|                                 "type": "integer" | ||||
|                             } | ||||
|                         } | ||||
|                     ] | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     """ | ||||
|     ) | ||||
|  | ||||
|     assert captured.out.strip() == expected.strip() | ||||
|  | ||||
|  | ||||
| def test_show_oad_of_sub_app(cmd_line, capfd): | ||||
|     args = cmd_line.parse_args(["tests.test_oas.test_cmd.sample:sub_app"]) | ||||
|     args.func(args) | ||||
|     captured = capfd.readouterr() | ||||
|     expected = dedent( | ||||
|         """ | ||||
|         { | ||||
|         "openapi": "3.0.0", | ||||
|         "paths": { | ||||
|             "/sub-app/route-2/{b}": { | ||||
|                 "post": { | ||||
|                     "parameters": [ | ||||
|                         { | ||||
|                             "in": "path", | ||||
|                             "name": "b", | ||||
|                             "required": true, | ||||
|                             "schema": { | ||||
|                                 "title": "b", | ||||
|                                 "type": "integer" | ||||
|                             } | ||||
|                         } | ||||
|                     ] | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     """ | ||||
|     ) | ||||
|  | ||||
|     assert captured.out.strip() == expected.strip() | ||||
|  | ||||
|  | ||||
| def test_show_oad_of_a_callable(cmd_line, capfd): | ||||
|     args = cmd_line.parse_args(["tests.test_oas.test_cmd.sample:make_app()"]) | ||||
|     args.func(args) | ||||
|     captured = capfd.readouterr() | ||||
|     expected = dedent( | ||||
|         """ | ||||
|         { | ||||
|         "openapi": "3.0.0", | ||||
|         "paths": { | ||||
|             "/route-3/{a}": { | ||||
|                 "get": { | ||||
|                     "parameters": [ | ||||
|                         { | ||||
|                             "in": "path", | ||||
|                             "name": "a", | ||||
|                             "required": true, | ||||
|                             "schema": { | ||||
|                                 "title": "a", | ||||
|                                 "type": "integer" | ||||
|                             } | ||||
|                         } | ||||
|                     ] | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     """ | ||||
|     ) | ||||
|  | ||||
|     assert captured.out.strip() == expected.strip() | ||||
							
								
								
									
										0
									
								
								tests/test_oas/test_struct/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								tests/test_oas/test_struct/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										54
									
								
								tests/test_oas/test_struct/test_info.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								tests/test_oas/test_struct/test_info.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| import pytest | ||||
|  | ||||
| from aiohttp_pydantic.oas.struct import OpenApiSpec3 | ||||
|  | ||||
|  | ||||
| def test_info_title(): | ||||
|     oas = OpenApiSpec3() | ||||
|     assert oas.info.title is None | ||||
|     oas.info.title = "Info Title" | ||||
|     assert oas.info.title == "Info Title" | ||||
|     assert oas.spec == {"info": {"title": "Info Title"}, "openapi": "3.0.0"} | ||||
|  | ||||
|  | ||||
| def test_info_description(): | ||||
|     oas = OpenApiSpec3() | ||||
|     assert oas.info.description is None | ||||
|     oas.info.description = "info description" | ||||
|     assert oas.info.description == "info description" | ||||
|     assert oas.spec == {"info": {"description": "info description"}, "openapi": "3.0.0"} | ||||
|  | ||||
|  | ||||
| def test_info_version(): | ||||
|     oas = OpenApiSpec3() | ||||
|     assert oas.info.version is None | ||||
|     oas.info.version = "3.14" | ||||
|     assert oas.info.version == "3.14" | ||||
|     assert oas.spec == {"info": {"version": "3.14"}, "openapi": "3.0.0"} | ||||
|  | ||||
|  | ||||
| def test_info_terms_of_service(): | ||||
|     oas = OpenApiSpec3() | ||||
|     assert oas.info.terms_of_service is None | ||||
|     oas.info.terms_of_service = "http://example.com/terms/" | ||||
|     assert oas.info.terms_of_service == "http://example.com/terms/" | ||||
|     assert oas.spec == { | ||||
|         "info": {"termsOfService": "http://example.com/terms/"}, | ||||
|         "openapi": "3.0.0", | ||||
|     } | ||||
|  | ||||
|  | ||||
| @pytest.mark.skip("Not yet implemented") | ||||
| def test_info_license(): | ||||
|     oas = OpenApiSpec3() | ||||
|     oas.info.license.name = "Apache 2.0" | ||||
|     oas.info.license.url = "https://www.apache.org/licenses/LICENSE-2.0.html" | ||||
|     assert oas.spec == { | ||||
|         "info": { | ||||
|             "license": { | ||||
|                 "name": "Apache 2.0", | ||||
|                 "url": "https://www.apache.org/licenses/LICENSE-2.0.html", | ||||
|             } | ||||
|         }, | ||||
|         "openapi": "3.0.0", | ||||
|     } | ||||
							
								
								
									
										124
									
								
								tests/test_oas/test_struct/test_paths.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								tests/test_oas/test_struct/test_paths.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,124 @@ | ||||
| from aiohttp_pydantic.oas.struct import OpenApiSpec3 | ||||
|  | ||||
|  | ||||
| def test_paths_description(): | ||||
|     oas = OpenApiSpec3() | ||||
|     oas.paths["/users/{id}"].description = "This route ..." | ||||
|     assert oas.spec == { | ||||
|         "openapi": "3.0.0", | ||||
|         "paths": {"/users/{id}": {"description": "This route ..."}}, | ||||
|     } | ||||
|  | ||||
|  | ||||
| def test_paths_get(): | ||||
|     oas = OpenApiSpec3() | ||||
|     oas.paths["/users/{id}"].get | ||||
|     assert oas.spec == {"openapi": "3.0.0", "paths": {"/users/{id}": {"get": {}}}} | ||||
|  | ||||
|  | ||||
| def test_paths_operation_description(): | ||||
|     oas = OpenApiSpec3() | ||||
|     operation = oas.paths["/users/{id}"].get | ||||
|     operation.description = "Long descriptions ..." | ||||
|     assert oas.spec == { | ||||
|         "openapi": "3.0.0", | ||||
|         "paths": {"/users/{id}": {"get": {"description": "Long descriptions ..."}}}, | ||||
|     } | ||||
|  | ||||
|  | ||||
| def test_paths_operation_summary(): | ||||
|     oas = OpenApiSpec3() | ||||
|     operation = oas.paths["/users/{id}"].get | ||||
|     operation.summary = "Updates a pet in the store with form data" | ||||
|     assert oas.spec == { | ||||
|         "openapi": "3.0.0", | ||||
|         "paths": { | ||||
|             "/users/{id}": { | ||||
|                 "get": {"summary": "Updates a pet in the store with form data"} | ||||
|             } | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|  | ||||
| def test_paths_operation_parameters(): | ||||
|     oas = OpenApiSpec3() | ||||
|     operation = oas.paths["/users/{petId}"].get | ||||
|     parameter = operation.parameters[0] | ||||
|     parameter.name = "petId" | ||||
|     parameter.description = "ID of pet that needs to be updated" | ||||
|     parameter.in_ = "path" | ||||
|     parameter.required = True | ||||
|  | ||||
|     assert oas.spec == { | ||||
|         "openapi": "3.0.0", | ||||
|         "paths": { | ||||
|             "/users/{petId}": { | ||||
|                 "get": { | ||||
|                     "parameters": [ | ||||
|                         { | ||||
|                             "description": "ID of pet that needs to be updated", | ||||
|                             "in": "path", | ||||
|                             "name": "petId", | ||||
|                             "required": True, | ||||
|                         } | ||||
|                     ] | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|  | ||||
| def test_paths_operation_requestBody(): | ||||
|     oas = OpenApiSpec3() | ||||
|     request_body = oas.paths["/users/{petId}"].get.request_body | ||||
|     request_body.description = "user to add to the system" | ||||
|     request_body.content = { | ||||
|         "application/json": { | ||||
|             "schema": {"$ref": "#/components/schemas/User"}, | ||||
|             "examples": { | ||||
|                 "user": { | ||||
|                     "summary": "User Example", | ||||
|                     "externalValue": "http://foo.bar/examples/user-example.json", | ||||
|                 } | ||||
|             }, | ||||
|         } | ||||
|     } | ||||
|     request_body.required = True | ||||
|     assert oas.spec == { | ||||
|         "openapi": "3.0.0", | ||||
|         "paths": { | ||||
|             "/users/{petId}": { | ||||
|                 "get": { | ||||
|                     "requestBody": { | ||||
|                         "content": { | ||||
|                             "application/json": { | ||||
|                                 "examples": { | ||||
|                                     "user": { | ||||
|                                         "externalValue": "http://foo.bar/examples/user-example.json", | ||||
|                                         "summary": "User Example", | ||||
|                                     } | ||||
|                                 }, | ||||
|                                 "schema": {"$ref": "#/components/schemas/User"}, | ||||
|                             } | ||||
|                         }, | ||||
|                         "description": "user to add to the system", | ||||
|                         "required": True, | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|  | ||||
| def test_paths_operation_responses(): | ||||
|     oas = OpenApiSpec3() | ||||
|     response = oas.paths["/users/{petId}"].get.responses[200] | ||||
|     response.description = "A complex object array response" | ||||
|     response.content = { | ||||
|         "application/json": { | ||||
|             "schema": { | ||||
|                 "type": "array", | ||||
|                 "items": {"$ref": "#/components/schemas/VeryComplexType"}, | ||||
|             } | ||||
|         } | ||||
|     } | ||||
							
								
								
									
										36
									
								
								tests/test_oas/test_struct/test_servers.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								tests/test_oas/test_struct/test_servers.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| import pytest | ||||
|  | ||||
| from aiohttp_pydantic.oas.struct import OpenApiSpec3 | ||||
|  | ||||
|  | ||||
| def test_sever_url(): | ||||
|     oas = OpenApiSpec3() | ||||
|     oas.servers[0].url = "https://development.gigantic-server.com/v1" | ||||
|     oas.servers[1].url = "https://development.gigantic-server.com/v2" | ||||
|     assert oas.spec == { | ||||
|         "openapi": "3.0.0", | ||||
|         "servers": [ | ||||
|             {"url": "https://development.gigantic-server.com/v1"}, | ||||
|             {"url": "https://development.gigantic-server.com/v2"}, | ||||
|         ], | ||||
|     } | ||||
|  | ||||
|  | ||||
| def test_sever_description(): | ||||
|     oas = OpenApiSpec3() | ||||
|     oas.servers[0].url = "https://development.gigantic-server.com/v1" | ||||
|     oas.servers[0].description = "Development server" | ||||
|     assert oas.spec == { | ||||
|         "openapi": "3.0.0", | ||||
|         "servers": [ | ||||
|             { | ||||
|                 "url": "https://development.gigantic-server.com/v1", | ||||
|                 "description": "Development server", | ||||
|             } | ||||
|         ], | ||||
|     } | ||||
|  | ||||
|  | ||||
| @pytest.mark.skip("Not yet implemented") | ||||
| def test_sever_variables(): | ||||
|     oas = OpenApiSpec3() | ||||
							
								
								
									
										363
									
								
								tests/test_oas/test_view.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										363
									
								
								tests/test_oas/test_view.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,363 @@ | ||||
| from enum import Enum | ||||
| from typing import List, Optional, Union | ||||
| from uuid import UUID | ||||
|  | ||||
| import pytest | ||||
| from aiohttp import web | ||||
| from pydantic.main import BaseModel | ||||
|  | ||||
| from aiohttp_pydantic import PydanticView, oas | ||||
| from aiohttp_pydantic.oas.typing import r200, r201, r204, r404 | ||||
|  | ||||
|  | ||||
| class Color(str, Enum): | ||||
|     RED = "red" | ||||
|     GREEN = "green" | ||||
|     PINK = "pink" | ||||
|  | ||||
|  | ||||
| class Toy(BaseModel): | ||||
|     name: str | ||||
|     color: Color | ||||
|  | ||||
|  | ||||
| class Pet(BaseModel): | ||||
|     id: int | ||||
|     name: str | ||||
|     toys: List[Toy] | ||||
|  | ||||
|  | ||||
| class PetCollectionView(PydanticView): | ||||
|     async def get( | ||||
|         self, format: str, name: Optional[str] = None, *, promo: Optional[UUID] = None | ||||
|     ) -> r200[List[Pet]]: | ||||
|         """ | ||||
|         Get a list of pets | ||||
|         """ | ||||
|         return web.json_response() | ||||
|  | ||||
|     async def post(self, pet: Pet) -> r201[Pet]: | ||||
|         """Create a Pet""" | ||||
|         return web.json_response() | ||||
|  | ||||
|  | ||||
| class PetItemView(PydanticView): | ||||
|     async def get(self, id: int, /) -> Union[r200[Pet], r404]: | ||||
|         return web.json_response() | ||||
|  | ||||
|     async def put(self, id: int, /, pet: Pet): | ||||
|         return web.json_response() | ||||
|  | ||||
|     async def delete(self, id: int, /) -> r204: | ||||
|         return web.json_response() | ||||
|  | ||||
|  | ||||
| @pytest.fixture | ||||
| async def generated_oas(aiohttp_client, loop) -> web.Application: | ||||
|     app = web.Application() | ||||
|     app.router.add_view("/pets", PetCollectionView) | ||||
|     app.router.add_view("/pets/{id}", PetItemView) | ||||
|     oas.setup(app) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     response = await client.get("/oas/spec") | ||||
|     assert response.status == 200 | ||||
|     assert response.content_type == "application/json" | ||||
|     return await response.json() | ||||
|  | ||||
|  | ||||
| async def test_generated_oas_should_have_components_schemas(generated_oas): | ||||
|     assert generated_oas["components"]["schemas"] == { | ||||
|         "Color": { | ||||
|             "description": "An enumeration.", | ||||
|             "enum": ["red", "green", "pink"], | ||||
|             "title": "Color", | ||||
|             "type": "string", | ||||
|         }, | ||||
|         "Toy": { | ||||
|             "properties": { | ||||
|                 "color": {"$ref": "#/components/schemas/Color"}, | ||||
|                 "name": {"title": "Name", "type": "string"}, | ||||
|             }, | ||||
|             "required": ["name", "color"], | ||||
|             "title": "Toy", | ||||
|             "type": "object", | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|  | ||||
| async def test_generated_oas_should_have_pets_paths(generated_oas): | ||||
|     assert "/pets" in generated_oas["paths"] | ||||
|  | ||||
|  | ||||
| async def test_pets_route_should_have_get_method(generated_oas): | ||||
|     assert generated_oas["paths"]["/pets"]["get"] == { | ||||
|         "description": "Get a list of pets", | ||||
|         "parameters": [ | ||||
|             { | ||||
|                 "in": "query", | ||||
|                 "name": "format", | ||||
|                 "required": True, | ||||
|                 "schema": {"title": "format", "type": "string"}, | ||||
|             }, | ||||
|             { | ||||
|                 "in": "query", | ||||
|                 "name": "name", | ||||
|                 "required": False, | ||||
|                 "schema": {"title": "name", "type": "string"}, | ||||
|             }, | ||||
|             { | ||||
|                 "in": "header", | ||||
|                 "name": "promo", | ||||
|                 "required": False, | ||||
|                 "schema": {"format": "uuid", "title": "promo", "type": "string"}, | ||||
|             }, | ||||
|         ], | ||||
|         "responses": { | ||||
|             "200": { | ||||
|                 "content": { | ||||
|                     "application/json": { | ||||
|                         "schema": { | ||||
|                             "items": { | ||||
|                                 "definitions": { | ||||
|                                     "Color": { | ||||
|                                         "description": "An enumeration.", | ||||
|                                         "enum": ["red", "green", "pink"], | ||||
|                                         "title": "Color", | ||||
|                                         "type": "string", | ||||
|                                     }, | ||||
|                                     "Toy": { | ||||
|                                         "properties": { | ||||
|                                             "color": { | ||||
|                                                 "$ref": "#/components/schemas/Color" | ||||
|                                             }, | ||||
|                                             "name": {"title": "Name", "type": "string"}, | ||||
|                                         }, | ||||
|                                         "required": ["name", "color"], | ||||
|                                         "title": "Toy", | ||||
|                                         "type": "object", | ||||
|                                     }, | ||||
|                                 }, | ||||
|                                 "properties": { | ||||
|                                     "id": {"title": "Id", "type": "integer"}, | ||||
|                                     "name": {"title": "Name", "type": "string"}, | ||||
|                                     "toys": { | ||||
|                                         "items": {"$ref": "#/components/schemas/Toy"}, | ||||
|                                         "title": "Toys", | ||||
|                                         "type": "array", | ||||
|                                     }, | ||||
|                                 }, | ||||
|                                 "required": ["id", "name", "toys"], | ||||
|                                 "title": "Pet", | ||||
|                                 "type": "object", | ||||
|                             }, | ||||
|                             "type": "array", | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|  | ||||
| async def test_pets_route_should_have_post_method(generated_oas): | ||||
|     assert generated_oas["paths"]["/pets"]["post"] == { | ||||
|         "description": "Create a Pet", | ||||
|         "requestBody": { | ||||
|             "content": { | ||||
|                 "application/json": { | ||||
|                     "schema": { | ||||
|                         "definitions": { | ||||
|                             "Color": { | ||||
|                                 "description": "An enumeration.", | ||||
|                                 "enum": ["red", "green", "pink"], | ||||
|                                 "title": "Color", | ||||
|                                 "type": "string", | ||||
|                             }, | ||||
|                             "Toy": { | ||||
|                                 "properties": { | ||||
|                                     "color": {"$ref": "#/components/schemas/Color"}, | ||||
|                                     "name": {"title": "Name", "type": "string"}, | ||||
|                                 }, | ||||
|                                 "required": ["name", "color"], | ||||
|                                 "title": "Toy", | ||||
|                                 "type": "object", | ||||
|                             }, | ||||
|                         }, | ||||
|                         "properties": { | ||||
|                             "id": {"title": "Id", "type": "integer"}, | ||||
|                             "name": {"title": "Name", "type": "string"}, | ||||
|                             "toys": { | ||||
|                                 "items": {"$ref": "#/components/schemas/Toy"}, | ||||
|                                 "title": "Toys", | ||||
|                                 "type": "array", | ||||
|                             }, | ||||
|                         }, | ||||
|                         "required": ["id", "name", "toys"], | ||||
|                         "title": "Pet", | ||||
|                         "type": "object", | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         "responses": { | ||||
|             "201": { | ||||
|                 "content": { | ||||
|                     "application/json": { | ||||
|                         "schema": { | ||||
|                             "definitions": { | ||||
|                                 "Color": { | ||||
|                                     "description": "An enumeration.", | ||||
|                                     "enum": ["red", "green", "pink"], | ||||
|                                     "title": "Color", | ||||
|                                     "type": "string", | ||||
|                                 }, | ||||
|                                 "Toy": { | ||||
|                                     "properties": { | ||||
|                                         "color": {"$ref": "#/components/schemas/Color"}, | ||||
|                                         "name": {"title": "Name", "type": "string"}, | ||||
|                                     }, | ||||
|                                     "required": ["name", "color"], | ||||
|                                     "title": "Toy", | ||||
|                                     "type": "object", | ||||
|                                 }, | ||||
|                             }, | ||||
|                             "properties": { | ||||
|                                 "id": {"title": "Id", "type": "integer"}, | ||||
|                                 "name": {"title": "Name", "type": "string"}, | ||||
|                                 "toys": { | ||||
|                                     "items": {"$ref": "#/components/schemas/Toy"}, | ||||
|                                     "title": "Toys", | ||||
|                                     "type": "array", | ||||
|                                 }, | ||||
|                             }, | ||||
|                             "required": ["id", "name", "toys"], | ||||
|                             "title": "Pet", | ||||
|                             "type": "object", | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|  | ||||
| async def test_generated_oas_should_have_pets_id_paths(generated_oas): | ||||
|     assert "/pets/{id}" in generated_oas["paths"] | ||||
|  | ||||
|  | ||||
| async def test_pets_id_route_should_have_delete_method(generated_oas): | ||||
|     assert generated_oas["paths"]["/pets/{id}"]["delete"] == { | ||||
|         "parameters": [ | ||||
|             { | ||||
|                 "required": True, | ||||
|                 "in": "path", | ||||
|                 "name": "id", | ||||
|                 "schema": {"title": "id", "type": "integer"}, | ||||
|             } | ||||
|         ], | ||||
|         "responses": {"204": {"content": {}}}, | ||||
|     } | ||||
|  | ||||
|  | ||||
| async def test_pets_id_route_should_have_get_method(generated_oas): | ||||
|     assert generated_oas["paths"]["/pets/{id}"]["get"] == { | ||||
|         "parameters": [ | ||||
|             { | ||||
|                 "in": "path", | ||||
|                 "name": "id", | ||||
|                 "required": True, | ||||
|                 "schema": {"title": "id", "type": "integer"}, | ||||
|             } | ||||
|         ], | ||||
|         "responses": { | ||||
|             "200": { | ||||
|                 "content": { | ||||
|                     "application/json": { | ||||
|                         "schema": { | ||||
|                             "definitions": { | ||||
|                                 "Color": { | ||||
|                                     "description": "An enumeration.", | ||||
|                                     "enum": ["red", "green", "pink"], | ||||
|                                     "title": "Color", | ||||
|                                     "type": "string", | ||||
|                                 }, | ||||
|                                 "Toy": { | ||||
|                                     "properties": { | ||||
|                                         "color": {"$ref": "#/components/schemas/Color"}, | ||||
|                                         "name": {"title": "Name", "type": "string"}, | ||||
|                                     }, | ||||
|                                     "required": ["name", "color"], | ||||
|                                     "title": "Toy", | ||||
|                                     "type": "object", | ||||
|                                 }, | ||||
|                             }, | ||||
|                             "properties": { | ||||
|                                 "id": {"title": "Id", "type": "integer"}, | ||||
|                                 "name": {"title": "Name", "type": "string"}, | ||||
|                                 "toys": { | ||||
|                                     "items": {"$ref": "#/components/schemas/Toy"}, | ||||
|                                     "title": "Toys", | ||||
|                                     "type": "array", | ||||
|                                 }, | ||||
|                             }, | ||||
|                             "required": ["id", "name", "toys"], | ||||
|                             "title": "Pet", | ||||
|                             "type": "object", | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             }, | ||||
|             "404": {"content": {}}, | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|  | ||||
| async def test_pets_id_route_should_have_put_method(generated_oas): | ||||
|     assert generated_oas["paths"]["/pets/{id}"]["put"] == { | ||||
|         "parameters": [ | ||||
|             { | ||||
|                 "in": "path", | ||||
|                 "name": "id", | ||||
|                 "required": True, | ||||
|                 "schema": {"title": "id", "type": "integer"}, | ||||
|             } | ||||
|         ], | ||||
|         "requestBody": { | ||||
|             "content": { | ||||
|                 "application/json": { | ||||
|                     "schema": { | ||||
|                         "definitions": { | ||||
|                             "Color": { | ||||
|                                 "description": "An enumeration.", | ||||
|                                 "enum": ["red", "green", "pink"], | ||||
|                                 "title": "Color", | ||||
|                                 "type": "string", | ||||
|                             }, | ||||
|                             "Toy": { | ||||
|                                 "properties": { | ||||
|                                     "color": {"$ref": "#/components/schemas/Color"}, | ||||
|                                     "name": {"title": "Name", "type": "string"}, | ||||
|                                 }, | ||||
|                                 "required": ["name", "color"], | ||||
|                                 "title": "Toy", | ||||
|                                 "type": "object", | ||||
|                             }, | ||||
|                         }, | ||||
|                         "properties": { | ||||
|                             "id": {"title": "Id", "type": "integer"}, | ||||
|                             "name": {"title": "Name", "type": "string"}, | ||||
|                             "toys": { | ||||
|                                 "items": {"$ref": "#/components/schemas/Toy"}, | ||||
|                                 "title": "Toys", | ||||
|                                 "type": "array", | ||||
|                             }, | ||||
|                         }, | ||||
|                         "required": ["id", "name", "toys"], | ||||
|                         "title": "Pet", | ||||
|                         "type": "object", | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|     } | ||||
| @@ -1,7 +1,9 @@ | ||||
| from aiohttp_pydantic.injectors import _parse_func_signature | ||||
| from pydantic import BaseModel | ||||
| from uuid import UUID | ||||
|  | ||||
| from pydantic import BaseModel | ||||
|  | ||||
| from aiohttp_pydantic.injectors import _parse_func_signature | ||||
|  | ||||
|  | ||||
| class User(BaseModel): | ||||
|     firstname: str | ||||
| @@ -9,7 +11,6 @@ class User(BaseModel): | ||||
|  | ||||
|  | ||||
| def test_parse_func_signature(): | ||||
|  | ||||
|     def body_only(self, user: User): | ||||
|         pass | ||||
|  | ||||
| @@ -37,13 +38,42 @@ def test_parse_func_signature(): | ||||
|     def path_body_qs_and_header(self, id: str, /, user: User, page: int, *, auth: UUID): | ||||
|         pass | ||||
|  | ||||
|     assert _parse_func_signature(body_only) == ({}, {'user': User}, {},  {}) | ||||
|     assert _parse_func_signature(path_only) == ({'id': str}, {}, {},  {}) | ||||
|     assert _parse_func_signature(qs_only) == ({}, {}, {'page': int},  {}) | ||||
|     assert _parse_func_signature(header_only) == ({}, {}, {},  {'auth': UUID}) | ||||
|     assert _parse_func_signature(path_and_qs) == ({'id': str}, {}, {'page': int},  {}) | ||||
|     assert _parse_func_signature(path_and_header) == ({'id': str}, {}, {},  {'auth': UUID}) | ||||
|     assert _parse_func_signature(qs_and_header) == ({}, {}, {'page': int},  {'auth': UUID}) | ||||
|     assert _parse_func_signature(path_qs_and_header) == ({'id': str}, {}, {'page': int},  {'auth': UUID}) | ||||
|     assert _parse_func_signature(path_body_qs_and_header) == ({'id': str}, {'user': User}, {'page': int},  {'auth': UUID}) | ||||
|  | ||||
|     assert _parse_func_signature(body_only) == ({}, {"user": User}, {}, {}, {}) | ||||
|     assert _parse_func_signature(path_only) == ({"id": str}, {}, {}, {}, {}) | ||||
|     assert _parse_func_signature(qs_only) == ({}, {}, {"page": int}, {}, {}) | ||||
|     assert _parse_func_signature(header_only) == ({}, {}, {}, {"auth": UUID}, {}) | ||||
|     assert _parse_func_signature(path_and_qs) == ( | ||||
|         {"id": str}, | ||||
|         {}, | ||||
|         {"page": int}, | ||||
|         {}, | ||||
|         {}, | ||||
|     ) | ||||
|     assert _parse_func_signature(path_and_header) == ( | ||||
|         {"id": str}, | ||||
|         {}, | ||||
|         {}, | ||||
|         {"auth": UUID}, | ||||
|         {}, | ||||
|     ) | ||||
|     assert _parse_func_signature(qs_and_header) == ( | ||||
|         {}, | ||||
|         {}, | ||||
|         {"page": int}, | ||||
|         {"auth": UUID}, | ||||
|         {}, | ||||
|     ) | ||||
|     assert _parse_func_signature(path_qs_and_header) == ( | ||||
|         {"id": str}, | ||||
|         {}, | ||||
|         {"page": int}, | ||||
|         {"auth": UUID}, | ||||
|         {}, | ||||
|     ) | ||||
|     assert _parse_func_signature(path_body_qs_and_header) == ( | ||||
|         {"id": str}, | ||||
|         {"user": User}, | ||||
|         {"page": int}, | ||||
|         {"auth": UUID}, | ||||
|         {}, | ||||
|     ) | ||||
|   | ||||
| @@ -1,6 +1,8 @@ | ||||
| from pydantic import BaseModel | ||||
| from typing import Optional | ||||
|  | ||||
| from aiohttp import web | ||||
| from pydantic import BaseModel | ||||
|  | ||||
| from aiohttp_pydantic import PydanticView | ||||
|  | ||||
|  | ||||
| @@ -10,43 +12,56 @@ class ArticleModel(BaseModel): | ||||
|  | ||||
|  | ||||
| class ArticleView(PydanticView): | ||||
|  | ||||
|     async def post(self, article: ArticleModel): | ||||
|         return web.json_response(article.dict()) | ||||
|  | ||||
|  | ||||
| async def test_post_an_article_without_required_field_should_return_an_error_message(aiohttp_client, loop): | ||||
| async def test_post_an_article_without_required_field_should_return_an_error_message( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article', ArticleView) | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.post('/article', json={}) | ||||
|     resp = await client.post("/article", json={}) | ||||
|     assert resp.status == 400 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == [{'loc': ['name'], | ||||
|                                   'msg': 'field required', | ||||
|                                   'type': 'value_error.missing'}] | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == [ | ||||
|         { | ||||
|             "in": "body", | ||||
|             "loc": ["name"], | ||||
|             "msg": "field required", | ||||
|             "type": "value_error.missing", | ||||
|         } | ||||
|     ] | ||||
|  | ||||
|  | ||||
| async def test_post_an_article_with_wrong_type_field_should_return_an_error_message(aiohttp_client, loop): | ||||
| async def test_post_an_article_with_wrong_type_field_should_return_an_error_message( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article', ArticleView) | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.post('/article', json={'name': 'foo', 'nb_page': 'foo'}) | ||||
|     resp = await client.post("/article", json={"name": "foo", "nb_page": "foo"}) | ||||
|     assert resp.status == 400 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == [{'loc': ['nb_page'], | ||||
|                                   'msg': 'value is not a valid integer', | ||||
|                                   'type': 'type_error.integer'}] | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == [ | ||||
|         { | ||||
|             "in": "body", | ||||
|             "loc": ["nb_page"], | ||||
|             "msg": "value is not a valid integer", | ||||
|             "type": "type_error.integer", | ||||
|         } | ||||
|     ] | ||||
|  | ||||
|  | ||||
| async def test_post_a_valid_article_should_return_the_parsed_type(aiohttp_client, loop): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article', ArticleView) | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.post('/article', json={'name': 'foo', 'nb_page': 3}) | ||||
|     resp = await client.post("/article", json={"name": "foo", "nb_page": 3}) | ||||
|     assert resp.status == 200 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == {'name': 'foo', 'nb_page': 3} | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == {"name": "foo", "nb_page": 3} | ||||
|   | ||||
| @@ -1,11 +1,13 @@ | ||||
| from aiohttp import web | ||||
| from aiohttp_pydantic import PydanticView | ||||
| from datetime import datetime | ||||
| import json | ||||
| from datetime import datetime | ||||
| from enum import Enum | ||||
|  | ||||
| from aiohttp import web | ||||
|  | ||||
| from aiohttp_pydantic import PydanticView | ||||
|  | ||||
|  | ||||
| class JSONEncoder(json.JSONEncoder): | ||||
|  | ||||
|     def default(self, o): | ||||
|         if isinstance(o, datetime): | ||||
|             return o.isoformat() | ||||
| @@ -14,54 +16,121 @@ class JSONEncoder(json.JSONEncoder): | ||||
|  | ||||
|  | ||||
| class ArticleView(PydanticView): | ||||
|  | ||||
|     async def get(self, *, signature_expired: datetime): | ||||
|         return web.json_response({'signature': signature_expired}, dumps=JSONEncoder().encode) | ||||
|         return web.json_response( | ||||
|             {"signature": signature_expired}, dumps=JSONEncoder().encode | ||||
|         ) | ||||
|  | ||||
|  | ||||
| async def test_get_article_without_required_header_should_return_an_error_message(aiohttp_client, loop): | ||||
| class FormatEnum(str, Enum): | ||||
|     UTM = "UMT" | ||||
|     MGRS = "MGRS" | ||||
|  | ||||
|  | ||||
| class ViewWithEnumType(PydanticView): | ||||
|     async def get(self, *, format: FormatEnum): | ||||
|         return web.json_response({"format": format}, dumps=JSONEncoder().encode) | ||||
|  | ||||
|  | ||||
| async def test_get_article_without_required_header_should_return_an_error_message( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article', ArticleView) | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get('/article', headers={}) | ||||
|     resp = await client.get("/article", headers={}) | ||||
|     assert resp.status == 400 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == [{'loc': ['signature_expired'], | ||||
|                                   'msg': 'field required', | ||||
|                                   'type': 'value_error.missing'}] | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == [ | ||||
|         { | ||||
|             "in": "headers", | ||||
|             "loc": ["signature_expired"], | ||||
|             "msg": "field required", | ||||
|             "type": "value_error.missing", | ||||
|         } | ||||
|     ] | ||||
|  | ||||
|  | ||||
| async def test_get_article_with_wrong_header_type_should_return_an_error_message(aiohttp_client, loop): | ||||
| async def test_get_article_with_wrong_header_type_should_return_an_error_message( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article', ArticleView) | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get('/article', headers={'signature_expired': 'foo'}) | ||||
|     resp = await client.get("/article", headers={"signature_expired": "foo"}) | ||||
|     assert resp.status == 400 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == [{'loc': ['signature_expired'], | ||||
|                                   'msg': 'invalid datetime format', | ||||
|                                   'type': 'value_error.datetime'}] | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == [ | ||||
|         { | ||||
|             "in": "headers", | ||||
|             "loc": ["signature_expired"], | ||||
|             "msg": "invalid datetime format", | ||||
|             "type": "value_error.datetime", | ||||
|         } | ||||
|     ] | ||||
|  | ||||
|  | ||||
| async def test_get_article_with_valid_header_should_return_the_parsed_type(aiohttp_client, loop): | ||||
| async def test_get_article_with_valid_header_should_return_the_parsed_type( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article', ArticleView) | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get('/article', headers={'signature_expired': '2020-10-04T18:01:00'}) | ||||
|     resp = await client.get( | ||||
|         "/article", headers={"signature_expired": "2020-10-04T18:01:00"} | ||||
|     ) | ||||
|     assert resp.status == 200 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == {'signature': '2020-10-04T18:01:00'} | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == {"signature": "2020-10-04T18:01:00"} | ||||
|  | ||||
|  | ||||
| async def test_get_article_with_valid_header_containing_hyphen_should_be_returned(aiohttp_client, loop): | ||||
| async def test_get_article_with_valid_header_containing_hyphen_should_be_returned( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article', ArticleView) | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get('/article', headers={'Signature-Expired': '2020-10-04T18:01:00'}) | ||||
|     resp = await client.get( | ||||
|         "/article", headers={"Signature-Expired": "2020-10-04T18:01:00"} | ||||
|     ) | ||||
|     assert resp.status == 200 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == {'signature': '2020-10-04T18:01:00'} | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == {"signature": "2020-10-04T18:01:00"} | ||||
|  | ||||
|  | ||||
| async def test_wrong_value_to_header_defined_with_str_enum(aiohttp_client, loop): | ||||
|     app = web.Application() | ||||
|     app.router.add_view("/coord", ViewWithEnumType) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get("/coord", headers={"format": "WGS84"}) | ||||
|     assert ( | ||||
|         await resp.json() | ||||
|         == [ | ||||
|             { | ||||
|                 "ctx": {"enum_values": ["UMT", "MGRS"]}, | ||||
|                 "in": "headers", | ||||
|                 "loc": ["format"], | ||||
|                 "msg": "value is not a valid enumeration member; permitted: 'UMT', 'MGRS'", | ||||
|                 "type": "type_error.enum", | ||||
|             } | ||||
|         ] | ||||
|         != {"signature": "2020-10-04T18:01:00"} | ||||
|     ) | ||||
|     assert resp.status == 400 | ||||
|     assert resp.content_type == "application/json" | ||||
|  | ||||
|  | ||||
| async def test_correct_value_to_header_defined_with_str_enum(aiohttp_client, loop): | ||||
|     app = web.Application() | ||||
|     app.router.add_view("/coord", ViewWithEnumType) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get("/coord", headers={"format": "UMT"}) | ||||
|     assert await resp.json() == {"format": "UMT"} | ||||
|     assert resp.status == 200 | ||||
|     assert resp.content_type == "application/json" | ||||
|   | ||||
| @@ -1,20 +1,41 @@ | ||||
| from aiohttp import web | ||||
|  | ||||
| from aiohttp_pydantic import PydanticView | ||||
|  | ||||
|  | ||||
| class ArticleView(PydanticView): | ||||
|  | ||||
|     async def get(self, author_id: str, tag: str, date: int, /): | ||||
|         return web.json_response({'path': [author_id, tag, date]}) | ||||
|         return web.json_response({"path": [author_id, tag, date]}) | ||||
|  | ||||
|  | ||||
| async def test_get_article_without_required_qs_should_return_an_error_message(aiohttp_client, loop): | ||||
| async def test_get_article_with_correct_path_parameters_should_return_parameters_in_path( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article/{author_id}/tag/{tag}/before/{date}', ArticleView) | ||||
|     app.router.add_view("/article/{author_id}/tag/{tag}/before/{date}", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get('/article/1234/tag/music/before/1980') | ||||
|     resp = await client.get("/article/1234/tag/music/before/1980") | ||||
|     assert resp.status == 200 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == {'path': ['1234', 'music', 1980]} | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == {"path": ["1234", "music", 1980]} | ||||
|  | ||||
|  | ||||
| async def test_get_article_with_wrong_path_parameters_should_return_error( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view("/article/{author_id}/tag/{tag}/before/{date}", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get("/article/1234/tag/music/before/now") | ||||
|     assert resp.status == 400 | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == [ | ||||
|         { | ||||
|             "in": "path", | ||||
|             "loc": ["date"], | ||||
|             "msg": "value is not a valid integer", | ||||
|             "type": "type_error.integer", | ||||
|         } | ||||
|     ] | ||||
|   | ||||
| @@ -1,45 +1,82 @@ | ||||
| from typing import Optional | ||||
|  | ||||
| from aiohttp import web | ||||
|  | ||||
| from aiohttp_pydantic import PydanticView | ||||
|  | ||||
|  | ||||
| class ArticleView(PydanticView): | ||||
|  | ||||
|     async def get(self, with_comments: bool): | ||||
|         return web.json_response({'with_comments': with_comments}) | ||||
|     async def get( | ||||
|         self, with_comments: bool, age: Optional[int] = None, nb_items: int = 7 | ||||
|     ): | ||||
|         return web.json_response( | ||||
|             {"with_comments": with_comments, "age": age, "nb_items": nb_items} | ||||
|         ) | ||||
|  | ||||
|  | ||||
| async def test_get_article_without_required_qs_should_return_an_error_message(aiohttp_client, loop): | ||||
| async def test_get_article_without_required_qs_should_return_an_error_message( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article', ArticleView) | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get('/article') | ||||
|     resp = await client.get("/article") | ||||
|     assert resp.status == 400 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == [{'loc': ['with_comments'], | ||||
|                                   'msg': 'field required', | ||||
|                                   'type': 'value_error.missing'}] | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == [ | ||||
|         { | ||||
|             "in": "query string", | ||||
|             "loc": ["with_comments"], | ||||
|             "msg": "field required", | ||||
|             "type": "value_error.missing", | ||||
|         } | ||||
|     ] | ||||
|  | ||||
|  | ||||
| async def test_get_article_with_wrong_qs_type_should_return_an_error_message(aiohttp_client, loop): | ||||
| async def test_get_article_with_wrong_qs_type_should_return_an_error_message( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article', ArticleView) | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get('/article', params={'with_comments': 'foo'}) | ||||
|     resp = await client.get("/article", params={"with_comments": "foo"}) | ||||
|     assert resp.status == 400 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == [{'loc': ['with_comments'], | ||||
|                                   'msg': 'value could not be parsed to a boolean', | ||||
|                                   'type': 'type_error.bool'}] | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == [ | ||||
|         { | ||||
|             "in": "query string", | ||||
|             "loc": ["with_comments"], | ||||
|             "msg": "value could not be parsed to a boolean", | ||||
|             "type": "type_error.bool", | ||||
|         } | ||||
|     ] | ||||
|  | ||||
|  | ||||
| async def test_get_article_with_valid_qs_should_return_the_parsed_type(aiohttp_client, loop): | ||||
| async def test_get_article_with_valid_qs_should_return_the_parsed_type( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view('/article', ArticleView) | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|     resp = await client.get('/article', params={'with_comments': 'yes'}) | ||||
|  | ||||
|     resp = await client.get("/article", params={"with_comments": "yes", "age": 3}) | ||||
|     assert resp.status == 200 | ||||
|     assert resp.content_type == 'application/json' | ||||
|     assert await resp.json() == {'with_comments': True} | ||||
|     assert resp.content_type == "application/json" | ||||
|     assert await resp.json() == {"with_comments": True, "age": 3, "nb_items": 7} | ||||
|  | ||||
|  | ||||
| async def test_get_article_with_valid_qs_and_omitted_optional_should_return_default_value( | ||||
|     aiohttp_client, loop | ||||
| ): | ||||
|     app = web.Application() | ||||
|     app.router.add_view("/article", ArticleView) | ||||
|  | ||||
|     client = await aiohttp_client(app) | ||||
|  | ||||
|     resp = await client.get("/article", params={"with_comments": "yes"}) | ||||
|     assert await resp.json() == {"with_comments": True, "age": None, "nb_items": 7} | ||||
|     assert resp.status == 200 | ||||
|     assert resp.content_type == "application/json" | ||||
|   | ||||
		Reference in New Issue
	
	Block a user