为啥 Flask WTForms 和 WTForms-SQLAlchemy QuerySelectField 会产生太多无法解包的值?

Posted

技术标签:

【中文标题】为啥 Flask WTForms 和 WTForms-SQLAlchemy QuerySelectField 会产生太多无法解包的值?【英文标题】:Why does Flask WTForms and WTForms-SQLAlchemy QuerySelectField produce too many values to unpack?为什么 Flask WTForms 和 WTForms-SQLAlchemy QuerySelectField 会产生太多无法解包的值? 【发布时间】:2018-06-29 09:47:57 【问题描述】:

我有一个非常基本的 Flask 应用程序:

from flask import Flask, render_template
from flask_wtf import FlaskForm
from wtforms_sqlalchemy.fields import QuerySelectField
from wtforms.validators import DataRequired
from flask_sqlalchemy import SQLAlchemy
from wtforms import StringField

db = SQLAlchemy()


app = Flask(__name__)
app.config['SQLALCHEMY_URI'] = 'sqlite:///:memory:'
app.config['SECRET_KEY'] = 'fnord'
db.init_app(app)


class Section(db.Model):
    id = db.Column(db.Integer, primary_key = True)
    name = db.Column(db.Text, nullable = False)
    subject_id = db.Column(db.Integer, db.ForeignKey('subject.id'))
    subject = db.relationship('Subject', back_populates='sections')


class Subject(db.Model):
    id = db.Column(db.Integer, primary_key = True)
    name = db.Column(db.Text, nullable = False)
    sections = db.relationship('Section', back_populates='subject')


def do_something():
    return Subject.query


class SelectionForm(FlaskForm):
    name = StringField('Section Name', validators=[DataRequired()])
    subject = QuerySelectField('Subject', query_factory=do_something, allow_blank=False, get_label='name')

with app.app_context():
    db.create_all()
    db.session.add(Subject(id=1, name='Science'))
    db.session.add(Subject(id=2, name='Math'))
    db.session.commit()

@app.route('/')
def main():
    form = SelectionForm()
    return render_template('index.html', form=SelectionForm())

app.run('0.0.0.0', port=5000, debug=True)

使用非常简单的模板:

<!DOCTYPE html>
<html>
  <body>
     form.csrf_token 
     form.name 
     form.subject 
  </body>
</html>

问题是我得到了ValueError: too many values to unpack (expected 2)。这显然不是我所期望的——据我所知,我正在遵循我在网上看到的示例,但这里有些不同。问题是,无论 wtforms_sqlalchemy/fields.py 文件中发生了什么,它都会得到(&lt;class '__main__.Subject'&gt;, (1), None),而不是我预期的可能是别的东西,因为它试图分配给cls, key,可能(&lt;class '__main__.Subject'&gt;, 1)

那么我在这里做错了什么?

【问题讨论】:

查看本期QuerySelectField returning ValueError: too many values to unpack (expected 2) @pjcunningham 就是这样,谢谢! 【参考方案1】:

基于 pjcunningham 链接的补丁,由于 wtforms 尚未发布该更新,我继续创建自己的猴子补丁:

import wtforms_sqlalchemy.fields as f                                                                                                                                                                                                                                                                    
def get_pk_from_identity(obj):
    cls, key = f.identity_key(instance=obj)[:2]
    return ':'.join(f.text_type(x) for x in key)                             
f.get_pk_from_identity = get_pk_from_identity

在我的代码库中添加它现在就足够了,直到他们删除了与最新版本的 SQLAlchemy 一起使用的版本。

【讨论】:

以上是关于为啥 Flask WTForms 和 WTForms-SQLAlchemy QuerySelectField 会产生太多无法解包的值?的主要内容,如果未能解决你的问题,请参考以下文章

Flask WTForm BooleanField用法

为啥 Flask WTForms 和 WTForms-SQLAlchemy QuerySelectField 会产生太多无法解包的值?

Flask WTForm表单编程

wtform 表单示例

如何将flask_sqlalchemy orm中的数据添加到WTForm FieldList?

为 wtforms 字段赋值