2 changed files with 194 additions and 1 deletions
@ -0,0 +1,192 @@ |
|||
from typing import Any, Iterator, Set, Type |
|||
|
|||
import fastapi.openapi.utils |
|||
import fastapi.utils |
|||
import pydantic.schema |
|||
from fastapi import FastAPI |
|||
from pydantic import BaseModel |
|||
from starlette.testclient import TestClient |
|||
|
|||
|
|||
class Address(BaseModel): |
|||
""" |
|||
This is a public description of an Address |
|||
\f |
|||
You can't see this part of the docstring, it's private! |
|||
""" |
|||
|
|||
line_1: str |
|||
city: str |
|||
state_province: str |
|||
|
|||
|
|||
class Facility(BaseModel): |
|||
id: str |
|||
address: Address |
|||
|
|||
|
|||
app = FastAPI() |
|||
|
|||
client = TestClient(app) |
|||
|
|||
|
|||
@app.get("/facilities/{facility_id}") |
|||
def get_facility(facility_id: str) -> Facility: |
|||
... |
|||
|
|||
|
|||
openapi_schema = { |
|||
"components": { |
|||
"schemas": { |
|||
"Address": { |
|||
# NOTE: the description of this model shows only the public-facing text, before the `\f` in docstring |
|||
"description": "This is a public " "description of an Address\n", |
|||
"properties": { |
|||
"city": {"title": "City", "type": "string"}, |
|||
"line_1": {"title": "Line " "1", "type": "string"}, |
|||
"state_province": {"title": "State " "Province", "type": "string"}, |
|||
}, |
|||
"required": ["line_1", "city", "state_province"], |
|||
"title": "Address", |
|||
"type": "object", |
|||
}, |
|||
"Facility": { |
|||
"properties": { |
|||
"address": {"$ref": "#/components/schemas/Address"}, |
|||
"id": {"title": "Id", "type": "string"}, |
|||
}, |
|||
"required": ["id", "address"], |
|||
"title": "Facility", |
|||
"type": "object", |
|||
}, |
|||
"HTTPValidationError": { |
|||
"properties": { |
|||
"detail": { |
|||
"items": {"$ref": "#/components/schemas/ValidationError"}, |
|||
"title": "Detail", |
|||
"type": "array", |
|||
} |
|||
}, |
|||
"title": "HTTPValidationError", |
|||
"type": "object", |
|||
}, |
|||
"ValidationError": { |
|||
"properties": { |
|||
"loc": { |
|||
"items": {"anyOf": [{"type": "string"}, {"type": "integer"}]}, |
|||
"title": "Location", |
|||
"type": "array", |
|||
}, |
|||
"msg": {"title": "Message", "type": "string"}, |
|||
"type": {"title": "Error " "Type", "type": "string"}, |
|||
}, |
|||
"required": ["loc", "msg", "type"], |
|||
"title": "ValidationError", |
|||
"type": "object", |
|||
}, |
|||
} |
|||
}, |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"openapi": "3.0.2", |
|||
"paths": { |
|||
"/facilities/{facility_id}": { |
|||
"get": { |
|||
"operationId": "get_facility_facilities__facility_id__get", |
|||
"parameters": [ |
|||
{ |
|||
"in": "path", |
|||
"name": "facility_id", |
|||
"required": True, |
|||
"schema": {"title": "Facility " "Id", "type": "string"}, |
|||
} |
|||
], |
|||
"responses": { |
|||
"200": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/Facility"} |
|||
} |
|||
}, |
|||
"description": "Successful " "Response", |
|||
}, |
|||
"422": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
"description": "Validation " "Error", |
|||
}, |
|||
}, |
|||
"summary": "Get Facility", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
|
|||
|
|||
def test_openapi_schema(): |
|||
""" |
|||
Sanity check to ensure our app's openapi schema renders as we expect |
|||
""" |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == openapi_schema |
|||
|
|||
|
|||
class SortedTypeSet(Set[Type[Any]]): |
|||
""" |
|||
Set of Types whose `__iter__()` method yields results sorted by the type names |
|||
""" |
|||
|
|||
def __init__(self, seq: Set[Type[Any]], *, sort_reversed: bool): |
|||
""" |
|||
:param seq: Initial members of this set |
|||
:param sort_reversed: If true, reverse-order the sorting by type name during iteration |
|||
""" |
|||
super().__init__(seq) |
|||
self.sort_reversed = sort_reversed |
|||
|
|||
def __iter__(self) -> Iterator[Type[Any]]: |
|||
members_sorted = sorted( |
|||
super().__iter__(), |
|||
key=lambda type_: type_.__name__, |
|||
reverse=self.sort_reversed, |
|||
) |
|||
for member in members_sorted: |
|||
yield member |
|||
|
|||
|
|||
def test_model_description_escaped_with_formfeed(): |
|||
""" |
|||
Ensure that openapi model descriptions that originate from Pydantic docstrings always truncate the docstring to text |
|||
that falls before the formfeed (\f) character. This feature was introduced in (https://github.com/tiangolo/fastapi/pull/3032). |
|||
When originally introduced, there was a possibility that the truncation may be ignored depending on the order in which |
|||
the models got processed. This created non-deterministic errors, since Pydantic model processing uses unordered sets |
|||
and model ordering may differ from one invocation to the next. |
|||
|
|||
This test verifies that (\f) escape of docstrings works in all possible orderings of our two Pydantic model classes. |
|||
""" |
|||
flat_models = fastapi.openapi.utils.get_flat_models_from_routes(app.routes) |
|||
model_name_map = pydantic.schema.get_model_name_map(flat_models) |
|||
|
|||
expected_address_description = "This is a public description of an Address\n" |
|||
|
|||
models_when_sorted_asc = fastapi.utils.get_model_definitions( |
|||
flat_models=SortedTypeSet(flat_models, sort_reversed=False), |
|||
model_name_map=model_name_map, |
|||
) |
|||
assert ( |
|||
models_when_sorted_asc["Address"]["description"] == expected_address_description |
|||
) |
|||
|
|||
models_when_sorted_desc = fastapi.utils.get_model_definitions( |
|||
flat_models=SortedTypeSet(flat_models, sort_reversed=True), |
|||
model_name_map=model_name_map, |
|||
) |
|||
assert ( |
|||
models_when_sorted_desc["Address"]["description"] |
|||
== expected_address_description |
|||
) |
Loading…
Reference in new issue