From 94ee932351ac25a40ae9eb0adcc2de7376fe1b9e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sebasti=C3=A1n=20Ram=C3=ADrez?= Date: Sun, 1 Mar 2020 22:30:58 +0100 Subject: [PATCH] :sparkles: Add ORJSONResponse (#1065) * :sparkles: Add ORJSONResponse * :memo: Add tutorial using ORJSONResponse * :white_check_mark: Add test for ORJSONResponse * :memo: Update index.md --- README.md | 1 + docs/advanced/custom-response.md | 22 +++++++++--- docs/index.md | 1 + docs/src/custom_response/tutorial001b.py | 9 +++++ fastapi/responses.py | 15 ++++++++ .../test_custom_response/test_tutorial001b.py | 36 +++++++++++++++++++ 6 files changed, 80 insertions(+), 4 deletions(-) create mode 100644 docs/src/custom_response/tutorial001b.py create mode 100644 tests/test_tutorial/test_custom_response/test_tutorial001b.py diff --git a/README.md b/README.md index 9f9c44097..8801c1af5 100644 --- a/README.md +++ b/README.md @@ -398,6 +398,7 @@ Used by Starlette: Used by FastAPI / Starlette: * uvicorn - for the server that loads and serves your application. +* orjson - Required if you want to use `ORJSONResponse`. You can install all of these with `pip install fastapi[all]`. diff --git a/docs/advanced/custom-response.md b/docs/advanced/custom-response.md index 5e4504023..21e7abee4 100644 --- a/docs/advanced/custom-response.md +++ b/docs/advanced/custom-response.md @@ -13,14 +13,14 @@ And if that `Response` has a JSON media type (`application/json`), like is the c !!! note If you use a response class with no media type, FastAPI will expect your response to have no content, so it will not document the response format in its generated OpenAPI docs. -## Use `UJSONResponse` +## Use `ORJSONResponse` -For example, if you are squeezing performance, you can install and use `ujson` and set the response to be `UJSONResponse`. +For example, if you are squeezing performance, you can install and use `orjson` and set the response to be `ORJSONResponse`. Import the `Response` class (sub-class) you want to use and declare it in the *path operation decorator*. ```Python hl_lines="2 7" -{!./src/custom_response/tutorial001.py!} +{!./src/custom_response/tutorial001b.py!} ``` !!! info @@ -30,6 +30,9 @@ Import the `Response` class (sub-class) you want to use and declare it in the *p And it will be documented as such in OpenAPI. +!!! tip + The `ORJSONResponse` is currently only available in FastAPI, not in Starlette. + ## HTML Response To return a response with HTML directly from **FastAPI**, use `HTMLResponse`. @@ -134,13 +137,24 @@ Takes some data and returns an `application/json` encoded response. This is the default response used in **FastAPI**, as you read above. +### `ORJSONResponse` + +A fast alternative JSON response using `orjson`, as you read above. + ### `UJSONResponse` -An alternative JSON response using `ujson` for faster serialization as you read above. +An alternative JSON response using `ujson`. !!! warning `ujson` is less careful than Python's built-in implementation in how it handles some edge-cases. +```Python hl_lines="2 7" +{!./src/custom_response/tutorial001.py!} +``` + +!!! tip + It's possible that `ORJSONResponse` might be a faster alternative. + ### `RedirectResponse` Returns an HTTP redirect. Uses a 307 status code (Temporary Redirect) by default. diff --git a/docs/index.md b/docs/index.md index 9f9c44097..8801c1af5 100644 --- a/docs/index.md +++ b/docs/index.md @@ -398,6 +398,7 @@ Used by Starlette: Used by FastAPI / Starlette: * uvicorn - for the server that loads and serves your application. +* orjson - Required if you want to use `ORJSONResponse`. You can install all of these with `pip install fastapi[all]`. diff --git a/docs/src/custom_response/tutorial001b.py b/docs/src/custom_response/tutorial001b.py new file mode 100644 index 000000000..e98460372 --- /dev/null +++ b/docs/src/custom_response/tutorial001b.py @@ -0,0 +1,9 @@ +from fastapi import FastAPI +from fastapi.responses import ORJSONResponse + +app = FastAPI() + + +@app.get("/items/", response_class=ORJSONResponse) +async def read_items(): + return [{"item_id": "Foo"}] diff --git a/fastapi/responses.py b/fastapi/responses.py index 69bf5c91c..0aeff61d0 100644 --- a/fastapi/responses.py +++ b/fastapi/responses.py @@ -1,3 +1,5 @@ +from typing import Any + from starlette.responses import FileResponse # noqa from starlette.responses import HTMLResponse # noqa from starlette.responses import JSONResponse # noqa @@ -6,3 +8,16 @@ from starlette.responses import RedirectResponse # noqa from starlette.responses import Response # noqa from starlette.responses import StreamingResponse # noqa from starlette.responses import UJSONResponse # noqa + +try: + import orjson +except ImportError: # pragma: nocover + orjson = None # type: ignore + + +class ORJSONResponse(JSONResponse): + media_type = "application/json" + + def render(self, content: Any) -> bytes: + assert orjson is not None, "orjson must be installed to use ORJSONResponse" + return orjson.dumps(content) diff --git a/tests/test_tutorial/test_custom_response/test_tutorial001b.py b/tests/test_tutorial/test_custom_response/test_tutorial001b.py new file mode 100644 index 000000000..08f372897 --- /dev/null +++ b/tests/test_tutorial/test_custom_response/test_tutorial001b.py @@ -0,0 +1,36 @@ +from fastapi.testclient import TestClient + +from custom_response.tutorial001b 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", + } + } + }, +} + + +def test_openapi_schema(): + response = client.get("/openapi.json") + assert response.status_code == 200 + assert response.json() == openapi_schema + + +def test_get_custom_response(): + response = client.get("/items/") + assert response.status_code == 200 + assert response.json() == [{"item_id": "Foo"}]