Django REST Framework之频率限制的使用


Posted in Python onSeptember 29, 2019

开放平台的API接口调用需要限制其频率,以节约服务器资源和避免恶意的频繁调用

使用

自定义频率限制组件:utils/thottle.py

class MyThrottle(BaseThrottle):
 
  def __init__(self):
    self.history = None
 
  def allow_request(self, request, view):
    # 实现限流的逻辑
    # 以IP限流
    # 访问列表 {IP: [time1, time2, time3]}
    # 1, 获取请求的IP地址
    ip = request.META.get("REMOTE_ADDR")
    # 2,判断IP地址是否在访问列表
    now = time.time()
    if ip not in VISIT_RECORD:
      # --1, 不在 需要给访问列表添加key,value
      VISIT_RECORD[ip] = [now,]
      return True
      # --2 在 需要把这个IP的访问记录 把当前时间加入到列表
    history = VISIT_RECORD[ip]
    history.insert(0, now)
    # 3, 确保列表里最新访问时间以及最老的访问时间差 是1分钟
    while history and history[0] - history[-1] > 60:
      history.pop()
    self.history = history
    # 4,得到列表长度,判断是否是允许的次数
    if len(history) > 3:
      return False
    else:
      return True
 
  def wait(self):
    # 返回需要再等多久才能访问
    time = 60 - (self.history[0] - self.history[-1])
    return time

views.py

class TestThrottle(APIView):
  throttle_classes = [MyThrottle, ]
   
  def get(self, request, *args, **kwargs):
    pass
  • allow_request() 方法内定义频率控制的实现
  • wait() 方法的返回值代表了距离下次允许访问还剩多久,单位:秒

全局使用

同样,需要配置setttings文件

EST_FRAMEWORK = {
  "DEFAULT_PERMISSION_CLASSES": [] # 默认的权限类
}

使用REST Framework提供的频率控制组件

DRF提供了四种频率控制组件:

  • SimpleRateThrottle
  • AnonRateThrottle
  • UserRateThrottle
  • ScopedRateThrottle

以SimpleRateThrottle为例:

settings.py

REST_FRAMEWORK = {
  'DEFAULT_THROTTLE_CLASSES':['api.utils.mythrottle.UserThrottle',]
  'DEFAULT_THROTTLE_RATES': {
    '未认证用户': '10/m',
    '已认证用户': '100/h',
  },
}

utils.thorttle.py

from rest_framework.views import APIView
from rest_framework.response import Response
from rest_framework import exceptions
from rest_framework.throttling import SimpleRateThrottle
 
class VisitThrottle(SimpleRateThrottle):
  scope = "未认证用户"
 
  def get_cache_key(self, request, view):
    return self.get_ident(request)
    
 
class UserThrottle(SimpleRateThrottle):
  scope = "已认证用户"
 
  def get_cache_key(self, request, view):
    return request.user # 认证通过后,认证方法authenticate的返回值之一

views.py

class TestThrottle(APIView):<br># 这样设置后,节流功能就会使用VisitThrottle类,而不会使用UserThrottle类
  throttle_classes = [VisitThrottle,]
   
  def get(self, request, *args, **kwargs):

这里使用的节流类是继承了SimplePateThrottle类,而这个类利用了django内置的缓存来存储访问记录。通过全局节流设置,所有的视图类默认是使用UserThrottle类进行节流,如果不想使用默认的类就自定义给throttle_classes属性变量赋值,如:“throttle_classes = [VisitThrottle,]”。

源码分析

1.为什么会使用“scope”属性变量,它有什么用?

Django REST Framework之频率限制的使用

由内置接口代码基本结构中可以看到,视图类TestThrottle继承了SimpleRateThrottle类,跳转到这个类中,就可以看到scope属性变量。

Django REST Framework之频率限制的使用

Django REST Framework之频率限制的使用

由“THROTTLE_RATES[self.scope]”知,scope一定是一个key值,而THROTTLE_RATES不就是在配置文件中所设置的变量吗?所以说scope代表的就是“未认证用户”和“已认证用户”这两个key值,而这两个key值代表的就是不同的节流方案。返回值就这这两个key值所对应的value值,具体是哪一个,那就看视图类TestTrottle中对scope属性变量的值是什么了,如果这个scope值不存在,就会抛出异常。

2.为什么会使用“get_cache_key”方法?该方法的返回值是什么?

在分析get_cache_key方法前,先分析一下SimpleRateThrottle类:

Django REST Framework之频率限制的使用

cache = default_cache 它表示的就是存储用户访问记录的缓存,而这个缓存正是django默认的缓存。

get_rate 方法,前面已经说过了,是用来获取节流方式的。

Django REST Framework之频率限制的使用

parse_rate方法,解析节流方式

Django REST Framework之频率限制的使用

allow_request方法,就是跟在自定义节流方法一样,是实现节流算法的。之所以会用内置节流方法,就是因为在这里,已经实现了节流算法。

Django REST Framework之频率限制的使用

wait方法,就是跟在自定义节流类中的wait方法一样,返回提示用户还有多长时间就可以再次访问了。

Django REST Framework之频率限制的使用

通过初始化方法,获取并解析好要使用的节流方式,供allow_request方法使用。

Django REST Framework之频率限制的使用

通过调用check_throttles方法,来调用allow_request方法,由上面关于allow_request截图来看,要完成它的功能,就必须通过get_cache_key方法获取到当前用户的唯一标识,所以get_cache_key应该返回唯一标识。

Django REST Framework之频率限制的使用

get_cache_key 方法,这就是在视图类TestThrottle中重写的方法。由上图可知,该方法是必须重写的,不然就会抛出异常。

3.为什么会使用“throttle_classes”属性变量,它有什么用?

Django REST Framework之频率限制的使用

通过查看dispatch方法中的intial方法可以看到调用的节流方法“check_throttles”。

Django REST Framework之频率限制的使用

这个for循环返回的一定是一个列表,类似于认证和授权的源码,那么这个列表一定是保存节流类的列表。

Django REST Framework之频率限制的使用

get_throttles方法返回的是一个列表生成式,而这里的throttle_classes就是在视图类TestThrottle中使用的,该变量就保存节流类对象。

4.“DEAFULT_THROTTLE_CLASSES”从哪里来?有什么用?

Django REST Framework之频率限制的使用

通过throttle_classes属性变量,跳转到该图,可以看到配置文件中说的“DEFAULT_THROTTLE_CLASSES”,它是用来通过配置文件settings来对全局节流类进行配置,功能等价于throttle_classes属性变量

5.“DEAFULT_THROTTLE_RATES”从哪里来?有什么用?

Django REST Framework之频率限制的使用

由SimpleRateThrottle类和上文对scope属性变量的分析可知,THROTTLE_RATES就是为了存储在配置文件中设置的不同的节流方法的。

综上所述,可以看出,在利用内置节流接口时,通过配置文件settings的设置和提供该接口所需的用户唯一标识外,不需要我们做再多的操作,这就形成了我们上文写的内置接口代码基本结构的样式。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python学习pygal绘制线图代码分享
Dec 09 Python
Python编程实现线性回归和批量梯度下降法代码实例
Jan 04 Python
使用python爬虫实现网络股票信息爬取的demo
Jan 05 Python
Python实现提取XML内容并保存到Excel中的方法
Sep 01 Python
8段用于数据清洗Python代码(小结)
Oct 31 Python
Python Tkinter模块 GUI 可视化实例
Nov 20 Python
如何更改 pandas dataframe 中两列的位置
Dec 27 Python
pycharm 实现本地写代码,服务器运行的操作
Jun 08 Python
python中的yield from语法快速学习
Nov 06 Python
python3 re返回形式总结
Nov 20 Python
详解Python Celery和RabbitMQ实战教程
Jan 20 Python
Matlab如何实现矩阵复制扩充
Jun 02 Python
基于Python实现大文件分割和命名脚本过程解析
Sep 29 #Python
Python2比较当前图片跟图库哪个图片相似的方法示例
Sep 28 #Python
python使用opencv实现马赛克效果示例
Sep 28 #Python
python打包成so文件过程解析
Sep 28 #Python
python基于FTP实现文件传输相关功能代码实例
Sep 28 #Python
python网络爬虫 Scrapy中selenium用法详解
Sep 28 #Python
在vscode中配置python环境过程解析
Sep 28 #Python
You might like
《破坏领主》销量已超100万 未来将继续开发新内容
2020/03/08 其他游戏
mysql_fetch_row,mysql_fetch_array,mysql_fetch_assoc的区别
2009/04/24 PHP
php 编写安全的代码时容易犯的错误小结
2010/05/20 PHP
PHP开发不能违背的安全规则 过滤用户输入
2011/05/01 PHP
PHP 字符串长度判断效率更高的方法
2014/03/02 PHP
Win7下手动安装apache2.2、php5.4笔记
2015/04/03 PHP
PHP生成腾讯云COS接口需要的请求签名
2018/05/20 PHP
phpstudy后门rce批量利用脚本的实现
2019/12/12 PHP
实例:尽可能写友好的Javascript代码
2006/10/09 Javascript
基于jQuery的淡入淡出可自动切换的幻灯插件打包下载
2010/09/15 Javascript
实现点击列表弹出列表索引的两种方式
2013/03/08 Javascript
JQuery性能优化的几点建议
2014/05/14 Javascript
基于jQuery实现复选框的全选 全不选 反选功能
2014/11/24 Javascript
javascript使用smipleChart实现简单图表
2015/01/02 Javascript
45个JavaScript编程注意事项、技巧大全
2015/02/11 Javascript
Zero Clipboard实现浏览器复制到剪贴板的方法(多个复制按钮)
2016/03/24 Javascript
jQuery Mobile动态刷新页面样式的实现方法
2016/05/28 Javascript
小程序实现带年月选取效果的日历
2018/06/27 Javascript
Vue-router 切换组件页面时进入进出动画方法
2018/09/01 Javascript
详解jQuery设置内容和属性
2019/04/11 jQuery
微信小程序使用前置摄像头拍照
2020/10/22 Javascript
如何在Express4.x中愉快地使用async的方法
2020/11/18 Javascript
python多线程编程方式分析示例详解
2013/12/06 Python
python实现简单socket通信的方法
2016/04/19 Python
python list删除元素时要注意的坑点分享
2018/04/18 Python
python3.7将代码打包成exe程序并添加图标的方法
2019/10/11 Python
python boto和boto3操作bucket的示例
2020/10/30 Python
英国PC组件和在线电脑商店:SCAN
2019/04/18 全球购物
澳大利亚设计的优质鞋类和适合澳大利亚生活方式的服装:Rivers
2019/04/23 全球购物
中国制造网:Made-in-China.com
2019/10/25 全球购物
美国健康和保健平台:healtop
2020/07/02 全球购物
办公室主任岗位职责
2013/11/08 职场文书
小孩百日宴答谢词
2014/01/15 职场文书
初中生自我评价
2014/02/01 职场文书
pytorch 实现多个Dataloader同时训练
2021/05/29 Python
Nginx四层负载均衡的配置指南
2021/06/11 Servers