Compare commits
	
		
			24 Commits
		
	
	
		
			v1.0.0
			...
			v1.5.1-fix
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | efbaaa5e6f | ||
|  | 6211c71875 | ||
|  | 5567d73952 | ||
|  | 67a95ec9c9 | ||
|  | 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/ | .idea/ | ||||||
| .pytest_cache | .pytest_cache | ||||||
| __pycache__ | __pycache__ | ||||||
| aiohttp_pydantic.egg-info/ | aiohttp_pydantic.egg-info/ | ||||||
| build/ | build/ | ||||||
|  | coverage.xml | ||||||
| dist/ | dist/ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,9 +2,14 @@ language: python | |||||||
| python: | python: | ||||||
| - '3.8' | - '3.8' | ||||||
| script: | script: | ||||||
| - pytest tests/ | - pytest --cov-report=xml --cov=aiohttp_pydantic tests/ | ||||||
| install: | 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: | deploy: | ||||||
|   provider: pypi |   provider: pypi | ||||||
|   username: __token__ |   username: __token__ | ||||||
|   | |||||||
							
								
								
									
										133
									
								
								README.rst
									
									
									
									
									
								
							
							
						
						
									
										133
									
								
								README.rst
									
									
									
									
									
								
							| @@ -1,6 +1,28 @@ | |||||||
| Aiohttp pydantic - Aiohttp View to validate and parse request | 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 | How to install | ||||||
| -------------- | -------------- | ||||||
|  |  | ||||||
| @@ -144,3 +166,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/ | .. _positional-only parameters: https://www.python.org/dev/peps/pep-0570/ | ||||||
| .. _pydantic Model: https://pydantic-docs.helpmanual.io/usage/models/ | .. _pydantic Model: https://pydantic-docs.helpmanual.io/usage/models/ | ||||||
| .. _keyword-only argument: https://www.python.org/dev/peps/pep-3102/ | .. _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 | from .view import PydanticView | ||||||
|  |  | ||||||
| __all__ = ("PydanticView",) | __version__ = "1.5.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 typing import Callable, Tuple | ||||||
|  |  | ||||||
|  | from aiohttp.web_exceptions import HTTPBadRequest | ||||||
| from aiohttp.web_request import BaseRequest | from aiohttp.web_request import BaseRequest | ||||||
| from pydantic import BaseModel | from pydantic import BaseModel | ||||||
| from inspect import signature |  | ||||||
|  |  | ||||||
|  | from .utils import is_pydantic_base_model | ||||||
| import abc |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AbstractInjector(metaclass=abc.ABCMeta): | class AbstractInjector(metaclass=abc.ABCMeta): | ||||||
| @@ -47,7 +49,13 @@ class BodyGetter(AbstractInjector): | |||||||
|         self.arg_name, self.model = next(iter(args_spec.items())) |         self.arg_name, self.model = next(iter(args_spec.items())) | ||||||
|  |  | ||||||
|     async def inject(self, request: BaseRequest, args_view: list, kwargs_view: dict): |     async def inject(self, request: BaseRequest, args_view: list, kwargs_view: dict): | ||||||
|         body = await request.json() |         try: | ||||||
|  |             body = await request.json() | ||||||
|  |         except JSONDecodeError: | ||||||
|  |             raise HTTPBadRequest( | ||||||
|  |                 text='{"error": "Malformed JSON"}', content_type="application/json" | ||||||
|  |             ) | ||||||
|  |  | ||||||
|         kwargs_view[self.arg_name] = self.model(**body) |         kwargs_view[self.arg_name] = self.model(**body) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -94,10 +102,13 @@ def _parse_func_signature(func: Callable) -> Tuple[dict, dict, dict, dict]: | |||||||
|         if param_name == "self": |         if param_name == "self": | ||||||
|             continue |             continue | ||||||
|  |  | ||||||
|  |         if param_spec.annotation == param_spec.empty: | ||||||
|  |             raise RuntimeError(f"The parameter {param_name} must have an annotation") | ||||||
|  |  | ||||||
|         if param_spec.kind is param_spec.POSITIONAL_ONLY: |         if param_spec.kind is param_spec.POSITIONAL_ONLY: | ||||||
|             path_args[param_name] = param_spec.annotation |             path_args[param_name] = param_spec.annotation | ||||||
|         elif param_spec.kind is param_spec.POSITIONAL_OR_KEYWORD: |         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 |                 body_args[param_name] = param_spec.annotation | ||||||
|             else: |             else: | ||||||
|                 qs_args[param_name] = param_spec.annotation |                 qs_args[param_name] = param_spec.annotation | ||||||
|   | |||||||
							
								
								
									
										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> | ||||||
							
								
								
									
										319
									
								
								aiohttp_pydantic/oas/struct.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										319
									
								
								aiohttp_pydantic/oas/struct.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,319 @@ | |||||||
|  | """ | ||||||
|  | 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 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 spec(self): | ||||||
|  |         return self._spec | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def definitions(self): | ||||||
|  |         self._spec.setdefault('definitions', {}) | ||||||
|  |         return self._spec['definitions'] | ||||||
							
								
								
									
										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, Optional, Type | ||||||
|  |  | ||||||
|  | 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") | ||||||
							
								
								
									
										189
									
								
								aiohttp_pydantic/oas/view.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										189
									
								
								aiohttp_pydantic/oas/view.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,189 @@ | |||||||
|  | import typing | ||||||
|  | from datetime import date, datetime | ||||||
|  | from inspect import getdoc | ||||||
|  | from itertools import count | ||||||
|  | from typing import List, Type | ||||||
|  | from uuid import UUID | ||||||
|  |  | ||||||
|  | from aiohttp.web import Response, json_response | ||||||
|  | from aiohttp.web_app import Application | ||||||
|  |  | ||||||
|  | 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 | ||||||
|  |  | ||||||
|  | JSON_SCHEMA_TYPES = { | ||||||
|  |     float: {"type": "number"}, | ||||||
|  |     str: {"type": "string"}, | ||||||
|  |     int: {"type": "integer"}, | ||||||
|  |     UUID: {"type": "string", "format": "uuid"}, | ||||||
|  |     bool: {"type": "boolean"}, | ||||||
|  |     datetime: {"type": "string", "format": "date-time"}, | ||||||
|  |     date: {"type": "string", "format": "date"}, | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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_operation, definitions): | ||||||
|  |         self._oas_operation = oas_operation | ||||||
|  |         self._definitions = definitions | ||||||
|  |  | ||||||
|  |     def _process_definitions(self, schema): | ||||||
|  |         if 'definitions' in schema: | ||||||
|  |             for k, v in schema['definitions'].items(): | ||||||
|  |                 self._definitions[k] = v | ||||||
|  |  | ||||||
|  |         return {i:schema[i] for i in schema if i!='definitions'} | ||||||
|  |  | ||||||
|  |     def _handle_pydantic_base_model(self, obj): | ||||||
|  |         if is_pydantic_base_model(obj): | ||||||
|  |             return self._process_definitions(obj.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_path: PathItem, http_method: str, view: Type[PydanticView], definitions: dict | ||||||
|  | ): | ||||||
|  |     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 = _parse_func_signature(handler) | ||||||
|  |     description = getdoc(handler) | ||||||
|  |     if description: | ||||||
|  |         oas_operation.description = description | ||||||
|  |  | ||||||
|  |     if body_args: | ||||||
|  |         oas_operation.request_body.content = { | ||||||
|  |             "application/json": {"schema": next(iter(body_args.values())).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_) | ||||||
|  |             if optional_type is None: | ||||||
|  |                 oas_operation.parameters[i].schema = JSON_SCHEMA_TYPES[type_] | ||||||
|  |                 oas_operation.parameters[i].required = True | ||||||
|  |             else: | ||||||
|  |                 oas_operation.parameters[i].schema = JSON_SCHEMA_TYPES[optional_type] | ||||||
|  |                 oas_operation.parameters[i].required = False | ||||||
|  |  | ||||||
|  |     return_type = handler.__annotations__.get("return") | ||||||
|  |     if return_type is not None: | ||||||
|  |         _OASResponseBuilder(oas_operation, definitions).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 is_pydantic_view(resource_route.handler): | ||||||
|  |                     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(path, method_name, view, oas.definitions) | ||||||
|  |                     else: | ||||||
|  |                         _add_http_method_to_oas(path, resource_route.method, view, oas.definitions) | ||||||
|  |  | ||||||
|  |     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() | ||||||
|  |  | ||||||
|  |     if request.scheme != request.headers.get('x-forwarded-proto', request.scheme): | ||||||
|  |         request = request.clone(scheme=request.headers['x-forwarded-proto']) | ||||||
|  |     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,21 +1,20 @@ | |||||||
|  | from functools import update_wrapper | ||||||
| from inspect import iscoroutinefunction | from inspect import iscoroutinefunction | ||||||
|  | from typing import Any, Callable, Generator, Iterable | ||||||
|  |  | ||||||
| from aiohttp.abc import AbstractView | from aiohttp.abc import AbstractView | ||||||
| from aiohttp.hdrs import METH_ALL | from aiohttp.hdrs import METH_ALL | ||||||
|  | from aiohttp.web import json_response | ||||||
| from aiohttp.web_exceptions import HTTPMethodNotAllowed | from aiohttp.web_exceptions import HTTPMethodNotAllowed | ||||||
| from aiohttp.web_response import StreamResponse | from aiohttp.web_response import StreamResponse | ||||||
| from pydantic import ValidationError | 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 ( | from .injectors import ( | ||||||
|     MatchInfoGetter, |  | ||||||
|     HeadersGetter, |  | ||||||
|     QueryGetter, |  | ||||||
|     BodyGetter, |  | ||||||
|     AbstractInjector, |     AbstractInjector, | ||||||
|  |     BodyGetter, | ||||||
|  |     HeadersGetter, | ||||||
|  |     MatchInfoGetter, | ||||||
|  |     QueryGetter, | ||||||
|     _parse_func_signature, |     _parse_func_signature, | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -34,21 +33,21 @@ class PydanticView(AbstractView): | |||||||
|         return self._iter().__await__() |         return self._iter().__await__() | ||||||
|  |  | ||||||
|     def __init_subclass__(cls, **kwargs): |     def __init_subclass__(cls, **kwargs): | ||||||
|         allowed_methods = { |         cls.allowed_methods = { | ||||||
|             meth_name for meth_name in METH_ALL if hasattr(cls, meth_name.lower()) |             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: |         for meth_name in METH_ALL: | ||||||
|             if meth_name not in allowed_methods: |             if meth_name not in cls.allowed_methods: | ||||||
|                 setattr(cls, meth_name.lower(), raise_not_allowed) |                 setattr(cls, meth_name.lower(), cls.raise_not_allowed) | ||||||
|             else: |             else: | ||||||
|                 handler = getattr(cls, meth_name.lower()) |                 handler = getattr(cls, meth_name.lower()) | ||||||
|                 decorated_handler = inject_params(handler, cls.parse_func_signature) |                 decorated_handler = inject_params(handler, cls.parse_func_signature) | ||||||
|                 setattr(cls, meth_name.lower(), decorated_handler) |                 setattr(cls, meth_name.lower(), decorated_handler) | ||||||
|  |  | ||||||
|  |     async def raise_not_allowed(self): | ||||||
|  |         raise HTTPMethodNotAllowed(self.request.method, self.allowed_methods) | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def parse_func_signature(func: Callable) -> Iterable[AbstractInjector]: |     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 = _parse_func_signature(func) | ||||||
| @@ -90,3 +89,13 @@ def inject_params( | |||||||
|  |  | ||||||
|     update_wrapper(wrapped_handler, handler) |     update_wrapper(wrapped_handler, handler) | ||||||
|     return wrapped_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) | ||||||
							
								
								
									
										48
									
								
								demo/model.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								demo/model.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | |||||||
|  | from pydantic import BaseModel | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Pet(BaseModel): | ||||||
|  |     id: int | ||||||
|  |     name: str | ||||||
|  |     age: int | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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 | ||||||
|  |     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 | from setuptools import setup | ||||||
|  |  | ||||||
|  | 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'] |  | ||||||
| ) |  | ||||||
|   | |||||||
| @@ -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 | ||||||
							
								
								
									
										120
									
								
								tests/test_oas/test_cmd/test_cmd.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								tests/test_oas/test_cmd/test_cmd.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,120 @@ | |||||||
|  | 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": { | ||||||
|  |                                 "type": "integer" | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     ] | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "/sub-app/route-2/{b}": { | ||||||
|  |                 "post": { | ||||||
|  |                     "parameters": [ | ||||||
|  |                         { | ||||||
|  |                             "in": "path", | ||||||
|  |                             "name": "b", | ||||||
|  |                             "required": true, | ||||||
|  |                             "schema": { | ||||||
|  |                                 "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": { | ||||||
|  |                                 "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": { | ||||||
|  |                                 "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() | ||||||
							
								
								
									
										218
									
								
								tests/test_oas/test_view.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										218
									
								
								tests/test_oas/test_view.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,218 @@ | |||||||
|  | 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 Pet(BaseModel): | ||||||
|  |     id: int | ||||||
|  |     name: str | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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_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": {"type": "string"}, | ||||||
|  |             }, | ||||||
|  |             { | ||||||
|  |                 "in": "query", | ||||||
|  |                 "name": "name", | ||||||
|  |                 "required": False, | ||||||
|  |                 "schema": {"type": "string"}, | ||||||
|  |             }, | ||||||
|  |             { | ||||||
|  |                 "in": "header", | ||||||
|  |                 "name": "promo", | ||||||
|  |                 "required": False, | ||||||
|  |                 "schema": {"format": "uuid", "type": "string"}, | ||||||
|  |             }, | ||||||
|  |         ], | ||||||
|  |         "responses": { | ||||||
|  |             "200": { | ||||||
|  |                 "content": { | ||||||
|  |                     "application/json": { | ||||||
|  |                         "schema": { | ||||||
|  |                             "items": { | ||||||
|  |                                 "properties": { | ||||||
|  |                                     "id": {"title": "Id", "type": "integer"}, | ||||||
|  |                                     "name": {"title": "Name", "type": "string"}, | ||||||
|  |                                 }, | ||||||
|  |                                 "required": ["id", "name"], | ||||||
|  |                                 "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": { | ||||||
|  |                         "title": "Pet", | ||||||
|  |                         "type": "object", | ||||||
|  |                         "properties": { | ||||||
|  |                             "id": {"title": "Id", "type": "integer"}, | ||||||
|  |                             "name": {"title": "Name", "type": "string"}, | ||||||
|  |                         }, | ||||||
|  |                         "required": ["id", "name"], | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |         "responses": { | ||||||
|  |             "201": { | ||||||
|  |                 "content": { | ||||||
|  |                     "application/json": { | ||||||
|  |                         "schema": { | ||||||
|  |                             "title": "Pet", | ||||||
|  |                             "type": "object", | ||||||
|  |                             "properties": { | ||||||
|  |                                 "id": {"title": "Id", "type": "integer"}, | ||||||
|  |                                 "name": {"title": "Name", "type": "string"}, | ||||||
|  |                             }, | ||||||
|  |                             "required": ["id", "name"], | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 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": {"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": {"type": "integer"}, | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         "responses": { | ||||||
|  |             "200": { | ||||||
|  |                 "content": { | ||||||
|  |                     "application/json": { | ||||||
|  |                         "schema": { | ||||||
|  |                             "properties": { | ||||||
|  |                                 "id": {"title": "Id", "type": "integer"}, | ||||||
|  |                                 "name": {"title": "Name", "type": "string"}, | ||||||
|  |                             }, | ||||||
|  |                             "required": ["id", "name"], | ||||||
|  |                             "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": {"type": "integer"}, | ||||||
|  |             } | ||||||
|  |         ], | ||||||
|  |         "requestBody": { | ||||||
|  |             "content": { | ||||||
|  |                 "application/json": { | ||||||
|  |                     "schema": { | ||||||
|  |                         "properties": { | ||||||
|  |                             "id": {"title": "Id", "type": "integer"}, | ||||||
|  |                             "name": {"title": "Name", "type": "string"}, | ||||||
|  |                         }, | ||||||
|  |                         "required": ["id", "name"], | ||||||
|  |                         "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 uuid import UUID | ||||||
|  |  | ||||||
|  | from pydantic import BaseModel | ||||||
|  |  | ||||||
|  | from aiohttp_pydantic.injectors import _parse_func_signature | ||||||
|  |  | ||||||
|  |  | ||||||
| class User(BaseModel): | class User(BaseModel): | ||||||
|     firstname: str |     firstname: str | ||||||
| @@ -9,7 +11,6 @@ class User(BaseModel): | |||||||
|  |  | ||||||
|  |  | ||||||
| def test_parse_func_signature(): | def test_parse_func_signature(): | ||||||
|  |  | ||||||
|     def body_only(self, user: User): |     def body_only(self, user: User): | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
| @@ -37,13 +38,32 @@ def test_parse_func_signature(): | |||||||
|     def path_body_qs_and_header(self, id: str, /, user: User, page: int, *, auth: UUID): |     def path_body_qs_and_header(self, id: str, /, user: User, page: int, *, auth: UUID): | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     assert _parse_func_signature(body_only) == ({}, {'user': User}, {},  {}) |     assert _parse_func_signature(body_only) == ({}, {"user": User}, {}, {}) | ||||||
|     assert _parse_func_signature(path_only) == ({'id': str}, {}, {},  {}) |     assert _parse_func_signature(path_only) == ({"id": str}, {}, {}, {}) | ||||||
|     assert _parse_func_signature(qs_only) == ({}, {}, {'page': int},  {}) |     assert _parse_func_signature(qs_only) == ({}, {}, {"page": int}, {}) | ||||||
|     assert _parse_func_signature(header_only) == ({}, {}, {},  {'auth': UUID}) |     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_qs) == ({"id": str}, {}, {"page": int}, {}) | ||||||
|     assert _parse_func_signature(path_and_header) == ({'id': str}, {}, {},  {'auth': UUID}) |     assert _parse_func_signature(path_and_header) == ( | ||||||
|     assert _parse_func_signature(qs_and_header) == ({}, {}, {'page': int},  {'auth': UUID}) |         {"id": str}, | ||||||
|     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}) |         {}, | ||||||
|  |         {"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 typing import Optional | ||||||
|  |  | ||||||
| from aiohttp import web | from aiohttp import web | ||||||
|  | from pydantic import BaseModel | ||||||
|  |  | ||||||
| from aiohttp_pydantic import PydanticView | from aiohttp_pydantic import PydanticView | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -10,43 +12,50 @@ class ArticleModel(BaseModel): | |||||||
|  |  | ||||||
|  |  | ||||||
| class ArticleView(PydanticView): | class ArticleView(PydanticView): | ||||||
|  |  | ||||||
|     async def post(self, article: ArticleModel): |     async def post(self, article: ArticleModel): | ||||||
|         return web.json_response(article.dict()) |         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 = web.Application() | ||||||
|     app.router.add_view('/article', ArticleView) |     app.router.add_view("/article", ArticleView) | ||||||
|  |  | ||||||
|     client = await aiohttp_client(app) |     client = await aiohttp_client(app) | ||||||
|     resp = await client.post('/article', json={}) |     resp = await client.post("/article", json={}) | ||||||
|     assert resp.status == 400 |     assert resp.status == 400 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == [{'loc': ['name'], |     assert await resp.json() == [ | ||||||
|                                   'msg': 'field required', |         {"loc": ["name"], "msg": "field required", "type": "value_error.missing"} | ||||||
|                                   '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 = web.Application() | ||||||
|     app.router.add_view('/article', ArticleView) |     app.router.add_view("/article", ArticleView) | ||||||
|  |  | ||||||
|     client = await aiohttp_client(app) |     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.status == 400 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == [{'loc': ['nb_page'], |     assert await resp.json() == [ | ||||||
|                                   'msg': 'value is not a valid integer', |         { | ||||||
|                                   'type': 'type_error.integer'}] |             "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): | async def test_post_a_valid_article_should_return_the_parsed_type(aiohttp_client, loop): | ||||||
|     app = web.Application() |     app = web.Application() | ||||||
|     app.router.add_view('/article', ArticleView) |     app.router.add_view("/article", ArticleView) | ||||||
|  |  | ||||||
|     client = await aiohttp_client(app) |     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.status == 200 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == {'name': 'foo', 'nb_page': 3} |     assert await resp.json() == {"name": "foo", "nb_page": 3} | ||||||
|   | |||||||
| @@ -1,11 +1,12 @@ | |||||||
| from aiohttp import web |  | ||||||
| from aiohttp_pydantic import PydanticView |  | ||||||
| from datetime import datetime |  | ||||||
| import json | import json | ||||||
|  | from datetime import datetime | ||||||
|  |  | ||||||
|  | from aiohttp import web | ||||||
|  |  | ||||||
|  | from aiohttp_pydantic import PydanticView | ||||||
|  |  | ||||||
|  |  | ||||||
| class JSONEncoder(json.JSONEncoder): | class JSONEncoder(json.JSONEncoder): | ||||||
|  |  | ||||||
|     def default(self, o): |     def default(self, o): | ||||||
|         if isinstance(o, datetime): |         if isinstance(o, datetime): | ||||||
|             return o.isoformat() |             return o.isoformat() | ||||||
| @@ -14,54 +15,75 @@ class JSONEncoder(json.JSONEncoder): | |||||||
|  |  | ||||||
|  |  | ||||||
| class ArticleView(PydanticView): | class ArticleView(PydanticView): | ||||||
|  |  | ||||||
|     async def get(self, *, signature_expired: datetime): |     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): | async def test_get_article_without_required_header_should_return_an_error_message( | ||||||
|  |     aiohttp_client, loop | ||||||
|  | ): | ||||||
|     app = web.Application() |     app = web.Application() | ||||||
|     app.router.add_view('/article', ArticleView) |     app.router.add_view("/article", ArticleView) | ||||||
|  |  | ||||||
|     client = await aiohttp_client(app) |     client = await aiohttp_client(app) | ||||||
|     resp = await client.get('/article', headers={}) |     resp = await client.get("/article", headers={}) | ||||||
|     assert resp.status == 400 |     assert resp.status == 400 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == [{'loc': ['signature_expired'], |     assert await resp.json() == [ | ||||||
|                                   'msg': 'field required', |         { | ||||||
|                                   'type': 'value_error.missing'}] |             "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 = web.Application() | ||||||
|     app.router.add_view('/article', ArticleView) |     app.router.add_view("/article", ArticleView) | ||||||
|  |  | ||||||
|     client = await aiohttp_client(app) |     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.status == 400 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == [{'loc': ['signature_expired'], |     assert await resp.json() == [ | ||||||
|                                   'msg': 'invalid datetime format', |         { | ||||||
|                                   'type': 'value_error.datetime'}] |             "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 = web.Application() | ||||||
|     app.router.add_view('/article', ArticleView) |     app.router.add_view("/article", ArticleView) | ||||||
|  |  | ||||||
|     client = await aiohttp_client(app) |     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.status == 200 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == {'signature': '2020-10-04T18:01:00'} |     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 = web.Application() | ||||||
|     app.router.add_view('/article', ArticleView) |     app.router.add_view("/article", ArticleView) | ||||||
|  |  | ||||||
|     client = await aiohttp_client(app) |     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.status == 200 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == {'signature': '2020-10-04T18:01:00'} |     assert await resp.json() == {"signature": "2020-10-04T18:01:00"} | ||||||
|   | |||||||
| @@ -1,20 +1,21 @@ | |||||||
| from aiohttp import web | from aiohttp import web | ||||||
|  |  | ||||||
| from aiohttp_pydantic import PydanticView | from aiohttp_pydantic import PydanticView | ||||||
|  |  | ||||||
|  |  | ||||||
| class ArticleView(PydanticView): | class ArticleView(PydanticView): | ||||||
|  |     async def get(self, author_id: str, tag: str, date: int, /): | ||||||
|     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_without_required_qs_should_return_an_error_message( | ||||||
|  |     aiohttp_client, loop | ||||||
|  | ): | ||||||
|     app = web.Application() |     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) |     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.status == 200 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == {'path': ['1234', 'music', 1980]} |     assert await resp.json() == {"path": ["1234", "music", 1980]} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,45 +1,76 @@ | |||||||
|  | from typing import Optional | ||||||
|  |  | ||||||
| from aiohttp import web | from aiohttp import web | ||||||
|  |  | ||||||
| from aiohttp_pydantic import PydanticView | from aiohttp_pydantic import PydanticView | ||||||
|  |  | ||||||
|  |  | ||||||
| class ArticleView(PydanticView): | class ArticleView(PydanticView): | ||||||
|  |     async def get(self, with_comments: bool, age: Optional[int] = None): | ||||||
|     async def get(self, with_comments: bool): |         return web.json_response({"with_comments": with_comments, "age": age}) | ||||||
|         return web.json_response({'with_comments': with_comments}) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| 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 = web.Application() | ||||||
|     app.router.add_view('/article', ArticleView) |     app.router.add_view("/article", ArticleView) | ||||||
|  |  | ||||||
|     client = await aiohttp_client(app) |     client = await aiohttp_client(app) | ||||||
|     resp = await client.get('/article') |     resp = await client.get("/article") | ||||||
|     assert resp.status == 400 |     assert resp.status == 400 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == [{'loc': ['with_comments'], |     assert await resp.json() == [ | ||||||
|                                   'msg': 'field required', |         { | ||||||
|                                   'type': 'value_error.missing'}] |             "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 = web.Application() | ||||||
|     app.router.add_view('/article', ArticleView) |     app.router.add_view("/article", ArticleView) | ||||||
|  |  | ||||||
|     client = await aiohttp_client(app) |     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.status == 400 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == [{'loc': ['with_comments'], |     assert await resp.json() == [ | ||||||
|                                   'msg': 'value could not be parsed to a boolean', |         { | ||||||
|                                   'type': 'type_error.bool'}] |             "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 = web.Application() | ||||||
|     app.router.add_view('/article', ArticleView) |     app.router.add_view("/article", ArticleView) | ||||||
|  |  | ||||||
|     client = await aiohttp_client(app) |     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.status == 200 | ||||||
|     assert resp.content_type == 'application/json' |     assert resp.content_type == "application/json" | ||||||
|     assert await resp.json() == {'with_comments': True} |     assert await resp.json() == {"with_comments": True, "age": 3} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def test_get_article_with_valid_qs_and_omitted_optional_should_return_none( | ||||||
|  |     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 resp.status == 200 | ||||||
|  |     assert resp.content_type == "application/json" | ||||||
|  |     assert await resp.json() == {"with_comments": True, "age": None} | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user