FastAPIMiddleTechnical
Как работает система type hints в FastAPI с Optional-полями и типами Union?
Optional[X] эквивалентен Union[X, None] и делает поле необязательным со значением None по умолчанию. FastAPI читает аннотации через Pydantic: Union генерирует anyOf в OpenAPI, Optional добавляет null в схему. В Python 3.10+ можно писать X | None.
Type hints, Optional и Union в FastAPI
FastAPI использует Python type hints для трёх вещей сразу: валидация входных данных (через Pydantic), сериализация ответов и генерация OpenAPI-схемы. Понимание Optional и Union критично для корректной работы всего этого.
Optional[X] — поле может быть None
from typing import Optional
from pydantic import BaseModel
class JobFilter(BaseModel):
title: str # обязательно, не может быть None
location: Optional[str] = None # необязательно, по умолчанию None
min_salary: Optional[int] = None
В OpenAPI это транслируется в "location": {"anyOf": [{"type": "string"}, {"type": "null"}]}.
Разница: Optional без default vs с default
from pydantic import BaseModel
from typing import Optional
class Example(BaseModel):
a: Optional[str] # обязательное поле, но может быть null
b: Optional[str] = None # необязательное поле, default = None
# Example(a=None) — OK
# Example() — ошибка: a обязательно
# Example(b="hi") — OK
Union — несколько допустимых типов
from typing import Union
from pydantic import BaseModel
class TextBlock(BaseModel):
type: str = "text"
content: str
class ImageBlock(BaseModel):
type: str = "image"
url: str
alt: Optional[str] = None
class Message(BaseModel):
blocks: list[Union[TextBlock, ImageBlock]]
Discriminated Union — быстрее и точнее
from typing import Annotated, Union, Literal
from pydantic import BaseModel, Field
class TextBlock(BaseModel):
type: Literal["text"]
content: str
class ImageBlock(BaseModel):
type: Literal["image"]
url: str
Block = Annotated[
Union[TextBlock, ImageBlock],
Field(discriminator="type"),
]
class Message(BaseModel):
block: Block
# Pydantic выберет нужный тип по полю "type" без перебора вариантов
Python 3.10+ синтаксис
class Item(BaseModel):
name: str
description: str | None = None # то же что Optional[str] = None
price: int | float # то же что Union[int, float]
Union в Query-параметрах
from fastapi import FastAPI, Query
from typing import Annotated
app = FastAPI()
@app.get("/search/")
async def search(
q: Annotated[str | None, Query(min_length=2)] = None,
) -> dict:
return {"query": q}
Подводные камни
- В Pydantic v2
Optional[str]без default делает поле обязательным (значение должно быть передано, пусть иNone). В v1 поведение было другим — частый источник миграционных багов. Union[str, int]в Pydantic v2 попытается привести значение к первому подходящему типу; порядок элементов в Union важен.- OpenAPI генерирует
anyOfдля Union — не все клиенты кодогенерации поддерживают это корректно. - Discriminated Union требует, чтобы дискриминирующее поле было типа
Literal— обычныйstrне сработает. - При использовании
|(Python 3.10+) в аргументах FastAPI убедитесь, что версия Pydantic >= 2 и Python >= 3.10; в старых окружениях получитеTypeError. - Тип
Anyиз typing отключает валидацию Pydantic для конкретного поля — не используйте без крайней необходимости. - Response model с
Unionиfrom_attributes=Trueможет некорректно выбрать тип, если ORM-объект совместим с несколькими вариантами — используйте discriminated union. Optional[list[str]]иlist[Optional[str]]— разные типы: первый означает «список или None», второй — «список, элементы которого могут быть None».
Common mistakes
- Описывать type hints optional union только как термин и не показывать механизм на минимальном примере.
- Игнорировать ошибки, пустые данные, конкурентный доступ или границы транзакции.
- Не связывать поведение с официальным контрактом FastAPI и реальной эксплуатацией.
What the interviewer is testing
- Объясняет type hints optional union через последовательность действий, а не через набор ключевых слов.
- Приводит короткий кодовый пример или production-сценарий с ожидаемым поведением.
- Называет хотя бы один риск: производительность, безопасность, транзакции, память или сопровождение.