committed by
GitHub
13 changed files with 487 additions and 1 deletions
After Width: | Height: | Size: 66 KiB |
@ -0,0 +1,8 @@ |
|||
from fastapi import FastAPI |
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
@app.get("/items/", openapi_extra={"x-aperture-labs-portal": "blue"}) |
|||
async def read_items(): |
|||
return [{"item_id": "portal-gun"}] |
@ -0,0 +1,41 @@ |
|||
from fastapi import FastAPI, Request |
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
def magic_data_reader(raw_body: bytes): |
|||
return { |
|||
"size": len(raw_body), |
|||
"content": { |
|||
"name": "Maaaagic", |
|||
"price": 42, |
|||
"description": "Just kiddin', no magic here. ✨", |
|||
}, |
|||
} |
|||
|
|||
|
|||
@app.post( |
|||
"/items/", |
|||
openapi_extra={ |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"required": ["name", "price"], |
|||
"type": "object", |
|||
"properties": { |
|||
"name": {"type": "string"}, |
|||
"price": {"type": "number"}, |
|||
"description": {"type": "string"}, |
|||
}, |
|||
} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
}, |
|||
) |
|||
async def create_item(request: Request): |
|||
raw_body = await request.body() |
|||
data = magic_data_reader(raw_body) |
|||
return data |
@ -0,0 +1,34 @@ |
|||
from typing import List |
|||
|
|||
import yaml |
|||
from fastapi import FastAPI, HTTPException, Request |
|||
from pydantic import BaseModel, ValidationError |
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
class Item(BaseModel): |
|||
name: str |
|||
tags: List[str] |
|||
|
|||
|
|||
@app.post( |
|||
"/items/", |
|||
openapi_extra={ |
|||
"requestBody": { |
|||
"content": {"application/x-yaml": {"schema": Item.schema()}}, |
|||
"required": True, |
|||
}, |
|||
}, |
|||
) |
|||
async def create_item(request: Request): |
|||
raw_body = await request.body() |
|||
try: |
|||
data = yaml.safe_load(raw_body) |
|||
except yaml.YAMLError: |
|||
raise HTTPException(status_code=422, detail="Invalid YAML") |
|||
try: |
|||
item = Item.parse_obj(data) |
|||
except ValidationError as e: |
|||
raise HTTPException(status_code=422, detail=e.errors()) |
|||
return item |
@ -0,0 +1,45 @@ |
|||
from fastapi import FastAPI |
|||
from fastapi.testclient import TestClient |
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
@app.get("/", openapi_extra={"x-custom-extension": "value"}) |
|||
def route_with_extras(): |
|||
return {} |
|||
|
|||
|
|||
client = TestClient(app) |
|||
|
|||
|
|||
openapi_schema = { |
|||
"openapi": "3.0.2", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/": { |
|||
"get": { |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
}, |
|||
"summary": "Route With Extras", |
|||
"operationId": "route_with_extras__get", |
|||
"x-custom-extension": "value", |
|||
} |
|||
}, |
|||
}, |
|||
} |
|||
|
|||
|
|||
def test_openapi(): |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == openapi_schema |
|||
|
|||
|
|||
def test_get_route(): |
|||
response = client.get("/") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == {} |
@ -0,0 +1,36 @@ |
|||
from fastapi.testclient import TestClient |
|||
|
|||
from docs_src.path_operation_advanced_configuration.tutorial005 import app |
|||
|
|||
client = TestClient(app) |
|||
|
|||
openapi_schema = { |
|||
"openapi": "3.0.2", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"get": { |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
} |
|||
}, |
|||
"summary": "Read Items", |
|||
"operationId": "read_items_items__get", |
|||
"x-aperture-labs-portal": "blue", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
|
|||
|
|||
def test_openapi_schema(): |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == openapi_schema |
|||
|
|||
|
|||
def test_get(): |
|||
response = client.get("/items/") |
|||
assert response.status_code == 200, response.text |
@ -0,0 +1,59 @@ |
|||
from fastapi.testclient import TestClient |
|||
|
|||
from docs_src.path_operation_advanced_configuration.tutorial006 import app |
|||
|
|||
client = TestClient(app) |
|||
|
|||
openapi_schema = { |
|||
"openapi": "3.0.2", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"post": { |
|||
"summary": "Create Item", |
|||
"operationId": "create_item_items__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"required": ["name", "price"], |
|||
"type": "object", |
|||
"properties": { |
|||
"name": {"type": "string"}, |
|||
"price": {"type": "number"}, |
|||
"description": {"type": "string"}, |
|||
}, |
|||
} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
} |
|||
}, |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
|
|||
|
|||
def test_openapi_schema(): |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == openapi_schema |
|||
|
|||
|
|||
def test_post(): |
|||
response = client.post("/items/", data=b"this is actually not validated") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == { |
|||
"size": 30, |
|||
"content": { |
|||
"name": "Maaaagic", |
|||
"price": 42, |
|||
"description": "Just kiddin', no magic here. ✨", |
|||
}, |
|||
} |
@ -0,0 +1,97 @@ |
|||
from fastapi.testclient import TestClient |
|||
|
|||
from docs_src.path_operation_advanced_configuration.tutorial007 import app |
|||
|
|||
client = TestClient(app) |
|||
|
|||
openapi_schema = { |
|||
"openapi": "3.0.2", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"post": { |
|||
"summary": "Create Item", |
|||
"operationId": "create_item_items__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/x-yaml": { |
|||
"schema": { |
|||
"title": "Item", |
|||
"required": ["name", "tags"], |
|||
"type": "object", |
|||
"properties": { |
|||
"name": {"title": "Name", "type": "string"}, |
|||
"tags": { |
|||
"title": "Tags", |
|||
"type": "array", |
|||
"items": {"type": "string"}, |
|||
}, |
|||
}, |
|||
} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
} |
|||
}, |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
|
|||
|
|||
def test_openapi_schema(): |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == openapi_schema |
|||
|
|||
|
|||
def test_post(): |
|||
yaml_data = """ |
|||
name: Deadpoolio |
|||
tags: |
|||
- x-force |
|||
- x-men |
|||
- x-avengers |
|||
""" |
|||
response = client.post("/items/", data=yaml_data) |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == { |
|||
"name": "Deadpoolio", |
|||
"tags": ["x-force", "x-men", "x-avengers"], |
|||
} |
|||
|
|||
|
|||
def test_post_broken_yaml(): |
|||
yaml_data = """ |
|||
name: Deadpoolio |
|||
tags: |
|||
x - x-force |
|||
x - x-men |
|||
x - x-avengers |
|||
""" |
|||
response = client.post("/items/", data=yaml_data) |
|||
assert response.status_code == 422, response.text |
|||
assert response.json() == {"detail": "Invalid YAML"} |
|||
|
|||
|
|||
def test_post_invalid(): |
|||
yaml_data = """ |
|||
name: Deadpoolio |
|||
tags: |
|||
- x-force |
|||
- x-men |
|||
- x-avengers |
|||
- sneaky: object |
|||
""" |
|||
response = client.post("/items/", data=yaml_data) |
|||
assert response.status_code == 422, response.text |
|||
assert response.json() == { |
|||
"detail": [ |
|||
{"loc": ["tags", 3], "msg": "str type expected", "type": "type_error.str"} |
|||
] |
|||
} |
Loading…
Reference in new issue