Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Release v0.9.0 #87

Merged
merged 25 commits into from
Sep 17, 2019
Merged
Show file tree
Hide file tree
Changes from 24 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
bd69db4
Parse extended operation description
janbuchar May 28, 2019
f66ed90
Merge pull request #73 from iterait/operation-descriptions
May 29, 2019
e020e8c
Prototype of new error handler API
janbuchar May 29, 2019
812b7b9
Handle forward references in a forward-compatible way
janbuchar Jun 27, 2019
29e254b
Add type annotations and doc
janbuchar Jul 29, 2019
0c11838
Merge branch 'better-annotation-processing' into exception-documentation
janbuchar Jul 30, 2019
7ac9811
Documentation of error responses inferred from :raises in doc blocks
janbuchar Aug 7, 2019
8d21355
Merge pull request #75 from iterait/better-annotation-processing
Aug 13, 2019
32bb7aa
Merge branch 'dev' into exception-documentation
Aug 13, 2019
5b6e4cf
Merge pull request #78 from iterait/exception-documentation
Aug 13, 2019
3bdc937
Add test for AioHTTP handler without request parameter
janbuchar Aug 13, 2019
cca3686
Refactor decorator handling
janbuchar Aug 15, 2019
34f7687
Add query string parameter tests
janbuchar Aug 15, 2019
c75ee87
Remove dead code
janbuchar Aug 15, 2019
d76d609
Add test of security spec generation
janbuchar Aug 26, 2019
31a915c
Add test of security enforcement
janbuchar Aug 26, 2019
29bbc7e
Add nocover where it's reasonable
janbuchar Aug 26, 2019
816d23c
Improve test coverage
janbuchar Aug 26, 2019
764f591
Fill in missing documentation
janbuchar Sep 17, 2019
a7eda6d
Add missing newlines
janbuchar Sep 17, 2019
60625b1
Code format
janbuchar Sep 17, 2019
cad31b1
Merge pull request #82 from iterait/decorators-refactor
Sep 17, 2019
9293388
Bump version
janbuchar Sep 17, 2019
8078952
Merge pull request #86 from iterait/version-bump
petrbel Sep 17, 2019
c3c9d43
Merge branch 'master' into dev
petrbel Sep 17, 2019
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion apistrap/__init__.py
Original file line number Diff line number Diff line change
@@ -1 +1 @@
__version__ = "0.6.0"
__version__ = "0.9.0"
497 changes: 208 additions & 289 deletions apistrap/aiohttp.py

Large diffs are not rendered by default.

290 changes: 31 additions & 259 deletions apistrap/decorators.py
Original file line number Diff line number Diff line change
@@ -1,307 +1,79 @@
from __future__ import annotations

import abc
import inspect
from functools import wraps
from typing import TYPE_CHECKING, Callable, Optional, Sequence, Type, Union
from dataclasses import dataclass
from typing import Optional, Sequence, Type, Union

from schematics import Model
from schematics.exceptions import DataError

from apistrap.errors import ApiClientError, InvalidFieldsError
from apistrap.examples import ExamplesMixin, model_examples_to_openapi_dict
from apistrap.schematics_converters import schematics_model_to_schema_object
from apistrap.tags import TagData
from apistrap.types import FileResponse

if TYPE_CHECKING:
from apistrap.extension import Apistrap


def _ensure_specs_dict(func: Callable):
if not hasattr(func, "specs_dict"):
func.specs_dict = {"parameters": [], "responses": {}}


def _add_ignored_param(func: Callable, arg: str):
if not hasattr(func, "_ignored_params"):
setattr(func, "_ignored_params", [])

func._ignored_params.append(arg)


def _get_wrapped_function(func: Callable):
class IgnoreDecorator:
"""
Get the actual function from a decorated function. This could end up in a loop on horribly mangled functions.
Marks an endpoint as ignored so that Apistrap doesn't include it in the specification.
"""

wrapped = getattr(func, "__wrapped__", None)

if wrapped is None:
return func

return _get_wrapped_function(wrapped)


@dataclass(frozen=True)
class IgnoreParamsDecorator:
"""
A decorator that marks specified function parameters as ignored for the purposes of generating a specification
Marks specified function parameters as ignored for the purposes of generating a specification
"""

def __init__(self, ignored_params: Sequence[str]):
self._ignored_params = ignored_params

def __call__(self, wrapped_func):
for param in self._ignored_params:
_add_ignored_param(wrapped_func, param)
return wrapped_func
ignored_params: Sequence[str]


@dataclass(frozen=True)
class RespondsWithDecorator:
"""
A decorator that fills in response schemas in the Swagger specification. It also converts Schematics models returned
by view functions to JSON and validates them.
"""

outermost_decorators = {}
"""
Maps functions to the outermost RespondsWithDecorator so that we can perform checks for unknown response classes
when we get to the last decorator
Specifies the format of the response. The response is automatically validated by Apistrap.
"""

def __init__(
self,
apistrap: Apistrap,
response_class: Type[Model],
*,
code: int = 200,
description: Optional[str] = None,
mimetype: Optional[str] = None,
):
self._response_class = response_class
self._code = code
self._apistrap = apistrap
self._description = description or self._response_class.__name__
self._mimetype = mimetype

def __call__(self, wrapped_func: Callable):
_ensure_specs_dict(wrapped_func)

if self._response_class == FileResponse:
wrapped_func.specs_dict["responses"][str(self._code)] = {
"description": self._description or self._response_class.__name__,
"content": {
self._mimetype or "application/octet-stream": {"schema": {"type": "string", "format": "binary"}}
},
}
else:
wrapped_func.specs_dict["responses"][str(self._code)] = {
"description": self._description or self._response_class.__name__,
"content": {"application/json": {"schema": self._get_schema_object()}},
}

if issubclass(self._response_class, ExamplesMixin):
# fmt: off
wrapped_func.specs_dict["responses"][str(self._code)]["content"]["application/json"]["examples"] = \
model_examples_to_openapi_dict(self._response_class)
# fmt: on

innermost_func = _get_wrapped_function(wrapped_func)
self.outermost_decorators[innermost_func] = self

if inspect.iscoroutinefunction(wrapped_func):

@wraps(wrapped_func)
async def wrapper(*args, **kwargs):
response = await wrapped_func(*args, **kwargs)
is_last_decorator = self.outermost_decorators[innermost_func] == self
return await self._process_response(response, is_last_decorator, *args, **kwargs)
response_class: Type[Model]
code: int = 200
description: Optional[str] = None
mimetype: Optional[str] = None

else:

@wraps(wrapped_func)
def wrapper(*args, **kwargs):
response = wrapped_func(*args, **kwargs)
is_last_decorator = self.outermost_decorators[innermost_func] == self
return self._process_response(response, is_last_decorator)

return wrapper

def _get_schema_object(self):
return schematics_model_to_schema_object(self._response_class, self._apistrap)

def _process_response(self, response, is_last_decorator: bool, *args, **kwargs):
"""
Process a response received from an endpoint handler (i.e. send it)
:param response: the response to be processed
:param is_last_decorator: True if the current decorator is the outermost one
"""


class AcceptsDecorator(metaclass=abc.ABCMeta):
@dataclass(frozen=True)
class AcceptsDecorator:
"""
A decorator that validates request bodies against a schema and passes it as an argument to the view function.
The destination argument must be annotated with the request type.
Specifies the format of the request body and injects it as an argument to the view handler.
The destination parameter must be annotated with a corresponding type.
"""

def __init__(self, apistrap: Apistrap, request_class: Type[Model]):
self._apistrap = apistrap
self._request_class = request_class

def _find_parameter_by_request_class(self, signature: inspect.Signature) -> Optional[inspect.Parameter]:
for parameter in signature.parameters.values():
if isinstance(parameter.annotation, str):
if parameter.annotation == self._request_class.__qualname__:
return parameter
elif isinstance(parameter.annotation, type):
if issubclass(self._request_class, parameter.annotation):
return parameter
return None

def __call__(self, wrapped_func: Callable):
_ensure_specs_dict(wrapped_func)

# TODO parse title from param in docblock
wrapped_func.specs_dict["requestBody"] = {
"content": {
"application/json": {"schema": schematics_model_to_schema_object(self._request_class, self._apistrap)}
},
"required": True,
}

if issubclass(self._request_class, ExamplesMixin):
# fmt: off
wrapped_func.specs_dict["requestBody"]["content"]["application/json"]["examples"] = \
model_examples_to_openapi_dict(self._request_class)
# fmt: on

wrapped_func.specs_dict["x-codegen-request-body-name"] = "body"

signature = inspect.signature(wrapped_func)
request_arg = self._find_parameter_by_request_class(signature)

if request_arg is None:
raise TypeError(f"no argument of type `{self._request_class}` found")

if inspect.iscoroutinefunction(wrapped_func):

@wraps(wrapped_func)
async def wrapper(*args, **kwargs):
self._check_request_type(*args, **kwargs)
body = await self._get_request_json(*args, **kwargs)
kwargs = self._process_request_kwargs(body, signature, request_arg, *args, **kwargs)
return await wrapped_func(*args, **kwargs)

else:

@wraps(wrapped_func)
def wrapper(*args, **kwargs):
self._check_request_type(*args, **kwargs)
body = self._get_request_json()
kwargs = self._process_request_kwargs(body, signature, request_arg, *args, **kwargs)
return wrapped_func(*args, **kwargs)

_add_ignored_param(wrapper, request_arg.name)
return wrapper

def _check_request_type(self, *args, **kwargs):
if self._get_request_content_type(*args, **kwargs) != "application/json":
raise ApiClientError("Unsupported media type, JSON is expected")

def _process_request_kwargs(self, body, signature, request_arg, *args, **kwargs):
bound_args = signature.bind_partial(*args, **kwargs)
if request_arg.name not in bound_args.arguments:
request_object = self._request_class.__new__(self._request_class)

try:
request_object.__init__(body, validate=True, partial=False, strict=True)
except DataError as e:
raise InvalidFieldsError(e.errors) from e

new_kwargs = {request_arg.name: request_object}
new_kwargs.update(**kwargs)
return new_kwargs

@abc.abstractmethod
def _get_request_content_type(self, *args, **kwargs) -> str:
"""
Get the value of the Content-Type header of current request
"""

@abc.abstractmethod
def _get_request_json(self, *args, **kwargs):
"""
Get the JSON content of the request
"""
request_class: Type[Model]


@dataclass(frozen=True)
class AcceptsFileDecorator:
"""
A decorator used to declare that an endpoint accepts a file upload in the request body.
Declares that an endpoint accepts a file upload as the request body.
"""

def __init__(self, mime_type: str = None):
self.mime_type = mime_type or "application/octet-stream"

def __call__(self, wrapped_func: Callable):
_ensure_specs_dict(wrapped_func)

wrapped_func.specs_dict["requestBody"] = {
"content": {self.mime_type: {"schema": {"type": "string", "format": "binary"}}},
"required": True,
}
mime_type: str

return wrapped_func


class IgnoreDecorator:
@dataclass(frozen=True)
class AcceptsQueryStringDecorator:
"""
A decorator that marks an endpoint as ignored so that the extension won't include it in the specification.
Declares that an endpoint accepts query string parameters.
"""

def __call__(self, wrapped_func: Callable):
wrapped_func.apistrap_ignore = True
return wrapped_func
parameter_names: Sequence[str]


@dataclass(frozen=True)
class TagsDecorator:
"""
A decorator that adds tags to the OpenAPI specification of the decorated view function.
Adds tags to the OpenAPI specification of the decorated view function.
"""

def __init__(self, extension: Apistrap, tags: Sequence[Union[str, TagData]]):
self._tags = tags
self._extension = extension

def __call__(self, wrapped_func: Callable):
_ensure_specs_dict(wrapped_func)
wrapped_func.specs_dict.setdefault("tags", [])

for tag in self._tags:
wrapped_func.specs_dict["tags"].append(tag.name if isinstance(tag, TagData) else tag)

if isinstance(tag, TagData):
self._extension.add_tag_data(tag)

return wrapped_func
tags: Sequence[Union[str, TagData]]


@dataclass(frozen=True)
class SecurityDecorator:
"""
A decorator that enforces user authentication and authorization.
Enforces user authentication and authorization.
"""

def __init__(self, extension: Apistrap, scopes: Sequence[str]):
self._extension = extension
self._scopes = scopes

def __call__(self, wrapped_func: Callable):
_ensure_specs_dict(wrapped_func)
wrapped_func.specs_dict.setdefault("security", [])

for scheme in self._extension.security_schemes:
wrapped_func.specs_dict["security"].append({scheme.name: [*map(str, self._scopes)]})

wrapped_func = scheme.enforcer(self._scopes)(wrapped_func)

return wrapped_func
scopes: Sequence[str]
8 changes: 6 additions & 2 deletions apistrap/errors.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,8 +34,12 @@ class UnexpectedResponseError(ApiServerError):
An exception raised when a view function returns a response of an unexpected type
"""

def __init__(self, response_class: type):
super().__init__(f"Unexpected response class: `{response_class.__name__}`")
def __init__(self, response_class: type, code: int = None):
msg = f"Unexpected response class: `{response_class.__name__}`"
if code is not None:
msg += f" (status code {code})"

super().__init__(msg)


class InvalidResponseError(ApiServerError):
Expand Down
3 changes: 2 additions & 1 deletion apistrap/examples.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,8 @@ def get_examples(cls: __class__) -> List[ModelExample[__class__]]:
:return: a list of example objects of this class
"""

raise NotImplementedError() # Using abc would lead to metaclass conflicts with Schematics
# Using abc would lead to metaclass conflicts with Schematics
raise NotImplementedError() # pragma: no cover


def model_examples_to_openapi_dict(model: Type[ExamplesMixin]) -> Dict[str, Any]:
Expand Down
Loading