diff --git a/docs/img/tutorial/additional-responses/image01.png b/docs/img/tutorial/additional-responses/image01.png new file mode 100644 index 000000000..b69b9c7df Binary files /dev/null and b/docs/img/tutorial/additional-responses/image01.png differ diff --git a/docs/img/tutorial/bigger-applications/image01.png b/docs/img/tutorial/bigger-applications/image01.png index 78544b403..f31195934 100644 Binary files a/docs/img/tutorial/bigger-applications/image01.png and b/docs/img/tutorial/bigger-applications/image01.png differ diff --git a/docs/src/additional_responses/tutorial001.py b/docs/src/additional_responses/tutorial001.py new file mode 100644 index 000000000..dfb0fc91e --- /dev/null +++ b/docs/src/additional_responses/tutorial001.py @@ -0,0 +1,23 @@ +from fastapi import FastAPI +from pydantic import BaseModel +from starlette.responses import JSONResponse + + +class Item(BaseModel): + id: str + value: str + + +class Message(BaseModel): + message: str + + +app = FastAPI() + + +@app.get("/items/{item_id}", response_model=Item, responses={404: {"model": Message}}) +async def read_item(item_id: str): + if item_id == "foo": + return {"id": "foo", "value": "there goes my hero"} + else: + return JSONResponse(status_code=404, content={"message": "Item not found"}) diff --git a/docs/src/additional_responses/tutorial002.py b/docs/src/additional_responses/tutorial002.py new file mode 100644 index 000000000..bb19c8938 --- /dev/null +++ b/docs/src/additional_responses/tutorial002.py @@ -0,0 +1,28 @@ +from fastapi import FastAPI +from pydantic import BaseModel +from starlette.responses import FileResponse + + +class Item(BaseModel): + id: str + value: str + + +app = FastAPI() + + +@app.get( + "/items/{item_id}", + response_model=Item, + responses={ + 200: { + "content": {"image/png": {}}, + "description": "Return the JSON item or an image.", + } + }, +) +async def read_item(item_id: str, img: bool = None): + if img: + return FileResponse("image.png", media_type="image/png") + else: + return {"id": "foo", "value": "there goes my hero"} diff --git a/docs/src/additional_responses/tutorial003.py b/docs/src/additional_responses/tutorial003.py new file mode 100644 index 000000000..bca3ba93e --- /dev/null +++ b/docs/src/additional_responses/tutorial003.py @@ -0,0 +1,37 @@ +from fastapi import FastAPI +from pydantic import BaseModel +from starlette.responses import JSONResponse + + +class Item(BaseModel): + id: str + value: str + + +class Message(BaseModel): + message: str + + +app = FastAPI() + + +@app.get( + "/items/{item_id}", + response_model=Item, + responses={ + 404: {"model": Message, "description": "The item was not found"}, + 200: { + "description": "Item requested by ID", + "content": { + "application/json": { + "example": {"id": "bar", "value": "The bar tenders"} + } + }, + }, + }, +) +async def read_item(item_id: str): + if item_id == "foo": + return {"id": "foo", "value": "there goes my hero"} + else: + return JSONResponse(status_code=404, content={"message": "Item not found"}) diff --git a/docs/src/additional_responses/tutorial004.py b/docs/src/additional_responses/tutorial004.py new file mode 100644 index 000000000..f7314ba8e --- /dev/null +++ b/docs/src/additional_responses/tutorial004.py @@ -0,0 +1,30 @@ +from fastapi import FastAPI +from pydantic import BaseModel +from starlette.responses import FileResponse + + +class Item(BaseModel): + id: str + value: str + + +responses = { + 404: {"description": "Item not found"}, + 302: {"description": "The item was moved"}, + 403: {"description": "Not enough privileges"}, +} + + +app = FastAPI() + + +@app.get( + "/items/{item_id}", + response_model=Item, + responses={**responses, 200: {"content": {"image/png": {}}}}, +) +async def read_item(item_id: str, img: bool = None): + if img: + return FileResponse("image.png", media_type="image/png") + else: + return {"id": "foo", "value": "there goes my hero"} diff --git a/docs/src/bigger_applications/app/main.py b/docs/src/bigger_applications/app/main.py index 4a12da434..2cebd4244 100644 --- a/docs/src/bigger_applications/app/main.py +++ b/docs/src/bigger_applications/app/main.py @@ -5,4 +5,9 @@ from .routers import items, users app = FastAPI() app.include_router(users.router) -app.include_router(items.router, prefix="/items", tags=["items"]) +app.include_router( + items.router, + prefix="/items", + tags=["items"], + responses={404: {"description": "Not found"}}, +) diff --git a/docs/src/bigger_applications/app/routers/items.py b/docs/src/bigger_applications/app/routers/items.py index 46a241902..de5d9b645 100644 --- a/docs/src/bigger_applications/app/routers/items.py +++ b/docs/src/bigger_applications/app/routers/items.py @@ -1,4 +1,4 @@ -from fastapi import APIRouter +from fastapi import APIRouter, HTTPException router = APIRouter() @@ -11,3 +11,14 @@ async def read_items(): @router.get("/{item_id}") async def read_item(item_id: str): return {"name": "Fake Specific Item", "item_id": item_id} + + +@router.put( + "/{item_id}", + tags=["custom"], + responses={403: {"description": "Operation forbidden"}}, +) +async def update_item(item_id: str): + if item_id != "foo": + raise HTTPException(status_code=403, detail="You can only update the item: foo") + return {"item_id": item_id, "name": "The Fighters"} diff --git a/docs/tutorial/additional-responses.md b/docs/tutorial/additional-responses.md new file mode 100644 index 000000000..a74e431f8 --- /dev/null +++ b/docs/tutorial/additional-responses.md @@ -0,0 +1,235 @@ +!!! warning + This is a rather advanced topic. + + If you are starting with **FastAPI**, you might not need this. + +You can declare additional responses, with additional status codes, media types, descriptions, etc. + +Those additional responses will be included in the OpenAPI schema, so they will also appear in the API docs. + +But for those additional responses you have to make sure you return a `Response` like `JSONResponse` directly, with your status code and content. + +## Additional Response with `model` + +You can pass to your *path operation decorators* a parameter `responses`. + +It receives a `dict`, the keys are status codes for each response, like `200`, and the values are other `dict`s with the information for each of them. + +Each of those response `dict`s can have a key `model`, containing a Pydantic model, just like `response_model`. + +**FastAPI** will take that model, generate its JSON Schema and include it in the correct place in OpenAPI. + +For example, to declare another response with a status code `404` and a Pydantic model `Message`, you can write: + + +```Python hl_lines="18 23" +{!./src/additional_responses/tutorial001.py!} +``` + +!!! note + Have in mind that you have to return the `JSONResponse` directly. + +!!! info + The `model` key is not part of OpenAPI. + + **FastAPI** will take the Pydantic model from there, generate the `JSON Schema`, and put it in the correct place. + + The correct place is: + + * In the key `content`, that has as value another JSON object (`dict`) that contains: + * A key with the media type, e.g. `application/json`, that contains as value another JSON object, that contains: + * A key `schema`, that has as the value the JSON Schema from the model, here's the correct place. + * **FastAPI** adds a reference here to the global JSON Schemas in another place in your OpenAPI instead of including it directly. This way, other applications and clients can use those JSON Schemas directly, provide better code generation tools, etc. + +The generated responses in the OpenAPI for this *path operation* will be: + +```JSON hl_lines="3 4 5 6 7 8 9 10 11 12" +{ + "responses": { + "404": { + "description": "Additional Response", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Message" + } + } + } + }, + "200": { + "description": "Successful Response", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Item" + } + } + } + }, + "422": { + "description": "Validation Error", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/HTTPValidationError" + } + } + } + } + } +} +``` + +The schemas are referenced to another place inside the OpenAPI schema: + +```JSON hl_lines="4 5 6 7 8 9 10 11 12 13 14 15 16" +{ + "components": { + "schemas": { + "Message": { + "title": "Message", + "required": [ + "message" + ], + "type": "object", + "properties": { + "message": { + "title": "Message", + "type": "string" + } + } + }, + "Item": { + "title": "Item", + "required": [ + "id", + "value" + ], + "type": "object", + "properties": { + "id": { + "title": "Id", + "type": "string" + }, + "value": { + "title": "Value", + "type": "string" + } + } + }, + "ValidationError": { + "title": "ValidationError", + "required": [ + "loc", + "msg", + "type" + ], + "type": "object", + "properties": { + "loc": { + "title": "Location", + "type": "array", + "items": { + "type": "string" + } + }, + "msg": { + "title": "Message", + "type": "string" + }, + "type": { + "title": "Error Type", + "type": "string" + } + } + }, + "HTTPValidationError": { + "title": "HTTPValidationError", + "type": "object", + "properties": { + "detail": { + "title": "Detail", + "type": "array", + "items": { + "$ref": "#/components/schemas/ValidationError" + } + } + } + } + } + } +} +``` + +## Additional media types for the main response + +You can use this same `responses` parameter to add different media types for the same main response. + +For example, you can add an additional media type of `image/png`, declaring that your *path operation* can return a JSON object (with media type `application/json`) or a PNG image: + +```Python hl_lines="17 18 19 20 21 22 23 24 28" +{!./src/additional_responses/tutorial002.py!} +``` + +!!! note + Notice that you have to return the image using a `FileResponse` directly. + +## Combining information + +You can also combine response information from multiple places, including the `response_model`, `status_code`, and `responses` parameters. + +You can declare a `response_model`, using the default status code `200` (or a custom one if you need), and then declare additional information for that same response in `responses`, directly in the OpenAPI schema. + +**FastAPI** will keep the additional information from `responses`, and combine it with the JSON Schema from your model. + +For example, you can declare a response with a status code `404` that uses a Pydantic model and has a custom `description`. + +And a response with a status code `200` that uses your `response_model`, but includes a custom `example`: + +```Python hl_lines="20 21 22 23 24 25 26 27 28 29 30 31" +{!./src/additional_responses/tutorial003.py!} +``` + +It will all be combined and included in your OpenAPI, and shown in the API docs: + + + + +## Combine predefined responses and custom ones + +You might want to have some predefined responses that apply to many *path operations*, but you want to combine them with custom responses needed by each *path operation*. + +For those cases, you can use the Python technique of "unpacking" a `dict` with `**dict_to_unpack`: + +```Python +old_dict = { + "old key": "old value", + "second old key": "second old value", +} +new_dict = {**old_dict, "new key": "new value"} +``` + +Here, `new_dict` will contain all the key-value pairs from `old_dict` plus the new key-value pair: + +```Python +{ + "old key": "old value", + "second old key": "second old value", + "new key": "new value", +} +``` + +You can use that technique to re-use some predefined responses in your *path operations* and combine them with additional custom ones. + +For example: + +```Python hl_lines="11 12 13 14 15 24" +{!./src/additional_responses/tutorial004.py!} +``` + +## More information about OpenAPI responses + +To see what exactly you can include in the responses, you can check these sections in the OpenAPI specification: + +* OpenAPI Responses Object, it includes the `Response Object`. +* OpenAPI Response Object, you can include anything from this directly in each response inside your `responses` parameter. Including `description`, `headers`, `content` (inside of this is that you declare different media types and JSON Schemas), and `links`. diff --git a/docs/tutorial/bigger-applications.md b/docs/tutorial/bigger-applications.md index 611c848bd..3bf406f3a 100644 --- a/docs/tutorial/bigger-applications.md +++ b/docs/tutorial/bigger-applications.md @@ -103,7 +103,17 @@ But let's say that this time we are more lazy. And we don't want to have to explicitly type `/items/` and `tags=["items"]` in every *path operation* (we will be able to do it later): -```Python hl_lines="6 11 16" +```Python hl_lines="6 11" +{!./src/bigger_applications/app/routers/items.py!} +``` + +### Add some custom `tags` and `responses` + +We are not adding the prefix `/items/` nor the `tags=["items"]` to add them later. + +But we can add custom `tags` and `responses` that will be applied to a specific *path operation*: + +```Python hl_lines="18 19" {!./src/bigger_applications/app/routers/items.py!} ``` @@ -192,7 +202,7 @@ So, to be able to use both of them in the same file, we import the submodules di Now, let's include the `router` from the submodule `users`: -```Python hl_lines="8" +```Python hl_lines="7" {!./src/bigger_applications/app/main.py!} ``` @@ -217,7 +227,7 @@ It will include all the routes from that router as part of it. So it won't affect performance. -### Include an `APIRouter` with a prefix +### Include an `APIRouter` with a `prefix`, `tags`, and `responses` Now, let's include the router form the `items` submodule. @@ -237,9 +247,11 @@ async def read_item(item_id: str): So, the prefix in this case would be `/items`. -And we can also add a list of `tags` that will be applied to all the *path operations* included in this router: +We can also add a list of `tags` that will be applied to all the *path operations* included in this router. + +And we can add predefined `responses` that will be included in all the *path operations* too. -```Python hl_lines="9" +```Python hl_lines="8 9 10 11 12 13" {!./src/bigger_applications/app/main.py!} ``` @@ -250,12 +262,18 @@ The end result is that the item paths are now: ...as we intended. -And they are marked with a list of tags that contain a single string `"items"`. +They will be marked with a list of tags that contain a single string `"items"`. + +The *path operation* that declared a `"custom"` tag will have both tags, `items` and `custom`. These "tags" are especially useful for the automatic interactive documentation systems (using OpenAPI). +And all of them will include the the predefined `responses`. + +The *path operation* that declared a custom `403` response will have both the predefined responses (`404`) and the `403` declared in it directly. + !!! check - The `prefix` and `tags` parameters are (as in many other cases) just a feature from **FastAPI** to help you avoid code duplication. + The `prefix`, `tags`, and `responses` parameters are (as in many other cases) just a feature from **FastAPI** to help you avoid code duplication. !!! tip diff --git a/fastapi/applications.py b/fastapi/applications.py index 9e9cf7126..b8387babd 100644 --- a/fastapi/applications.py +++ b/fastapi/applications.py @@ -1,8 +1,7 @@ -from typing import Any, Callable, Dict, List, Optional, Type +from typing import Any, Callable, Dict, List, Optional, Type, Union from fastapi import routing from fastapi.openapi.docs import get_redoc_html, get_swagger_ui_html -from fastapi.openapi.models import AdditionalResponse from fastapi.openapi.utils import get_openapi from pydantic import BaseModel from starlette.applications import Starlette @@ -115,7 +114,7 @@ class FastAPI(Starlette): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, methods: List[str] = None, operation_id: str = None, @@ -132,7 +131,7 @@ class FastAPI(Starlette): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=methods, operation_id=operation_id, @@ -151,7 +150,7 @@ class FastAPI(Starlette): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, methods: List[str] = None, operation_id: str = None, @@ -169,7 +168,7 @@ class FastAPI(Starlette): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=methods, operation_id=operation_id, @@ -187,10 +186,10 @@ class FastAPI(Starlette): *, prefix: str = "", tags: List[str] = None, - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, ) -> None: self.router.include_router( - router, prefix=prefix, tags=tags, additional_responses=additional_responses + router, prefix=prefix, tags=tags, responses=responses or {} ) def get( @@ -203,7 +202,7 @@ class FastAPI(Starlette): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -218,7 +217,7 @@ class FastAPI(Starlette): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, operation_id=operation_id, include_in_schema=include_in_schema, @@ -236,7 +235,7 @@ class FastAPI(Starlette): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -251,7 +250,7 @@ class FastAPI(Starlette): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, operation_id=operation_id, include_in_schema=include_in_schema, @@ -269,7 +268,7 @@ class FastAPI(Starlette): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -284,7 +283,7 @@ class FastAPI(Starlette): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, operation_id=operation_id, include_in_schema=include_in_schema, @@ -302,7 +301,7 @@ class FastAPI(Starlette): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -317,7 +316,7 @@ class FastAPI(Starlette): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, operation_id=operation_id, include_in_schema=include_in_schema, @@ -335,7 +334,7 @@ class FastAPI(Starlette): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -350,7 +349,7 @@ class FastAPI(Starlette): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, operation_id=operation_id, include_in_schema=include_in_schema, @@ -368,7 +367,7 @@ class FastAPI(Starlette): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -383,7 +382,7 @@ class FastAPI(Starlette): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, operation_id=operation_id, include_in_schema=include_in_schema, @@ -401,7 +400,7 @@ class FastAPI(Starlette): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -416,7 +415,7 @@ class FastAPI(Starlette): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, operation_id=operation_id, include_in_schema=include_in_schema, @@ -434,7 +433,7 @@ class FastAPI(Starlette): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -449,7 +448,7 @@ class FastAPI(Starlette): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, operation_id=operation_id, include_in_schema=include_in_schema, diff --git a/fastapi/openapi/models.py b/fastapi/openapi/models.py index 1be51a1a5..6572c7c07 100644 --- a/fastapi/openapi/models.py +++ b/fastapi/openapi/models.py @@ -3,7 +3,6 @@ from enum import Enum from typing import Any, Dict, List, Optional, Union from pydantic import BaseModel, Schema as PSchema -from pydantic.fields import Field from pydantic.types import UrlStr try: @@ -344,28 +343,6 @@ class Tag(BaseModel): externalDocs: Optional[ExternalDocumentation] = None -class BaseAdditionalResponse(BaseModel): - description: str - content_type: Optional[str] = None - - -class AdditionalResponse(BaseAdditionalResponse): - status_code: int = PSchema( - ..., ge=100, le=540, title="Status Code", description="HTTP status code" - ) - # NOTE: waiting for pydantic to allow `typing.Type[BasicModel]` type - # so, going for `Any` and extra validation on - # routing methods - models: List[Any] = PSchema([], title="Additional Response Models") - - -class AdditionalResponseDescription(BaseAdditionalResponse): - schema_field: Optional[Field] = None - - class Config: - arbitrary_types_allowed = True - - class OpenAPI(BaseModel): openapi: str info: Info diff --git a/fastapi/openapi/utils.py b/fastapi/openapi/utils.py index 9b0e4ed30..78175d841 100644 --- a/fastapi/openapi/utils.py +++ b/fastapi/openapi/utils.py @@ -178,6 +178,23 @@ def get_openapi_path( definitions[ "HTTPValidationError" ] = validation_error_response_definition + if route.responses: + for (additional_status_code, response) in route.responses.items(): + assert isinstance( + response, dict + ), "An additional response must be a dict" + field = route.response_fields.get(additional_status_code) + if field: + response_schema, _ = field_schema( + field, model_name_map=model_name_map, ref_prefix=REF_PREFIX + ) + response.setdefault("content", {}).setdefault( + "application/json", {} + )["schema"] = response_schema + response.setdefault("description", "Additional Response") + operation.setdefault("responses", {})[ + str(additional_status_code) + ] = response status_code = str(route.status_code) response_schema = {"type": "string"} if lenient_issubclass(route.content_type, JSONResponse): @@ -189,13 +206,14 @@ def get_openapi_path( ) else: response_schema = {} - content = {route.content_type.media_type: {"schema": response_schema}} - operation["responses"] = { - status_code: { - "description": route.response_description, - "content": content, - } - } + operation.setdefault("responses", {}).setdefault(status_code, {})[ + "description" + ] = route.response_description + operation.setdefault("responses", {}).setdefault( + status_code, {} + ).setdefault("content", {}).setdefault(route.content_type.media_type, {})[ + "schema" + ] = response_schema if all_route_params or route.body_field: operation["responses"][str(HTTP_422_UNPROCESSABLE_ENTITY)] = { "description": "Validation Error", @@ -205,24 +223,6 @@ def get_openapi_path( } }, } - for add_response_code, add_response in route.additional_responses.items(): - add_response_schema: Dict[str, Any] = {} - if ( - add_response.content_type or route.content_type.media_type - ) == "application/json" and add_response.schema_field is not None: - add_response_schema, _ = field_schema( - add_response.schema_field, - model_name_map=model_name_map, - ref_prefix=REF_PREFIX, - ) - add_content = { - add_response.content_type - or route.content_type.media_type: {"schema": add_response_schema} - } - operation["responses"][str(add_response_code)] = { - "description": add_response.description, - "content": add_content, - } path[method.lower()] = operation return path, security_schemes, definitions diff --git a/fastapi/routing.py b/fastapi/routing.py index 9dfd316ab..cc6cac79b 100644 --- a/fastapi/routing.py +++ b/fastapi/routing.py @@ -7,7 +7,6 @@ from fastapi import params from fastapi.dependencies.models import Dependant from fastapi.dependencies.utils import get_body_field, get_dependant, solve_dependencies from fastapi.encoders import jsonable_encoder -from fastapi.openapi.models import AdditionalResponse, AdditionalResponseDescription from fastapi.utils import UnconstrainedConfig from pydantic import BaseModel, Schema from pydantic.error_wrappers import ErrorWrapper, ValidationError @@ -105,7 +104,7 @@ class APIRoute(routing.Route): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, name: str = None, methods: List[str] = None, @@ -139,35 +138,30 @@ class APIRoute(routing.Route): self.summary = summary self.description = description or self.endpoint.__doc__ self.response_description = response_description - self.additional_responses: Dict[int, AdditionalResponseDescription] = {} - existed_codes = [self.status_code, 422] - for add_response in additional_responses: - assert ( - add_response.status_code not in existed_codes - ), f"(Duplicated Status Code): Response with status code [{add_response.status_code}] already defined!" - existed_codes.append(add_response.status_code) - response_models: List[Any] = [m for m in add_response.models] - schema_field = None - if ( - add_response.content_type == "application/json" - or lenient_issubclass(content_type, JSONResponse) - and len(response_models) - ): - schema_field = Field( - name=f"Additional_response_{add_response.status_code}", - type_=Union[tuple(response_models)], - class_validators=[], + self.responses = responses or {} + response_fields = {} + for additional_status_code, response in self.responses.items(): + assert isinstance(response, dict), "An additional response must be a dict" + model = response.get("model") + if model: + assert lenient_issubclass( + model, BaseModel + ), "A response model must be a Pydantic model" + response_name = f"Response_{additional_status_code}_{self.name}" + response_field = Field( + name=response_name, + type_=model, + class_validators=None, default=None, required=False, model_config=UnconstrainedConfig, schema=Schema(None), ) - add_resp_description = AdditionalResponseDescription( - description=add_response.description, - content_type=add_response.content_type, - schema_field=schema_field, - ) - self.additional_responses[add_response.status_code] = add_resp_description + response_fields[additional_status_code] = response_field + if response_fields: + self.response_fields: Dict[Union[int, str], Field] = response_fields + else: + self.response_fields = {} self.deprecated = deprecated if methods is None: methods = ["GET"] @@ -205,7 +199,7 @@ class APIRouter(routing.Router): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, methods: List[str] = None, operation_id: str = None, @@ -222,7 +216,7 @@ class APIRouter(routing.Router): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=methods, operation_id=operation_id, @@ -242,7 +236,7 @@ class APIRouter(routing.Router): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, methods: List[str] = None, operation_id: str = None, @@ -260,7 +254,7 @@ class APIRouter(routing.Router): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=methods, operation_id=operation_id, @@ -278,7 +272,7 @@ class APIRouter(routing.Router): *, prefix: str = "", tags: List[str] = None, - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, ) -> None: if prefix: assert prefix.startswith("/"), "A path prefix must start with '/'" @@ -287,6 +281,9 @@ class APIRouter(routing.Router): ), "A path prefix must not end with '/', as the routes will start with '/'" for route in router.routes: if isinstance(route, APIRoute): + if responses is None: + responses = {} + responses = {**responses, **route.responses} self.add_api_route( prefix + route.path, route.endpoint, @@ -296,7 +293,7 @@ class APIRouter(routing.Router): summary=route.summary, description=route.description, response_description=route.response_description, - additional_responses=additional_responses, + responses=responses, deprecated=route.deprecated, methods=route.methods, operation_id=route.operation_id, @@ -323,7 +320,7 @@ class APIRouter(routing.Router): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -338,7 +335,7 @@ class APIRouter(routing.Router): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=["GET"], operation_id=operation_id, @@ -357,7 +354,7 @@ class APIRouter(routing.Router): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -372,7 +369,7 @@ class APIRouter(routing.Router): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=["PUT"], operation_id=operation_id, @@ -391,7 +388,7 @@ class APIRouter(routing.Router): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -406,7 +403,7 @@ class APIRouter(routing.Router): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=["POST"], operation_id=operation_id, @@ -425,7 +422,7 @@ class APIRouter(routing.Router): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -440,7 +437,7 @@ class APIRouter(routing.Router): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=["DELETE"], operation_id=operation_id, @@ -459,7 +456,7 @@ class APIRouter(routing.Router): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -474,7 +471,7 @@ class APIRouter(routing.Router): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=["OPTIONS"], operation_id=operation_id, @@ -493,7 +490,7 @@ class APIRouter(routing.Router): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -508,7 +505,7 @@ class APIRouter(routing.Router): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=["HEAD"], operation_id=operation_id, @@ -527,7 +524,7 @@ class APIRouter(routing.Router): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -542,7 +539,7 @@ class APIRouter(routing.Router): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=["PATCH"], operation_id=operation_id, @@ -561,7 +558,7 @@ class APIRouter(routing.Router): summary: str = None, description: str = None, response_description: str = "Successful Response", - additional_responses: List[AdditionalResponse] = [], + responses: Dict[Union[int, str], Dict[str, Any]] = None, deprecated: bool = None, operation_id: str = None, include_in_schema: bool = True, @@ -576,7 +573,7 @@ class APIRouter(routing.Router): summary=summary, description=description, response_description=response_description, - additional_responses=additional_responses, + responses=responses or {}, deprecated=deprecated, methods=["TRACE"], operation_id=operation_id, diff --git a/fastapi/utils.py b/fastapi/utils.py index d0ffbb5d8..8eacb83d1 100644 --- a/fastapi/utils.py +++ b/fastapi/utils.py @@ -30,10 +30,8 @@ def get_flat_models_from_routes( body_fields_from_routes.append(route.body_field) if route.response_field: responses_from_routes.append(route.response_field) - if route.additional_responses: - for _, add_response in route.additional_responses.items(): - if add_response.schema_field is not None: - responses_from_routes.append(add_response.schema_field) + if route.response_fields: + responses_from_routes.extend(route.response_fields.values()) flat_models = get_flat_models_from_fields( body_fields_from_routes + responses_from_routes ) diff --git a/mkdocs.yml b/mkdocs.yml index 2760194af..e453323e8 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -44,6 +44,7 @@ nav: - Path Operation Configuration: 'tutorial/path-operation-configuration.md' - Path Operation Advanced Configuration: 'tutorial/path-operation-advanced-configuration.md' - Custom Response: 'tutorial/custom-response.md' + - Additional Responses: 'tutorial/additional-responses.md' - Dependencies: - First Steps: 'tutorial/dependencies/first-steps.md' - Classes as Dependencies: 'tutorial/dependencies/classes-as-dependencies.md' diff --git a/tests/test_additional_response_extra.py b/tests/test_additional_response_extra.py new file mode 100644 index 000000000..45afcc0e5 --- /dev/null +++ b/tests/test_additional_response_extra.py @@ -0,0 +1,52 @@ +from fastapi import APIRouter, FastAPI +from starlette.testclient import TestClient + +router = APIRouter() + +sub_router = APIRouter() + +app = FastAPI() + + +@sub_router.get("/") +def read_item(): + return {"id": "foo"} + + +router.include_router(sub_router, prefix="/items") + +app.include_router(router) + + +openapi_schema = { + "openapi": "3.0.2", + "info": {"title": "Fast API", "version": "0.1.0"}, + "paths": { + "/items/": { + "get": { + "responses": { + "200": { + "description": "Successful Response", + "content": {"application/json": {"schema": {}}}, + } + }, + "summary": "Read Item Get", + "operationId": "read_item_items__get", + } + } + }, +} + +client = TestClient(app) + + +def test_openapi_schema(): + response = client.get("/openapi.json") + assert response.status_code == 200 + assert response.json() == openapi_schema + + +def test_path_operation(): + response = client.get("/items/") + assert response.status_code == 200 + assert response.json() == {"id": "foo"} diff --git a/tests/test_additional_responses.py b/tests/test_additional_responses.py deleted file mode 100644 index fa46d3ccb..000000000 --- a/tests/test_additional_responses.py +++ /dev/null @@ -1,471 +0,0 @@ -import pytest -from fastapi import FastAPI -from fastapi.openapi.models import AdditionalResponse -from pydantic import BaseModel -from starlette.responses import JSONResponse -from starlette.testclient import TestClient - -app = FastAPI() - - -class Item(BaseModel): - name: str - price: float = None - - -class Response400(BaseModel): - """HTTP 4xx Response Schema""" - - title: str - detail: str - error_code: int # functional error ref - - -response_403 = AdditionalResponse( - status_code=403, description="Forbidden", models=[Response400] -) - -additional_responses = [response_403] - - -@app.api_route( - "/items/{item_id}", methods=["GET"], additional_responses=additional_responses -) -def get_items(item_id: str): - return {"item_id": item_id} - - -def get_not_decorated(item_id: str): - return {"item_id": item_id} - - -app.add_api_route( - "/items-not-decorated/{item_id}", - get_not_decorated, - additional_responses=additional_responses, -) - - -@app.delete("/items/{item_id}", additional_responses=additional_responses) -def delete_item(item_id: str, item: Item): - return {"item_id": item_id, "item": item} - - -@app.head("/items/{item_id}", additional_responses=additional_responses) -def head_item(item_id: str): - return JSONResponse(headers={"x-fastapi-item-id": item_id}) - - -@app.options("/items/{item_id}", additional_responses=additional_responses) -def options_item(item_id: str): - return JSONResponse(headers={"x-fastapi-item-id": item_id}) - - -@app.patch("/items/{item_id}", additional_responses=additional_responses) -def patch_item(item_id: str, item: Item): - return {"item_id": item_id, "item": item} - - -@app.trace("/items/{item_id}", additional_responses=additional_responses) -def trace_item(item_id: str): - return JSONResponse(media_type="message/http") - - -client = TestClient(app) - -openapi_schema = { - "openapi": "3.0.2", - "info": {"title": "Fast API", "version": "0.1.0"}, - "paths": { - "/items/{item_id}": { - "get": { - "responses": { - "200": { - "description": "Successful Response", - "content": {"application/json": {"schema": {}}}, - }, - "403": { - "description": "Forbidden", - "content": { - "application/json": { - "schema": {"$ref": "#/components/schemas/Response400"} - } - }, - }, - "422": { - "description": "Validation Error", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/HTTPValidationError" - } - } - }, - }, - }, - "summary": "Get Items Get", - "operationId": "get_items_items__item_id__get", - "parameters": [ - { - "required": True, - "schema": {"title": "Item_Id", "type": "string"}, - "name": "item_id", - "in": "path", - } - ], - }, - "delete": { - "responses": { - "200": { - "description": "Successful Response", - "content": {"application/json": {"schema": {}}}, - }, - "403": { - "description": "Forbidden", - "content": { - "application/json": { - "schema": {"$ref": "#/components/schemas/Response400"} - } - }, - }, - "422": { - "description": "Validation Error", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/HTTPValidationError" - } - } - }, - }, - }, - "summary": "Delete Item Delete", - "operationId": "delete_item_items__item_id__delete", - "parameters": [ - { - "required": True, - "schema": {"title": "Item_Id", "type": "string"}, - "name": "item_id", - "in": "path", - } - ], - "requestBody": { - "content": { - "application/json": { - "schema": {"$ref": "#/components/schemas/Item"} - } - }, - "required": True, - }, - }, - "options": { - "responses": { - "200": { - "description": "Successful Response", - "content": {"application/json": {"schema": {}}}, - }, - "403": { - "description": "Forbidden", - "content": { - "application/json": { - "schema": {"$ref": "#/components/schemas/Response400"} - } - }, - }, - "403": { - "description": "Forbidden", - "content": { - "application/json": { - "schema": {"$ref": "#/components/schemas/Response400"} - } - }, - }, - "422": { - "description": "Validation Error", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/HTTPValidationError" - } - } - }, - }, - }, - "summary": "Options Item Options", - "operationId": "options_item_items__item_id__options", - "parameters": [ - { - "required": True, - "schema": {"title": "Item_Id", "type": "string"}, - "name": "item_id", - "in": "path", - } - ], - }, - "head": { - "responses": { - "200": { - "description": "Successful Response", - "content": {"application/json": {"schema": {}}}, - }, - "403": { - "description": "Forbidden", - "content": { - "application/json": { - "schema": {"$ref": "#/components/schemas/Response400"} - } - }, - }, - "422": { - "description": "Validation Error", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/HTTPValidationError" - } - } - }, - }, - }, - "summary": "Head Item Head", - "operationId": "head_item_items__item_id__head", - "parameters": [ - { - "required": True, - "schema": {"title": "Item_Id", "type": "string"}, - "name": "item_id", - "in": "path", - } - ], - }, - "patch": { - "responses": { - "200": { - "description": "Successful Response", - "content": {"application/json": {"schema": {}}}, - }, - "403": { - "description": "Forbidden", - "content": { - "application/json": { - "schema": {"$ref": "#/components/schemas/Response400"} - } - }, - }, - "422": { - "description": "Validation Error", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/HTTPValidationError" - } - } - }, - }, - }, - "summary": "Patch Item Patch", - "operationId": "patch_item_items__item_id__patch", - "parameters": [ - { - "required": True, - "schema": {"title": "Item_Id", "type": "string"}, - "name": "item_id", - "in": "path", - } - ], - "requestBody": { - "content": { - "application/json": { - "schema": {"$ref": "#/components/schemas/Item"} - } - }, - "required": True, - }, - }, - "trace": { - "responses": { - "200": { - "description": "Successful Response", - "content": {"application/json": {"schema": {}}}, - }, - "403": { - "description": "Forbidden", - "content": { - "application/json": { - "schema": {"$ref": "#/components/schemas/Response400"} - } - }, - }, - "422": { - "description": "Validation Error", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/HTTPValidationError" - } - } - }, - }, - }, - "summary": "Trace Item Trace", - "operationId": "trace_item_items__item_id__trace", - "parameters": [ - { - "required": True, - "schema": {"title": "Item_Id", "type": "string"}, - "name": "item_id", - "in": "path", - } - ], - }, - }, - "/items-not-decorated/{item_id}": { - "get": { - "responses": { - "200": { - "description": "Successful Response", - "content": {"application/json": {"schema": {}}}, - }, - "403": { - "description": "Forbidden", - "content": { - "application/json": { - "schema": {"$ref": "#/components/schemas/Response400"} - } - }, - }, - "422": { - "description": "Validation Error", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/HTTPValidationError" - } - } - }, - }, - }, - "summary": "Get Not Decorated Get", - "operationId": "get_not_decorated_items-not-decorated__item_id__get", - "parameters": [ - { - "required": True, - "schema": {"title": "Item_Id", "type": "string"}, - "name": "item_id", - "in": "path", - } - ], - } - }, - }, - "components": { - "schemas": { - "Item": { - "title": "Item", - "required": ["name"], - "type": "object", - "properties": { - "name": {"title": "Name", "type": "string"}, - "price": {"title": "Price", "type": "number"}, - }, - }, - "Response400": { - "title": "Response400", - "description": "HTTP 4xx Response Schema", - "required": ["title", "detail", "error_code"], - "type": "object", - "properties": { - "title": {"title": "Title", "type": "string"}, - "detail": {"title": "Detail", "type": "string"}, - "error_code": {"title": "Error_Code", "type": "integer"}, - }, - }, - "ValidationError": { - "title": "ValidationError", - "required": ["loc", "msg", "type"], - "type": "object", - "properties": { - "loc": { - "title": "Location", - "type": "array", - "items": {"type": "string"}, - }, - "msg": {"title": "Message", "type": "string"}, - "type": {"title": "Error Type", "type": "string"}, - }, - }, - "HTTPValidationError": { - "title": "HTTPValidationError", - "type": "object", - "properties": { - "detail": { - "title": "Detail", - "type": "array", - "items": {"$ref": "#/components/schemas/ValidationError"}, - } - }, - }, - } - }, -} - - -def test_uncompatible_response_model_undecorated(): - app = FastAPI() - - class NotBaseModel: - pass - - response_403 = AdditionalResponse( - status_code=403, description="Forbidden", models=[NotBaseModel] - ) - with pytest.raises(RuntimeError): - app.add_api_route("/", get_not_decorated, additional_responses=[response_403]) - - -def test_openapi_schema(): - response = client.get("/openapi.json") - assert response.status_code == 200 - assert response.json() == openapi_schema - - -def test_get_api_route(): - response = client.get("/items/foo") - assert response.status_code == 200 - assert response.json() == {"item_id": "foo"} - - -def test_get_api_route_not_decorated(): - response = client.get("/items-not-decorated/foo") - assert response.status_code == 200 - assert response.json() == {"item_id": "foo"} - - -def test_delete(): - response = client.delete("/items/foo", json={"name": "Foo"}) - assert response.status_code == 200 - assert response.json() == {"item_id": "foo", "item": {"name": "Foo", "price": None}} - - -def test_head(): - response = client.head("/items/foo") - assert response.status_code == 200 - assert response.headers["x-fastapi-item-id"] == "foo" - - -def test_options(): - response = client.options("/items/foo") - assert response.status_code == 200 - assert response.headers["x-fastapi-item-id"] == "foo" - - -def test_patch(): - response = client.patch("/items/foo", json={"name": "Foo"}) - assert response.status_code == 200 - assert response.json() == {"item_id": "foo", "item": {"name": "Foo", "price": None}} - - -def test_trace(): - response = client.request("trace", "/items/foo") - assert response.status_code == 200 - assert response.headers["content-type"] == "message/http" diff --git a/tests/test_tutorial/test_additional_responses/__init__.py b/tests/test_tutorial/test_additional_responses/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/tests/test_tutorial/test_additional_responses/test_tutorial001.py b/tests/test_tutorial/test_additional_responses/test_tutorial001.py new file mode 100644 index 000000000..e1ec1ed8c --- /dev/null +++ b/tests/test_tutorial/test_additional_responses/test_tutorial001.py @@ -0,0 +1,116 @@ +from starlette.testclient import TestClient + +from additional_responses.tutorial001 import app + +client = TestClient(app) + +openapi_schema = { + "openapi": "3.0.2", + "info": {"title": "Fast API", "version": "0.1.0"}, + "paths": { + "/items/{item_id}": { + "get": { + "responses": { + "404": { + "description": "Additional Response", + "content": { + "application/json": { + "schema": {"$ref": "#/components/schemas/Message"} + } + }, + }, + "200": { + "description": "Successful Response", + "content": { + "application/json": { + "schema": {"$ref": "#/components/schemas/Item"} + } + }, + }, + "422": { + "description": "Validation Error", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/HTTPValidationError" + } + } + }, + }, + }, + "summary": "Read Item Get", + "operationId": "read_item_items__item_id__get", + "parameters": [ + { + "required": True, + "schema": {"title": "Item_Id", "type": "string"}, + "name": "item_id", + "in": "path", + } + ], + } + } + }, + "components": { + "schemas": { + "Item": { + "title": "Item", + "required": ["id", "value"], + "type": "object", + "properties": { + "id": {"title": "Id", "type": "string"}, + "value": {"title": "Value", "type": "string"}, + }, + }, + "Message": { + "title": "Message", + "required": ["message"], + "type": "object", + "properties": {"message": {"title": "Message", "type": "string"}}, + }, + "ValidationError": { + "title": "ValidationError", + "required": ["loc", "msg", "type"], + "type": "object", + "properties": { + "loc": { + "title": "Location", + "type": "array", + "items": {"type": "string"}, + }, + "msg": {"title": "Message", "type": "string"}, + "type": {"title": "Error Type", "type": "string"}, + }, + }, + "HTTPValidationError": { + "title": "HTTPValidationError", + "type": "object", + "properties": { + "detail": { + "title": "Detail", + "type": "array", + "items": {"$ref": "#/components/schemas/ValidationError"}, + } + }, + }, + } + }, +} + + +def test_openapi_schema(): + response = client.get("/openapi.json") + assert response.status_code == 200 + assert response.json() == openapi_schema + + +def test_path_operation(): + response = client.get("/items/foo") + assert response.status_code == 200 + assert response.json() == {"id": "foo", "value": "there goes my hero"} + + +def test_path_operation_not_found(): + response = client.get("/items/bar") + assert response.status_code == 404 + assert response.json() == {"message": "Item not found"} diff --git a/tests/test_tutorial/test_additional_responses/test_tutorial002.py b/tests/test_tutorial/test_additional_responses/test_tutorial002.py new file mode 100644 index 000000000..b45a148d4 --- /dev/null +++ b/tests/test_tutorial/test_additional_responses/test_tutorial002.py @@ -0,0 +1,115 @@ +import os +import shutil + +from starlette.testclient import TestClient + +from additional_responses.tutorial002 import app + +client = TestClient(app) + +openapi_schema = { + "openapi": "3.0.2", + "info": {"title": "Fast API", "version": "0.1.0"}, + "paths": { + "/items/{item_id}": { + "get": { + "responses": { + "200": { + "description": "Successful Response", + "content": { + "image/png": {}, + "application/json": { + "schema": {"$ref": "#/components/schemas/Item"} + }, + }, + }, + "422": { + "description": "Validation Error", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/HTTPValidationError" + } + } + }, + }, + }, + "summary": "Read Item Get", + "operationId": "read_item_items__item_id__get", + "parameters": [ + { + "required": True, + "schema": {"title": "Item_Id", "type": "string"}, + "name": "item_id", + "in": "path", + }, + { + "required": False, + "schema": {"title": "Img", "type": "boolean"}, + "name": "img", + "in": "query", + }, + ], + } + } + }, + "components": { + "schemas": { + "Item": { + "title": "Item", + "required": ["id", "value"], + "type": "object", + "properties": { + "id": {"title": "Id", "type": "string"}, + "value": {"title": "Value", "type": "string"}, + }, + }, + "ValidationError": { + "title": "ValidationError", + "required": ["loc", "msg", "type"], + "type": "object", + "properties": { + "loc": { + "title": "Location", + "type": "array", + "items": {"type": "string"}, + }, + "msg": {"title": "Message", "type": "string"}, + "type": {"title": "Error Type", "type": "string"}, + }, + }, + "HTTPValidationError": { + "title": "HTTPValidationError", + "type": "object", + "properties": { + "detail": { + "title": "Detail", + "type": "array", + "items": {"$ref": "#/components/schemas/ValidationError"}, + } + }, + }, + } + }, +} + + +def test_openapi_schema(): + response = client.get("/openapi.json") + assert response.status_code == 200 + assert response.json() == openapi_schema + + +def test_path_operation(): + response = client.get("/items/foo") + assert response.status_code == 200 + assert response.json() == {"id": "foo", "value": "there goes my hero"} + + +def test_path_operation_img(): + shutil.copy("./docs/img/favicon.png", "./image.png") + response = client.get("/items/foo?img=1") + assert response.status_code == 200 + assert response.headers["Content-Type"] == "image/png" + assert len(response.content) + os.remove("./image.png") diff --git a/tests/test_tutorial/test_additional_responses/test_tutorial003.py b/tests/test_tutorial/test_additional_responses/test_tutorial003.py new file mode 100644 index 000000000..09dc8fd3c --- /dev/null +++ b/tests/test_tutorial/test_additional_responses/test_tutorial003.py @@ -0,0 +1,117 @@ +from starlette.testclient import TestClient + +from additional_responses.tutorial003 import app + +client = TestClient(app) + +openapi_schema = { + "openapi": "3.0.2", + "info": {"title": "Fast API", "version": "0.1.0"}, + "paths": { + "/items/{item_id}": { + "get": { + "responses": { + "404": { + "description": "The item was not found", + "content": { + "application/json": { + "schema": {"$ref": "#/components/schemas/Message"} + } + }, + }, + "200": { + "description": "Successful Response", + "content": { + "application/json": { + "schema": {"$ref": "#/components/schemas/Item"}, + "example": {"id": "bar", "value": "The bar tenders"}, + } + }, + }, + "422": { + "description": "Validation Error", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/HTTPValidationError" + } + } + }, + }, + }, + "summary": "Read Item Get", + "operationId": "read_item_items__item_id__get", + "parameters": [ + { + "required": True, + "schema": {"title": "Item_Id", "type": "string"}, + "name": "item_id", + "in": "path", + } + ], + } + } + }, + "components": { + "schemas": { + "Item": { + "title": "Item", + "required": ["id", "value"], + "type": "object", + "properties": { + "id": {"title": "Id", "type": "string"}, + "value": {"title": "Value", "type": "string"}, + }, + }, + "Message": { + "title": "Message", + "required": ["message"], + "type": "object", + "properties": {"message": {"title": "Message", "type": "string"}}, + }, + "ValidationError": { + "title": "ValidationError", + "required": ["loc", "msg", "type"], + "type": "object", + "properties": { + "loc": { + "title": "Location", + "type": "array", + "items": {"type": "string"}, + }, + "msg": {"title": "Message", "type": "string"}, + "type": {"title": "Error Type", "type": "string"}, + }, + }, + "HTTPValidationError": { + "title": "HTTPValidationError", + "type": "object", + "properties": { + "detail": { + "title": "Detail", + "type": "array", + "items": {"$ref": "#/components/schemas/ValidationError"}, + } + }, + }, + } + }, +} + + +def test_openapi_schema(): + response = client.get("/openapi.json") + assert response.status_code == 200 + assert response.json() == openapi_schema + + +def test_path_operation(): + response = client.get("/items/foo") + assert response.status_code == 200 + assert response.json() == {"id": "foo", "value": "there goes my hero"} + + +def test_path_operation_not_found(): + response = client.get("/items/bar") + assert response.status_code == 404 + assert response.json() == {"message": "Item not found"} diff --git a/tests/test_tutorial/test_additional_responses/test_tutorial004.py b/tests/test_tutorial/test_additional_responses/test_tutorial004.py new file mode 100644 index 000000000..91b5ca959 --- /dev/null +++ b/tests/test_tutorial/test_additional_responses/test_tutorial004.py @@ -0,0 +1,118 @@ +import os +import shutil + +from starlette.testclient import TestClient + +from additional_responses.tutorial004 import app + +client = TestClient(app) + +openapi_schema = { + "openapi": "3.0.2", + "info": {"title": "Fast API", "version": "0.1.0"}, + "paths": { + "/items/{item_id}": { + "get": { + "responses": { + "404": {"description": "Item not found"}, + "302": {"description": "The item was moved"}, + "403": {"description": "Not enough privileges"}, + "200": { + "description": "Successful Response", + "content": { + "image/png": {}, + "application/json": { + "schema": {"$ref": "#/components/schemas/Item"} + }, + }, + }, + "422": { + "description": "Validation Error", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/HTTPValidationError" + } + } + }, + }, + }, + "summary": "Read Item Get", + "operationId": "read_item_items__item_id__get", + "parameters": [ + { + "required": True, + "schema": {"title": "Item_Id", "type": "string"}, + "name": "item_id", + "in": "path", + }, + { + "required": False, + "schema": {"title": "Img", "type": "boolean"}, + "name": "img", + "in": "query", + }, + ], + } + } + }, + "components": { + "schemas": { + "Item": { + "title": "Item", + "required": ["id", "value"], + "type": "object", + "properties": { + "id": {"title": "Id", "type": "string"}, + "value": {"title": "Value", "type": "string"}, + }, + }, + "ValidationError": { + "title": "ValidationError", + "required": ["loc", "msg", "type"], + "type": "object", + "properties": { + "loc": { + "title": "Location", + "type": "array", + "items": {"type": "string"}, + }, + "msg": {"title": "Message", "type": "string"}, + "type": {"title": "Error Type", "type": "string"}, + }, + }, + "HTTPValidationError": { + "title": "HTTPValidationError", + "type": "object", + "properties": { + "detail": { + "title": "Detail", + "type": "array", + "items": {"$ref": "#/components/schemas/ValidationError"}, + } + }, + }, + } + }, +} + + +def test_openapi_schema(): + response = client.get("/openapi.json") + assert response.status_code == 200 + assert response.json() == openapi_schema + + +def test_path_operation(): + response = client.get("/items/foo") + assert response.status_code == 200 + assert response.json() == {"id": "foo", "value": "there goes my hero"} + + +def test_path_operation_img(): + shutil.copy("./docs/img/favicon.png", "./image.png") + response = client.get("/items/foo?img=1") + assert response.status_code == 200 + assert response.headers["Content-Type"] == "image/png" + assert len(response.content) + os.remove("./image.png") diff --git a/tests/test_tutorial/test_bigger_applications/test_main.py b/tests/test_tutorial/test_bigger_applications/test_main.py index eb68c4492..fd11a77a9 100644 --- a/tests/test_tutorial/test_bigger_applications/test_main.py +++ b/tests/test_tutorial/test_bigger_applications/test_main.py @@ -69,10 +69,11 @@ openapi_schema = { "/items/": { "get": { "responses": { + "404": {"description": "Not found"}, "200": { "description": "Successful Response", "content": {"application/json": {"schema": {}}}, - } + }, }, "tags": ["items"], "summary": "Read Items Get", @@ -82,6 +83,7 @@ openapi_schema = { "/items/{item_id}": { "get": { "responses": { + "404": {"description": "Not found"}, "200": { "description": "Successful Response", "content": {"application/json": {"schema": {}}}, @@ -108,7 +110,38 @@ openapi_schema = { "in": "path", } ], - } + }, + "put": { + "responses": { + "404": {"description": "Not found"}, + "403": {"description": "Operation forbidden"}, + "200": { + "description": "Successful Response", + "content": {"application/json": {"schema": {}}}, + }, + "422": { + "description": "Validation Error", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/HTTPValidationError" + } + } + }, + }, + }, + "tags": ["custom", "items"], + "summary": "Update Item Put", + "operationId": "update_item_items__item_id__put", + "parameters": [ + { + "required": True, + "schema": {"title": "Item_Id", "type": "string"}, + "name": "item_id", + "in": "path", + } + ], + }, }, }, "components": { @@ -158,3 +191,15 @@ def test_get_path(path, expected_status, expected_response): response = client.get(path) assert response.status_code == expected_status assert response.json() == expected_response + + +def test_put(): + response = client.put("/items/foo") + assert response.status_code == 200 + assert response.json() == {"item_id": "foo", "name": "The Fighters"} + + +def test_put_forbidden(): + response = client.put("/items/bar") + assert response.status_code == 403 + assert response.json() == {"detail": "You can only update the item: foo"}