2

我正在尝试使用 Python3.4、Flask 和 MongoEngine 创建一个简单的 web 应用程序。

尝试使用 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 类是 MongoEngineBaseDocumentDocumentMetaclass classes. /usr/local/lib/python3.4/dist-packages/flask_mongoengine/wtf/orm.py读取的第 223-224 行:

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

当我尝试手动检查我的 Post 类时,它似乎是 a 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)
4

2 回答 2

1

我发现同样的问题。

当我在异常之前打印所有局部变量时,我发现当前模型满足条件

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

但接下来不会。下一个模型是相关模型的路径。在你的例子中是 database.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

于 2014-09-03T10:03:54.783 回答
0

您是否正在导入初始化应用程序的数据库实例?

下面是一个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()
于 2014-08-26T21:27:14.193 回答