committed by
GitHub
2 changed files with 13 additions and 502 deletions
@ -1,498 +0,0 @@ |
|||||
from unittest.mock import patch |
|
||||
|
|
||||
import pytest |
|
||||
from dirty_equals import IsDict |
|
||||
from fastapi.testclient import TestClient |
|
||||
|
|
||||
from ...utils import needs_py310 |
|
||||
|
|
||||
|
|
||||
@pytest.fixture |
|
||||
def client(): |
|
||||
from docs_src.body.tutorial001_py310 import app |
|
||||
|
|
||||
client = TestClient(app) |
|
||||
return client |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_body_float(client: TestClient): |
|
||||
response = client.post("/items/", json={"name": "Foo", "price": 50.5}) |
|
||||
assert response.status_code == 200 |
|
||||
assert response.json() == { |
|
||||
"name": "Foo", |
|
||||
"price": 50.5, |
|
||||
"description": None, |
|
||||
"tax": None, |
|
||||
} |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_post_with_str_float(client: TestClient): |
|
||||
response = client.post("/items/", json={"name": "Foo", "price": "50.5"}) |
|
||||
assert response.status_code == 200 |
|
||||
assert response.json() == { |
|
||||
"name": "Foo", |
|
||||
"price": 50.5, |
|
||||
"description": None, |
|
||||
"tax": None, |
|
||||
} |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_post_with_str_float_description(client: TestClient): |
|
||||
response = client.post( |
|
||||
"/items/", json={"name": "Foo", "price": "50.5", "description": "Some Foo"} |
|
||||
) |
|
||||
assert response.status_code == 200 |
|
||||
assert response.json() == { |
|
||||
"name": "Foo", |
|
||||
"price": 50.5, |
|
||||
"description": "Some Foo", |
|
||||
"tax": None, |
|
||||
} |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_post_with_str_float_description_tax(client: TestClient): |
|
||||
response = client.post( |
|
||||
"/items/", |
|
||||
json={"name": "Foo", "price": "50.5", "description": "Some Foo", "tax": 0.3}, |
|
||||
) |
|
||||
assert response.status_code == 200 |
|
||||
assert response.json() == { |
|
||||
"name": "Foo", |
|
||||
"price": 50.5, |
|
||||
"description": "Some Foo", |
|
||||
"tax": 0.3, |
|
||||
} |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_post_with_only_name(client: TestClient): |
|
||||
response = client.post("/items/", json={"name": "Foo"}) |
|
||||
assert response.status_code == 422 |
|
||||
assert response.json() == IsDict( |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"type": "missing", |
|
||||
"loc": ["body", "price"], |
|
||||
"msg": "Field required", |
|
||||
"input": {"name": "Foo"}, |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) | IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"loc": ["body", "price"], |
|
||||
"msg": "field required", |
|
||||
"type": "value_error.missing", |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_post_with_only_name_price(client: TestClient): |
|
||||
response = client.post("/items/", json={"name": "Foo", "price": "twenty"}) |
|
||||
assert response.status_code == 422 |
|
||||
assert response.json() == IsDict( |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"type": "float_parsing", |
|
||||
"loc": ["body", "price"], |
|
||||
"msg": "Input should be a valid number, unable to parse string as a number", |
|
||||
"input": "twenty", |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) | IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"loc": ["body", "price"], |
|
||||
"msg": "value is not a valid float", |
|
||||
"type": "type_error.float", |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_post_with_no_data(client: TestClient): |
|
||||
response = client.post("/items/", json={}) |
|
||||
assert response.status_code == 422 |
|
||||
assert response.json() == IsDict( |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"type": "missing", |
|
||||
"loc": ["body", "name"], |
|
||||
"msg": "Field required", |
|
||||
"input": {}, |
|
||||
}, |
|
||||
{ |
|
||||
"type": "missing", |
|
||||
"loc": ["body", "price"], |
|
||||
"msg": "Field required", |
|
||||
"input": {}, |
|
||||
}, |
|
||||
] |
|
||||
} |
|
||||
) | IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"loc": ["body", "name"], |
|
||||
"msg": "field required", |
|
||||
"type": "value_error.missing", |
|
||||
}, |
|
||||
{ |
|
||||
"loc": ["body", "price"], |
|
||||
"msg": "field required", |
|
||||
"type": "value_error.missing", |
|
||||
}, |
|
||||
] |
|
||||
} |
|
||||
) |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_post_with_none(client: TestClient): |
|
||||
response = client.post("/items/", json=None) |
|
||||
assert response.status_code == 422 |
|
||||
assert response.json() == IsDict( |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"type": "missing", |
|
||||
"loc": ["body"], |
|
||||
"msg": "Field required", |
|
||||
"input": None, |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) | IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"loc": ["body"], |
|
||||
"msg": "field required", |
|
||||
"type": "value_error.missing", |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_post_broken_body(client: TestClient): |
|
||||
response = client.post( |
|
||||
"/items/", |
|
||||
headers={"content-type": "application/json"}, |
|
||||
content="{some broken json}", |
|
||||
) |
|
||||
assert response.status_code == 422, response.text |
|
||||
assert response.json() == IsDict( |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"type": "json_invalid", |
|
||||
"loc": ["body", 1], |
|
||||
"msg": "JSON decode error", |
|
||||
"input": {}, |
|
||||
"ctx": { |
|
||||
"error": "Expecting property name enclosed in double quotes" |
|
||||
}, |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) | IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"loc": ["body", 1], |
|
||||
"msg": "Expecting property name enclosed in double quotes: line 1 column 2 (char 1)", |
|
||||
"type": "value_error.jsondecode", |
|
||||
"ctx": { |
|
||||
"msg": "Expecting property name enclosed in double quotes", |
|
||||
"doc": "{some broken json}", |
|
||||
"pos": 1, |
|
||||
"lineno": 1, |
|
||||
"colno": 2, |
|
||||
}, |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_post_form_for_json(client: TestClient): |
|
||||
response = client.post("/items/", data={"name": "Foo", "price": 50.5}) |
|
||||
assert response.status_code == 422, response.text |
|
||||
assert response.json() == IsDict( |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"type": "model_attributes_type", |
|
||||
"loc": ["body"], |
|
||||
"msg": "Input should be a valid dictionary or object to extract fields from", |
|
||||
"input": "name=Foo&price=50.5", |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) | IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"loc": ["body"], |
|
||||
"msg": "value is not a valid dict", |
|
||||
"type": "type_error.dict", |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_explicit_content_type(client: TestClient): |
|
||||
response = client.post( |
|
||||
"/items/", |
|
||||
content='{"name": "Foo", "price": 50.5}', |
|
||||
headers={"Content-Type": "application/json"}, |
|
||||
) |
|
||||
assert response.status_code == 200, response.text |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_geo_json(client: TestClient): |
|
||||
response = client.post( |
|
||||
"/items/", |
|
||||
content='{"name": "Foo", "price": 50.5}', |
|
||||
headers={"Content-Type": "application/geo+json"}, |
|
||||
) |
|
||||
assert response.status_code == 200, response.text |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_no_content_type_is_json(client: TestClient): |
|
||||
response = client.post( |
|
||||
"/items/", |
|
||||
content='{"name": "Foo", "price": 50.5}', |
|
||||
) |
|
||||
assert response.status_code == 200, response.text |
|
||||
assert response.json() == { |
|
||||
"name": "Foo", |
|
||||
"description": None, |
|
||||
"price": 50.5, |
|
||||
"tax": None, |
|
||||
} |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_wrong_headers(client: TestClient): |
|
||||
data = '{"name": "Foo", "price": 50.5}' |
|
||||
response = client.post( |
|
||||
"/items/", content=data, headers={"Content-Type": "text/plain"} |
|
||||
) |
|
||||
assert response.status_code == 422, response.text |
|
||||
assert response.json() == IsDict( |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"type": "model_attributes_type", |
|
||||
"loc": ["body"], |
|
||||
"msg": "Input should be a valid dictionary or object to extract fields from", |
|
||||
"input": '{"name": "Foo", "price": 50.5}', |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) | IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"loc": ["body"], |
|
||||
"msg": "value is not a valid dict", |
|
||||
"type": "type_error.dict", |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) |
|
||||
|
|
||||
response = client.post( |
|
||||
"/items/", content=data, headers={"Content-Type": "application/geo+json-seq"} |
|
||||
) |
|
||||
assert response.status_code == 422, response.text |
|
||||
assert response.json() == IsDict( |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"type": "model_attributes_type", |
|
||||
"loc": ["body"], |
|
||||
"msg": "Input should be a valid dictionary or object to extract fields from", |
|
||||
"input": '{"name": "Foo", "price": 50.5}', |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) | IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"loc": ["body"], |
|
||||
"msg": "value is not a valid dict", |
|
||||
"type": "type_error.dict", |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) |
|
||||
response = client.post( |
|
||||
"/items/", content=data, headers={"Content-Type": "application/not-really-json"} |
|
||||
) |
|
||||
assert response.status_code == 422, response.text |
|
||||
assert response.json() == IsDict( |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"type": "model_attributes_type", |
|
||||
"loc": ["body"], |
|
||||
"msg": "Input should be a valid dictionary or object to extract fields from", |
|
||||
"input": '{"name": "Foo", "price": 50.5}', |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) | IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{ |
|
||||
"detail": [ |
|
||||
{ |
|
||||
"loc": ["body"], |
|
||||
"msg": "value is not a valid dict", |
|
||||
"type": "type_error.dict", |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
) |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_other_exceptions(client: TestClient): |
|
||||
with patch("json.loads", side_effect=Exception): |
|
||||
response = client.post("/items/", json={"test": "test2"}) |
|
||||
assert response.status_code == 400, response.text |
|
||||
|
|
||||
|
|
||||
@needs_py310 |
|
||||
def test_openapi_schema(client: TestClient): |
|
||||
response = client.get("/openapi.json") |
|
||||
assert response.status_code == 200, response.text |
|
||||
assert response.json() == { |
|
||||
"openapi": "3.1.0", |
|
||||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|
||||
"paths": { |
|
||||
"/items/": { |
|
||||
"post": { |
|
||||
"responses": { |
|
||||
"200": { |
|
||||
"description": "Successful Response", |
|
||||
"content": {"application/json": {"schema": {}}}, |
|
||||
}, |
|
||||
"422": { |
|
||||
"description": "Validation Error", |
|
||||
"content": { |
|
||||
"application/json": { |
|
||||
"schema": { |
|
||||
"$ref": "#/components/schemas/HTTPValidationError" |
|
||||
} |
|
||||
} |
|
||||
}, |
|
||||
}, |
|
||||
}, |
|
||||
"summary": "Create Item", |
|
||||
"operationId": "create_item_items__post", |
|
||||
"requestBody": { |
|
||||
"content": { |
|
||||
"application/json": { |
|
||||
"schema": {"$ref": "#/components/schemas/Item"} |
|
||||
} |
|
||||
}, |
|
||||
"required": True, |
|
||||
}, |
|
||||
} |
|
||||
} |
|
||||
}, |
|
||||
"components": { |
|
||||
"schemas": { |
|
||||
"Item": { |
|
||||
"title": "Item", |
|
||||
"required": ["name", "price"], |
|
||||
"type": "object", |
|
||||
"properties": { |
|
||||
"name": {"title": "Name", "type": "string"}, |
|
||||
"price": {"title": "Price", "type": "number"}, |
|
||||
"description": IsDict( |
|
||||
{ |
|
||||
"title": "Description", |
|
||||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|
||||
} |
|
||||
) |
|
||||
| IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{"title": "Description", "type": "string"} |
|
||||
), |
|
||||
"tax": IsDict( |
|
||||
{ |
|
||||
"title": "Tax", |
|
||||
"anyOf": [{"type": "number"}, {"type": "null"}], |
|
||||
} |
|
||||
) |
|
||||
| IsDict( |
|
||||
# TODO: remove when deprecating Pydantic v1 |
|
||||
{"title": "Tax", "type": "number"} |
|
||||
), |
|
||||
}, |
|
||||
}, |
|
||||
"ValidationError": { |
|
||||
"title": "ValidationError", |
|
||||
"required": ["loc", "msg", "type"], |
|
||||
"type": "object", |
|
||||
"properties": { |
|
||||
"loc": { |
|
||||
"title": "Location", |
|
||||
"type": "array", |
|
||||
"items": { |
|
||||
"anyOf": [{"type": "string"}, {"type": "integer"}] |
|
||||
}, |
|
||||
}, |
|
||||
"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"}, |
|
||||
} |
|
||||
}, |
|
||||
}, |
|
||||
} |
|
||||
}, |
|
||||
} |
|
Loading…
Reference in new issue