繁体   English   中英

如何让 wtforms 获取 json 并将数据插入到表单 object 中?

[英]How to get wtforms to take json and insert data into the form object?

情况:

我在前端使用 React 和 Flask api 服务器。 我想将数据从 React 发送到 api,完成后我想使用 WTForms 在处理数据之前对数据运行验证。 这个问题可能看起来类似于CSRF Protection with Flask/WTForms and React ,但这并没有回答问题,请看一下我已经付出了很多努力来写一个好问题。

我有的

目前,数据作为 json object 成功发送,其中键与 wtform 结构中的名称匹配,目的是让 wtforms 获取该 json 数据并将其插入 object 并从那里正常处理

JSON object 正在发送

{'username': 'tster', 'fullname': 'Tester test', 'phone': '038287827216', 'email': 'test@example.com', 'password': 'Tester1010', 'repeatPassword': 'Tester1010'}

整个概念是从 React 组件发送它并将其添加到 wtforms 以便进行相应的验证和使用。

Python 代码:

我尝试了多种不同的方法,这是最新的:


    @bp.route('/api/register_user', methods=['POST', 'GET'])
    def register():
        """ End-point to register users """ 
        
        if request.method == 'GET':
            return ('', {'csrf_token': generate_csrf()})
        elif request.method == 'POST':
            req = request.get_json(force=True)
            if validate_csrf(request.headers['X-Csrftoken']):
    
                print('validated')
            return{"message": "posted"}

这个概念是,如果我不能让它与 wtforms 一起工作,我可以手动生成 csrf_token,验证它,一旦验证通过,就对数据运行手动验证,而不是依赖 wtforms,但是它似乎没有相应地验证。 这取自https://flask-wtf.readthedocs.io/en/stable/api.html#flask_wtf.csrf.generate_csrf

在此之前我试过:


    @bp.route('/api/register_user', methods=['POST', 'GET'])
    def register():
        """ End-point to register users """ 
        
        form = RegistrationForm()
    
        print(request.headers)
        if request.method == 'GET':
            print(form.csrf_token._value())
            return ('', {'csrf_token': form.csrf_token._value()})
        elif request.method == 'POST':
            req = request.get_json(force=True)
            form = RegistrationForm.from_json(req)
            print(form.username)
            return{"message": "posted"}
        else:
            return { 'errors': form.errors }

这背后的想法是像其他情况一样发送 csrf_token,因为在之前的链接中我读到:

生成 CSRF 令牌。 令牌是为请求缓存的,因此多次调用此 function 将生成相同的令牌。

time_limit -- 令牌有效的秒数。 默认值为 WTF_CSRF_TIME_LIMIT 或 3600 秒(60 分钟)。

读完这篇文章后,我从逻辑上认为这是有道理的,因为我基本上可以从标准表单 object 中获取 csrf_token,因为您通常会使用它们进行操作,然后当请求方法为 POST 时,我可以重新呈现表单并从中插入数据请求中的 JSON,我从https://wtforms-json.readthedocs.io/en/latest/得到了这个想法

然而,这会遇到相同的行为,即无法访问 object 形式的数据,从以下位置接收到多个错误:

print(form.username.data) AttributeError: 'tuple' object 没有属性 'data'

print(form.username) (<UnboundField(StringField, ('username',), {'validators': [<wtforms.validators.DataRequired object at 0x000002A3B600A6D0>]})>,)

因此,我什至无法访问我插入到表单 object 中的数据。(如果它像我假设的那样插入的话。)

在此之前:

我尝试了使用 wtforms 的标准操作方式:


    @bp.route('/api/register_user', methods=['POST', 'GET'])
    def register():
        """ End-point to register users """ 
        
        form = RegistrationForm()
    
        print(request.headers)
        if request.method == 'GET':
            print(form.csrf_token._value())
            return ('', {'csrf_token': form.csrf_token._value()})
        elif form.validate_on_submit():
            req = request.get_json(force=True)
            return{"message": "posted"}
        else:
            return { 'errors': form.errors }

现在它从未通过验证,这很奇怪,因为验证器只是 DataRequired()。 然而,当这样测试时它确实通过了 form.is_submitted() ,但我在打印时遇到了同样的错误:

print(form.username.data) AttributeError: 'tuple' object 没有属性 'data'

print(form.username) (<UnboundField(StringField, ('username',), {'validators': [<wtforms.validators.DataRequired object at 0x000002A3B600A6D0>]})>,)

同样,反应代码必须稍微改变才能工作,否则它会发回错误的请求,请求必须是:


     handleSubmit = (details) => {
    
            const finalSend =  {
                
                'csrf_token': this.state.csrf_token,
                'username': this.state.usernameInput,
                'fullname': this.state.fullnameInput,
                'phone': this.state.phoneInput,
                'email': this.state.emailInput,
                'password': this.state.passwordInput,
                'repeatPassword': this.state.repeatPasswordInput
    
            }
            axios({
                method: 'post',
                url: '/api/register_user',
                data: finalSend,
                headers: {
                  'content-type': 'application/json'
                }
              })
            .then(res => res.json()).catch(e => console.log(e));
        }

问题

所以毕竟我基本上要问的是:

  • 如何让 wtforms 将发送的数据作为 JSON object?
  • 与 WTForms 分开操作会更好吗?如果是的话,如何操作?
  • 如果我执行上述操作,我该如何设置 CSRF 安全性,就像我最近尝试的那样,但它没有按需要工作。

额外需要:

组件发送的JSON形式为:


    {'csrf_token': 'ImQ5MjhlY2VlYzM5Zjg0NmY4ZTg0NDk5ZjNlMjlkNzVlZGM4OGZhY2Ui.YBPbuw.D6BW8XpwEyXySCaBBeS0jIKYabU', 'username': 'charlie', 'fullname': 'charlie char', 'phone': '344444444', 'email': 'cindy@example.com', 'password': 'charlieandcindy', 'repeatPassword': 'charlieandcindy'}

我正在使用的 WTForm:


    class RegistrationForm(FlaskForm):
    
        username = StringField('username', validators=[DataRequired()]), 
        fullname = StringField('fullname', validators=[DataRequired()]), 
        email = BooleanField('email', validators=[DataRequired(), Email()]), 
        phone = StringField('phone', validators=[DataRequired()]), 
        password = StringField('password', validators=[DataRequired()]),
        repeatPassword = StringField('repeatPassword', validators=[DataRequired(), EqualTo('password')])
    
        def validate_username(self, username):
            print("validating username")

我还咨询了以下网站寻找解决方案: Flask wtforms - 'UnboundField' object 不可调用,动态字段无法正确初始化https://wtforms.readthedocs.io/en/2.3.x/forms/ https: //wtforms-json.readthedocs.io/en/latest/ Flask-WTF set time limit on CSRF token https://flask-wtf.readthedocs.io/en/stable/api.html#flask_wtf.csrf.generate_csrf 怎么做我从带有 JavaScript 的元标记中获取信息? https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API/Using_Fetch#uploading_json_data https://flask-wtf.readthedocs.io/en/stable/csrf.html#javascript-requests CSRF 保护使用 Flask/WTForms 和 React

我再一次尽我最大的努力找到解决方案,并付出了很多努力来写一个好问题,如果你需要更多,请问。

编辑

在这方面做了更多工作,我又尝试了一些 wtforms_json 并让它稍微填充,但它只填充 repeatPassword 字段,我不确定为什么。

为此,我不得不关闭 csrf 验证并在路由中使用以下代码:


    @bp.route('/api/register_user', methods=['POST'])
    def register():
        """ End-point to register users """ 
        json = request.get_json()
        form = RegistrationForm.from_json(json)
        print(request.get_json())
        print(form.data)
        if form.validate():
            print("validated")
            return{"message": "posted"}
        else:
            return { 'errors': form.errors }

打印 form.data 时的结果是设置了 repeatPassword 但没有别的,也不知道为什么......这个进步的来源在这里WTForms-JSON not working with FormFields

实际上,我在没有意识到您使用了wtforms_json扩展的情况下尝试了您的解决方案。 我想看看是否可以找到解决方案而无需添加额外的 flask 扩展名。

我的版本:

from werkzeug.datastructures import ImmutableMultiDict

@bp.route('/api/register_user', methods=['GET','POST'])
def register():
    """ End-point to register users """ 
    reg_json = request.get_json()
    form_input = ImmutableMultiDict(reg_json)
    form = RegistrationForm(form_input)
    
    if form.validate():
        user = Users(
            username = form.username.data,
            name = form.fullname.data,
            phone = form.phone.data,
            email = form.email.data,
            password = guard.hash_password(form.password.data),
            roles = 'User'
        )
        user.save()
        return{"message": "inserted successfully"}
    else:
        return { 'errors': form.errors }

打破它:

我发现了一些使用flask-login的示例,其中客户端将表单数据作为表单而不是 JSON 数据发送。 代码看起来像这样:

form = RegistrationForm(request.form)

我继续深入研究源代码并意识到request.form输入(到RegistrationForm构造函数中)的数据结构是ImmutableMultiDict

https://pythonise.com/series/learning-flask/the-flask-request-object (向下滚动到查询字符串)

ImmutableMultiDict可以在werkzeug扩展中找到,该扩展可以通过安装flask方便地安装。 所以基本上,我通过从请求中传递数据创建了一个ImmutableMultiDict object:

reg_json = request.get_json()
...
form_input = ImmutableMultiDict(reg_json)

并将其用作RegistrationForm的输入:

form = RegistrationForm(form_input)

现在您可以通过POST请求从不同的客户端(例如使用 React 开发的前端)接收 JSON 数据。 您也不必担心与过时的扩展保持同步。

希望这可以帮助!

干杯。

我也找到了答案。

为了做到这一点,我最终使用了 json 方法中的 wtforms_json,如下所示:

    @bp.route('/api/register_user', methods=['GET','POST'])
    def register():
        """ End-point to register users """ 
        reg_json = request.get_json()
        form = RegistrationForm.from_json(reg_json)
        
        if form.validate():
            user = Users(
                username = form.username.data,
                name = form.fullname.data,
                phone = form.phone.data,
                email = form.email.data,
                password = guard.hash_password(form.password.data),
                roles = 'User'
            )
            user.save()
            return{"message": "inserted successfully"}
        else:
            return { 'errors': form.errors }

以及不得不调整forms:


    class RegistrationForm(FlaskForm):
    
        username = StringField('username', validators=[DataRequired()]) 
        fullname = StringField('fullname', validators=[DataRequired()])
        email = StringField('email', validators=[DataRequired()])
        phone = StringField('phone', validators=[DataRequired()])
        password = StringField('password', validators=[DataRequired()])
        repeatPassword = StringField('repeatPassword', validators=[DataRequired(), EqualTo('password')])
    
        def validate_username(self, username):
            """ Ensure email isn't in use """
    
            user = Users.objects(username=username.data).first()
            if user != None:
                raise ValidationError('Please use a different username.')

唯一的缺点是我不得不关闭配置中的 csrf_token:

WTF_CSRF_ENABLED = False

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM