Browse Source
* ✨ Allow disabling docs UIs by disabling openapi_url * 📝 Add docs for disabling OpenAPI and docs in prod or other environments * ✅ Add tests for disabling OpenAPI and docspull/1423/head
committed by
GitHub
7 changed files with 122 additions and 4 deletions
@ -0,0 +1,58 @@ |
|||
# Conditional OpenAPI |
|||
|
|||
If you needed to, you could use settings and environment variables to configure OpenAPI conditionally depending on the environment, and even disable it entirely. |
|||
|
|||
## About security, APIs, and docs |
|||
|
|||
Hiding your documentation user interfaces in production *shouldn't* be the way to protect your API. |
|||
|
|||
That doesn't add any extra security to your API, the *path operations* will still be available where they are. |
|||
|
|||
If there's a security flaw in your code, it will still exist. |
|||
|
|||
Hiding the documentation just makes it more difficult to understand how to interact with your API, and could make it more difficult for you to debug it in production. It could be considered simply a form of <a href="https://en.wikipedia.org/wiki/Security_through_obscurity" class="external-link" target="_blank">Security through obscurity</a>. |
|||
|
|||
If you want to secure your API, there are several better things you can do, for example: |
|||
|
|||
* Make sure you have well defined Pydantic models for your request bodies and responses. |
|||
* Configure any required permissions and roles using dependencies. |
|||
* Never store plaintext passwords, only password hashes. |
|||
* Implement and use well-known cryptographic tools, like Passlib and JWT tokens, etc. |
|||
* Add more granular permission controls with OAuth2 scopes where needed. |
|||
* ...etc. |
|||
|
|||
Nevertheless, you might have a very specific use case where you really need to disable the API docs for some environment (e.g. for production) or depending on configurations from environment variables. |
|||
|
|||
## Conditional OpenAPI from settings and env vars |
|||
|
|||
You can easily use the same Pydantic settings to configure your generated OpenAPI and the docs UIs. |
|||
|
|||
For example: |
|||
|
|||
```Python hl_lines="6 11" |
|||
{!../../../docs_src/conditional_openapi/tutorial001.py!} |
|||
``` |
|||
|
|||
Here we declare the setting `openapi_url` with the same default of `"/openapi.json"`. |
|||
|
|||
And then we use it when creating the `FastAPI` app. |
|||
|
|||
Then you could disable OpenAPI (including the UI docs) by setting the environment variable `OPENAPI_URL` to the empty string, like: |
|||
|
|||
<div class="termy"> |
|||
|
|||
```console |
|||
$ OPENAPI_URL= uvicorn main:app |
|||
|
|||
<span style="color: green;">INFO</span>: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit) |
|||
``` |
|||
|
|||
</div> |
|||
|
|||
Then if you go to the URLs at `/openapi.json`, `/docs`, or `/redoc` you will just get a `404 Not Found` error like: |
|||
|
|||
```JSON |
|||
{ |
|||
"detail": "Not Found" |
|||
} |
|||
``` |
@ -0,0 +1,16 @@ |
|||
from fastapi import FastAPI |
|||
from pydantic import BaseSettings |
|||
|
|||
|
|||
class Settings(BaseSettings): |
|||
openapi_url: str = "/openapi.json" |
|||
|
|||
|
|||
settings = Settings() |
|||
|
|||
app = FastAPI(openapi_url=settings.openapi_url) |
|||
|
|||
|
|||
@app.get("/") |
|||
def root(): |
|||
return {"message": "Hello World"} |
@ -0,0 +1,46 @@ |
|||
import importlib |
|||
|
|||
from fastapi.testclient import TestClient |
|||
|
|||
from conditional_openapi import tutorial001 |
|||
|
|||
openapi_schema = { |
|||
"openapi": "3.0.2", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/": { |
|||
"get": { |
|||
"summary": "Root", |
|||
"operationId": "root__get", |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
} |
|||
}, |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
|
|||
|
|||
def test_default_openapi(): |
|||
client = TestClient(tutorial001.app) |
|||
response = client.get("/openapi.json") |
|||
assert response.json() == openapi_schema |
|||
response = client.get("/docs") |
|||
assert response.status_code == 200, response.text |
|||
response = client.get("/redoc") |
|||
assert response.status_code == 200, response.text |
|||
|
|||
|
|||
def test_disable_openapi(monkeypatch): |
|||
monkeypatch.setenv("OPENAPI_URL", "") |
|||
importlib.reload(tutorial001) |
|||
client = TestClient(tutorial001.app) |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 404, response.text |
|||
response = client.get("/docs") |
|||
assert response.status_code == 404, response.text |
|||
response = client.get("/redoc") |
|||
assert response.status_code == 404, response.text |
Loading…
Reference in new issue