24 changed files with 1057 additions and 606 deletions
After Width: | Height: | Size: 80 KiB |
Before Width: | Height: | Size: 60 KiB After Width: | Height: | Size: 76 KiB |
@ -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"}) |
@ -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"} |
@ -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"}) |
@ -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"} |
@ -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: |
|||
|
|||
<img src="/img/tutorial/additional-responses/image01.png"> |
|||
|
|||
|
|||
## 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: |
|||
|
|||
* <a href="https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#responsesObject" target="_blank">OpenAPI Responses Object</a>, it includes the `Response Object`. |
|||
* <a href="https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#responseObject" target="_blank">OpenAPI Response Object</a>, 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`. |
@ -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"} |
@ -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" |
@ -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"} |
@ -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") |
@ -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"} |
@ -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") |
Loading…
Reference in new issue