diff --git a/docs/en/docs/tutorial/body.md b/docs/en/docs/tutorial/body.md index baa013809..f5a75dfe0 100644 --- a/docs/en/docs/tutorial/body.md +++ b/docs/en/docs/tutorial/body.md @@ -9,9 +9,11 @@ Your API almost always has to send a **response** body. But clients don't necess To declare a **request** body, you use Pydantic models with all their power and benefits. !!! info - You cannot send a request body using a `GET` operation (HTTP method). + To send data, you should use one of: `POST` (the more common), `PUT`, `DELETE` or `PATCH`. - To send data, you have to use one of: `POST` (the more common), `PUT`, `DELETE` or `PATCH`. + Sending a body with a `GET` request has an undefined behavior in the specifications, nevertheless, it is supported by FastAPI, only for very complex/extreme use cases. + + As it is discouraged, the interactive docs with Swagger UI won't show the documentation for the body when using `GET`, and proxies in the middle might not support it. ## Import Pydantic's `BaseModel` diff --git a/fastapi/dependencies/utils.py b/fastapi/dependencies/utils.py index 7fb7347c5..a09ad2512 100644 --- a/fastapi/dependencies/utils.py +++ b/fastapi/dependencies/utils.py @@ -246,7 +246,9 @@ def get_typed_signature(call: Callable) -> inspect.Signature: def get_typed_annotation(param: inspect.Parameter, globalns: Dict[str, Any]) -> Any: annotation = param.annotation if isinstance(annotation, str): - annotation = ForwardRef(annotation) + # Temporary ignore type + # Ref: https://github.com/samuelcolvin/pydantic/issues/1738 + annotation = ForwardRef(annotation) # type: ignore annotation = evaluate_forwardref(annotation, globalns, globalns) return annotation diff --git a/fastapi/openapi/constants.py b/fastapi/openapi/constants.py index bba050a1a..000011a1f 100644 --- a/fastapi/openapi/constants.py +++ b/fastapi/openapi/constants.py @@ -1,3 +1,3 @@ -METHODS_WITH_BODY = set(("POST", "PUT", "DELETE", "PATCH")) +METHODS_WITH_BODY = set(("GET", "HEAD", "POST", "PUT", "DELETE", "PATCH")) STATUS_CODES_WITH_NO_BODY = set((100, 101, 102, 103, 204, 304)) REF_PREFIX = "#/components/schemas/" diff --git a/tests/test_get_request_body.py b/tests/test_get_request_body.py new file mode 100644 index 000000000..348aee5f9 --- /dev/null +++ b/tests/test_get_request_body.py @@ -0,0 +1,108 @@ +from fastapi import FastAPI +from fastapi.testclient import TestClient +from pydantic import BaseModel + +app = FastAPI() + + +class Product(BaseModel): + name: str + description: str = None + price: float + + +@app.get("/product") +async def create_item(product: Product): + return product + + +client = TestClient(app) + + +openapi_schema = { + "openapi": "3.0.2", + "info": {"title": "FastAPI", "version": "0.1.0"}, + "paths": { + "/product": { + "get": { + "summary": "Create Item", + "operationId": "create_item_product_get", + "requestBody": { + "content": { + "application/json": { + "schema": {"$ref": "#/components/schemas/Product"} + } + }, + "required": True, + }, + "responses": { + "200": { + "description": "Successful Response", + "content": {"application/json": {"schema": {}}}, + }, + "422": { + "description": "Validation Error", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/HTTPValidationError" + } + } + }, + }, + }, + } + } + }, + "components": { + "schemas": { + "HTTPValidationError": { + "title": "HTTPValidationError", + "type": "object", + "properties": { + "detail": { + "title": "Detail", + "type": "array", + "items": {"$ref": "#/components/schemas/ValidationError"}, + } + }, + }, + "Product": { + "title": "Product", + "required": ["name", "price"], + "type": "object", + "properties": { + "name": {"title": "Name", "type": "string"}, + "description": {"title": "Description", "type": "string"}, + "price": {"title": "Price", "type": "number"}, + }, + }, + "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"}, + }, + }, + } + }, +} + + +def test_openapi_schema(): + response = client.get("/openapi.json") + assert response.status_code == 200, response.text + assert response.json() == openapi_schema + + +def test_get_with_body(): + body = {"name": "Foo", "description": "Some description", "price": 5.5} + response = client.get("/product", json=body) + assert response.json() == body