WTForms是如何工作的?

WTForms是一个Python语言的请求数据验证库,功能强大、且支持自定义验证器。

1.初体验

首先用一个简单的示例,来体验一下WTForms的效果。

from wtforms import Form, StringField, IntegerField
from wtforms.validators import DataRequired, NumberRange, length


class UserForm(Form):
    name = StringField(label='姓名', validators=[DataRequired(message='can not be empty'), length(min=2, max=6, message='姓名必须2-6个字符')])
    age = IntegerField(label='年龄', validators=[DataRequired(message='can not be empty'), NumberRange(min=0, max=10, message='年龄必须为0-10')])


if __name__ == '__main__':
    data = {'name': 'zhangsan', 'age': 11}
    form = UserForm(data=data)
    print(form.validate())
    print(form.errors)

执行结果如下, errors中包含了所有的错误提示。

False
{'name': ['姓名必须2-6个字符'], 'age': ['年龄必须为0-10']}

2.Form

作为一个有追求的开发者,我们不能满足于仅仅知道怎么用这个库,有必要思考下这究竟是怎样实现的。

首先从UserForm的基类Form着手

class Form(with_metaclass(FormMeta, BaseForm)):

Form又使用了一个函数with_metaclass的返回结果作为基类,函数with_metaclasswtforms.compat.py

def with_metaclass(meta, base=object):
    return meta("NewBase", (base,), {})

所以Form的基类其实就是

FormMeta("NewBase", (BaseForm,), {})

3.BaseForm

BaseForm是一个基础"Form"类,它提供了"Form"的核心行为的代理。

# 其中部分代码太长,用pass替代,如有需要直接查看源码
class BaseForm(object):
    def __init__(self, fields, prefix='', meta=DefaultMeta()):
        pass

    def __iter__(self):
        """Iterate form fields in creation order."""
        return iter(itervalues(self._fields))

    def __contains__(self, name):
        """ Returns `True` if the named field is a member of this form. """
        return (name in self._fields)

    def __getitem__(self, name):
        """ Dict-style access to this form's fields."""
        return self._fields[name]

    def __setitem__(self, name, value):
        """ Bind a field to this form. """
        self._fields[name] = value.bind(form=self, name=name, prefix=self._prefix)

    def __delitem__(self, name):
        """ Remove a field from this form. """
        del self._fields[name]

    def _get_translations(self):
        pass

    def populate_obj(self, obj):
        pass

    def process(self, formdata=None, obj=None, data=None, **kwargs):
        pass

    def validate(self, extra_validators=None):
        pass

    @property
    def data(self):
        return dict((name, f.data) for name, f in iteritems(self._fields))

    @property
    def errors(self):
        if self._errors is None:
            self._errors = dict((name, f.errors) for name, f in iteritems(self._fields) if f.errors)
        return self._errors

4.FormMeta

FormMeta(type)是一个元类,用来动态创建Form及其字段列表(关键),为了便于分析,可以在FormMeta的call函数中断点调试,能够清晰地看到Field的绑定过程。

# 部分代码较长,使用pass替代,如有需要直接查看源码
class FormMeta(type):
    def __init__(cls, name, bases, attrs):
        type.__init__(cls, name, bases, attrs)
        cls._unbound_fields = None
        cls._wtforms_meta = None

    def __call__(cls, *args, **kwargs):
        pass

    def __setattr__(cls, name, value):
        pass

    def __delattr__(cls, name):
        pass

这里使用到了Python中元类的技巧。

5.Python中的元类编程

通过FormMeta这个元类,动态创建了BaseForm的子类Form,并且控制了Form的子类实例化的时候能够按照FormMeta中的逻辑绑定Field。

这里顺便提一下Python中的元类编程,元类编程的关键是type这个类。

查看type的源码,发现type有3种构造函数,其中type(name, bases, dict) -> a new type用来动态创建一个新的类,创建类的时候可以指定新类继承的基类(bases),以及自身的属性和函数(dict)。

接下来我们用一个最简单的例子来说明元类编程。

class A:
    a = 1

    @staticmethod
    def test_a():
        return 'aaa'


class B:
    b = 2


class CMeta(type):
    pass


if __name__ == '__main__':
    C = CMeta("C", (A, B), {'x': 123})
    c = C()
    print(c)
    print(c.a)
    print(c.b)
    print(c.test_a())
    print(c.x)

输出结果如下

<__main__.C object at 0x1023668d0>
1
2
aaa
123

这里动态地创建了类C,并且继承了A、B,当然直接使用type而非type的子类CMeta也是可以的,只不过使用CMeta可以在创建类的时候做更多的控制。

6.Basic Field

上面提到了Form的子类,也就是一开始的例子中我们定义的UserForm,在示例化的时候会按照FormMeta中定义的逻辑绑定Field。

诸如StringField、IntegerField等都是Field的子类,Field中关键的函数_run_validation_chain执行校验链,对所有的validators逐一校验。

    def _run_validation_chain(self, form, validators):
        
        for validator in validators:
            try:
                validator(form, self)
            except StopValidation as e:
                if e.args and e.args[0]:
                    self.errors.append(e.args[0])
                return True
            except ValueError as e:
                self.errors.append(e.args[0])

        return False

7.Custom Field

关于Custom Fields这里不进行阐述,可自行查看官方文档。

8.Built-in validators

这里以DataRequired这个validator为例,其中call魔法函数中定义了具体的校验逻辑,使其成为一个可调用对象,并且2个参数分别为form和field。而Field中_run_validation_chain函数中有一句validator(form, self)正好就是在Field中遍历validator,并执行validator的call中的校验逻辑。

class DataRequired(object):

    field_flags = ('required', )

    def __init__(self, message=None):
        self.message = message

    def __call__(self, form, field):
        if not field.data or isinstance(field.data, string_types) and not field.data.strip():
            if self.message is None:
                message = field.gettext('This field is required.')
            else:
                message = self.message

            field.errors[:] = []
            raise StopValidation(message)

9.Custorm validator

通过源代码的分析可知,只要我们定义的类中有call魔法函数,传入两个参数form、field,并在其中给定具体的校验逻辑,如果校验失败时,raise ValueError或其子类,Field就能捕获异常,并将error追加到该字段的错误列表中。

官方文档中也给出了自定义validator的示例

class Length(object):
    def __init__(self, min=-1, max=-1, message=None):
        self.min = min
        self.max = max
        if not message:
            message = u'Field must be between %i and %i characters long.' % (min, max)
        self.message = message

    def __call__(self, form, field):
        l = field.data and len(field.data) or 0
        if l < self.min or self.max != -1 and l > self.max:
            raise ValidationError(self.message)

当然,除了call的方式,使用闭包函数自定义validator也是可以的,不过还是建议可调用对象的方式。

10.总结

一句话总结WTForms的层级结构:Form中绑定Filed,Field中通过构造函数定义需要的validators。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270