Подтвердить что ты не робот

У меня возникают проблемы с wtforms selectfields, когда я использую POST с Flask

Я новичок в wtforms и flask, и я возился с selectfields и получил ошибку. Сама форма отлично работает без поля выбора, но с этим я получаю следующую ошибку:

Ошибка:

....fields.py", line 386, in pre_validate
    for v, _ in self.choices: TypeError: 'NoneType' object is not iterable

Я вижу поле выбора так, чтобы оно отображалось. Я подозреваю, что как-то идентификатор не проверяется должным образом в POST и не возвращает его. Или это как-то связано с возвратом моего кортежа selectfield? Также поле ID, которое я использую, вытягивается из автоматического ключа GAE ndb(). Id(), который довольно длинный и неприятный. Может быть, длина id, используемая для поля выбора слишком длинная?

Googling не предоставил много с точки зрения точной проблемы, поэтому подумал, что я бы разместил здесь. Соответствующий код ниже. Если я что-то пропустил, дайте мне знать

view.py code:

@app.route('/new/post', methods = ['GET', 'POST'])
@login_required
def new_post():

    form = PostForm()
    if form.validate_on_submit():
        post = Post(title = form.title.data,
                    content = form.content.data,
                    hometest = form.hometest.data,
                    author = users.get_current_user())
        post.put()
        flash('Post saved on database.')
        return redirect(url_for('list_posts'))
    form.hometest.choices = [ (h.key.id(),h.homename)for h in Home.query()]

    return render_template('new_post.html', form=form)

myforms.py

class PostForm(Form):
    title = wtf.TextField('Title', validators=[validators.Required()])
    content = wtf.TextAreaField('Content', validators=[validators.Required()])
    hometest = wtf.SelectField(u'Home Name List', coerce=int,validators=[validators.optional()])

new_post.html:

{% extends "base.html" %}

{% block content %}
    <h1 id="">Write a post</h1>
    <form action="{{ url_for('new_post') }}" method="post" accept-charset="utf-8">
        {{ form.csrf_token }}
        <p>
            <label for="title">{{ form.title.label }}</label><br />
            {{ form.title|safe }}<br />
            {% if form.title.errors %}
            <ul class="errors">
                {% for error in form.title.errors %}
                <li>{{ error }}</li>
                {% endfor %}
            </ul>
            {% endif %}
        </p>
        <p>
            <label for="title">{{form.hometest.label}}</label><br/>
            {{form.hometest}}
            {% if form.hometest.errors %}
        <ul class="errors">
            {% for error in form.hometest.errors %}
            <li>{{ error }}</li>
            {% endfor %}
        </ul>
        {% endif %}
        </p>
        <p>
            <label for="title">{{ form.content.label }}</label><br />
            {{ form.content|safe }}<br />

            {% if form.content.errors %}
            <ul class="errors">
                {% for error in form.content.errors %}
                <li>{{ error }}</li>
                {% endfor %}
            </ul>
            {% endif %}
        </p>
        <p><input type="submit" value="Save post"/></p>
    </form>
{% endblock %}
4b9b3361

Ответ 1

Вам нужно задать свой выбор, прежде чем вы вызовете validate_on_submit, поскольку form.validate попытается подтвердить предоставленное значение (если оно есть) в отношении списка вариантов (который None перед установкой choices):

form = PostForm()
form.hometest.choices = [(h.key.id(), h.homename) for h in Home.query()]

if form.validate_on_submit():
    # form is valid, continue

Ответ 2

Вам следует предоставить аргумент choices=[...], например

wtf.SelectField(u'Home Name List',
                choices=[(1, 'Label 1'),
                         (2, 'Label 2')],
                coerce=int,
                validators=[validators.optional()])