Flask-WTForms的FileField无法验证问题

3

我正在尝试使用Flask和WTForms的FileField表单字段构建一个应用程序,以将文件上传到Web服务器。帖子成功传递,但我很好奇为什么每次form.validate_on_submit()都会失败,即使特定验证器都成功了。这是我的表单(forms.py)、应用程序(main.py)和HTML模板(upload.html)的代码。

### forms.py   

from flask.ext.wtf import Form
from flask.ext.wtf.html5 import EmailField
from flask.ext.wtf.file import FileField, FileRequired, FileAllowed
from wtforms import validators, ValidationError, SubmitField

class UploadForm(Form):
  presentation = FileField('Presentation in Image Format', validators=[FileRequired(), FileAllowed(['jpg', 'png'], 'Images only!')])
  submit = SubmitField("Send")



### main.py

from forms import UploadForm
from flask import render_template, url_for, redirect, send_from_directory

@app.route('/upload/', methods=('GET', 'POST'))
  def upload():
    form = UploadForm()
    if form.validate_on_submit():
      filename = secure_filename(form.presentation.file.filename)
      print filename
      form.presentation.file.save(os.path.join('uploads/', filename))
      return redirect(url_for('uploads', filename=filename))
    filename = None
    return render_template('upload.html', form=form, filename=filename)

@app.route('/uploads/<filename>')
  def uploaded_file(filename):
    return send_from_directory(app.config['UPLOAD_FOLDER'], filename)



### upload.html
   {% for message in form.presentation.errors %}
     <div class="flash">{{ message }}</div>
   {% endfor %}
   <form action="/upload/" method="POST" enctype="multipart/form-data">
     {{ form.presentation.label }}
     {{ form.presentation }}
     {{ form.submit}}
   </form>

有人知道为什么这个验证不起作用吗?或者我不应该使用validate_on_submit()吗?

使用form.validate()替代form.validate_on_submit()并将form = UploadForm()更改为form = UploadForm(request.form)。 - Odai Al-Ghamdi
我尝试了这个,但它没有起作用。而且,这次它返回了一个错误消息(先前没有任何错误消息就可以发布),其中“None”是文件名,因为它没有经过验证。 - user2408069
1个回答

9

Flask-WTF默认启用CRSF,如果您打印form.errors,将会得到一个提示信息,告诉您需要CSRF令牌。

解决方法很简单,在您的模板中加入{{ form.csrf_token }},或禁用表单的CSRF,但您真的不应该这样做。

<form action="/upload/" method="POST" enctype="multipart/form-data">
 {{ form.presentation.label }}
 {{ form.presentation }}
 {{ form.csrf_token }}
 {{ form.submit}}
</form>

还有一种快速的方法可以将所有隐藏字段添加到表单hidden_tags中:

<form action="/upload/" method="POST" enctype="multipart/form-data">
 {{ form.presentation.label }}
 {{ form.presentation }}
 {{ form.hidden_tag() }}
 {{ form.submit}}
</form>

1
很遗憾,我没有足够的积分来投赞成票。不过已经接受为正确答案了。 - user2408069

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接