• Django DRF JWT 认证


    1. Token

    首先大致了解一下什么是 Token

    • Token 是一种客户端认证机制、令牌,是一个经过加密的字符串,安全性强,支持跨域

    • 用户第一次登录,服务器通过数据库校验其用户名和密码是否合法,则再生成一个token串,服务端会返回Token给前端,前端可以在每次请求的时候带上Token证明自己的合法地位

    • Token 的生成一般是采用uuid保证唯一性,当用户登录时为其生成唯一的token,存储一般保存在数据库中

    在这里插入图片描述
    在这里插入图片描述

    2. Json Web Token

    Json Web Token 简称 JWT,其本质就是 token 认证机制。

    • Json web token (JWT),是为了在网络应用环境间传递声明而执行的一种基于JSON的开放标准
    • 该 token 被设计为紧凑且安全的,特别适用于分布式站点的单点登录(SSO)场景。
    • JWT 的声明一般被用来在身份提供者和服务提供者间传递被认证的用户身份信息,以便于从资源服务器获取资源,也可以增加一些额外的其它业务逻辑所必须的声明信息,该 token 也可直接被用于认证,也可被加密。
      在这里插入图片描述

    在这里插入图片描述

    2.1 JWT的构成

    JWT 就是一段字符串,由三段信息构成的,将这三段信息文本用.链接一起就构成了JWT 字符串

    • 第一部分我们称它为头部(header),可以存放公司信息,加密方式等等
    • 第二部分我们称其为载荷(payload, 类似于飞机上承载的物品),可以存放用户信息
    • 第三部分是签证(signature),是把第一段和第二段通过某种加密方式 + 秘钥加密得到一个字符串。同样的,验证就是拿到第一段和第二段使用同样的加密方式+秘钥重新加密,将得到字符串跟第三段比较,如果一样,表示没有被篡改,如果不一样,表明被篡改了,token 不能用了

    2.1.1 header

    jwt 的头部承载两部分信息:

    • 声明类型,这里是jwt
    • 声明加密的算法 通常直接使用 HMAC SHA256

    完整的头部类似于下面的 JSON:

    {
      'typ': 'JWT',
      'alg': 'HS256'
    }
    
    • 1
    • 2
    • 3
    • 4

    然后将头部进行base64加密(该加密是可以对称解密的),构成了第一部分

    • 对称加密:加密和解密都使用同一个秘钥
    • 非对称加密:加密用公钥,解密用私钥

    加密后的结果

    eyJ0eXAiOiAiSldUIiwgImFsZyI6ICJIUzI1NiJ9
    
    • 1

    2.1.2 base64

    Base64是网络上最常见的用于传输8Bit字节码的编码方式之一
    Base64就是一种基于64个可打印字符来表示二进制数据的方法。
    Base64编码是从二进制到字符的过程,可用于在HTTP环境下传递较长的标识信息。
    采用Base64编码具有不可读性,需要解码后才能阅读。
    
    • 1
    • 2
    • 3
    • 4

    base64的长度一定是4的倍数,如果不到用 = 号补齐

    s = json.dumps(s)
    
    # 对字符串进行编码,
    base64.b64encode(s.encode('utf8'))
    base64.b64encode(bytes(s, encoding='utf8'))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    s = b'eyJ0eXAiOiAiSldUIiwgImFsZyI6ICJIUzI1NiJ9'
    
    # 对使用 b64 编码的字符串进行解码,注意要是 byte 类型。
    base64.b64decode(s)
    
    • 1
    • 2
    • 3
    • 4

    2.1.2 payload

    载荷就是存放有效信息的地方。这个名字像是特指飞机上承载的货品,这些有效信息包含三个部分

    • 标准中注册的声明
    • 公共的声明
    • 私有的声明

    标准中注册的声明 (建议但不强制使用) :

    • iss: jwt签发者
    • sub: jwt所面向的用户
    • aud: 接收jwt的一方
    • exp: jwt的过期时间,这个过期时间必须要大于签发时间
    • nbf: 定义在什么时间之前,该jwt都是不可用的.
    • iat: jwt的签发时间
    • jti: jwt的唯一身份标识,主要用来作为一次性token,从而回避时序攻击。

    公共的声明 : 公共的声明可以添加任何的信息,一般添加用户的相关信息或其他业务需要的必要信息,但不建议添加敏感信息,因为该部分在客户端可解密

    私有的声明 : 私有声明是提供者和消费者所共同定义的声明,一般不建议存放敏感信息,因为base64是对称解密的,意味着该部分信息可以归类为明文信息。

    定义一个payload:

    {
      "sub": "1234567890",
      "name": "xwx",
      "admin": True
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5

    然后将其进行 base64 加密,得到 JWT 的第二部分

    b'eyJzdWIiOiAiMTIzNDU2Nzg5MCIsICJuYW1lIjogImxxeiIsICJhZG1pbiI6IHRydWV9'
    
    • 1

    2.1.3 signature

    JWT的第三部分是一个签证信息,这个签证信息由三部分组成:

    • header (base64 编码后的)
    • payload (base64 编码后的)
    • secret

    这个部分需要 base64 加密后的 header 和 payload 使用.连接组成的字符串
    然后通过 header 中声明的加密方式进行加盐 secret 组合加密,然后就构成了 jwt 的第三部分

    最终的样子类似于

    eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWV9.TJVA95OrM7E2cBab30RMHrHDcEfxjoYZgeFONFh7HgQ
    
    • 1

    注意secret 是保存在服务器端的,jwt 的签发生成也是在服务器端的,secret 就是用来进行 jwt 的签发和 jwt 的验证,所以,它就是服务端的私钥,在任何场景都不应该流露出去。一旦客户端得知这个 secret, 那就意味着客户端是可以自我签发 jwt 了。

    2.2 签发与校验

    • 签发:根据登录请求提交来的 账号 + 密码 + 设备信息 签发 token
    • 校验:根据客户端带token的请求 反解出 user 对象

    3. drf-jwt

    关于签发和核验JWT,我们可以使用Django REST framework JWT扩展来完成。

    pip3 install djangorestframework-jwt
    
    • 1

    除了 djangorestframework-jwt 还有 djangorestframework-simplejwt,俩者用法很相似。

    快速使用

    JWT 使用的表是 django auth 的 user 表

    • 1 创建超级用户
    python3 manage.py createsuperuser
    
    • 1
    • 配置路由urls.py
    from django.urls import path
    from rest_framework_jwt.views import obtain_jwt_token
    
    urlpatterns = [
        path('login/', obtain_jwt_token),
    ]
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • postman 测试
      向后端接口发送 post 请求,携带用户名密码,即可看到生成的 token
      在这里插入图片描述

    修改返回格式

    • 编写函数
    def jwt_response_payload_handler(token, user=None, request=None):
        return {'code': 100, 'msg': '登陆成功', 'token': token, 'username': user.username}
    
    • 1
    • 2
    • 配置文件中配置
    JWT_AUTH = {
        'JWT_RESPONSE_PAYLOAD_HANDLER': 'app01.utils.jwt_response_payload_handler',
    }
    
    • 1
    • 2
    • 3
    • 修改后如下所示
      在这里插入图片描述

    jwt 验证

    • 导入认证类、权限类
    from rest_framework_jwt.authentication import JSONWebTokenAuthentication
    from rest_framework.permissions import IsAuthenticated
    
    • 1
    • 2
    • 在视图函数中添加认证类、权限类
    class BookView(ViewSet):
        authentication_classes = [JSONWebTokenAuthentication,]
        permission_classes = [IsAuthenticated,]  
    
    • 1
    • 2
    • 3
    • 向视图函数发生 get 请求,并且需要在请求头中携带 token,请求头键名是 Authorization,键值是jwt token串,必须要以 jwt 开头。
    请求头中 key:  Authorization
    请求头的 value:jwt token串
    
    • 1
    • 2
    • 验证失败会有以下错误:
      在这里插入图片描述在这里插入图片描述

    4. 自定义用户表签发 token

    在前面使用的方式都需要依赖 Django auth 提供的用户表或者继承了该表的拓展表。我们可以自定义一个用户表进行签发 token。

    • 用户表
    class User(models.Model):
        username = models.CharField(max_length=32)
        password = models.CharField(max_length=32)
    
    • 1
    • 2
    • 3
    • 视图函数
    from app01 import models
    from rest_framework_jwt.settings import api_settings
    jwt_payload_handler = api_settings.JWT_PAYLOAD_HANDLER
    jwt_encode_handler = api_settings.JWT_ENCODE_HANDLER
    
    class User(APIView):
        def post(self, request):
            username = request.data.get('username')
            password = request.data.get('password')
            user = models.User.objects.filter(username=username, password=password).first()
            if user:
                # 登录成功签发token
                payload = jwt_payload_handler(user)  # 根据当前登录用户获取荷载
                print(payload)
                token = jwt_encode_handler(payload)  # 根据荷载生成token
    
                return Response({'code': 100, 'msg': '登录成功', 'token': token})
            else:
                return Response({'code': 101, 'msg': '用户名或密码错误'})
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20

    在这里插入图片描述

    5. 自定义认证类验证 token

    • 认证类
    from rest_framework_jwt.settings import api_settings
    from rest_framework import exceptions
    from app01 import models
    
    jwt_decode_handler = api_settings.JWT_DECODE_HANDLER
    jwt_get_username_from_payload = api_settings.JWT_PAYLOAD_GET_USERNAME_HANDLER
    
    
    class LoginAuth(BaseAuthentication):
        def authenticate(self, request):
        
            # 1 取出 token
            jwt_value = request.META.get('HTTP_TOKEN')
            
            # 2 验证token是否合法
            # try:
            #     payload = jwt_decode_handler(jwt_value)
            # except jwt.ExpiredSignature:
            #     msg = 'token过期了'
            #     raise exceptions.AuthenticationFailed(msg)
            # except jwt.DecodeError:
            #     msg = 'token解码错误'
            #     raise exceptions.AuthenticationFailed(msg)
            # except jwt.InvalidTokenError:
            #     msg = '解析token未知错误'
            #     raise exceptions.AuthenticationFailed(msg)
            
            try:
                payload = jwt_decode_handler(jwt_value)
            except Exception:
                raise exceptions.AuthenticationFailed('token错误')
    
            print(payload)  # 荷载 user_id
            
            user = models.User.objects.filter(pk=payload['user_id']).first()
            return user, jwt_value
            
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 视图函数

    添加认证类

    authentication_classes = [LoginAuth, ]
    
    • 1
    • 使用的时候,键名为 token,键值为 token 串,并且不需要加上 jwt
      在这里插入图片描述
  • 相关阅读:
    close excel by keyword 根据关键字关闭 excel 窗口 xlwings 方式实现
    【分布式计算:原理、算法和系统】第三章 逻辑时间【待补全】
    数据库操作语言:DML(data management lauguage)
    HDU_3234
    十大免费好用的视频软件推荐,新手小白必备
    SpringBoot中post请求报405错误排坑
    高德地图设置电子围栏
    走进Redis-扯扯集群
    【C++】C++学习记录.DAY5-结构体的使用
    【6 - 完结】Sql Server - 郝斌(identity、视图、事务、索引、存储过程、触发器、游标、TL_SQL)
  • 原文地址:https://blog.csdn.net/m0_58987515/article/details/125434684