這篇文章的主要目的是分享學習<<深入理解Flask>>這本書中遇到的坑,可以使遇到相同問題的同學少走彎路。大家有遇到或解決過相關問題的,也歡迎分享出來。
坑一:
在把博客從sqlite重構到mongodb時,在登陸成功后想要訪問只有登陸用戶才能訪問的頁面時,遇到了下面的錯誤:
in decorated_view elif not current_user.is_authenticated: File "/usr/local/lib/python3.4/dist-packages/werkzeug/local.py", line 343, in __getattr__ return getattr(self._get_current_object(), name)AttributeError: 'BaseQuerySet' object has no attribute 'is_authenticated'
我是在登陸成功后,準備寫新的博客時遇到的這個錯誤??梢钥吹剑@個函數(shù)用login_required裝飾器進行了裝飾,因此訪問這個頁面要求有登陸信息:
@blog_blue進一步查看裝飾器的實現(xiàn):def login_required(func): @wraps(func) def decorated_view(*args, **kwargs): if request.method in EXEMPT_METHODS: return func(*args, **kwargs) elif current_app.login_manager._login_disabled: return func(*args, **kwargs) elif not current_user.is_authenticated: return current_app.login_manager.unauthorized() return func(*args, **kwargs) return decorated_view可以看到出錯的語句就是在訪問current_user.is_authenticated時報錯的.錯誤提示說沒有'is_authenticated'屬性,查看User的models定義:
class User(mongo.Document): ... def is_authenticated(self): if isinstance(self, AnonymousUserMixin): return False else: return True也是定義了這個屬性的,再仔細看出錯提示是說 'BaseQuerySet'沒有這個屬性,為什么會報這個類錯誤呢,我們的User應該不是這個類啊,因此查看'current_user'的具體實現(xiàn):current_user = LocalProxy(lambda: _get_user())一路跟蹤,可以看出最終的user是在下面設置的:flask_login/login_manager.py:
def reload_user(self, user=None): ctx = _request_ctx_stack.top if user is None: user_id = session.get('user_id') if user_id is None: ctx.user = self.anonymous_user() else: if self.user_callback is None: raise Exception( "No user_loader has been installed for this " "LoginManager. Add one with the " "'LoginManager.user_loader' decorator.") user = self.user_callback(user_id)...通過user_id獲取user,而這個user_callback就是通過user_id獲取user的關鍵,進一步看這個函數(shù)是在何時設置的:def user_loader(self, callback): ''' This sets the callback for reloading a user from the session. The function you set should take a user ID (a ``unicode``) and return a user object, or ``None`` if the user does not exist. :param callback: The callback for retrieving a user object. :type callback: callable ''' self.user_callback = callback return callback前面使用sqlalchemy時曾在extensions.py中設置過這個函數(shù):@login_manager.user_loaderdef load_user(userid): from .models import User return User.objects(id=userid)可以看到,當時使用的是sqlite,因此這樣返回的就是User對象,而現(xiàn)在使用mongodb后,這樣返回的只是一個'BaseQuerySet',因此出現(xiàn)了問題。修改成下面后,問題解決:@login_manager.user_loaderdef load_user(userid): from .models import User return User.objects(id=userid).first()問題的根本原因使用mongodb后,所有使用相關數(shù)據(jù)庫操作的代碼都要去適配。
新聞熱點
疑難解答