FastAPI 单个参数主体导致 Pydantic 验证错误
FastAPI Single Parameter Body cause Pydantic Validation Error
我有一个 POST FastAPI 方法。我不想构建 class 或查询字符串。所以,我决定应用 Body()
方法。
@app.post("/test-single-int")
async def test_single_int(
t: int = Body(...)
):
pass
这是请求
POST http://localhost:8000/test-single-int/
{
"t": 10
}
这是回复
HTTP/1.1 422 Unprocessable Entity
date: Fri, 22 May 2020 10:00:16 GMT
server: uvicorn
content-length: 83
content-type: application/json
connection: close
{
"detail": [
{
"loc": [
"body",
"s"
],
"msg": "str type expected",
"type": "type_error.str"
}
]
}
然而,在尝试了很多样本之后,我发现如果我有一个以上的样本,它们也不会出错Body()
。例如,
@app.post("/test-multi-mix")
async def test_multi_param(
s: str = Body(...),
t: int = Body(...),
):
pass
要求
POST http://localhost:8000/test-multi-mix/
{
"s": "test",
"t": 10
}
回应
HTTP/1.1 200 OK
date: Fri, 22 May 2020 10:16:12 GMT
server: uvicorn
content-length: 4
content-type: application/json
connection: close
null
有人知道我的实现吗?有错吗?这不是最佳实践吗?或者这是一个错误?
这不是错误,它是 Body 的行为方式,它存在于 "extending" 请求参数如何记录 outlines:
class Item(BaseModel):
name: str
class User(BaseModel):
username: str
full_name: str = None
@app.put("/items/{item_id}")
async def update_item(
*,
item_id: int,
item: Item,
user: User,
importance: int = Body(..., gt=0),
q: str = None
):
pass
此视图的有效请求正文为:
{
"item": {
"name": "Foo",
"tax": 3.2
},
"user": {
"username": "dave",
"full_name": "Dave Grohl"
},
"importance": 5
}
如果你真的想单独使用 Body,你必须指定 embed=True
,这个按预期工作:
@app.put("/items/{item_id}")
async def update_item(
*,
item_id:int,
importance: int = Body(..., gt=0, embed=True),
q: str = None
):
pass
要使用 FastApi 从正文中获取任何数据:
@app.post("/someurl")
async def someMethod(body: dict):
return body
我有一个 POST FastAPI 方法。我不想构建 class 或查询字符串。所以,我决定应用 Body()
方法。
@app.post("/test-single-int")
async def test_single_int(
t: int = Body(...)
):
pass
这是请求
POST http://localhost:8000/test-single-int/
{
"t": 10
}
这是回复
HTTP/1.1 422 Unprocessable Entity
date: Fri, 22 May 2020 10:00:16 GMT
server: uvicorn
content-length: 83
content-type: application/json
connection: close
{
"detail": [
{
"loc": [
"body",
"s"
],
"msg": "str type expected",
"type": "type_error.str"
}
]
}
然而,在尝试了很多样本之后,我发现如果我有一个以上的样本,它们也不会出错Body()
。例如,
@app.post("/test-multi-mix")
async def test_multi_param(
s: str = Body(...),
t: int = Body(...),
):
pass
要求
POST http://localhost:8000/test-multi-mix/
{
"s": "test",
"t": 10
}
回应
HTTP/1.1 200 OK
date: Fri, 22 May 2020 10:16:12 GMT
server: uvicorn
content-length: 4
content-type: application/json
connection: close
null
有人知道我的实现吗?有错吗?这不是最佳实践吗?或者这是一个错误?
这不是错误,它是 Body 的行为方式,它存在于 "extending" 请求参数如何记录 outlines:
class Item(BaseModel):
name: str
class User(BaseModel):
username: str
full_name: str = None
@app.put("/items/{item_id}")
async def update_item(
*,
item_id: int,
item: Item,
user: User,
importance: int = Body(..., gt=0),
q: str = None
):
pass
此视图的有效请求正文为:
{
"item": {
"name": "Foo",
"tax": 3.2
},
"user": {
"username": "dave",
"full_name": "Dave Grohl"
},
"importance": 5
}
如果你真的想单独使用 Body,你必须指定 embed=True
,这个按预期工作:
@app.put("/items/{item_id}")
async def update_item(
*,
item_id:int,
importance: int = Body(..., gt=0, embed=True),
q: str = None
):
pass
要使用 FastApi 从正文中获取任何数据:
@app.post("/someurl")
async def someMethod(body: dict):
return body