zoukankan      html  css  js  c++  java
  • Django之中间件

      一.django中间件()

        1.是什么:django 中间件类似django 门户 保安

        请求的时候需要先经过中间件才能到达django 后端(urls,vies,templates,)

        响应走的时候也是需要经过中间件才能到达web服务网关接口

        django 中间件的默认七个门户

    ]
    
    MIDDLEWARE = [
        'django.middleware.security.SecurityMiddleware',
        'django.contrib.sessions.middleware.SessionMiddleware',
        'django.middleware.com]mon.CommonMiddleware',
        'django.middleware.csrf.CsrfViewMiddleware',
        'django.contrib.auth.middleware.AuthenticationMiddleware',
        'django.contrib.messages.middleware.MessageMiddleware',
        'django.middleware.clickjacking.XFrameOptionsMiddleware',
    ]
    from django.utils.deprecation import MiddlewareMixin
    from django.shortcuts import render,HttpResponse
    
    class Mymidd1(MiddlewareMixin):
        def process_request(selfs, request):
            print('我是第一个中间件里面的process_request方法')
            # 注意:这里是从上到下执行
            return HttpResponse('OK')
        def process_response(self, request, response):
            print('我是第一个中间件里面的process_response响应方法')
            return response
    
    
    class Mymidd2(MiddlewareMixin):
        def process_request(selfs, request):
            print('我是第二个中间件里面的process_request方法')
            # 注意:这里是从上到下执行
    
        def process_response(self, request, response):
            print('我是第二个中间件里面的process_response响应方法')
    
            return response
    
    
    class Mymidd3(MiddlewareMixin):
        def process_request(selfs, request):
            print('我是第三个中间件里面的process_request方法')
            # 注意:这里是从上到下执行
    
        def process_response(self, request, response):
            print('我是第三个中间件里面的process_response响应方法')
            return response
    我是第一个中间件里面的process_request方法
    
    [25/Sep/2019 20:11:54] "GET /test1/ HTTP/1.1" 200 2
    我是第一个中间件里面的process_response响应方法

    如果在请求体中直接有相应结果 就会触发 process_response()  执同级别的process_request( ) 中额结果返回  必须通过process_response() 方法返回 return response 

        

    from django.utils.deprecation import MiddlewareMixin
    from django.shortcuts import render,HttpResponse
    
    class Mymidd1(MiddlewareMixin):
        def process_request(selfs, request):
            print('我是第一个中间件里面的process_request方法')
            # 注意:这里是从上到下执行
    
        def process_response(self, request, response):
            print('我是第一个中间件里面的process_response响应方法')
            return response
    
    
    class Mymidd2(MiddlewareMixin):
        def process_request(selfs, request):
            print('我是第二个中间件里面的process_request方法')
            # 注意:这里是从上到下执行
            return HttpResponse('OK')
        def process_response(self, request, response):
            print('我是第二个中间件里面的process_response响应方法')
    
            return response
    
    
    class Mymidd3(MiddlewareMixin):
        def process_request(selfs, request):
            print('我是第三个中间件里面的process_request方法')
            # 注意:这里是从上到下执行
    
        def process_response(self, request, response):
            print('我是第三个中间件里面的process_response响应方法')
            return response

    我们再次将

    return HttpResponse('OK') 执行procses_request() 函数体中

    我是第一个中间件里面的process_request方法
    我是第二个中间件里面的process_request方法   # 也就是说只要请求中 我们拿到return Httpresponse('') 对象就不会再往下进行请求 通过process_response() 方法进行返回 但会执行同级别的request 请求 


    我是第二个中间件里面的process_response响应方法
    我是第一个中间件里面的process_response响应方法

    总结:浏览器再向服务端发送请求时时通过中间件的执行顺序是从上 往下 一个 个 进行 执行 请求 数据 如果没有 再 往下一层 直到 process_request() 中的一层

    返回了一个 HttpResponse  对象  而 拿到的结果 会往我们redis  存放一份数据  一份返回到浏览器 方便下次在进行同样的数据 请求 redis 可以直接 返回 减少访问数据库的压力

    2.为什么

        (1)中间件的存在就是为了帮我们做检验 一次响应一次请求  如果数据不符合者直接在中间件就给你过滤,不会再走数据库 这样减少数据库的访问 减少数据库的压力  

        (2)网站全局的身份验证,访问频率的限制,权限的校验...要是涉及到全局的校验咱们都要可以在中间件取完成

        (3)django 的中间件是所有web框架中 做得最好的

        3.怎么做

        需要我们掌握的方法有

        1.process_request()方法

        2.process_response()方法

        需要了解的方法

       

    要了解的方法
                            3.process_view()
                                1.在路由匹配成功执行视图函数之前 触发
                                
                            4.process_exception()
                                1.当你的视图函数报错时  就会自动执行
                                
                            5.process_template_response()
                                1.当你返回的HttpResponse对象中必须包含render属性才会触发
                                def index(request):
                                    print('我是index视图函数')
                                    def render():
                                        return HttpResponse('什么鬼玩意')
                                    obj = HttpResponse('index')
                                    obj.render = render
                                    return obj

    总结:

        

      二.基于中间件的解决跨站请求伪造

        1.钓鱼网站伪造

    后端代码

    def transfer(request):
        # 其他网站会向我们中国银行的网站访问一个页面 然后给用户输入 name='' 和 vlue= ;写的是我们自己的账号
        # 我们一旦提交 相当于向钓鱼网站进行转账 仿真页面
        # 真实网站
        if request.method == 'POST':
            username = request.POST.get('username')
            money = request.POST.get('money')
            to_name = request.POST.get('to_name')
            print(' %s向%s 转账%s '%(username,to_name,money))
    
        return render(request,'transfer.html')

    前端代码

    </head>
    <body>
    {#{% csrf_token %}#}
        <h1>中国银行</h1>
        <form action="" method="post">
            <p>username:<input type="text" name="username"></p>
            <p>money:<input type="text" name="money"></p>
    #1 伪造的网站向中国银行请求一个一摸一样的页面
    # 2伪造得的网站在input 框内做手脚 将用户提交的name='to_name'
    属性去除 相当于 后端没办法识别
    
    # 4 然后我们自己设置一个input 框 有name='to_name' 属性 将value=‘ccoo’ 目标账户设置为默认
            <p>to_name:<input type="text" name='to_name'></p>
            <input type="submit">
        </form>
    
    </body>
    </html>

    伪造网站

    def transfer(request):
        # 其他网站会向我们中国银行的网站访问一个页面 然后给用户输入 name='' 和 vlaue= ;写的是我们自己的账号
        # 我们一旦提交 相当于向钓鱼网站进行转账 仿真页面
        # 真实网站
        # 只需跳转页面即可
        return render(request,'transfer.html')

    前端代码

    {#{% csrf_token %}#}
        <h1>钓鱼网站</h1>
            这里就是向中国银行请求一个页面 然后开始设置input 框
        <form action=" http://127.0.0.1:8000/transfer/" method="post">
            <p>username:<input type="text" name="username"></p>
            <p>money:<input type="text" name="money"></p>
    {#        不设置name属性#}
            <p>to_name:<input type="text"></p>
            <input type="text" name="to_name" value="hhh" style="display: none">
            <input type="submit">
        </form>
    
    </body>
    </html>

        2.如何解决:Django 跨站请求伪造

        直接在form表单 {% csrf_token%} 

       <form action="" method="post">
              {% csrf_token %}
            <p>username:<input type="text" name="username"></p>
            <p>money:<input type="text" name="money"></p>
            <p>to_name:<input type="text" name='to_name'></p>
            <input type="submit">
        </form>

          form 表单 {%csrf_token%}

          ajax 的三种处理csrf 的方法

          第一种ajax 提交form 表单 数据

       data:{'username':'koko','csrfmiddlewaretoken':$('[name=csrfmiddlewaretoken]').val()},   注意 value 的来源 是form  的%csrf_token%{}表单
    <body>
    {#{% csrf_token %}#}
        <h1>中国银行</h1>
    
        <form action="" method="post">
              {% csrf_token %}
            <p>username:<input type="text" name="username"></p>
            <p>money:<input type="text" name="money"></p>
            <p>to_name:<input type="text" name='to_name'></p>
    {#        <input type="submit" class="c1">#}
        </form>
    <button class="c1">ajax提交</button>
    </body>
    
    
    
    <script>
        $('.c1').on('click',function () {
            $.ajax({
                url:'', // 路径这里有三种 类似:action 可以写全路径 默认不写诗当前地址
                type:'post',  //methon
                data:{'username':'koko','csrfmiddlewaretoken':$('[name=csrfmiddlewaretoken]').val()},
                success:function (data) {
                    alert(data)
    
                }
            })
    
        })
    
    
    </script>
    </html>

    第二种

      data:{'username':'koko','csrfmiddlewaretoken':'{{ csrf_token }}'},   
    </head>
    <body> {#{% csrf_token %}#} <h1>中国银行</h1> <form action="" method="post"> {% csrf_token %} <p>username:<input type="text" name="username"></p> <p>money:<input type="text" name="money"></p> <p>to_name:<input type="text" name='to_name'></p> {# <input type="submit" class="c1">#} </form> <button class="c1">ajax提交</button> </body> <script> $('.c1').on('click',function () { $.ajax({ url:'', // 路径这里有三种 类似:action 可以写全路径 默认不写诗当前地址 type:'post', //methon {#data:{'username':'koko','csrfmiddlewaretoken':$('[name=csrfmiddlewaretoken]').val()},#} data:{'username':'koko','csrfmiddlewaretoken':'{{ csrf_token }}'}, success:function (data) { alert(data) } }) }) </script> </html>

    第三种

     在script 的上方直接引入

    {% load static  %}
    <script src="{% static 'ajax_js.js' %}"></script>
     // 第三种方式 :直接引入js文件
    </head>
    <body>
    {#{% csrf_token %}#}
        <h1>中国银行</h1>
    
        <form action="" method="post">
              {% csrf_token %}
            <p>username:<input type="text" name="username"></p>
            <p>money:<input type="text" name="money"></p>
            <p>to_name:<input type="text" name='to_name'></p>
    {#        <input type="submit" class="c1">#}
        </form>
    <button class="c1">ajax提交</button>
    </body>
    
    
    {% load static  %}
    <script src="{% static 'ajax_js.js' %}"></script>
    <script>
    
        $('.c1').on('click',function () {
            $.ajax({
                url:'', // 路径这里有三种 类似:action 可以写全路径 默认不写诗当前地址
                type:'post',  //methon
                {#data:{'username':'koko','csrfmiddlewaretoken':$('[name=csrfmiddlewaretoken]').val()},#}
                {#data:{'username':'koko','csrfmiddlewaretoken':'{{ csrf_token }}'},#}
                // 第三种方式 :直接引入js文件
                data:{'username':'koko'},
                success:function (data) {
                    alert(data)
    
                }
            })
    
        })
    
    
    </script>
    </html>

      三.跨站请求伪造象关装饰器

    如果是csrf_protect 那么有三种方式


    1.当你网站全局都需要校验csrf的时候 有几个不需要校验该如何处理
    2.当你网站全局不校验csrf的时候 有几个需要校验又该如何处理

    from django.views import View
    from django.utils.decorators import method_decorator
    from django.views.decorators.csrf import csrf_exempt,csrf_protect
    # 如果是接受跨站请求的保护 则用csrf_protect
    # 这是保护的第三种方法
    # @method_decorator(csrf_protect,name='post')   # 指名道姓
    class MyView(View):
        # 这是第二种保护
        # 任务的分发
        # @method_decorator(csrf_protect)
        def dispatch(self, request, *args, **kwargs):
            res = super().dispatch(request,*args,**kwargs)
            return res
    
        def get(self,request):
    
            return HttpResponse('get')
        # AttributeError at /my_views/
        # 'MyView' object has no attribute 'COOKIES'
        # @csrf_protect 也就是说我将后端的中间件注释 掉他就不会 阻拦我 发送post 数据凡是我放这个保护的装饰器 就会将这个post
        # 阻拦数据传数
        # 这是第一种
        # @method_decorator(csrf_protect)
        def post(self,request):
    
            return HttpResponse('post')

    如果是csrf_exempt 只有两种(只能给dispatch装)   特例

    # @method_decorator(csrf_protect,name='post')   # 指名道姓
    class MyView(View):
        # 这是第二种保护
        # 任务的分发
    
        # 不保护数据的安全习性
        # @method_decorator(csrf_exempt,name='dispatch')
        def dispatch(self, request, *args, **kwargs):
            res = super().dispatch(request,*args,**kwargs)
            return res
        
        def get(self,request):
    
            return HttpResponse('get')
        # AttributeError at /my_views/
        # 'MyView' object has no attribute 'COOKIES'
        # @csrf_protect 也就是说我将后端的中间件注释 掉他就不会 阻拦我 发送post 数据凡是我放这个保护的装饰器 就会将这个post
        # 阻拦数据传数
        # 这是第一种
        # @method_decorator(csrf_protect)
        # @method_decorator(csrf_exempt,name='post')
        # 放这里是不行的 起不到效果
        def post(self,request):
    
            return HttpResponse('post')

      四.auth模块

      注意事项:如果用了auto模块就必须用全套

      1.创建超级用户和普通用户

    普通用户和超级用户的区别是 超级用户可以进行django 后台admin 的管理 

      2.创建普通用户

      普通用户

      增

    from django.contrib.auth.models import User
    
    
    def register(request):
    
        if request.method == 'POST':
            username = request.POST.get('username')
            password = request.POST.get('password')
    
            user_obj = User.objects.filter(username=username)
            if not user_obj:
            # 创建普通用户和超级用户
            # 注意:别再用create 的方法这里的是auth模块
                User.objects.create_user(username=username, password=password)  #  创建普通用户
                # User.objects.create_superuser(username=username, password=password)  # 这是创建超级用户
                return HttpResponse('注册成功')
        return render(request,'login.html')

      删

      用户注销

      

    @login_required  # 无参装饰器
    def logout(request):
        # request.session.flush()  # 没问题
        auth.logout(request)  # ok
        return HttpResponse('注销成功')

      改

    # 装饰器
    from django.contrib.auth.decorators import  login_required
    
    # 修改密码
    @login_required  #  # 自动校验当前用户是否登录  如果没有登录 默认跳转到 一个莫名其妙的登陆页面
    def set_password(request):
        if request.method == 'POST':
            old_pwd = request.POST.get('old_pwd')
            print(old_pwd)
            new_pwd = request.POST.get('new_pwd')
            # 如何检验当前输入的密码和数据库的密码一致
            is_valid = request.user.check_password(old_pwd)
            # 会将当前输入的密码进行自动加密 然后去数据库中比对当前的用户的密码
            if is_valid:
                request.user.set_password(new_pwd)
                # 修改密码后一定记得save() 保存
                request.user.save()
                return HttpResponse('修改成功')
    
            else:
                return HttpResponse('密码有误')

      查 

        is_valid = request.user.check_password(old_pwd)
            # 会将当前输入的密码进行自动加密 然后去数据库中比对当前的用户的密码

      1.用户登录

    def login(request):
        if request.method == 'POST':
            username = request.POST.get('username')
            password = request.POST.get('password')
            user_obj = auth.authenticate(username=username,password=password)  # 必须用户auth 模块 因为数据库的密码是秘文
            if user_obj:
                # 保存sessioon 值
                auth.login(request,user_obj)  # 将用户的状态保存在session中
                #  """只要执行了这一句话  你就可以在后端任意位置通过request.user获取到当前用户对象"""
                return HttpResponse('登录成功')
            else:
                return HttpResponse('用户不存在')
        # 先返回一个页面
        return render(request,'register.html')

      

    2.# 判断用户是否登录
    def check_auth(request):
        print(request.user)  # koko
        print(request.user.is_authenticated)  # 判断用户是否登录CallableBool(True)
        return HttpResponse('ok')

      检验用户是否登陆 auth 装饰器

     1装饰器的局部设置 

    @login_required(login_url='/login/')  # 如果没有登录 我们需要指定一个登陆的网址让他进行跳转 局部设置
    那这样不是很麻分每个函数都的加
    2.装饰器的全局设置
    别急 我们有更好的方法 就是在settings 中 设置全局的LOGIN_URL = 'login
    # auth登陆认证装饰器 跳转的url
    LOGIN_URL = 'login'
    
    
    # 告诉django  orm不再使用auth默认的表  而是使用你自定义的表
    AUTH_USER_MODEL = 'app01.Userinfo'  # '应用名.类名'

      3.自定义userinfo 将不会在django 表中创建auth_user  可以增强拓展性 添加新字段

      如何做

    自定义auth_user表
    from django.contrib.auth.models import AbstractUser
    # Create your models here.
    # 第一种 使用一对一关系 不考虑

    # 第二种方式 使用类的继承
    class Userinfo(AbstractUser):
    # 千万不要跟原来表中的字段重复 只能创新
    phone = models.BigIntegerField()
    avatar = models.CharField(max_length=32)

    # 一定要在配置文件中 告诉django
    # 告诉django orm不再使用auth默认的表 而是使用你自定义的表
    AUTH_USER_MODEL = 'app01.Userinfo' # '应用名.类名'    # 注意了哈项目中的mdels  创建Userinfo(Abs) 继承我们数据库的表 必须在settings 中告诉django ORM 不在使用默认的表 而是使用咱们自定义的表


    1.执行数据库迁移命令
    所有的auth模块功能 全部都基于你创建的表
    而不再使用auth_user

      怎么用

    五.settings中的中间件原理 原型的思路 如何实现同时发送给不同功能函数 实现统一的接口的功能 类似鸭子类型 

    设计思想(****************) 插拔式

    六.序列化组件DRF 框架

        

  • 相关阅读:
    [VBA]根据身份证号码计算年龄的Excel函数
    [VBA]发布一个计算桩号之差的Excel自定义函数(VBA)
    [VBA]用一个简单例子说明如何在Excel中自定义函数
    元素定位工具:Chrome浏览器ChroPath插件
    68 个 Python 内置函数
    Python错误重试
    jenkins 中展示漂亮的 HTML 测试报告
    python解决接口数据使用了RSA加密和签名
    Python装饰器用法
    Pycharm中文版教程
  • 原文地址:https://www.cnblogs.com/mofujin/p/11587092.html
Copyright © 2011-2022 走看看