Python 3.7+ 想验证一个对象是否符合 typing.Dict[str, int] 应该怎么做?

2019-10-09 20:40:44 +08:00
 ClericPy

主要想做的就是

  1. 拿到一个对象, cast 出来这个对象是否符合 typing 里那些奇怪类型, Union Optional 等, 尤其是 Dict[str, int] 时候要判断每一个 key value 的类型
  2. 拿到不符合的类型时候, 自动转换一下, 比如 '1.1' 需要 float 类型, 自动转成 1.1

找过一大些资料都失败了, 目前结论如下

  1. mypy 没法直接看源码, 然后看文档也没找到那种代码里 import mypy 然后做 cast 的功能

  2. typing.cast 在标准库那个是预留给 linter 用的, 实际用不上

  3. 目前已经尝试过 __origin____args__ 来做了, 效果还算不错, 但是感觉自己一点点拆包有点傻, 尤其是自己把不符合的先 _alias, 然后对 _GenericAlias 做上述俩魔术方法, 遇到了 _SpecialForm 直接凉

mypy 官网提到的 MonkeyType (Python 3) and PyAnnotate (type comments only). 还没试

5059 次点击
所在节点    Python
16 条回复
ipwx
2019-10-09 20:44:15 +08:00
1、你是想要做什么呢?如果是想要在每个函数的入口这么干,就太影响程序的性能了。
2、如果是想要对用户的输入(比如 API parameters )进行验证,可以使用 pydantic。

https://pydantic-docs.helpmanual.io/
ClericPy
2019-10-09 20:46:56 +08:00
@ipwx pydantic 看过了, 不过当时只看了它的那些 BaseSchema 用法, 就是自定义, 不过我要处理的是 typing 里的内置类型(或者复合类型)
我只是想对某些值做一下类型校验, mypy 上没找到, 官方文档也是说 typing.cast 太影响性能所以什么都不做, 只留给 linters 做猴子补丁
我看看那些 mypy 以外的有没有什么办法吧
ClericPy
2019-10-09 20:49:12 +08:00
@ipwx 好吧, 我傻了, 还是用 pydantic 吧
说白了其实我就是想知道下 typing 里那么多复杂的类型, 怎么做验证, 学习的目的
skinny
2019-10-09 20:50:56 +08:00
那是给 IDE 用的
ClericPy
2019-10-09 20:52:09 +08:00
@skinny 我就是打算研究研究有什么自带的内置方法可以让我验证么, 毕竟 isinstance 是报错的
ClericPy
2019-10-09 20:56:56 +08:00
我先研究研究用 type 动态构造 BaseModel 子类

from pydantic import BaseModel
import typing
import inspect


def test(a: int, b: typing.Dict[str, int]):
pass


sigs = inspect.signature(test)

kwargs = {p.name: p.annotation for p in sigs.parameters.values()}
Trim21
2019-10-09 21:19:55 +08:00
看了 pydantic 源码,他是这么干的

https://gist.github.com/Trim21/910601a17fbeaa07bd203a93afce6131

我没仔细研究 field.validate 的第二个参数是干啥用的
ClericPy
2019-10-09 21:32:44 +08:00
@Trim21 我之前第一选择是 pydantic , 后来因为看文档不仔细给弃用了, 结果仔细看了下文档, 觉得真香, 把我 3 个多小时写的垃圾全弃用了...

gist 我这边污染打不开, 好容易换 192.30.253.118 结果说 404..
Trim21
2019-10-09 21:34:00 +08:00
@ClericPy #8


import pydantic.validators
from pydantic.fields import Field

from typing import Dict, Any

field = Field(name='d',
type_=Dict[str, int],
class_validators=None,
model_config=pydantic.BaseConfig)
raw, errs = field.validate({'key': 'value'}, {'a': '1'}, loc='loc')
print(raw, errs)
# {'key': 'value'}, [<pydantic.error_wrappers.ErrorWrapper object at 0x00000199F1B9BE58>]
ClericPy
2019-10-09 21:38:58 +08:00
@Trim21 感谢,找了半天没找到,vscode 对这些相对路径的跳转太差了。。。已解决
lolizeppelin
2019-10-09 21:58:42 +08:00
你们思维真是死板啊

这种标准类型直接 jsonschema 校验不就完了 非要纠结到语言的新功能上
ClericPy
2019-10-09 22:11:43 +08:00
@lolizeppelin 感谢提醒, 我都快忘了那个库了, 刚听说 jsonschema 可以支持 Union Optional Dict[str,int]。
呃,show me your code?
so1n
2019-10-09 23:25:53 +08:00
只处理过简单的,要__annotations__属性或者 inspect.signature
ClericPy
2019-10-09 23:52:21 +08:00
@so1n 我上面提到的 `__origin__ 和 __args__` 就是...
watsy0007
2019-10-11 13:34:34 +08:00
@ClericPy 改用 pydantic 搞定很多类型严重和转换问题.
ClericPy
2019-10-11 14:42:25 +08:00
@watsy0007 库是好库, 这两天看他们家源码是真特么看的想吐...

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/607604

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX