Python Schema一种优雅的数据验证方式

Python Schema使用说明

项目地址:GitHubpython

Schema是什么?

无论咱们作什么应用,只要和用户输入打交道,就有一个原则--永远不要相信用户的输入数据。意味着咱们要对用户输入进行严格的验证,web开发时通常输入数据都以JSON形式发送到后端API,API要对输入数据作验证。通常我都是加不少判断,各类if,致使代码很丑陋,能不能有一种方式比较优雅的验证用户数据呢?Schema就派上用场了。git

Schema很是简单,也就几百行的代码,最核心的类就一个:Schemagithub

1. 给Schema类传入类型(intstrfloat等)

例如:web

from schema import Schema

Schema(int).validate(10)
10
Schema(int).validate('10')
SchemaUnexpectedTypeError: '10' should be instance of 'int'

可见Schema会去验证validate方法传入的对象是否是所指定的类型,是则返回传入的数据,不然抛出一个SchemaError的异常(SchemaUnexpectedTypeErrorSchemaError的子类)。后端

2. 给Schema类传入可调用的对象(函数、带__call__的类等)

例如:函数

Schema(lambda x: 0<x<10).validate(5)
5
Schema(lambda x: 0<x<10).validate(57)
SchemaError: <lambda>(57) should evaluate to True

可见Schema会把validate方法传入的值传入到对应的函数里面做为参数,若是函数返回值为True则返回输入数据,不然抛出异常。lua

3. 给Schema类传入带有validate方法的对象

Schema也内置了一些类(UseAndOr等等),这些类的实例都带有validate方法,亦可做为Schema的参数传入,例如:code

from schema import Schema, And

# And表明两个条件必须同时知足
Schema(And(str, lambda s: len(s) > 2)).validate('abcd')
'abcd'

4. 给Schema类传入容器对象(listtupleset等)

例如:对象

Schema([int, float]).validate([1, 2, 3, 4.0])
[1, 2, 3, 4.0]

至关于,对于[1, 2, 3, 4.0]当中的任何一个元素,必须是int或者float才行(注意是or的关系)开发

5. 给Schema传入一个字典对象(大部分使用Schema的场景都是传入字典对象,这个很重要)

Schema({'name': str, 'age': int}).validate({'name': 'foobar', 'age': 18})
{'age': 18, 'name': 'foobar'}
Schema({'name': str, 'age': int}).validate({'name': 'foobar'})
SchemaMissingKeyError: Missing keys: 'age'

首先,明确两个概念,Schema类传入的字典,称之为模式字典valdiate方法传入的字典称之为数据字典

首先,Schema会判断, 模式字典和数据字典的key是否彻底同样,不同的话直接抛出异常。若是同样,就去拿数据字典的value去验证模式字典相应的value,若是数据字典的所有value均可以验证经过的话才返回数据,不然抛出异常,是否是感受这种验证顿时感受清爽了呢?

6. faqs

  1. Schema传入字典很好用,可是我有的数据是可选的,也就是说有的key能够不提供怎么办?
from schema import Optional, Schema


Schema({'name': str, Optional('age'): int}).validate({'name': 'foobar'})
{'name': 'foobar'}
Schema({'name': str, Optional('age', default=18): int}).validate({'name': 'foobar'})
{'age': 18, 'name': 'foobar'}
  1. 我想让Schema只验证传入字典中的一部分数据,能够有多余的key可是不要抱错,怎么作?
Schema({'name': str, 'age': int}, ignore_extra_keys=True).validate({'name': 'foobar', 'age': 100, 'sex': 'male'})
{'age': 100, 'name': 'foobar'}
  1. Schema抛出的异常信息不是很友好,我想自定义错误信息,怎么办?

Schema自带的类(UseAndOrRegexSchema等)都有一个参数error,能够自定义错误信息

Schema({'name': str, 'age': Use(int, error='年龄必须是整数')}).validate({'name': 'foobar', 'age': 'abc'})
SchemaError: 年龄必须是整数
相关文章
相关标签/搜索