Browse Source
Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> Co-authored-by: Yurii Motov <yurii.motov.monte@gmail.com> Co-authored-by: Sebastián Ramírez <tiangolo@gmail.com>pull/10647/merge
committed by
GitHub
2 changed files with 182 additions and 1 deletions
@ -0,0 +1,180 @@ |
|||
from typing import Any, Iterator, Set, Type |
|||
|
|||
import fastapi._compat |
|||
import fastapi.openapi.utils |
|||
import pydantic.schema |
|||
import pytest |
|||
from fastapi import FastAPI |
|||
from pydantic import BaseModel |
|||
from starlette.testclient import TestClient |
|||
|
|||
from .utils import needs_pydanticv1 |
|||
|
|||
|
|||
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.1.0", |
|||
"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): |
|||
""" |
|||
Set of Types whose `__iter__()` method yields results sorted by the type names |
|||
""" |
|||
|
|||
def __init__(self, seq: Set[Type[Any]], *, sort_reversed: bool): |
|||
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, |
|||
) |
|||
yield from members_sorted |
|||
|
|||
|
|||
@needs_pydanticv1 |
|||
@pytest.mark.parametrize("sort_reversed", [True, False]) |
|||
def test_model_description_escaped_with_formfeed(sort_reversed: bool): |
|||
""" |
|||
Regression test for bug fixed by https://github.com/fastapi/fastapi/pull/6039. |
|||
|
|||
Test `get_model_definitions` with models passed in different order. |
|||
""" |
|||
all_fields = fastapi.openapi.utils.get_fields_from_routes(app.routes) |
|||
|
|||
flat_models = fastapi._compat.get_flat_models_from_fields( |
|||
all_fields, known_models=set() |
|||
) |
|||
model_name_map = pydantic.schema.get_model_name_map(flat_models) |
|||
|
|||
expected_address_description = "This is a public description of an Address\n" |
|||
|
|||
models = fastapi._compat.get_model_definitions( |
|||
flat_models=SortedTypeSet(flat_models, sort_reversed=sort_reversed), |
|||
model_name_map=model_name_map, |
|||
) |
|||
assert models["Address"]["description"] == expected_address_description |
Loading…
Reference in new issue