文档章节

flask-wtf validate error

cdsc
 cdsc
发布于 2017/07/22 19:27
字数 701
阅读 43
收藏 0
点赞 0
评论 0

描述

使用request.args.get方法获取web前段数据填充flask-wtf表单时,由于获取的值为unicode编码,导致flask-wtf验证失败。

问题排查

flask-wtf 后台的Form表单, selectField字段关联数据库,代码如下:

class ApplyReourceForm(FlaskForm):
    username = StringField('User Name')
    apply_pool_id = SelectField('Apply Pool Name', coerce=int)
    apply_date = StringField('Apply Date')
    audit_user_id = SelectField('Audit User Name', coerce=int)
    apply_reason = TextAreaField('Apply Reason')
    submit = SubmitField('Submit')
    save = SubmitField('Save')

    ''' SelectField字段关联数据库,在此初始化选项value值(其中User,ResourcePool 为数据库模型) '''
    def __init__(self, *args, **kwargs):
        super(ApplyReourceForm, self).__init__(*args, **kwargs)
        self.apply_pool_id.choices = [(pool.id, pool.name) for pool in
                                      ResourcePool.query.all()]
        self.audit_user_id.choices = [(audit_user.id, audit_user.username) for audit_user in
                                      User.query.all()]

调用view代码

@yarn.route('/apply', methods=['GET', 'POST'])
@login_required
def apply_resource_pool():
    resource_id = request.args.get('resource_id')
    form = ApplyReourceForm()
    form.username.data = current_user.username # form表单值预填充
    form.apply_pool_id.data = resource_id  #  form表单值预填充,值来自web前台
    form.apply_date.data = time.strftime('%Y-%m-%d %H:%M:%S')
    if form.validate_on_submit():
        if form.save.data:
            print 'click save button'
        elif form.submit.data:
            print 'click submit button'
    return render_template('yarn/apply_resource_pool.html', form=form)

前台form表单验证总是不通过,查看validate_on_submit 类,分别打印is_submitted() 及 validate()值

def validate_on_submit(self):
    """Call :meth:`validate` only if the form is submitted.
    This is a shortcut for ``form.is_submitted() and form.validate()``.
    """
    print 'submit: %s' % self.is_submitted()
    print 'validate: %s' % self.validate()
    return self.is_submitted() and self.validate()

执行结果:
submit: True
validate: False

触发了submit方法,但是表单验证未通过,继续查看validate方法:

def validate(self):
    """
    Validates the form by calling `validate` on each field, passing any
    extra `Form.validate_<fieldname>` validators to the field validator.
    """
    extra = {}
    for name in self._fields:
        inline = getattr(self.__class__, 'validate_%s' % name, None)
        if inline is not None:
            extra[name] = [inline]

    return super(Form, self).validate(extra) # 返回在这一句

继续查看super.validate()方法,在循环校验字段处依次打印每个字段的校验结果:

def validate(self, extra_validators=None):
    """
    Validates the form by calling `validate` on each field.

    :param extra_validators:
        If provided, is a dict mapping field names to a sequence of
        callables which will be passed as extra validators to the field's
        `validate` method.

    Returns `True` if no errors occur.
    """
    self._errors = None
    success = True
    for name, field in iteritems(self._fields):
        if extra_validators is not None and name in extra_validators:
            extra = extra_validators[name]
        else:
            extra = tuple()
        print extra  # 传入的校验方法
        if not field.validate(self, extra):
            print '------------------------ %s -------error--------------------------' % name
            success = False
    return success

执行结果:
()
()
------------------------ apply_pool_id -------error--------------------------
()
()
()
()
()
()

目前可以看出是apply_pool_id校验失败导致表单无法通过验证,在view处查看提交时表单的字段值及类型,代码如下:

print 'apply_pool_id: %s, %s' % (form.apply_pool_id.data, type(form.apply_pool_id.data))
print 'audit_user_id: %s, %s' % (form.audit_user_id.data, type(form.audit_user_id.data))

执行结果:
apply_pool_id: 1, <type 'unicode'>
audit_user_id: 1, <type 'int'>

发现apply_pool_id值类型为unicode与定义的int不符,导致表单验证失败,于是修改view视图,将参数强制转换为int类型:

resource_id = int(request.args.get('resource_id'))

再次提交表单,日志信息:
submit: True
()
()
()
()
()
()
()
()
validate: True
()
()
()
()
()
()
()
()
click submit button
apply_pool_id: 1, <type 'int'>
audit_user_id: 1, <type 'int'>

问题解决

发现验证已通过,以前没遇到过这种情况,网上也没找到对应的问题,所以在此记录一下。 PS:遇到问题还是要多去熟悉源码。

© 著作权归作者所有

共有 人打赏支持
cdsc
粉丝 0
博文 11
码字总数 7395
作品 0
深圳
程序员
设计模式 Template模式

在面向对象系统的分析与设计过程中经常会遇到这样一种情况: 一个流程可以分为A,B,C….多个步骤,但在具体的步骤中可能会有不同的情况出现,这时就Template模式就非常有用:

caoxinyu0205 ⋅ 2016/09/03 ⋅ 0

网站后端.Flask.实战-社交博客开发-flask-login认证用户?

用户模型 1.用户登录后认证状态需要被记录,这样浏览不同的页面才能记住这个状态,flask-login专门用来管理用户认证系统中的认证状态,且不依赖特定的认证机制 2.用户模型必须实现isauthenticat...

满满李 ⋅ 2016/06/26 ⋅ 0

(三)Flask 学习 —— web 表单

web 表单 回顾 在上一章节中,我们定义了一个简单的模板,使用占位符来虚拟了暂未实现的部分,比如用户以及文章等。 在本章我们将要讲述应用程序的特性之一–表单,我们将会详细讨论如何使用...

水果糖 ⋅ 2016/02/25 ⋅ 0

falsk-web 表单

web 表单 回顾 在上一章节中,我们定义了一个简单的模板,使用占位符来虚拟了暂未实现的部分,比如用户以及文章等。 在本章我们将要讲述应用程序的特性之一–表单,我们将会详细讨论如何使用...

joker_bug ⋅ 2017/07/12 ⋅ 0

最牛「CSRF防护」,带你进入大虾们的圈子!

简单理解 CSRF 什么是 CSRF? CSRF,通常称为跨站请求伪造,英文名 Cross-site request forgery 缩写 CSRF,是一种对网站的恶意攻击。一个跨站请求伪造攻击迫使登录用户的浏览器将伪造的HTT...

OneAPM蓝海讯通 ⋅ 2016/01/18 ⋅ 0

Flask 101: How to Add a Search Form

In our last article, we added a database to our Flask web application but didn’t have a way to add anything to our database. We also didn’t have a way to view anything, so, ba......

Mike Driscoll ⋅ 2017/12/14 ⋅ 0

Flask之WTForms验证

有时候我们经常会跟前端提交的表单数据打交道,每次手写验证规则真的很繁琐。Flask-WTF扩展模块提供了前端表单自动生成和前端表单提交数据验证的功能。 WTForms支持的HTML标准字段 WTForms常...

daibaiyang119 ⋅ 2017/11/30 ⋅ 0

Flask之WTForms验证

有时候我们经常会跟前端提交的表单数据打交道,每次手写验证规则真的很繁琐。Flask-WTF扩展模块提供了前端表单自动生成和前端表单提交数据验证的功能。 WTForms支持的HTML标准字段 WTForms常...

daibaiyang119 ⋅ 2017/11/30 ⋅ 0

flask-wtf扩展中FormField报错:'csrf_token': ['CSRF token missing']

在使用flask_wtf时使用的这个结构FieldList(FormField(SomeForm)) 报错: 'csrf_token': ['CSRF token missing'] 因为FormField也是form,flaskwtf中引入的Form默认开启了csrf验证(flaskwtf...

tinyhare ⋅ 2016/09/27 ⋅ 0

多语言网页输出工具--Flask-WTF-Polyglot

Flask-WTF-Polyglot 是 Flask-WTF 公司为多语言网页输出提供 PolyglotForm 类的工具。 代码示例: from flaskwtfpolyglot import PolyglotFormfrom wtforms import BooleanField class MyFo......

大胖森 ⋅ 2015/03/31 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

Greys Java在线问题诊断工具

Greys是一个JVM进程执行过程中的异常诊断工具。 在不中断程序执行的情况下轻松完成JVM相关问题排查工作 目标群体 有时候突然一个问题反馈上来,需要入参才能完成定位,但恰恰没有任何日志。回...

素雷 ⋅ 28分钟前 ⋅ 0

git从远程仓库拉取代码的常用指令

一种(比较麻烦的)拉代码的方法 git clone //克隆代码库,与远程代码库的主干建立连接,如果主干已经在就不用再clone啦,克隆路径为当前路径下的新创建的文件夹 git checkout -b //本地建立...

Helios51 ⋅ 42分钟前 ⋅ 0

005. 深入JVM学习—Java堆内存参数调整

1. JVM整体内存调整图解(调优关键) 实际上每一块子内存区域都会存在一部分可变伸缩区域,其基本流程:如果内存空间不足,则在可变的范围之内扩大内存空间,当一段时间之后,内存空间不紧张...

影狼 ⋅ 47分钟前 ⋅ 0

内存障碍: 软件黑客的硬件视图

此文为笔者近日有幸看到的一则关于计算机底层内存障碍的学术论文,并翻译(机译)而来[自认为翻译的还行],若读者想要英文原版的论文话,给我留言,我发给你。 内存障碍: 软件黑客的硬件视图...

Romane ⋅ 今天 ⋅ 0

SpringCloud 微服务 (七) 服务通信 Feign

壹 继续第(六)篇RestTemplate篇 做到现在,本机上已经有注册中心: eureka, 服务:client、order、product 继续在order中实现通信向product服务,使用Feign方式 下面记录学习和遇到的问题 贰 or...

___大侠 ⋅ 今天 ⋅ 0

gitee、github上issue标签方案

目录 [TOC] issue生命周期 st=>start: 开始e=>end: 结束op0=>operation: 新建issueop1=>operation: 评审issueop2=>operation: 任务负责人执行任务cond1=>condition: 是否通过?op3=>o......

lovewinner ⋅ 今天 ⋅ 0

浅谈mysql的索引设计原则以及常见索引的区别

索引定义:是一个单独的,存储在磁盘上的数据库结构,其包含着对数据表里所有记录的引用指针. 数据库索引的设计原则: 为了使索引的使用效率更高,在创建索引时,必须考虑在哪些字段上创建索...

屌丝男神 ⋅ 今天 ⋅ 0

String,StringBuilder,StringBuffer三者的区别

这三个类之间的区别主要是在两个方面,即运行速度和线程安全这两方面。 首先说运行速度,或者说是, 1.执行速度 在这方面运行速度快慢为:StringBuilder(线程不安全,可变) > StringBuffer...

时刻在奔跑 ⋅ 今天 ⋅ 0

java以太坊开发 - web3j使用钱包进行转账

首先载入钱包,然后利用账户凭证操作受控交易Transfer进行转账: Web3j web3 = Web3j.build(new HttpService()); // defaults to http://localhost:8545/Credentials credentials = Wallet......

以太坊教程 ⋅ 今天 ⋅ 0

Oracle全文检索配置与实践

Oracle全文检索配置与实践

微小宝 ⋅ 今天 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部