- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我的路线:
@router.get('/check/{value}', status_code=200)
def ranks_check(value: BasicInput = Depends()):
"""
Test endpoint
"""
return value
我的模型:
class BasicInput:
"""
Get Confidence to score Input class
"""
value: int
@validator('value')
def check_if_value_in_range(cls, v):
if not 0 < v < 1000001:
raise ValueError('Value Exceeded Limit')
我需要做的:
我需要验证输入并在出现 ValueError 时引发 HTTP 400。
我知道我可以使用 Pydantic 的 Field
类型完成整数验证,并在路由函数本身中运行 check_if_value_in_range
。我正在寻找使用该模型的解决方案。
最佳答案
请参阅 Raise an HTTPException in your code 上的 FastAPI 文档:
HTTPException
is a normal Python exception with additional data relevant for APIs.Because it's a Python exception, you don't
return
it, youraise
it.This also means that if you are inside a utility function that you are calling inside of your path operation function, and you raise the
HTTPException
from inside of that utility function, it won't run the rest of the code in the path operation function, it will terminate that request right away and send the HTTP error from theHTTPException
to the client.
from fastapi.exceptions import HTTPException
class BasicInput(BaseModel):
value: int
@validator("value")
def check_if_value_in_range(cls, v):
if not 0 < v < 1000001:
# raise ValueError("Value Exceeded Limit")
raise HTTPException(status_code=400, detail="value exceeded limit")
return v
$ curl -i -XGET localhost:8000/check/1000002
HTTP/1.1 400 Bad Request
...
{"detail":"value exceeded limit"}
$ curl -i -XGET localhost:8000/check/42
HTTP/1.1 200 OK
...
{"value":42}
但要使其正常工作,您当前的代码中有一些问题需要修复:
如果您使用 Pydantic 的 validator装饰器,那么你的类需要是 Pydantic BaseModel
或 a Pydantic dataclass
.
class BasicInput(BaseModel): # <--------------------
value: int
如果您还在 validator
函数上添加 print
语句或断点,您会看到它实际上从未被调用,因为它不是 Pydantic BaseModel
。
if
条件为 False
时验证器函数缺少处理(当 v
在范围内有效时).来自 Pydantic 文档:
validators should either return the parsed value or raise a
ValueError
,TypeError
, orAssertionError
(assert
statements may be used).
class BasicInput(BaseModel):
value: int
@validator("value")
def check_if_value_in_range(cls, v):
if not 0 < v < 1000001:
raise ValueError("Value Exceeded Limit")
# v is good
return v # <--------------------
在您的原始代码中,您 mentioned that you were getting "nothing" .我假设您得到的是针对有效和无效 value
的空 {}
响应:
$ curl -XGET localhost:8000/check/42
{}
$ curl -XGET localhost:8000/check/1000002
{}
这是因为在路由函数中,value
是BasicInput
类,不是{value}
路径值或者BasicInput.value
整数值。
@router.get("/check/{value}", status_code=200)
def ranks_check(value: BasicInput = Depends()):
print(type(value)) # <class 'main.BasicInput'>
return value
您实际上得到的“空”响应是 FastAPI 应用其 jsonable_encoder 的结果到 BasicInput
类。当 FastAPI 到达您的路由函数的 返回值
时,它将使用 jsonable_encoder
将其转换为 JSONResponse
。请参阅 Return a Response Directly 上的文档.
在内部,由于 BasicInput
不是 Pydantic BaseModel
或可迭代对象(例如 dict
类对象),它会产生一个空字典 {}
。 (您可以检查 code for jsonable_encoder
,但这是因为 dict(obj)
和 vars(obj)
)。
因此,如 MatsLindh's answer 中所述,FastAPI 请求/响应上下文中使用的模型通常是 Pydantic BaseModel
的子类,在正确地对其进行子类化之后,您现在应该得到非空响应:
$ curl -XGET localhost:8000/check/42
{"value":42}
$ curl -XGET localhost:8000/check/1000002
Internal Server Error
...
pydantic.error_wrappers.ValidationError: 1 validation error for BasicInput
value
Value Exceeded Limit (type=value_error)
修复后,只需按照 FastAPI 的 Raise an HTTPException in your code 中的指南进行操作即可(正如我在此答案开头提到的)直接从模型中引发 HTTP 错误。
但我同意 MatsLindh's comment ,这不是一个“好的”做法,因为它违反了SRP/Single Responsibility Principle .您的模型正在做两件事:验证您的输入和引发适当的 HTTP 错误响应。
FastAPI 应该处理您的请求和响应,而 Pydantic 应该代表您的模型和数据。您让 FastAPI 接受请求,将其传递给 Pydantic 以验证和存储模型数据,然后让 FastAPI 将结果转换为适当的响应。您的预期解决方案也令人困惑,因为对于有效/成功案例,它是在路由函数中处理的,但对于无效/错误案例,它是在模型验证函数中处理的。
FastAPI 已经知道如何捕获 Pydantic ValidationError
并将它们转换为适当的 HTTP 错误响应。在这种情况下,FastAPI 返回 HTTP 500 内部服务器错误。
如果目的是提供特定于模型的错误响应,您应该在模型上保留 raise ValueError
,然后覆盖 FastAPI 的默认验证错误处理程序。请参阅有关 Override the default exception handlers 的部分.
这是我推荐的代码:
from fastapi import APIRouter, Depends, FastAPI
from fastapi.exceptions import ValidationError
from fastapi.responses import JSONResponse
from pydantic import BaseModel, validator
# MODELS
class BasicInput(BaseModel):
value: int
@validator("value")
def check_if_value_in_range(cls, v):
if not 0 < v < 1000001:
raise ValueError("Value Exceeded Limit")
return v
# VIEWS
api = FastAPI()
router = APIRouter()
@api.exception_handler(ValidationError)
async def validation_exception_handler(request, exc: ValidationError):
return JSONResponse(status_code=400, content={"error": str(exc)})
@router.get("/check/{value}")
def ranks_check(value: BasicInput = Depends()):
return value
api.include_router(router)
$ curl -i -XGET localhost:8000/check/42
HTTP/1.1 200 OK
...
{"value":42}
$ curl -i -XGET localhost:8000/check/1000002
HTTP/1.1 400 Bad Request
...
{"error":"1 validation error for BasicInput\nvalue\n Value Exceeded Limit (type=value_error)"}
关于python - 如何在模型中引发 HTTP 400?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/68770979/
我正在处理一组标记为 160 个组的 173k 点。我想通过合并最接近的(到 9 或 10 个组)来减少组/集群的数量。我搜索过 sklearn 或类似的库,但没有成功。 我猜它只是通过 knn 聚类
我有一个扁平数字列表,这些数字逻辑上以 3 为一组,其中每个三元组是 (number, __ignored, flag[0 or 1]),例如: [7,56,1, 8,0,0, 2,0,0, 6,1,
我正在使用 pipenv 来管理我的包。我想编写一个 python 脚本来调用另一个使用不同虚拟环境(VE)的 python 脚本。 如何运行使用 VE1 的 python 脚本 1 并调用另一个 p
假设我有一个文件 script.py 位于 path = "foo/bar/script.py"。我正在寻找一种在 Python 中通过函数 execute_script() 从我的主要 Python
这听起来像是谜语或笑话,但实际上我还没有找到这个问题的答案。 问题到底是什么? 我想运行 2 个脚本。在第一个脚本中,我调用另一个脚本,但我希望它们继续并行,而不是在两个单独的线程中。主要是我不希望第
我有一个带有 python 2.5.5 的软件。我想发送一个命令,该命令将在 python 2.7.5 中启动一个脚本,然后继续执行该脚本。 我试过用 #!python2.7.5 和http://re
我在 python 命令行(使用 python 2.7)中,并尝试运行 Python 脚本。我的操作系统是 Windows 7。我已将我的目录设置为包含我所有脚本的文件夹,使用: os.chdir("
剧透:部分解决(见最后)。 以下是使用 Python 嵌入的代码示例: #include int main(int argc, char** argv) { Py_SetPythonHome
假设我有以下列表,对应于及时的股票价格: prices = [1, 3, 7, 10, 9, 8, 5, 3, 6, 8, 12, 9, 6, 10, 13, 8, 4, 11] 我想确定以下总体上最
所以我试图在选择某个单选按钮时更改此框架的背景。 我的框架位于一个类中,并且单选按钮的功能位于该类之外。 (这样我就可以在所有其他框架上调用它们。) 问题是每当我选择单选按钮时都会出现以下错误: co
我正在尝试将字符串与 python 中的正则表达式进行比较,如下所示, #!/usr/bin/env python3 import re str1 = "Expecting property name
考虑以下原型(prototype) Boost.Python 模块,该模块从单独的 C++ 头文件中引入类“D”。 /* file: a/b.cpp */ BOOST_PYTHON_MODULE(c)
如何编写一个程序来“识别函数调用的行号?” python 检查模块提供了定位行号的选项,但是, def di(): return inspect.currentframe().f_back.f_l
我已经使用 macports 安装了 Python 2.7,并且由于我的 $PATH 变量,这就是我输入 $ python 时得到的变量。然而,virtualenv 默认使用 Python 2.6,除
我只想问如何加快 python 上的 re.search 速度。 我有一个很长的字符串行,长度为 176861(即带有一些符号的字母数字字符),我使用此函数测试了该行以进行研究: def getExe
list1= [u'%app%%General%%Council%', u'%people%', u'%people%%Regional%%Council%%Mandate%', u'%ppp%%Ge
这个问题在这里已经有了答案: Is it Pythonic to use list comprehensions for just side effects? (7 个答案) 关闭 4 个月前。 告
我想用 Python 将两个列表组合成一个列表,方法如下: a = [1,1,1,2,2,2,3,3,3,3] b= ["Sun", "is", "bright", "June","and" ,"Ju
我正在运行带有最新 Boost 发行版 (1.55.0) 的 Mac OS X 10.8.4 (Darwin 12.4.0)。我正在按照说明 here构建包含在我的发行版中的教程 Boost-Pyth
学习 Python,我正在尝试制作一个没有任何第 3 方库的网络抓取工具,这样过程对我来说并没有简化,而且我知道我在做什么。我浏览了一些在线资源,但所有这些都让我对某些事情感到困惑。 html 看起来
我是一名优秀的程序员,十分优秀!