Skip to content

Commit 07d59c0

Browse files
committed
AWS API Gateway with Amazon Lambda integrations support
1 parent 0898d87 commit 07d59c0

18 files changed

+789
-128
lines changed

docs/integrations.rst

+42
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,48 @@ Integrations
33

44
Openapi-core integrates with your popular libraries and frameworks. Each integration offers different levels of integration that help validate and unmarshal your request and response data.
55

6+
Amazon API Gateway
7+
------------------
8+
9+
This section describes integration with `Amazon API Gateway <https://aws.amazon.com/api-gateway/>`__. It is useful for `AWS Lambda integrations <https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-develop-integrations-lambda.html>`__ where functions handle events from API Gateway.
10+
11+
Low level
12+
~~~~~~~~~
13+
14+
You can use ``APIGatewayEventOpenAPIRequest`` as an API Gateway event request factory:
15+
16+
.. code-block:: python
17+
18+
from openapi_core import unmarshal_request
19+
from openapi_core.contrib.aws import APIGatewayEventOpenAPIRequest
20+
21+
openapi_request = APIGatewayEventOpenAPIRequest(event)
22+
result = unmarshal_request(openapi_request, spec=spec)
23+
24+
You can use ``APIGatewayEventResponseOpenAPIResponse`` as an API Gateway (HTTP API) event response factory:
25+
26+
.. code-block:: python
27+
28+
from openapi_core import unmarshal_response
29+
from openapi_core.contrib.aws import APIGatewayEventResponseOpenAPIResponse
30+
31+
openapi_response = APIGatewayEventResponseOpenAPIResponse(response)
32+
result = unmarshal_response(openapi_request, openapi_response, spec=spec)
33+
34+
API Gateway have special ``ANY`` method that catches all HTTP methods. It's specified as `x-amazon-apigateway-any-method <https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-swagger-extensions-any-method.html>__` OpenAPI extension. If you use the extension, you want to define ``path_finder_cls`` to be ``APIGatewayPathFinder``:
35+
36+
.. code-block:: python
37+
38+
from openapi_core.contrib.aws import APIGatewayPathFinder
39+
40+
result = unmarshal_response(
41+
openapi_request,
42+
openapi_response,
43+
spec=spec,
44+
path_finder_cls=APIGatewayPathFinder,
45+
)
46+
47+
648
Bottle
749
------
850

openapi_core/contrib/aws/__init__.py

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
"""OpenAPI core contrib aws module"""
2+
from openapi_core.contrib.aws.finders import APIGatewayPathFinder
3+
from openapi_core.contrib.aws.requests import APIGatewayEventOpenAPIRequest
4+
from openapi_core.contrib.aws.responses import (
5+
APIGatewayEventResponseOpenAPIResponse,
6+
)
7+
8+
__all__ = [
9+
"APIGatewayEventOpenAPIRequest",
10+
"APIGatewayEventResponseOpenAPIResponse",
11+
"APIGatewayPathFinder",
12+
]

openapi_core/contrib/aws/datatypes.py

+61
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
from typing import Dict
2+
from typing import List
3+
from typing import Optional
4+
5+
from pydantic.dataclasses import dataclass
6+
7+
8+
class APIGatewayEventConfig:
9+
extra = "allow"
10+
11+
12+
@dataclass(config=APIGatewayEventConfig, kw_only=True)
13+
class BaseAPIGatewayEvent:
14+
path: str
15+
httpMethod: str
16+
headers: Dict[str, str]
17+
queryStringParameters: Optional[Dict[str, str]] = None
18+
isBase64Encoded: Optional[bool] = None
19+
body: Optional[str] = None
20+
pathParameters: Optional[Dict[str, str]] = None
21+
stageVariables: Optional[Dict[str, str]] = None
22+
23+
24+
@dataclass(config=APIGatewayEventConfig, kw_only=True)
25+
class APIGatewayEvent(BaseAPIGatewayEvent):
26+
"""AWS API Gateway event"""
27+
28+
resource: str
29+
multiValueHeaders: Dict[str, List[str]]
30+
version: Optional[str] = "1.0"
31+
multiValueQueryStringParameters: Optional[Dict[str, List[str]]] = None
32+
33+
34+
@dataclass(config=APIGatewayEventConfig, kw_only=True)
35+
class APIGatewayEventV2(BaseAPIGatewayEvent):
36+
"""AWS API Gateway event v2"""
37+
38+
version: str
39+
routeKey: str
40+
rawPath: str
41+
rawQueryString: str
42+
cookies: Optional[List[str]] = None
43+
44+
45+
@dataclass(config=APIGatewayEventConfig, kw_only=True)
46+
class APIGatewayEventResponse:
47+
"""AWS API Gateway event response"""
48+
49+
isBase64Encoded: bool
50+
statusCode: int
51+
headers: Dict[str, str]
52+
multiValueHeaders: Dict[str, List[str]]
53+
body: str
54+
55+
56+
@dataclass(config=APIGatewayEventConfig, kw_only=True)
57+
class APIGatewayEventV2Response:
58+
"""AWS API Gateway event v2 response"""
59+
60+
isBase64Encoded: bool = False
61+
statusCode: int = 200

openapi_core/contrib/aws/finders.py

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
from openapi_core.templating.paths.finders import APICallPathFinder
2+
from openapi_core.templating.paths.iterators import AnyMethodOperationsIterator
3+
4+
5+
class APIGatewayPathFinder(APICallPathFinder):
6+
operations_iterator = AnyMethodOperationsIterator(
7+
any_method="x-amazon-apigateway-any-method",
8+
)

openapi_core/contrib/aws/requests.py

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
from typing import Optional
2+
3+
from werkzeug.datastructures import Headers
4+
from werkzeug.datastructures import ImmutableMultiDict
5+
6+
from openapi_core.contrib.aws.datatypes import APIGatewayEvent
7+
from openapi_core.contrib.aws.typing import APIGatewayEventDict
8+
from openapi_core.datatypes import RequestParameters
9+
10+
11+
class APIGatewayEventOpenAPIRequest:
12+
"""
13+
Converts an API Gateway event to an OpenAPI request
14+
"""
15+
16+
def __init__(self, event: APIGatewayEventDict):
17+
self.event = APIGatewayEvent(**event)
18+
19+
self.parameters = RequestParameters(
20+
query=ImmutableMultiDict(self.event.queryStringParameters),
21+
header=Headers(self.event.headers),
22+
cookie=ImmutableMultiDict(),
23+
)
24+
25+
@property
26+
def host_url(self) -> str:
27+
proto = self.event.headers["X-Forwarded-Proto"]
28+
host = self.event.headers["Host"]
29+
return "://".join([proto, host])
30+
31+
@property
32+
def path(self) -> str:
33+
return self.event.resource
34+
35+
@property
36+
def method(self) -> str:
37+
return self.event.httpMethod.lower()
38+
39+
@property
40+
def body(self) -> Optional[str]:
41+
return self.event.body
42+
43+
@property
44+
def mimetype(self) -> str:
45+
return self.event.headers.get("Content-Type", "")

openapi_core/contrib/aws/responses.py

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
from werkzeug.datastructures import Headers
2+
3+
from openapi_core.contrib.aws.datatypes import APIGatewayEventResponse
4+
from openapi_core.contrib.aws.typing import APIGatewayEventResponseDict
5+
6+
7+
class APIGatewayEventResponseOpenAPIResponse:
8+
"""
9+
Converts an API Gateway event response to an OpenAPI request
10+
"""
11+
12+
def __init__(self, response: APIGatewayEventResponseDict):
13+
self.response = APIGatewayEventResponse(**response)
14+
15+
@property
16+
def data(self) -> str:
17+
return self.response.body
18+
19+
@property
20+
def status_code(self) -> int:
21+
return self.response.statusCode
22+
23+
@property
24+
def headers(self) -> Headers:
25+
return Headers(self.response.headers)
26+
27+
@property
28+
def mimetype(self) -> str:
29+
content_type = self.response.headers.get("Content-Type", "")
30+
assert isinstance(content_type, str)
31+
return content_type

openapi_core/contrib/aws/typing.py

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
from typing import Any
2+
from typing import Mapping
3+
4+
APIGatewayEventDict = Mapping[str, Any]
5+
APIGatewayEventResponseDict = Mapping[str, Any]
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
from openapi_core.templating.paths.finders import APICallPathFinder
2+
from openapi_core.templating.paths.finders import WebhookPathFinder
3+
4+
__all__ = [
5+
"APICallPathFinder",
6+
"WebhookPathFinder",
7+
]

0 commit comments

Comments
 (0)