zoukankan      html  css  js  c++  java
  • DRF 版本 认证

    DRF的版本

    版本控制是做什么用的, 我们为什么要用

    首先我们要知道我们的版本是干嘛用的呢~~大家都知道我们开发项目是有多个版本的~~

    随着我们项目的更新~版本就越来越多~~我们不可能新的版本出了~以前旧的版本就不进行维护了~~~

    那我们就需要对版本进行控制~~这个DRF也给我们提供了一些封装好的版本控制方法~~

    版本控制怎么用

    之前我们学视图的时候知道APIView,也知道APIView返回View中的view函数,然后调用的dispatch方法~

    那我们现在看下dispatch方法~~看下它都做了什么~~

    执行self.initial方法之前是各种赋值,包括request的重新封装赋值,下面是路由的分发,那我们看下这个方法都做了什么~~

    我们可以看到,我们的version版本信息赋值给了 request.version  版本控制方案赋值给了 request.versioning_scheme~~

    其实这个版本控制方案~就是我们配置的版本控制的类~~

    也就是说,APIView通过这个方法初始化自己提供的组件~~

    我们接下来看看框架提供了哪些版本的控制方法~~在rest_framework.versioning里~~

    框架一共给我们提供了这几个版本控制的方法~~我们在这里只演示一个~~因为基本配置都是一样的~~

    详细用法

    我们看下放在URL上携带版本信息怎么配置~~

    REST_FRAMEWORK = {
        # 默认使用的版本控制类
        'DEFAULT_VERSIONING_CLASS': 'rest_framework.versioning.URLPathVersioning',
        # 允许的版本
        'ALLOWED_VERSIONS': ['v1', 'v2'],
        # 版本使用的参数名称
        'VERSION_PARAM': 'version',
        # 默认使用的版本
        'DEFAULT_VERSION': 'v1',
    }
    settings.py
    urlpatterns = [
        url(r"^versions", MyView.as_view()),
        url(r"^(?P<version>[v1|v2]+)/test01", TestView.as_view()),
    ]
    urls.py
    class TestView(APIView):
        def get(self, request, *args, **kwargs):
            print(request.versioning_scheme)
            ret = request.version
            if ret == "v1":
                return Response("版本v1的信息")
            elif ret == "v2":
                return Response("版本v2的信息")
            else:
                return Response("根本就匹配不到这个路由")
    
    测试视图
    测试视图

    DRF的认证

    认证是干嘛的呢~

    我们都知道~我们可以在网站上登录~然后可以有个人中心,对自己信息就行修改~~~

    但是我们每次给服务器发请求,由于Http的无状态,导致我们每次都是新的请求~~

    那么服务端需要对每次来的请求进行认证,看用户是否登录,以及登录用户是谁~~

    那么我们服务器对每个请求进行认证的时候,不可能在每个视图函数中都写认证~~~

    一定是把认证逻辑抽离出来~~以前我们可能会加装饰器~或者中间件~~那我们看看DRF框架给我们提供了什么~~~

    认证怎么用

    上面讲版本的时候我们知道~在dispatch方法里~执行了initial方法~~那里初始化了我们的版本~~

    如果我们细心我们能看到~版本的下面其实就是我们的认证,权限,频率组件了~~

    我们先看看我们的认证组件~~

    我们进去我们的认证看下~~

    我们这个权限组件返回的是request.user,那我们这里的request是新的还是旧的呢~~

    我们的initial是在我们request重新赋值之后的~所以这里的request是新的~也就是Request类实例对象~~

    那这个user一定是一个静态方法~我们进去看看~~

    我没在这里反复的截图跳转页面~~大家可以尝试着自己去找~~要耐心~~细心~~

    我们通过上面基本可以知道我们的认证类一定要实现的方法~~以及返回值类型~~以及配置的参数authentication_classes~

    下面我们来看看具体用法~~~

    认证的详细用法

    我们先写个认证的小demo~~我们先建一个用户表~字段为用户名以及对应的token值~~

    from django.db import models
    
    # Create your models here.
    
    class User(models.Model):
        name = models.CharField(max_length=32)
        pwd = models.CharField(max_length=32)
        token = models.UUIDField(null=True, blank=True)
        type = models.IntegerField(choices=((1, "普通用户"), (2, "vip"),(3, "svip")), default=1)
    models.py
    from .models import User
    from rest_framework.exceptions import AuthenticationFailed
    from rest_framework.authentication import BaseAuthentication
    
    
    class MyAuth(BaseAuthentication):
        def authenticate(self, request):
            # 认证逻辑
            # 拿到前端传过来的token
            # 判断token是否存在
            token = request.query_params.get("token", "")
            if not token:
                raise AuthenticationFailed("缺少token")
            user_obj = User.objects.filter(token=token).first()
            if not user_obj:
                raise AuthenticationFailed("token不合法")
            # request.user  request.auth
            return (user_obj, token)
    auth.py
    from django.shortcuts import render
    from rest_framework.views import APIView
    from rest_framework.viewsets import ViewSetMixin
    from rest_framework import versioning
    from rest_framework.response import Response
    from .models import User
    from .auth import MyAuth
    from .permission import MyPermission
    import uuid
    from rest_framework import permissions
    from .throttle import MyThrottle, DRFThrottle
    from rest_framework import throttling
    
    # Create your views here.
    
    class LoginView(APIView):
        def post(self, request):
            name = request.data.get("name", "")
            pwd = request.data.get("pwd", "")
            # 校验用户名和密码是否正确
            user_obj = User.objects.filter(name=name, pwd=pwd).first()
            if user_obj:
                user_obj.token = uuid.uuid4()
                user_obj.save()
                return Response(user_obj.token)
            else:
                return Response("用户名或密码错误")
    
    
    class TestView(APIView):
        authentication_classes = [MyAuth, ]
        def get(self, request):
            return Response("测试认证组件")
    
    
    class TestPermission(APIView):
        authentication_classes = [MyAuth, ]
        permission_classes = [MyPermission, ]
        throttle_classes = [DRFThrottle, ]
    
        def get(self, request):
            return Response("vip用户能看的电影")
    views.py
    REST_FRAMEWORK = {
    
        # 配置全局认证
        'DEFAULT_AUTHENTICATION_CLASSES': ["BRQP.utils.MyAuth", ]
    }
    全局配置认证
    幻想毫无价值,计划渺如尘埃,目标不可能达到。这一切的一切毫无意义——除非我们付诸行动。
  • 相关阅读:
    windows服务启动有界面的程序
    [转发]读取txt防止读到乱码--自动根据文件编码进行读取
    线程UI同步
    SQL2012导出的脚本没有if exists判断
    power designer 设计数据库生成到oracle数据库
    经典三层,单元工作模式出错的解决方案
    EF ObjectStateManager 中已存在具有同一键的对象。ObjectStateManager 无法跟踪具有相同键的多个对象
    Oracle 11g 客户端连接 oracle 10g 服务端,乱码问题
    EF 连接oracle 基础连接失败 问题解决
    vs2010 oraclelient 引用问题
  • 原文地址:https://www.cnblogs.com/TodayWind/p/13898123.html
Copyright © 2011-2022 走看看