OrF*_*man 8 python enums list pydantic fastapi
我想创建 pydantic 模型来验证用户表单。我的模型值之一应该从名称列表中验证。我成功地使用枚举创建模型如下:
from enum import Enum
class Fruit(str, Enum):
APPLE = 'apple'
BANANA = 'banana'
MELON = 'melon'
from pydantic import BaseModel
class UserForm(BaseModel):
fruit: Fruit
name: str
Run Code Online (Sandbox Code Playgroud)
现在我想将枚举切换到我的代码中的值列表:
fruit = ['apple','banana','melon']
Run Code Online (Sandbox Code Playgroud)
我怎样才能做到这一点?
tnx
aah*_*nik 23
我提出一个优雅的解决方案。
from pydantic import BaseModel
from typing import List
from enum import Enum
class Fruit(str, Enum):
APPLE = 'apple'
BANANA = 'banana'
MELON = 'melon'
class UserForm(BaseModel):
fruits: List[Fruit]
name: str
Run Code Online (Sandbox Code Playgroud)
就是这样。
检查上面的代码:
将上面的代码放在一个文件中main.py。
跑步
python -i main.py
Run Code Online (Sandbox Code Playgroud)
>>> uf = UserForm(fruits=['apple','banana'],name='hello')
>>> uf
UserForm(fruits=[<Fruit.APPLE: 'apple'>, <Fruit.BANANA: 'banana'>], name='hello')
>>> af = UserForm(fruits=['monkey','apple'],name='hello')
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "pydantic/main.py", line 400, in pydantic.main.BaseModel.__init__
pydantic.error_wrappers.ValidationError: 1 validation error for UserForm
fruits -> 0
value is not a valid enumeration member; permitted: 'apple', 'banana', 'melon' (type=type_error.enum; enum_values=[<Fruit.APPLE: 'apple'>, <Fruit.BANANA: 'banana'>, <Fruit.MELON: 'melon'>])
>>>
Run Code Online (Sandbox Code Playgroud)
pydantic 会引发错误,而monkey水果中则不会。
ale*_*ame 22
您也可以通过Literal类型列表来完成此操作。就像这样:
import pydantic
from typing import Literal, List
class M(pydantic.BaseModel):
fruits: List[Literal["apple", "orange"]]
print(M.parse_obj({"fruits":["apple", "orange"]})) # OK fruits=['apple', 'orange']
print(M.parse_obj({"fruits":["apple", "orange", "potato"]})) # Error unexpected value potato
Run Code Online (Sandbox Code Playgroud)
您可以validator通过以下方式使用:
from pydantic import BaseModel, ValidationError, validator
class UserForm(BaseModel):
fruit: str
name: str
@validator('fruit')
def fruit_must_be_in_fruits(cls,fruit):
fruits=['apple','banana','melon']
if fruit not in fruits:
raise ValueError(f'must be in {fruits}')
return fruit
try:
UserForm(fruit="apple",name="apple")
except ValidationError as e:
print(e)
Run Code Online (Sandbox Code Playgroud)
如果不符合条件,它将引发验证错误。
您可以通过枚举的字典获取有关枚举的信息.__members__- 在这里您可以简单地迭代它的键:
from enum import Enum
class Fruit(str, Enum):
APPLE = 'apple'
BANANA = 'banana'
MELON = 'melon'
# only need __members__ if you need more infos about it
print(Fruit.__members__)
# you do not need the __members__ if you just want the keys
print([name.lower() for name in Fruit])
Run Code Online (Sandbox Code Playgroud)
输出:
# enums __members__ dictionary
{'APPLE': <Fruit.APPLE: 'apple'>,
'BANANA': <Fruit.BANANA: 'banana'>,
'MELON': <Fruit.MELON: 'melon'>}
# lower keys
['apple', 'banana', 'melon']
Run Code Online (Sandbox Code Playgroud)