TypeError при создании WTForm с MongoEngine и Flask

Я пытаюсь создать простое веб-приложение, используя Python3.4, Flask и MongoEngine.

У меня возникла проблема при попытке создать форму с помощью WTForms.

У меня есть инициализированное приложение MongoEngine, называемое базой данных:

database = MongoEngine(app)

Я создал класс модели под названием Post:

class Post(database.Document):
    author = database.StringField(
        default='David Y. Stephenson', max_length=255, required=True
    )
    body = database.StringField(required=True)
    comments = database.ListField(
        database.EmbeddedDocumentField('Comment')
    )
    slug = database.StringField(max_length=255, required=True, unique=True)
    tease = database.StringField(max_length=255, required=True)
    time = database.DateTimeField(
        default=datetime.datetime.now, required=True
    )
    title = database.StringField(max_length=255, required=True, unique=True)

У меня есть маршрут для создания формы с использованием этого класса:

@app.route('/blog/new')
def new():
    form = model_form(Post)
    return render_template('new_blog.html', form=form)

Когда я запускаю этот маршрут, я получаю TypeError:

TypeError: model must be a mongoengine Document schema

Глядя на код WTForm, кажется, что класс Post должен быть экземпляром MongoEngine BaseDocument или DocumentMetaclass classes. Строки 223-224 из /usr/local/lib/python3.4/dist-packages/flask_mongoengine/wtf/orm.py гласят:

from mongoengine.base import BaseDocument, DocumentMetaclass
if not isinstance(model, (BaseDocument, DocumentMetaclass)):
    raise TypeError('model must be a mongoengine Document schema'

Когда я пытаюсь вручную проверить свой класс Post, он кажется DocumentMetaclass, но не BaseDocument. Добавление

app.logger.debug(isinstance(Post, mongoengine.base.BaseDocument))
app.logger.debug(isinstance(Post, mongoengine.base.DocumentMetaclass))

в мой код возвращается:

--------------------------------------------------------------------------------
DEBUG in __init__ [__init__.py:57]:
False
--------------------------------------------------------------------------------
--------------------------------------------------------------------------------
DEBUG in __init__ [__init__.py:58]:
True
--------------------------------------------------------------------------------

Что я делаю неправильно?

Полный текст моего файла __init__.py:

import datetime
from flask import Flask
from flask import render_template
from flask.ext.mongoengine import MongoEngine
from flask.ext.mongoengine.wtf import model_form
import mongoengine.base


app = Flask(__name__)


app.config['MONGODB_SETTINGS'] = {
    'db': 'davidystephenson',
    'host': 'ds059908.mongolab.com',
    'username': 'david',
    'password': 'opensecret',
    'port': 59908,
}
database = MongoEngine(app)


class Post(database.Document):
    author = database.StringField(
        default='David Y. Stephenson', max_length=255, required=True
    )
    body = database.StringField(required=True)
    comments = database.ListField(
        database.EmbeddedDocumentField('Comment')
    )
    slug = database.StringField(max_length=255, required=True, unique=True)
    tease = database.StringField(max_length=255, required=True)
    time = database.DateTimeField(
        default=datetime.datetime.now, required=True
    )
    title = database.StringField(max_length=255, required=True, unique=True)


@app.route('/')
def index():
    return render_template('index.html')


@app.route('/blog/')
def blog():
    posts = Post.objects.all()
    return render_template('blog.html', posts=posts)


@app.route('/blog/<slug>')
def post(slug):
    post = Post.objects.get(slug=slug)
    return render_template('post.html', post=post)


@app.route('/blog/new')
def new():
    app.logger.debug(isinstance(Post, mongoengine.base.BaseDocument))
    app.logger.debug(isinstance(Post, mongoengine.base.DocumentMetaclass))
    form = model_form(Post)
    return render_template('new_blog.html', form=form)

if __name__ == '__main__':
    app.run(debug=True)

person David Y. Stephenson    schedule 25.08.2014    source источник


Ответы (2)


Я нахожу ту же проблему.

Когда я напечатал все локальные переменные перед исключением, я обнаружил, что текущая модель удовлетворяет условию

if not isinstance(model, (BaseDocument, DocumentMetaclass)):

но следующего не будет. Следующей моделью был путь к родственной модели. В вашем примере это база данных.EmbeddedDocumentField('Comment')

Исключение возникает, когда свойство списка ListField определено строковым путем к указанной модели.

comments = database.ListField(
    database.EmbeddedDocumentField('Comment')
)

вместо этого определите с классом модели

comments = database.ListField(
    database.EmbeddedDocumentField(Comment)
)

Я решаю эту ошибку в моем запросе на включение в проекте Flask-Admin. https://github.com/mrjoes/flask-admin/pull/645

person Alexey    schedule 03.09.2014
comment
Я думаю, что это ошибка в Flask-Admin, но вы можете решить проблему, импортировав модель комментариев и установив ее как класс database.EmbeddedDocumentField(Comment). Вчера я создал задачу github.com/mrjoes/flask-admin/issues/636 - person Alexey; 04.09.2014

Вы импортируете экземпляр базы данных, с помощью которого вы инициализировали приложение?

Вот пример того, как может выглядеть файл models.py. Мне кажется, что ваша программа не знает, что вы имеете в виду, когда пишете database.Document, поэтому я подозреваю, что это ваша проблема:

from app import db # <- Do you have this line in the same file as your class?

class Person(db.DynamicDocument):

    # Meta variables.
    meta = {
        'collection': 'people'
    }

# Document variables.
age = db.IntField()
name = db.StringField()
person David K.    schedule 26.08.2014
comment
И, конечно же, db.Document и db.DynamicDocument должны работать нормально. Это, так сказать, варианты на одну и ту же тему. - person David K.; 27.08.2014
comment
Да. Я даже попытался переместить класс Post в init.py, чтобы попытаться исправить это. Я добавил в свой вопрос полный текст init.py. - person David Y. Stephenson; 27.08.2014
comment
Интересно. В этом случае у меня нет для вас прямого ответа, а скорее подход — я бы рекомендовал сначала создать радикально упрощенную версию этого приложения, просто чтобы основы работали. Затем, если вы посещаете веб-страницу, а она все еще ломается и дает трассировку стека, я бы использовал консоль (предоставленную на странице отладчика) для исследования объектов на каждом этапе, чтобы увидеть, что происходит. - person David K.; 27.08.2014