Marc Laventure 21 hours ago
committed by GitHub
parent
commit
f36dc30f48
No known key found for this signature in database GPG Key ID: B5690EEEBB952194
  1. 2
      .github/DISCUSSION_TEMPLATE/questions.yml
  2. 20
      README.md
  3. 7
      docs/en/docs/features.md
  4. 5
      docs/en/docs/how-to/custom-docs-ui-assets.md
  5. BIN
      docs/en/docs/img/index/index-07-scalar-simple.png
  6. 0
      docs/en/docs/img/index/index-08-scalar-02.png
  7. 4
      docs/en/docs/reference/openapi/docs.md
  8. 12
      docs_src/custom_docs_ui/tutorial001.py
  9. 12
      docs_src/custom_docs_ui/tutorial002.py
  10. 39
      fastapi/applications.py
  11. 91
      fastapi/openapi/docs.py
  12. 7
      tests/test_application.py
  13. 6
      tests/test_tutorial/test_custom_docs_ui/test_tutorial001.py
  14. 6
      tests/test_tutorial/test_custom_docs_ui/test_tutorial002.py

2
.github/DISCUSSION_TEMPLATE/questions.yml

@ -42,6 +42,8 @@ body:
required: true
- label: I already checked if it is not related to FastAPI but to [ReDoc](https://github.com/Redocly/redoc).
required: true
- label: I already checked if it is not related to FastAPI but to [Scalar](https://github.com/scalar/scalar).
required: true
- type: checkboxes
id: help
attributes:

20
README.md

@ -268,12 +268,20 @@ You will see the automatic interactive API documentation (provided by <a href="h
### Alternative API docs
And now, go to <a href="http://127.0.0.1:8000/redoc" class="external-link" target="_blank">http://127.0.0.1:8000/redoc</a>.
Here are some alternatives to Swagger UI: Redoc & Scalar.
Go to <a href="http://127.0.0.1:8000/redoc" class="external-link" target="_blank">http://127.0.0.1:8000/redoc</a>.
You will see the alternative automatic documentation (provided by <a href="https://github.com/Rebilly/ReDoc" class="external-link" target="_blank">ReDoc</a>):
![ReDoc](https://fastapi.tiangolo.com/img/index/index-02-redoc-simple.png)
And now, go to <a href="http://127.0.0.1:8000/scalar" class="external-link" target="_blank">http://127.0.0.1:8000/scalar</a>.
You will see the alternative automatic documentation (provided by <a href="https://github.com/scalar/scalar" class="external-link" target="_blank">Scalar</a>):
![Scalar](https://fastapi.tiangolo.com/img/index/index-07-scalar-simple.png)
## Example upgrade
Now modify the file `main.py` to receive a body from a `PUT` request.
@ -330,12 +338,20 @@ Now go to <a href="http://127.0.0.1:8000/docs" class="external-link" target="_bl
### Alternative API docs upgrade
And now, go to <a href="http://127.0.0.1:8000/redoc" class="external-link" target="_blank">http://127.0.0.1:8000/redoc</a>.
Here are some alternatives to Swagger UI: Redoc & Scalar.
Go to <a href="http://127.0.0.1:8000/redoc" class="external-link" target="_blank">http://127.0.0.1:8000/redoc</a>.
* The alternative documentation will also reflect the new query parameter and body:
![ReDoc](https://fastapi.tiangolo.com/img/index/index-06-redoc-02.png)
And now, go to <a href="http://127.0.0.1:8000/scalar" class="external-link" target="_blank">http://127.0.0.1:8000/scalar</a>.
* The alternative documentation will also reflect the new query parameter and body:
![ReDoc](https://fastapi.tiangolo.com/img/index/index-08-scalar-02.png)
### Recap
In summary, you declare **once** the types of parameters, body, etc. as function parameters.

7
docs/en/docs/features.md

@ -19,10 +19,15 @@ Interactive API documentation and exploration web user interfaces. As the framew
![Swagger UI interaction](https://fastapi.tiangolo.com/img/index/index-03-swagger-02.png)
* Alternative API documentation with <a href="https://github.com/Rebilly/ReDoc" class="external-link" target="_blank"><strong>ReDoc</strong></a>.
* Alternative API documentation with Redoc or Scalar
* <a href="https://github.com/Rebilly/ReDoc" class="external-link" target="_blank"><strong>ReDoc</strong></a>.
![ReDoc](https://fastapi.tiangolo.com/img/index/index-06-redoc-02.png)
* <a href="https://github.com/scalar/scalar" class="external-link" target="_blank"><strong>Scalar</strong></a>.
![Scalar](https://fastapi.tiangolo.com/img/index/index-08-scalar-02.png)
### Just Modern Python
It's all based on standard **Python type** declarations (thanks to Pydantic). No new syntax to learn. Just standard modern Python.

5
docs/en/docs/how-to/custom-docs-ui-assets.md

@ -100,6 +100,10 @@ And **ReDoc** uses the file:
* <a href="https://cdn.jsdelivr.net/npm/redoc@next/bundles/redoc.standalone.js" class="external-link" target="_blank">`redoc.standalone.js`</a>
And **Scalar** uses the file:
* <a href="https://cdn.jsdelivr.net/npm/@scalar/api-reference" class="external-link" target="_blank">`scalar.standalone.js`</a>
After that, your file structure could look like:
```
@ -109,6 +113,7 @@ After that, your file structure could look like:
│   ├── main.py
└── static
├── redoc.standalone.js
├── scalar.standalone.js
├── swagger-ui-bundle.js
└── swagger-ui.css
```

BIN
docs/en/docs/img/index/index-07-scalar-simple.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 169 KiB

0
docs/en/docs/img/index/index-08-scalar-02.png

4
docs/en/docs/reference/openapi/docs.md

@ -1,11 +1,13 @@
# OpenAPI `docs`
Utilities to handle OpenAPI automatic UI documentation, including Swagger UI (by default at `/docs`) and ReDoc (by default at `/redoc`).
Utilities to handle OpenAPI automatic UI documentation, including Swagger UI (by default at `/docs`), ReDoc (by default at `/redoc`) & Scalar (by default at `/scalar`).
::: fastapi.openapi.docs.get_swagger_ui_html
::: fastapi.openapi.docs.get_redoc_html
::: fastapi.openapi.docs.get_scalar_html
::: fastapi.openapi.docs.get_swagger_ui_oauth2_redirect_html
::: fastapi.openapi.docs.swagger_ui_default_parameters

12
docs_src/custom_docs_ui/tutorial001.py

@ -1,11 +1,12 @@
from fastapi import FastAPI
from fastapi.openapi.docs import (
get_redoc_html,
get_scalar_html,
get_swagger_ui_html,
get_swagger_ui_oauth2_redirect_html,
)
app = FastAPI(docs_url=None, redoc_url=None)
app = FastAPI(docs_url=None, redoc_url=None, scalar_url=None)
@app.get("/docs", include_in_schema=False)
@ -33,6 +34,15 @@ async def redoc_html():
)
@app.get("/scalar", include_in_schema=False)
async def scalar_html():
return get_scalar_html(
openapi_url=app.openapi_url,
title=app.title + " - Scalar",
scalar_js_url="https://cdn.jsdelivr.net/npm/@scalar/api-reference",
)
@app.get("/users/{username}")
async def read_user(username: str):
return {"message": f"Hello {username}"}

12
docs_src/custom_docs_ui/tutorial002.py

@ -1,12 +1,13 @@
from fastapi import FastAPI
from fastapi.openapi.docs import (
get_redoc_html,
get_scalar_html,
get_swagger_ui_html,
get_swagger_ui_oauth2_redirect_html,
)
from fastapi.staticfiles import StaticFiles
app = FastAPI(docs_url=None, redoc_url=None)
app = FastAPI(docs_url=None, redoc_url=None, scalar_url=None)
app.mount("/static", StaticFiles(directory="static"), name="static")
@ -36,6 +37,15 @@ async def redoc_html():
)
@app.get("/scalar", include_in_schema=False)
async def scalar_html():
return get_scalar_html(
openapi_url=app.openapi_url,
title=app.title + " - Scalar",
scalar_js_url="/static/scalar.standalone.js",
)
@app.get("/users/{username}")
async def read_user(username: str):
return {"message": f"Hello {username}"}

39
fastapi/applications.py

@ -24,6 +24,7 @@ from fastapi.exceptions import RequestValidationError, WebSocketRequestValidatio
from fastapi.logger import logger
from fastapi.openapi.docs import (
get_redoc_html,
get_scalar_html,
get_swagger_ui_html,
get_swagger_ui_oauth2_redirect_html,
)
@ -415,7 +416,7 @@ class FastAPI(Starlette):
```python
from fastapi import FastAPI
app = FastAPI(docs_url="/documentation", redoc_url=None)
app = FastAPI(docs_url="/documentation", redoc_url=None, scalar_url=None)
```
"""
),
@ -444,6 +445,30 @@ class FastAPI(Starlette):
"""
),
] = "/redoc",
scalar_url: Annotated[
Optional[str],
Doc(
"""
The path to the alternative automatic interactive API documentation
provided by Scalar.
The default URL is `/scalar`. You can disable it by setting it to `None`.
If `openapi_url` is set to `None`, this will be automatically disabled.
Read more in the
[FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#docs-urls).
**Example**
```python
from fastapi import FastAPI
app = FastAPI(docs_url="/documentation", scalar_url="scalar-docs")
```
"""
),
] = "/scalar",
swagger_ui_oauth2_redirect_url: Annotated[
Optional[str],
Doc(
@ -833,6 +858,7 @@ class FastAPI(Starlette):
self.root_path_in_servers = root_path_in_servers
self.docs_url = docs_url
self.redoc_url = redoc_url
self.scalar_url = scalar_url
self.swagger_ui_oauth2_redirect_url = swagger_ui_oauth2_redirect_url
self.swagger_ui_init_oauth = swagger_ui_init_oauth
self.swagger_ui_parameters = swagger_ui_parameters
@ -1048,6 +1074,17 @@ class FastAPI(Starlette):
self.add_route(self.redoc_url, redoc_html, include_in_schema=False)
if self.openapi_url and self.scalar_url:
async def scalar_html(req: Request) -> HTMLResponse:
root_path = req.scope.get("root_path", "").rstrip("/")
openapi_url = root_path + self.openapi_url
return get_scalar_html(
openapi_url=openapi_url, title=self.title + " - Scalar"
)
self.add_route(self.scalar_url, scalar_html, include_in_schema=False)
async def __call__(self, scope: Scope, receive: Receive, send: Send) -> None:
if self.root_path:
scope["root_path"] = self.root_path

91
fastapi/openapi/docs.py

@ -158,6 +158,97 @@ def get_swagger_ui_html(
return HTMLResponse(html)
def get_scalar_html(
*,
openapi_url: Annotated[
str,
Doc(
"""
The OpenAPI URL that Scalar should load and use.
This is normally done automatically by FastAPI using the default URL
`/openapi.json`.
"""
),
],
title: Annotated[
str,
Doc(
"""
The HTML `<title>` content, normally shown in the browser tab.
"""
),
],
scalar_js_url: Annotated[
str,
Doc(
"""
The URL to use to load the Scalar JavaScript.
It is normally set to a CDN URL.
"""
),
] = "https://cdn.jsdelivr.net/npm/@scalar/api-reference",
scalar_proxy_url: Annotated[
str,
Doc(
"""
The URL to use to set the Scalar Proxy.
It is normally set to a Scalar API URL (https://api.scalar.com/request-proxy), but default is empty
"""
),
] = "",
scalar_favicon_url: Annotated[
str,
Doc(
"""
The URL of the favicon to use. It is normally shown in the browser tab.
"""
),
] = "https://fastapi.tiangolo.com/img/favicon.png",
) -> HTMLResponse:
"""
Generate and return the HTML response that loads Scalar for the alternative
API docs (normally served at `/scalar`).
You would only call this function yourself if you needed to override some parts,
for example the URLs to use to load Scalar's JavaScript and CSS.
Read more about it in the
[FastAPI docs for Custom Docs UI Static Assets (Self-Hosting)](https://fastapi.tiangolo.com/how-to/custom-docs-ui-assets/).
"""
html = f"""
<!DOCTYPE html>
<html>
<head>
<title>{title}</title>
<!-- needed for adaptive design -->
<meta charset="utf-8"/>
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="shortcut icon" href="{scalar_favicon_url}">
<style>
body {{
margin: 0;
padding: 0;
}}
</style>
</head>
<body>
<noscript>
Scalar requires Javascript to function. Please enable it to browse the documentation.
</noscript>
<script
id="api-reference"
data-url="{openapi_url}"
data-proxy-url="{scalar_proxy_url}"></script>
<script src="{scalar_js_url}"></script>
</body>
</html>
"""
return HTMLResponse(html)
def get_redoc_html(
*,
openapi_url: Annotated[

7
tests/test_application.py

@ -46,6 +46,13 @@ def test_redoc():
assert "redoc@next" in response.text
def test_scalar():
response = client.get("/scalar")
assert response.status_code == 200, response.text
assert response.headers["content-type"] == "text/html; charset=utf-8"
assert "@scalar/api-reference" in response.text
def test_enum_status_code_response():
response = client.get("/enum-status-code")
assert response.status_code == 201, response.text

6
tests/test_tutorial/test_custom_docs_ui/test_tutorial001.py

@ -36,6 +36,12 @@ def test_redoc_html(client: TestClient):
assert "https://unpkg.com/redoc@next/bundles/redoc.standalone.js" in response.text
def test_scalar_html(client: TestClient):
response = client.get("/scalar")
assert response.status_code == 200, response.text
assert "https://cdn.jsdelivr.net/npm/@scalar/api-reference" in response.text
def test_api(client: TestClient):
response = client.get("/users/john")
assert response.status_code == 200, response.text

6
tests/test_tutorial/test_custom_docs_ui/test_tutorial002.py

@ -36,6 +36,12 @@ def test_redoc_html(client: TestClient):
assert "/static/redoc.standalone.js" in response.text
def test_scalar_html(client: TestClient):
response = client.get("/scalar")
assert response.status_code == 200, response.text
assert "/static/scalar.standalone.js" in response.text
def test_api(client: TestClient):
response = client.get("/users/john")
assert response.status_code == 200, response.text

Loading…
Cancel
Save