반응형
지난 포스트에서는 FastAPI에서 HTTP API를 정의할 때 Path Operation과 Path parameter를 정의하는 방법에 대해 알아보았습니다. 이번 포스트에서는 Query parameter와 Request body를 정의하는 방법을 알아보겠습니다.
[Python/FastAPI] 1. API 정의 - HTTP Request (1)
FastAPI에서의 Query Parameter 정의
# query parameter 이용
@app.get("/items")
def read_item(page: int = 0, size: int = 10):
return items[page * size: (page + 1) * size]
# optional 설정
@app.get("/items/{item_id}")
def read_item(item_id: int, query: Union[str, None] = None):
if query:
return {"item_id": item_id, "query": query}
return {"item_id": item_id}
- 쿼리 파라미터는 URL에 포함되므로 항상 문자열이지만, 타입을 지정해주면 FastAPI가 검증 및 파싱해준다.
- 쿼리 파라미터는 경로에서 필수적으로 고정되는 부분이 아니기 때문에 선택적이므로 기본값을 지정해줄 수 있다.
- 기본값을 설정하면 파라미터 값이 없을 때 해당 값이 할당된다.
- 기본값을 None으로 설정하면 optional 파라미터로 선언할 수 있다.
- bool 타입으로 지정하면 아래의 값들을 bool 타입으로 변환해준다.
- 1, True, true, on, yes 등 → True
- 0, False, false, off, no 등 → False
- List 타입으로 지정하면 해당 쿼리 파라미터의 리스트 생성해준다.
- q: Union[List[str], None] = Query(default=["a", "b", "c"])
- …?q=foo&q=bar
- → q: [”foo”, “bar”]
- List 타입으로 지정하면서 기본값 Query를 지정해주지 않으면 Request body로 인식한다.
Query로 추가기능 사용하기
from fastapi import Query
...
# use Query as default
@app.get("/items")
async def read_items(q: Union[str, None] = Query(default=None, max_length=50)):
results = {"items": [{"item_id": "Foo"}, {"item_id": "Bar"}]}
if q:
results.update({"q", q})
return results
# q is required
@app.get("/items")
async def read_items(q: Union[str, None] = Query(max_length=50)):
results = {"items": [{"item_id": "Foo"}, {"item_id": "Bar"}]}
if q:
results.update({"q", q})
return results
- fastapi의 Query를 사용해서 쿼리 파라미터에 제약조건을 추가할 수 있다.
- 쿼리 파라미터는 항상 string이므로 max_length, min_length, regex를 설정할 수 있다.
- 이 방식은 Query() 자체를 해당 파라미터의 기본값으로 사용하게 되는데, 해당 쿼리 파라미터를 required하게 만드려면 Query() 내부에 default 옵션을 정의하지 않으면 된다.
- required 파라미터임을 명시적으로 표시하고 싶다면 아래의 방법들을 사용하면 된다.
- q: Union[str, None] = Query(default=..., max_length=50)
- q: Union[str, None] = Query(default=Required, max_length=50)
- (from pydantic import Required)
- 해당 파라미터의 입력명과, 파라미터 이름을 다르게 하려면 아래처럼 alias를 설정해주면 된다.
- q: Union[str, None] = Query(default=None, alias="item-query")
- → 요청에서는 item-query를 사용하고, 코드에서는 q를 사용한다.
- 파라미터 deprecated 시키기
- 더이상 사용하지 않는 쿼리 파라미터지만 클라이언트가 계속해서 해당 값을 요청에 포함하는경우, Query(…, deprecated=True) 값만 추가하면 deprecated 되었음을 문서에 표현해준다.
FastAPI에서 Request body 정의
body에 순수 list 또는 임의의 dict 받기
@app.post("/tags")
async def create_multiple_tags(tags: List[str]):
return tags
@app.post("/kv-store")
async def get_kv_store(kv: Dict[int, str]):
return kv
- List 또는 Dict 자체를 파라미터로 넣어 요청 바디로 받을 수 있다.
BaseModel로 Request body에 객체 받기
from pydantic import BaseModel
class Item(BaseModel):
name: str
description: Union[str, None] = None
price: float
tax: Union[str, None] = None
tags: List[str] = []
...
@app.post("/items")
async def create_item(item: Item):
return item
- pydantic의 BaseModel을 상속하는 서브 클래스를 생성한다.
- 파라미터의 경우와 마찬가지로 기본값을 설정할 수 있으며, 기본값이 None으로 설정되면 optional 조건을 줄 수 있다.
- 파라미터에 해당 서브 클래스 타입을 지정하면 FastAPI가 아래의 동작들을 수행한다.
- HTTP 요청 바디를 JSON으로 읽어온다.
- 데이터를 검증한다.
- 데이터가 invalid하면 어디가 문제인지에 대한 정보를 포함하여 에러를 반환한다.
- 정의한 파라미터로 데이터를 전달해준다. (item)
여러개의 BaseModel 파라미터
class Item(BaseModel):
name: str
description: Union[str, None] = None
price: float
tax: Union[str, None] = None
class User(BaseModel):
username: str
full_name: Union[str, None] = None
...
@app.post("/items")
async def create_item(item: Item, user: User):
return {"item": item, "user": user}
- 한 메소드에 대해 여러개의 파라미터가 BaseModel을 상속하는 타입이라면, FastAPI는 파라미터의 이름을 key로 사용하여 요청 바디를 파싱한다.
{
"item":
{ "name": "Foo", "description": "The pretender", "price": 42.0, "tax": 3.2 },
"user":
{ "username": "dave", "full_name": "Dave Grohl" }
}
→ "item" 부분을 item 파라미터에, "user" 부분을 user 파라미터에 각각 매핑해준다.
→ 한 개만 사용하는 경우에는 "item"과 같은 별도의 Key가 필요 없다.
Body로 Request body 받기
- singlular 타입의 값을 다른 객체 타입들과 함께 요청 바디로 받기 위해서는 fastapi의 Body를 사용하면 된다.
import fastapi import Body
...
@app.post("/items")
async def create_item(item: Item, user: User, count: int = Body()):
return {"item": item, "user": user, "count": count}
{
"item": { "name": "Foo", "description": "The pretender", "price": 42.0, "tax": 3.2 },
"user": { "username": "dave", "full_name": "Dave Grohl" },
"count": 5
}
- Path의 경우처럼 제약조건 추가도 가능하다.
- gt, ge, lt, le
- embed 설정을 통해서 요청 바디에 대한 파라미터가 하나만 존재하는 경우에도 key와 매핑되게 할 수 있다.
@app.post("/items")
async def create_item(item: Item = Body(embed=True)):
return item
{
"item": {
"name": "Foo", "description": "The pretender", "price": 42.0, "tax": 3.2
}
}
Field로 클래스 멤버 변수 검증하기
from pydantic import BaseModel, Field
...
class Item(BaseModel):
name: str
description: Union[str, None] = Field(default=None, title="The description of the item", max_length=300)
price: float = Field(gt=0, description="The price must be greater than zero")
tax: Union[str, None] = None
...
@app.post("/items")
async def create_item(item: Item):
return item
- pydantic의 Field도 fastapi의 Path, Query, Body처럼 동작한다.
중첩 모델 사용하기
from pydantic import BaseModel
class Image(BaseModel):
url: str
name: str
class Item(BaseModel):
name: str
description: Union[str, None] = None
price: float
tax: Union[str, None] = None
tags: List[str] = []
image: Union[Image, None] = None
...
@app.post("/items")
async def create_item(item: Item):
return item
{
"name": "Foo",
"description": "The pretender",
"price": 42.0,
"tax": 3.2,
"tags": ["rock", "metal", "bar"],
"image": { "url": "http://example.com/baz.jpg", "name": "The Foo live" }
}
- 클래스의 멤버 변수로 다른 클래스 타입을 갖는 변수를 선언하여 사용할 수 있다.
FastAPI에서의 Header와 Cookie
- fastapi의 Header를 이용해서 헤더값을 파라미터로 받을 수 있다.
from fastapi import Header
...
@app.get("/items/")
async def read_items(user_agent: Union[str, None] = Header(default=None)):
return {"User-Agent": user_agent}
- HTTP header에는 보통 - 문자를 사용하는데, 파이썬에서는 이를 변수 이름에 사용하지 않는다.
- 만약 이 설정을 사용하고 싶지 않다면 Header 옵션 중 convert_underscores를 False로 설정한다.
- → FastAPI는 파라미터의 _ 문자를 자동적으로 -로 변환하여 헤더값과 매핑한다.
@app.get("/items/")
async def read_items(x_token: Union[List[str], None] = Header(default=None)):
return {"X-Token values": x_token}
- 동일한 이름의 헤더가 여러개가 있는 경우를 위해 List 타입으로도 받을 수 있다.
from fastapi import Cookie
...
@app.get("/items/")
async def read_items(ads_id: Union[str, None] = Cookie(default=None)):
return {"ads_id": ads_id}
- fastapi의 Cookie를 이용해서 쿠키값을 파라미터로 받을 수 있다.
728x90
반응형
'기타' 카테고리의 다른 글
3년차 주니어 개발자가 읽어 본 [데이터 엔지니어를 위한 97가지 조언] (0) | 2023.11.30 |
---|---|
[Python/FastAPI] 1. API 정의 - HTTP Request (1) (0) | 2023.03.16 |
[Python/FastAPI] 0. Overview (0) | 2023.03.10 |
댓글