這篇文章的主要目的是分享學(xué)習(xí)<<深入理解Flask>>這本書中遇到的坑,可以使遇到相同問題的同學(xué)少走彎路。大家有遇到或解決過相關(guān)問題的,也歡迎分享出來。
坑一:
在把博客從sqlite重構(gòu)到mongodb時(shí),在登陸成功后想要訪問只有登陸用戶才能訪問的頁面時(shí),遇到了下面的錯(cuò)誤:
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'
我是在登陸成功后,準(zhǔn)備寫新的博客時(shí)遇到的這個(gè)錯(cuò)誤。可以看到,這個(gè)函數(shù)用login_required裝飾器進(jìn)行了裝飾,因此訪問這個(gè)頁面要求有登陸信息:
@blog_blue進(jìn)一步查看裝飾器的實(shí)現(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可以看到出錯(cuò)的語句就是在訪問current_user.is_authenticated時(shí)報(bào)錯(cuò)的.錯(cuò)誤提示說沒有'is_authenticated'屬性,查看User的models定義:
class User(mongo.Document): ... def is_authenticated(self): if isinstance(self, AnonymousUserMixin): return False else: return True也是定義了這個(gè)屬性的,再仔細(xì)看出錯(cuò)提示是說 'BaseQuerySet'沒有這個(gè)屬性,為什么會(huì)報(bào)這個(gè)類錯(cuò)誤呢,我們的User應(yīng)該不是這個(gè)類啊,因此查看'current_user'的具體實(shí)現(xiàn):current_user = LocalProxy(lambda: _get_user())一路跟蹤,可以看出最終的user是在下面設(shè)置的: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,而這個(gè)user_callback就是通過user_id獲取user的關(guān)鍵,進(jìn)一步看這個(gè)函數(shù)是在何時(shí)設(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時(shí)曾在extensions.py中設(shè)置過這個(gè)函數(shù):@login_manager.user_loaderdef load_user(userid): from .models import User return User.objects(id=userid)可以看到,當(dāng)時(shí)使用的是sqlite,因此這樣返回的就是User對(duì)象,而現(xiàn)在使用mongodb后,這樣返回的只是一個(gè)'BaseQuerySet',因此出現(xiàn)了問題。修改成下面后,問題解決:@login_manager.user_loaderdef load_user(userid): from .models import User return User.objects(id=userid).first()問題的根本原因使用mongodb后,所有使用相關(guān)數(shù)據(jù)庫操作的代碼都要去適配。
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注