Django從request請求到響應(yīng)的原理

前言

最近在研究學(xué)習(xí)django rest framework中的用戶登錄的權(quán)限和驗證方面基礎(chǔ),想結(jié)合源碼總結(jié)一下次询。

問題場景

在一個網(wǎng)頁中彰檬,比如博客,通過一個URL請求某個文章乔询,后臺系統(tǒng)中會判斷請求用戶中能否有權(quán)限進(jìn)行訪問蔬蕊,一般做法都會是獲取客戶端中的cookie信息,來進(jìn)行驗證是否有權(quán)限。

原理

我們從瀏覽器發(fā)出一個請求 Request岸夯,得到一個響應(yīng)后的內(nèi)容 HttpResponse 麻献,這個請求傳遞到 Django的過程是通過middleware來實現(xiàn),在settings.py中設(shè)置的middleware中的process_requestprocess_response依次執(zhí)行猜扮,接著再執(zhí)行后臺視圖函數(shù)勉吻。這篇文章主要講解用戶驗證就是SessionMiddlewareAuthenticationMiddleware

MIDDLEWARE = [
    'corsheaders.middleware.CorsMiddleware',
    'django.middleware.security.SecurityMiddleware',
    'django.contrib.sessions.middleware.SessionMiddleware',
    'django.middleware.common.CommonMiddleware',
    'django.middleware.csrf.CsrfViewMiddleware',
    'django.contrib.auth.middleware.AuthenticationMiddleware',
    'django.contrib.messages.middleware.MessageMiddleware',
    'django.middleware.clickjacking.XFrameOptionsMiddleware',
]
過程圖.png

步驟:

SessionMiddleware部分

/django/contrib/sessions/middleware.py
1.先依次加載上面中的middleware中有重載process_requestprocess_response
2.process_request會攔截用戶的cookie旅赢,把cookie中的信息放進(jìn)去request.session中

class SessionMiddleware(MiddlewareMixin):
    def __init__(self, get_response=None):
        self.get_response = get_response
        engine = import_module(settings.SESSION_ENGINE)
        self.SessionStore = engine.SessionStore

    def process_request(self, request):
        session_key = request.COOKIES.get(settings.SESSION_COOKIE_NAME)
        request.session = self.SessionStore(session_key)

    def process_response(self, request, response):
        """
        If request.session was modified, or if the configuration is to save the
        session every time, save the changes and set a session cookie or delete
        the session cookie if the session has been emptied.
        """
        try:
            accessed = request.session.accessed
            modified = request.session.modified
            empty = request.session.is_empty()
        except AttributeError:
            pass
        else:
            # First check if we need to delete this cookie.
            # The session should be deleted only if the session is entirely empty
            if settings.SESSION_COOKIE_NAME in request.COOKIES and empty:
                response.delete_cookie(
                    settings.SESSION_COOKIE_NAME,
                    path=settings.SESSION_COOKIE_PATH,
                    domain=settings.SESSION_COOKIE_DOMAIN,
                )
            else:
                if accessed:
                    patch_vary_headers(response, ('Cookie',))
                if (modified or settings.SESSION_SAVE_EVERY_REQUEST) and not empty:
                    if request.session.get_expire_at_browser_close():
                        max_age = None
                        expires = None
                    else:
                        max_age = request.session.get_expiry_age()
                        expires_time = time.time() + max_age
                        expires = http_date(expires_time)
                    # Save the session data and refresh the client cookie.
                    # Skip session save for 500 responses, refs #3881.
                    if response.status_code != 500:
                        try:
                            request.session.save()
                        except UpdateError:
                            raise SuspiciousOperation(
                                "The request's session was deleted before the "
                                "request completed. The user may have logged "
                                "out in a concurrent request, for example."
                            )
                        response.set_cookie(
                            settings.SESSION_COOKIE_NAME,
                            request.session.session_key, max_age=max_age,
                            expires=expires, domain=settings.SESSION_COOKIE_DOMAIN,
                            path=settings.SESSION_COOKIE_PATH,
                            secure=settings.SESSION_COOKIE_SECURE or None,
                            httponly=settings.SESSION_COOKIE_HTTPONLY or None,
                            samesite=settings.SESSION_COOKIE_SAMESITE,
                        )
        return response
AuthenticationMiddleware部分

/django/contrib/auth/middleware.py
1.先在process_request方法中判斷request中是否有session屬性字段齿桃,然后在get_uesr函數(shù)中調(diào)用auth.get_user方法,返回user煮盼,然后設(shè)置request.user
源碼:

def get_user(request):
    if not hasattr(request, '_cached_user'):
        request._cached_user = auth.get_user(request)
    return request._cached_user


class AuthenticationMiddleware(MiddlewareMixin):
    def process_request(self, request):
        assert hasattr(request, 'session'), (
            "The Django authentication middleware requires session middleware "
            "to be installed. Edit your MIDDLEWARE%s setting to insert "
            "'django.contrib.sessions.middleware.SessionMiddleware' before "
            "'django.contrib.auth.middleware.AuthenticationMiddleware'."
        ) % ("_CLASSES" if settings.MIDDLEWARE is None else "")
        request.user = SimpleLazyObject(lambda: get_user(request))
def get_user(request):
    """
    Return the user model instance associated with the given request session.
    If no user is retrieved, return an instance of `AnonymousUser`.
    """
    from .models import AnonymousUser
    user = None
    try:
        user_id = _get_user_session_key(request)
        backend_path = request.session[BACKEND_SESSION_KEY]
    except KeyError:
        pass
    else:
        if backend_path in settings.AUTHENTICATION_BACKENDS:
            backend = load_backend(backend_path)
            user = backend.get_user(user_id)
            # Verify the session
            if hasattr(user, 'get_session_auth_hash'):
                session_hash = request.session.get(HASH_SESSION_KEY)
                session_hash_verified = session_hash and constant_time_compare(
                    session_hash,
                    user.get_session_auth_hash()
                )
                if not session_hash_verified:
                    request.session.flush()
                    user = None

    return user or AnonymousUser()

總結(jié)

settings.py中的middleware是對每個用戶請求做處理短纵,而drf中 BasicAuthentication
SessionAuthentication僵控,TokenAuthentication是用來驗證用戶登錄信息的類香到,也就是用戶驗證信息上面的。對于TokenAuthentication驗證方式报破,這一篇文章有詳細(xì)介紹悠就。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌曹宴,老刑警劉巖篮绰,帶你破解...
    沈念sama閱讀 222,865評論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,296評論 3 399
  • 文/潘曉璐 我一進(jìn)店門宇立,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人自赔,你說我怎么就攤上這事妈嘹。” “怎么了绍妨?”我有些...
    開封第一講書人閱讀 169,631評論 0 364
  • 文/不壞的土叔 我叫張陵润脸,是天一觀的道長。 經(jīng)常有香客問我他去,道長毙驯,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 60,199評論 1 300
  • 正文 為了忘掉前任灾测,我火速辦了婚禮爆价,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己铭段,他們只是感情好骤宣,可當(dāng)我...
    茶點故事閱讀 69,196評論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著序愚,像睡著了一般憔披。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上爸吮,一...
    開封第一講書人閱讀 52,793評論 1 314
  • 那天芬膝,我揣著相機與錄音,去河邊找鬼形娇。 笑死锰霜,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的桐早。 我是一名探鬼主播癣缅,決...
    沈念sama閱讀 41,221評論 3 423
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼勘畔!你這毒婦竟也來了所灸?” 一聲冷哼從身側(cè)響起丽惶,我...
    開封第一講書人閱讀 40,174評論 0 277
  • 序言:老撾萬榮一對情侶失蹤炫七,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后钾唬,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體万哪,經(jīng)...
    沈念sama閱讀 46,699評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,770評論 3 343
  • 正文 我和宋清朗相戀三年抡秆,在試婚紗的時候發(fā)現(xiàn)自己被綠了奕巍。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,918評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡儒士,死狀恐怖的止,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情着撩,我是刑警寧澤诅福,帶...
    沈念sama閱讀 36,573評論 5 351
  • 正文 年R本政府宣布,位于F島的核電站拖叙,受9級特大地震影響氓润,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜薯鳍,卻給世界環(huán)境...
    茶點故事閱讀 42,255評論 3 336
  • 文/蒙蒙 一咖气、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦崩溪、人聲如沸浅役。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,749評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽担租。三九已至,卻和暖如春抵怎,著一層夾襖步出監(jiān)牢的瞬間奋救,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,862評論 1 274
  • 我被黑心中介騙來泰國打工反惕, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留尝艘,地道東北人。 一個月前我還...
    沈念sama閱讀 49,364評論 3 379
  • 正文 我出身青樓姿染,卻偏偏與公主長得像背亥,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子悬赏,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,926評論 2 361