flask 的表单提交基本用法( 以登陆页面为例 )

准备工作:

  • pip install flask-wtf (如有必要)
  • pip install flask-login (如有必要)

基本原理:

  • 基于 FlaskForm 类创建自定义的表单类,比如命名为: LoginForm
  • 自定义的表单类不仅可以在.py代码中使用封装的方法,还可以直接植入 jinja 的模板文件中,生成代码块。因此在做 render_template 时,将 LoginForm 的实例当做参数送入 jinja 模板即可
  • MethodView 的写法照旧,会用到 login_form.validate_on_submit() 方法来验证输入是否合规,如果不合规,直接重新渲染,login_form 会自动将报错信息带入 jinja 模板,将信息渲染出来给用户看
  • login_form 在 jinja 模板中生成的代码块不仅仅包含正常使用时看到的录入框,还包含有特定条件下才显示的报错信息
  • 针对每一个录入框的验证方法,是在 LoginForm 类的定义时,给每一个字段使用特定的类的构造方法时创建的。验证方法可能不止一个,因此是放在一个数组中传递给 validators 参数的
  • wtforms.validators 中有很多预设好的验证方法类,比如:DataRequired, Email, Length,直接用这些类创建实例,放入 validators 数组即可。
  • 输入验证问题还会涉及到防 CSRF 攻击带来的影响,处理不得当会造成 login_form.validate_on_submit() 的结果始终为 False,这是比较容易掉进去的坑。

创建 LoginForm 类

#encoding:utf8

from flask_wtf import FlaskForm
from wtforms import StringField,PasswordField
from wtforms.validators import DataRequired,Length,Email

class LoginForm(FlaskForm):
    email=StringField(u'邮箱',_name='email',validators=[
        DataRequired(u'必填'),
        Email(u'邮箱格式不合规')
    ])

    pswd=PasswordField(u'密码',_name='pswd',validators=[
        DataRequired(u'必填'),
        Length(6,20,u'长度必须在6-20字符之间')
    ])

  • "email=StringField..."中的 email 会在 jinjia 模板、 MethodView 中的获取用户提交的参数值时用到。 它是作为 LoginForm 的类变量存在的。
  • 创建录入框类( StringField / PasswordField )时
    • 第一个参数会在生成 jinja 模板时,填充到 label 中
    • 第二个参数 _name 会在生成 jinja 模板时,变成 input 标签的 name 值
    • 各种 validators 中第一个参数,会生成一些正常状态下隐藏的提示信息,仅当用户填写错误时才会在刷新页面后显示出来

将 LoginForm 定义的信息送入 jinja 模板

在 MethodView 的定义代码中,render_template 时,就可将其送入模板。

# encoding:utf8

from flask import render_template,jsonify
from flask.views import MethodView
from ..models.account.forms.login_form import LoginForm


class LoginView(MethodView):
    def __init__(self,template):
        self.template=template
        super(LoginView,self).__init__()
        # 在这里创建一个 LoginForm() 实例
        self.login_form = LoginForm()

    def get(self):
        # 在这里将 login_form 送入 jinja 模板
        return render_template(self.template,form=self.login_form)

    def post(self):
        if not self.login_form.validate_on_submit():
            # 在这里将 login_form 送入 jinja 模板,这次会让隐藏的报错信息显示出来
            return render_template(self.template,form=self.login_form)

        # 获取用户 POST 上来的值,是通过下面的方式
        # self.login_form.email 是 LoginForm 类的那个名叫 email 的类变量
        # 必须要通过 _value() 方法来获取值,否则获取到的是完整的 input 标签的 html 代码
        print self.login_form.email._value(), self.login_form.pswd._value()

        return jsonify({
            'msg':'success',
            'code':0
        })

预防 CSRF 带来的坑

缺省状态下,flask 为表单开启了预防 CSRF 攻击的功能。 如果我们要手工明确指定开启防 CSRF ,可以在创建 LoginForm 实例时,传入参数: csrf_enabled = False,即: login_form=LoginForm(csrf_enabled = True) ,但是这样一来很可能造成 MethodView 中的 validate_on_submit() 方法返回值始终是 False!

原因是:很可能你没有在 jinja 模板中放入 secretKey , 这样一来 flask 做 CSRF 验证时肯定是不能通过的。当然这个输入结果就会被判为无效!

解决的办法有两种:

  1. 关闭这个表单的 CSRF 验证,只需要创建 login_form 时:login_form=LoginForm(csrf_enabled = False) 即可
  2. 在 jinja 模板中,加入 secretKey 生成的验证码,具体的方法在后面讲述 jinja 模板代码时再说。

jinja 模板

 <form action={{url_for('main.view_login')}} method="POST" class="pure-form pure-form-stacked">
    <fieldset>
        <legend class="fs-28">登陆</legend>
    </fieldset>
    {{ form.csrf_token }}  /** 有这个,才会将 secretKey 生成的验证码放入其中,使得提交的信息能够通过 CSRF 验证 **/
    {{form.email.label}}
    {{form.email()}}<label>{{ form.email.errors[0] }}</label>
    {{form.pswd.label}}
    {{form.pswd()}}<label>{{ form.pswd.errors[0] }}</label>
    <button class="pure-button pure-button-primary" type="submit">提交</button>
</form>

可以看到,这个模板代码中输入框组合的结构是由三个元素组成的:

  1. label - 输入框名字,由 {{form.email.label}} 负责生成,其中的 email 是 LoginForm 类的类变量 email
  2. input - 由 {{form.email()}} 负责生成
  3. lable - 报错信息,由 <label>{{ form.email.errors[0] }}</label> 负责生成

前面讲述到的如果开启了防 CSRF 攻击的验证,模板代码中必须添加: {{ form.csrf_token }},它生成的 html 代码应该是这个样子的:

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

推荐阅读更多精彩内容