* root_path included in servers object instead of path prefix * ♻️ Refactor implementation of auto-including root_path in OpenAPI servers * 📝 Update docs and examples for Behind a Proxy, including servers * 📝 Update Extending OpenAPI as openapi_prefix is no longer needed * ✅ Add extra tests for root_path in servers and root_path_in_servers=False * 🍱 Update security docs images with relative token URL * 📝 Update security docs with relative token URL * 📝 Update example sources with relative token URLs * ✅ Update tests with relative tokens Co-authored-by: Sebastián Ramírez <[email protected]>pull/1703/head
Before Width: | Height: | Size: 59 KiB After Width: | Height: | Size: 28 KiB |
Before Width: | Height: | Size: 28 KiB After Width: | Height: | Size: 61 KiB |
After Width: | Height: | Size: 72 KiB |
Before Width: | Height: | Size: 79 KiB After Width: | Height: | Size: 88 KiB |
Before Width: | Height: | Size: 82 KiB After Width: | Height: | Size: 82 KiB |
Before Width: | Height: | Size: 79 KiB After Width: | Height: | Size: 80 KiB |
Before Width: | Height: | Size: 86 KiB After Width: | Height: | Size: 87 KiB |
Before Width: | Height: | Size: 79 KiB After Width: | Height: | Size: 76 KiB |
@ -0,0 +1,14 @@ |
|||||
|
from fastapi import FastAPI, Request |
||||
|
|
||||
|
app = FastAPI( |
||||
|
servers=[ |
||||
|
{"url": "https://stag.example.com", "description": "Staging environment"}, |
||||
|
{"url": "https://prod.example.com", "description": "Production environment"}, |
||||
|
], |
||||
|
root_path="/api/v1", |
||||
|
) |
||||
|
|
||||
|
|
||||
|
@app.get("/app") |
||||
|
def read_main(request: Request): |
||||
|
return {"message": "Hello World", "root_path": request.scope.get("root_path")} |
@ -0,0 +1,15 @@ |
|||||
|
from fastapi import FastAPI, Request |
||||
|
|
||||
|
app = FastAPI( |
||||
|
servers=[ |
||||
|
{"url": "https://stag.example.com", "description": "Staging environment"}, |
||||
|
{"url": "https://prod.example.com", "description": "Production environment"}, |
||||
|
], |
||||
|
root_path="/api/v1", |
||||
|
root_path_in_servers=False, |
||||
|
) |
||||
|
|
||||
|
|
||||
|
@app.get("/app") |
||||
|
def read_main(request: Request): |
||||
|
return {"message": "Hello World", "root_path": request.scope.get("root_path")} |
@ -0,0 +1,41 @@ |
|||||
|
from fastapi.testclient import TestClient |
||||
|
|
||||
|
from docs_src.behind_a_proxy.tutorial003 import app |
||||
|
|
||||
|
client = TestClient(app) |
||||
|
|
||||
|
openapi_schema = { |
||||
|
"openapi": "3.0.2", |
||||
|
"info": {"title": "FastAPI", "version": "0.1.0"}, |
||||
|
"servers": [ |
||||
|
{"url": "/api/v1"}, |
||||
|
{"url": "https://stag.example.com", "description": "Staging environment"}, |
||||
|
{"url": "https://prod.example.com", "description": "Production environment"}, |
||||
|
], |
||||
|
"paths": { |
||||
|
"/app": { |
||||
|
"get": { |
||||
|
"summary": "Read Main", |
||||
|
"operationId": "read_main_app_get", |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": {"application/json": {"schema": {}}}, |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
|
||||
|
|
||||
|
def test_openapi(): |
||||
|
response = client.get("/openapi.json") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == openapi_schema |
||||
|
|
||||
|
|
||||
|
def test_main(): |
||||
|
response = client.get("/app") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == {"message": "Hello World", "root_path": "/api/v1"} |
@ -0,0 +1,40 @@ |
|||||
|
from fastapi.testclient import TestClient |
||||
|
|
||||
|
from docs_src.behind_a_proxy.tutorial004 import app |
||||
|
|
||||
|
client = TestClient(app) |
||||
|
|
||||
|
openapi_schema = { |
||||
|
"openapi": "3.0.2", |
||||
|
"info": {"title": "FastAPI", "version": "0.1.0"}, |
||||
|
"servers": [ |
||||
|
{"url": "https://stag.example.com", "description": "Staging environment"}, |
||||
|
{"url": "https://prod.example.com", "description": "Production environment"}, |
||||
|
], |
||||
|
"paths": { |
||||
|
"/app": { |
||||
|
"get": { |
||||
|
"summary": "Read Main", |
||||
|
"operationId": "read_main_app_get", |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": {"application/json": {"schema": {}}}, |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
|
||||
|
|
||||
|
def test_openapi(): |
||||
|
response = client.get("/openapi.json") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == openapi_schema |
||||
|
|
||||
|
|
||||
|
def test_main(): |
||||
|
response = client.get("/app") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == {"message": "Hello World", "root_path": "/api/v1"} |