rest_framework -- 認證組件

来源:https://www.cnblogs.com/zhuchunyu/archive/2018/12/17/10133900.html
-Advertisement-
Play Games

#####認證組件##### 一、認證是什麼就不說了,某些網頁必須是用戶登陸之後,才能訪問的,所以這時候就需要用上認證組件。 你不用rest_framework的認證組件也行,這種認證的話,完全可以自己寫出來。 二、之前再寫APIView的時候,那裡提到過。 不記得在哪裡的話,先找dispatch方... ...


#####認證組件#####

一、認證是什麼就不說了,某些網頁必須是用戶登陸之後,才能訪問的,所以這時候就需要用上認證組件。
    你不用rest_framework的認證組件也行,這種認證的話,完全可以自己寫出來。


二、之前再寫APIView的時候,那裡提到過。
    不記得在哪裡的話,先找dispatch方法(APIView的記得),然後是self.initial(request, *args, **kwargs),
    最後找到self.perform_authentication(request)。看看這個方法的源碼:

        def perform_authentication(self, request):
            """
            Perform authentication on the incoming request.

            Note that if you override this and simply 'pass', then authentication
            will instead be performed lazily, the first time either
            `request.user` or `request.auth` is accessed.
            """
            request.user
        ## 回憶之前,此時的request是rest_framework進行封裝後的request了,所以找到的話,就去rest_framework中Request去找。
        ## 下麵附上request.user這個方法的源碼:

        @property
        def user(self):
            """
            Returns the user associated with the current request, as authenticated
            by the authentication classes provided to the request.
            """
            if not hasattr(self, '_user'):
                with wrap_attributeerrors():
                    self._authenticate()
            return self._user
        ## 它通過property將一個方法裝飾成一個屬性,此時self是request對象(rest_framework的),經過if判斷,執行了self._authenticate()
        ## 那我們繼續去看這個方法是什麼,附上源碼:

        def _authenticate(self):
            """
            Attempt to authenticate the request using each authentication instance
            in turn.
            """
            for authenticator in self.authenticators:
                try:
                    user_auth_tuple = authenticator.authenticate(self)
                except exceptions.APIException:
                    self._not_authenticated()
                    raise

                if user_auth_tuple is not None:
                    self._authenticator = authenticator
                    self.user, self.auth = user_auth_tuple
                    return

            self._not_authenticated()
        ### 此時的self也還是request對象(rest_framework的),self.authenticators這個是什麼?
        # authenticators它是request的一個屬性,那麼我們在哪裡生成了這個request對象呢?我們回到APIView的dispatch方法,找到這行代碼
        # request = self.initialize_request(request, *args, **kwargs),有沒有印象,得到一個rest_framework的request對象,
        # 下麵是self.initialize_request(request, *args, **kwargs)的源碼:

        def initialize_request(self, request, *args, **kwargs):
            """
            Returns the initial request object.
            """
            parser_context = self.get_parser_context(request)

            return Request(
                request,
                parsers=self.get_parsers(),
                authenticators=self.get_authenticators(),
                negotiator=self.get_content_negotiator(),
                parser_context=parser_context
            )
        authenticators=self.get_authenticators()   ---->> authenticators是一個裝著對象的列表
        那麼繼續看self.get_authenticators()這個方法到底做了些什麼,附上源碼:

        def get_authenticators(self):
            """
            Instantiates and returns the list of authenticators that this view can use.
            """
            return [auth() for auth in self.authentication_classes]
        返回的是一個列表,那麼self.authentication_classes應該就是列表(元組),此時self是視圖類的對象

    ####重點:面向對象屬性的查找順序,記住!!
        方式一:我們可以在當前視圖類中寫一個authentication_classes的列表(元組),裡面裝著一個一個的類,
               而這個類不是隨便的一個類,是進行認證驗證的類。
        方式二:當前視圖類中沒有authentication_classes這個屬性,那麼便會去APIView中去找該屬性,肯定能APIView中能夠找到該屬性
               authentication_classes = api_settings.DEFAULT_AUTHENTICATION_CLASSES -->> api_settings它是一個對象
               我們去看產生api_settings對象的類,其他的不說了,說說這段代碼:

                    @property
                    def user_settings(self):
                        if not hasattr(self, '_user_settings'):
                            self._user_settings = getattr(settings, 'REST_FRAMEWORK', {})
                        return self._user_settings
                    這裡的setting是通過from django.core import settings 導入的
               大概意思是:如果django的settings文件中有'REST_FRAMEWORK',那麼便會去那裡找DEFAULT_AUTHENTICATION_CLASSES這個屬性,
                          沒有的話,便會去rest_framework的settings文件中找DEFAULT_AUTHENTICATION_CLASSES,
               所以方式二可以這樣寫,在django的settings文件中寫上這樣的代碼
               REST_FRAMEWORK = {
                   'DEFAULT_AUTHENTICATION_CLASSES':[進行認證的類1,進行認證的類2],

                    }
        方式三:什麼都不寫,用rest_framework的settings文件中的DEFAULT_AUTHENTICATION_CLASSES

        好了,我們再回到self._authenticate()的源碼來看,for迴圈一個裝著對象的列表,所以authenticator就是一個對象,
        user_auth_tuple = authenticator.authenticate(self) --->>> 執行該對象的方法,將返回值賦給user_auth_tuple,
        我們使用前面的方式一,方式二,自己寫認證類的的話,那麼必須要有authenticate這個方法對吧,這個先放著,
        我們先看方式三,我猜rest_framework的settings文件中的DEFAULT_AUTHENTICATION_CLASSES里的認證類中,也肯定有authenticate方法,
        看看它是怎麼寫,我們跟著寫不就好了嘛?
        地址:from rest_framework import authentication
        看了下每個類中都有authenticate,傳來兩個參數,一個self,一個request,那我們自己寫的認證類也這樣寫。該方法的返回值將會賦值給user_auth_tuple,
        繼續回到def _authenticate(self)這個方法中,繼續看,如果返回值user_auth_tuple為None的話,將會繼續for迴圈,返回值為True的話,
        那麼這個返回值必須為一個元組,而且只能有兩個元素。執行for迴圈的過程中,authenticate這個方法沒有異常的話,那麼表示驗證成功。


    總結:上面把認證的整個流程都寫了一般,那麼需要些的東西我列出來,
          1、根據需求要求自己寫一個認證類,該類必須要有authenticate這個方法,繼承BaseAuthentication這個類
          2、驗證通過的話,返回None或者兩個元素的元組(列表也行)
          3、驗證不通過的話,拋異常,拋這個異常exceptions.APIException
          4、假如只想當前視圖類中使用認證功能的話,那麼在當前視圖類中添加authentication_classes屬性
          5、想全局都想進行認證功能,就在django的settings文件中添加
             REST_FRAMEWORK = {
                       'DEFAULT_AUTHENTICATION_CLASSES':[進行認證的類1,進行認證的類2],
                        }
          6、如果你既想全局配置,但是某個局部又不配置認證的話,那麼就是該視圖類中寫authentication_classes,值為[],就好了。

下麵寫個登陸驗證的例子把:test頁面必須登陸之後才能訪問

models文件:
class User(models.Model):
    nid = models.AutoField(primary_key=True)
    name = models.CharField(max_length=32)
    password = models.CharField(max_length=32)


class Token(models.Model):
    nid = models.AutoField(primary_key=True)
    token = models.UUIDField(max_length=64)
    user = models.OneToOneField(to='User')


urls文件:
    url(r'login/',views.Login.as_view()),
    url(r'test/',views.Test.as_view()),


views文件:
class Login(APIView):
    def post(self, request, *args, **kwargs):
        response = {'status': 100, 'msg': '登陸成功'}
        name = request.data.get('name')
        password = request.data.get('password')
        try:
            user = models.User.objects.get(name=name, password=password)
            token = uuid.uuid4()
            ret = models.Token.objects.filter(user=user)
            if ret:
                models.Token.objects.filter(user=user).update(token=token)
            else:
                models.Token.objects.create(token=token, user=user)
            response['token'] = token
        except Exception:
            response['status'] = 101
            response['msg'] = '用戶名或密碼錯誤'
        return JsonResponse(response)

class Test(APIView):
    authentication_classes = [LoginAuth,]
    def get(self, request, *args, **kwargs):
        return HttpResponse('test get')


auth_class.py文件:

class LoginAuth(BaseAuthentication):
    def authenticate(self, request):
        token = request.query_params.get('token', None)
        try:
            ret = models.Token.objects.get(token=token)
        except ObjectDoesNotExist:
            raise exceptions.APIException('請先進行登陸')
        return ret.user, ret


這裡推薦一個發送各種請求的軟體,postman

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在 Python2 中如要想要獲得用戶從命令行的輸入,可以使用 input() 和 raw_input() 兩個函數,那麼這兩者有什麼區別呢? 我們先藉助 help 函數來看下兩者的文檔註釋: 可以看出,raw_input() 返回的始終是一個“原始”(raw)字元串,並且去掉了行末的換行符。 值得 ...
  • 前面多次提到了正則串、正則表達式,那麼正則表達式究竟是符合什麼定義的字元串呢?正則表達式是編程語言處理字元串格式的一種邏輯式子,它利用若幹保留字元定義了形形色色的匹配規則,從而通過一個式子來覆蓋滿足了上述規則的所有字元串。正則表達式的保留字元主要有:圓括弧、方括弧、花括弧、豎線、橫線、點號、加號、星 ...
  • 1.Spring依賴的jar包下載網址: https://repo.spring.io/webapp/#/artifacts/browse/tree/General/libs-release-local/org/springframework/spring/4.0.0.RELEASE/spring- ...
  • 基礎 莫隊是啥? 暴力!! 莫隊的基礎是啥?會寫暴力!!! 啥叫莫隊 ...
  • 一.內置函數的補充 repr() 顯示出字元串的官方表示形式 chr() ord() 二.遞歸 二分法 ...
  • 面向對象 1.初始面向對象 面向過程: 一切以事務的發展流程為中心. 面向對象: 一切以對象為中心. 一切皆為對象. 具體的某一個事務就是對象 打比方: 面向過程 大象進冰箱 步驟: 第一步, 開門, 第二步, 裝大象, 第三步, 關門 面向對象: 大象, 你進冰箱. 此時主語是大象. 我操縱的是大 ...
  • 題意 一天做到兩道這種題目描述如此神仙的題也是夠了。真鍛煉語文能力。 題目的意思其實就是,給你一個序列,然後每次詢問一個區間。使得儘量 ...
  • 前言 感謝! 承蒙關照~ 中的 簡介筆記 簡介 1. 基本概念 2. 項目搭建 使用 進行開發 實現數據綁定和文件上傳 是開發 架構的通用方式. 前端控制器, 到`Controller View template`,用戶頁面. 前端控制器 前端控制器用於分發調度,而控制器用於業務數據抽取, 用於頁面 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...