drf框架中jwt認證,以及自定義jwt認證

来源:https://www.cnblogs.com/pythonywy/archive/2019/09/09/11494174.html
-Advertisement-
Play Games

0909自我總結 drf框架中jwt 一.模塊的安裝 :http://getblimp.github.io/django rest framework jwt/ 他是個第三方的開源項目 :`pip install djangorestframework jwt` 使用 設定好的jwt 測試介面:po ...


0909自我總結

drf框架中jwt

一.模塊的安裝

官方:http://getblimp.github.io/django-rest-framework-jwt/

他是個第三方的開源項目

安裝:pip install djangorestframework-jwt

使用自帶設定好的jwt

from django.urls import path
from rest_framework_jwt.views import obtain_jwt_token
urlpatterns = [
    path('login/', obtain_jwt_token),
]
'''
path('login/', obtain_jwt_token)其實相當於path('login/', ObtainJSONWebToken.as_view())
因為我們之間進源碼可以看到
obtain_jwt_token = ObtainJSONWebToken.as_view()     #獲得
refresh_jwt_token = RefreshJSONWebToken.as_view()   #刷新
verify_jwt_token = VerifyJSONWebToken.as_view()     #驗證
'''

測試介面:post請求

"""
postman發生post請求

介面:http://api.luffy.cn:8000/user/login/

數據:
{
    "username":"admin",
    "password":"admin"
}
"""

二.工作原理

"""
jwt:json web tokens 採用json格式在web上傳輸的 認證字元串

jwt字元串:頭.載荷.簽名

頭:公司基本信息、項目組基本信息、常規加密演算法名
載荷:用戶信息、過期時間
簽名:頭、載荷、秘鑰

{頭信息字典,採用base64加密演算法}.{載荷信息字典,採用base64加密演算法}.{頭加密串、載荷加密串、伺服器秘鑰,採用hs256加密演算法}

base64是可逆加密
hash256是不可逆加密
我們一般只會將賬號信息,過期時間放載荷裡面,一般把密碼什麼重要信息丟簽名裡面
"""

三.三大認證

session認證

系統自帶的

rest_framework.authentication.SessionAuthentication
ajax請求通過認證:
cookie中要攜帶 sessionid、csrftoken,請求頭中要攜帶 x-csrftoken

jwt認證

第三方

與session認證區別他不用再去查sessionid表,只要查user表就可以了

rest_framework_jwt.authentication.JSONWebTokenAuthentication
ajax請求通過認證:
請求頭中要攜帶 authorization,值為 jwt空格token

基於jwt、其它

自定義

1)自定義認證類,繼承BaseAuthentication(或其子類),重寫authenticate
2)authenticate中完成

  • 拿到認證標識 auth

  • 反解析出用戶 user

  • 前兩步操作失敗 返回None => 游客

  • 前兩步操作成功 返回user,auth => 登錄用戶

    :如果在某個分支拋出異常,直接定義失敗 => 非法用戶

四.自定義認證,基於jwt

其實就是在jwt的源碼基礎上進行相關的修改

最簡單的修改

from rest_framework.exceptions import AuthenticationFailed
import jwt
from rest_framework_jwt.authentication import BaseJSONWebTokenAuthentication
from rest_framework_jwt.authentication import jwt_decode_handler

from rest_framework.authentication import BaseAuthentication
def authenticate(self, request):
    auth = 從request中得到
    user = 從auth中得到
    if not user:
        return None
    return user, auth

如果我們自定製了一個許可權我們進行全局設置必須自己在setting把這個函數加進去

'DEFAULT_AUTHENTICATION_CLASSES': [
    '我們自定義認證函數的對象',
],

我們做局部設置就在我們自定義的類中添加

authentication_classes = [我們自定義認證函數的對象]

五.自定義許可權相關

也是改源碼

"""
系統:
1)AllowAny:允許所有用戶,校驗方法直接返回True
2)IsAuthenticated:只允許登錄用戶
    必須request.user和request.user.is_authenticated都通過
3)IsAuthenticatedOrReadOnly:游客只讀,登錄用戶無限制
    get、option、head 請求無限制
    前臺請求必須校驗 request.user和request.user.is_authenticated
4)IsAdminUser:是否是後臺用戶
    校驗 request.user和request.user.is_staff    is_staff(可以登錄後臺管理系統的用戶)
    

自定義:基於auth的Group與Permission表
1)自定義許可權類,繼承BasePermission,重寫has_permission
2)has_permission中完成
    拿到登錄用戶 user <= request.user
    校驗user的分組或是許可權
    前兩步操作失敗 返回False => 無許可權
    前兩步操作成功 返回True => 有許可權
"""
#根據用戶分組信息設置相關許可權
from rest_framework.permissions import BasePermission

class AdminPermission(BasePermission):
    # 繼承BasePermission,重寫has_permission
    def has_permission(self, request, view):
        # 有許可權,返回True
        # 無許可權,返回False
        user = request.user
        if not user:
            return False
        # 用戶是 管理員 分組 (管理員分組是Group表中的一條自定義記錄)
        if not user.groups.filter(name='管理員'):
            return False
        # 登錄的用戶必須是自定義管理員分組成員
        return True

如果我們自定製了一個許可權全局設置我們必須自己在setting把這個函數加進去

'DEFAULT_PERMISSION_CLASSES': [
    '我們自定義許可權函數的路徑',
],

我們做局部設置就在我們自定義的類中添加

permission_classes = [我們自定義認證函數的對象]

六.自定義訪問次數設置

"""
系統:
1)AnonRateThrottle:對同一IP游客的限制
2)UserRateThrottle:對同一IP登錄用戶的限制
必須在settings.py中
'DEFAULT_THROTTLE_RATES': {
    'user': '10/min',  # 登錄的用戶一分鐘可以訪問10次
    'anon': '3/min',  # 游客一分鐘可以訪問3次
}
在視圖類中:
class TempAPIView(APIView):
    ...
    throttle_classes = [AnonRateThrottle, UserRateThrottle]
    
    

自定義:基於auth的Group與Permission表
1)自定義頻率類,繼承SimpleRateThrottle,重寫get_cache_key,明確scope
    SimpleRateThrottle已經幫我們實現了 allow_request、wait
2)scope與settings.py的DEFAULT_THROTTLE_RATES配合使用
3)get_cache_key中完成
    拿到限制信息 ident <= request中獲取
    沒有限制信息 返回None => 不限制
    有限制信息 返回限制信息字元串 => 有限制
"""

自定義頻率類:一分鐘一個手機號只允許訪問一次介面

from rest_framework.throttling import SimpleRateThrottle

class ThreeMinRateThrottle(SimpleRateThrottle):
    scope = 'sms'
    def get_cache_key(self, request, view):
        # 對手機號頻率限制
        ident = request.data.get('mobile')
        if not ident:  # 為發現限制條件,返回None代表不進行頻率限制
            return None
        return self.cache_format % {
            'scope': self.scope,
            'ident': ident
        }
# settings.py 
'DEFAULT_THROTTLE_RATES': {
    'user': '10/min',  # 登錄的用戶一分鐘可以訪問10次 如果是
    'anon': '3/min',  # 游客一分鐘可以訪問3次
    'sms': '1/min',  #是我們自定義的,預設只提供user以及anon
}
在視圖層
class UserListAPIView(ListAPIView):
    throttle_classes = [我們自定義的方法路徑]

源碼里關於時間的一段代碼

    def parse_rate(self, rate):
        """
        Given the request rate string, return a two tuple of:
        <allowed number of requests>, <period of time in seconds>
        """
        if rate is None:
            return (None, None)
        num, period = rate.split('/')
        num_requests = int(num)
        duration = {'s': 1, 'm': 60, 'h': 3600, 'd': 86400}[period[0]]
        return (num_requests, duration)

這裡我們可以看出來是先/進行字元串切分然後取第一個字母所有我們這邊不一定用min代表分,只要開頭是m即可

七.全局設置有效時間以及jwt的名稱

import datetime
JWT_AUTH = {
    'JWT_EXPIRATION_DELTA': datetime.timedelta(seconds=30000),#d到期時間
    'JWT_AUTH_HEADER_PREFIX': 'TOKEN',  #我們傳參數的時候開頭自定義內容,註意點這裡必須與下麵的token中以宮格隔開
}

源碼中為

USER_SETTINGS = getattr(settings, 'JWT_AUTH', None)  #他是通過JWT_AUTH這個名字
......
'JWT_EXPIRATION_DELTA': datetime.timedelta(seconds=300),
'JWT_AUTH_HEADER_PREFIX': 'JWT',  系統預設以jwt開頭

八.關於jwt自定製獲取token

源碼在rest_framework_jwt.seriallizers.py中JSONWebTokenSerializer類

payload = jwt_payload_handler(user)

return {
    'token': jwt_encode_handler(payload),
    'user': user
}

我們如果自定義有幾個關鍵點把握就好了一個是jwt_payload_handler的方法

一個是 user對象

所有如果我們要在登入的時候拋出token

class LoginJWTAPIView(APIView):
    authentication_classes = ()
    permission_classes = ()
    def post(self, request, *args, **kwargs):
        # username可能攜帶的不止是用戶名,可能還是用戶的其它唯一標識 手機號 郵箱
        username = request.data.get('username')
        password = request.data.get('password')

        # 如果username匹配上手機號正則 => 可能是手機登錄
        if re.match(r'1[3-9][0-9]{9}', username):
            try:
                # 手動通過 user 簽發 jwt-token
                user = models.User.objects.get(mobile=username)
            except:
                return APIResponse(1, '該手機未註冊')

        # 郵箱登錄 等

        # 賬號登錄
        else:
            try:
                # 手動通過 user 簽發 jwt-token
                user = models.User.objects.get(username=username)
            except:
                return APIResponse(1, '該賬號未註冊')

        # 獲得用戶後,校驗密碼並簽發token
        if not user.check_password(password):
            return APIResponse(1, '密碼錯誤')
        payload = jwt_payload_handler(user)
        token = jwt_encode_handler(payload)
        return APIResponse(0, 'ok', results={
            'username': user.username,
            'mobile': user.mobile,
            'token': token
        })

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

-Advertisement-
Play Games
更多相關文章
  • 我想要一個Python函數,它接受一個字元串,並返回一個數組,其中數組中的每個項目都是一個字元,或者是另一個這樣的數組。嵌套數組在輸入字元串中以'('和以')'開頭標記。 因此,該函數將如下所示: 註意:我更喜歡純粹功能性的解決方案。 解決方案 和, ...
  • 一、方法在執行過程中是如何分配記憶體的,記憶體是如何變化的? 1.方法只定義,不調用,是不會執行的,並且在JVM中也不會給該方法分配”運行所屬“的記憶體空間,只有在調用這個方法的時候,才會動態的給這個方法分配所屬的記憶體空間。 2.在JVM記憶體劃分上有這樣三個主要的記憶體空間(當然除了這三塊之外還有其他的記憶體 ...
  • 這個月公司的項目有點忙,我又生病了,美術同事和我又有幾個周末都有事所以沒有來給我做資源 而我這邊也又遇到了瓶頸,目前是開始攻關飛行道具的部分 UE4的4.23在經歷了8個預覽版之後終於出正式版了,我也第一時間更新下來並且升級了工程 可破壞建築什麼的聽起來可能是不錯的效果,以後做戰爭游戲可能會大量用到 ...
  • isinstance() 判斷isinstance(obj,cls)中obj是否是cls類的對象 issubclass() 判斷issubclass(sub,super)中sub是否是super類的派生類 反射 反射就是用字元串類型的名字去操作變數,python中的一切事物皆為對象(都可以使用反射) ...
  • 今日所學: /* 2019.08.19開始學習,此為補檔。 */ 1.String類 實例化:①String name1 = "張三" ; ②String name2 = new String("李四") ; 2.==比較的是引用,equals比較的是具體內容。 String name3 = nam ...
  • 力爭清晰完整準確(逐步完善,持續更新) 1、String類為什麼是final的 首先分析String的源碼: 類被final關鍵字限定,說明它不可以被繼承,沒有子類。即持有一個String對象的引用,它必然是String類,而不會是其他的類。 value[]是用來存儲值的,被final關鍵字修飾,說 ...
  • 之前看過《深入瞭解Java虛擬機》感覺容易忘,今天寫一篇博客加深一下印象。 JVM的記憶體分配和垃圾回收(GC)主要發生在Java堆中。而Java堆根據對象的存活時間可以分為新生代和老年代,而新生代又細分為Eden區、From Survivor區、To Survivor區,這是由於新生代中的垃圾回收算 ...
  • 前言 這個周末被幾個技術博主的同一篇公眾號文章 "fastjson又被髮現漏洞,這次危害可導致服務癱瘓!" 刷屏,離之前漏洞事件沒多久,fastjson 又出現嚴重 Bug。目前項目中不少使用了 fastjson 做對象與JSON數據的轉換,又需要更新版本重新部署,可以說是費時費力。與此同時,也帶給 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...