2

我正在尝试使用 Flask-Mongoengine 和 Flask-Login 为应用程序编写授权代码。我收到了这个奇怪的错误:

File "/usr/lib/python3.5/site-packages/mongoengine/base/document.py", line 188, in __setattr__
    super(BaseDocument, self).__setattr__(name, value)
  File "/usr/lib/python3.5/site-packages/mongoengine/base/fields.py", line 132, in __set__
    if instance._initialised:
AttributeError: _initialised

我的模型.py:

from app import app, db
from flask.ext.login import LoginManager


login_manager = LoginManager()
login_manager.init_app(app)


class User(db.Document):
    email = db.StringField(required=True)
    first_name = db.StringField(max_lenght=40, required=True)
    last_name = db.StringField(max_lenght=40, required=True)
    password = db.StringField(required=True)

    def __init__(self, email, first_name, last_name, password):
        self.email = email
        self.first_name = first_name
        self.last_name = last_name
        self.password = password

    def is_authenticated(self):
        return True

    def is_active(self):
        return True

    def is_anonymous(self):
        return False

   def get_id(self):
        return self.email

我试过超班,但没有帮助。

https://github.com/MongoEngine/flask-mongoengine/issues/156

4

4 回答 4

5

Mongoengine 不需要你定义一个 __init__。

class User(db.Document):
    email = db.StringField(required=True)
    first_name = db.StringField(max_lenght=40, required=True)
    last_name = db.StringField(max_lenght=40, required=True)
    password = db.StringField(required=True)

    def clean(self):
        # clean will be called when you call .save()
        # You can do whatever you'd like to clean data before save
        self.password = str(self.password)

那么你所要做的就是

user = User(email='email@gmail', first_name='hello', last_name='there', password=2342143213)
user.save()
print(user.id)
于 2016-10-28T22:12:31.877 回答
3

这个问题的解决方法是你需要像这样调用超类的构造函数

class User(db.Document):
    email = db.StringField(required=True)
    first_name = db.StringField(max_lenght=40, required=True)
    last_name = db.StringField(max_lenght=40, required=True)
    password = db.StringField(required=True)

    def __init__(self, email, first_name, last_name, password, *args, **kwargs):
        super(User, self).__init__(*args, **kwargs)
        self.email = email
        self.first_name = first_name
        self.last_name = last_name
        self.password = password

    def is_authenticated(self):
        return True

    def is_active(self):
        return True

    def is_anonymous(self):
        return False

   def get_id(self):
        return self.email
于 2016-08-06T14:40:02.453 回答
0

未来的读者应该知道,在每次调用之前都会调用knittledan答案,因此如果它用于对模型中的密码字段进行哈希处理,例如:clean() save()

def clean(self):
    self.password = bcrypt.generate_password_hash(self.password).decode('utf-8')

密码将不断被覆盖,这是不希望的。一种解决方法是使用附加字段,例如:password_hashed = db.BooleanField(default=False)

def clean(self):
    if not self.password_hashed:
        self.password        = bcrypt.generate_password_hash(self.password).decode('utf-8')
        self.password_hashed = True
于 2018-03-20T07:53:52.047 回答
0

Super Class 尚未创建,您必须调用 super,如下所示;

def __init__(self, email, first_name, last_name, password):
    ***super().__init__()***
    self.email = email
    self.first_name = first_name
    self.last_name = last_name
    self.password = password
于 2022-01-07T22:10:15.363 回答