基本使用

  1. 频率:
  2. -限制每个ip地址一分钟访问10
  3. -写一个类
  4. from rest_framework.throttling import SimpleRateThrottle
  5. class VisitThrottle(SimpleRateThrottle):
  6. scope = 'xx' #一定要写它
  7. # 必须重写这个方法,返回什么,就以什么作为访问频率限制的东西,
  8. def get_cache_key(self, request, view):
  9. # return self.get_ident(request)
  10. return request.META.get('REMOTE_ADDR')
  11. -在setting中配置:
  12. 'DEFAULT_THROTTLE_RATES': {
  13. #这个key跟scope对应,value值3/m 3/h 4/d
  14. 'xx': '3/m'
  15. }
  16. -使用:
  17. -局部使用:
  18. throttle_classes = [VisitThrottle]
  19. -全局使用:
  20. "DEFAULT_THROTTLE_CLASSES":["app01.auth.VisitThrottle"],
  21. -问题?如果网站所有的接口都是一分钟访问10次,但是单独有一个接口要一分钟访问2次,如何做?

自定义频率类,自定义频率规则

自定义的逻辑

  1. 1)取出访问者ip
  2. 2)判断当前ip不在访问字典里,添加进去,并且直接返回True,表示第一次访问,在字典里,继续往下走
  3. 3)循环判断当前ip的列表,有值,并且当前时间减去列表的最后一个时间大于60s,把这种数据pop掉,这样列表中只有60s以内的访问时间,
  4. 4)判断,当列表小于3,说明一分钟以内访问不足三次,把当前时间插入到列表第一个位置,返回True,顺利通过
  5. 5)当大于等于3,说明一分钟内访问超过三次,返回False验证失败

代码实现

  1. class MyThrottles():
  2. VISIT_RECORD = {}
  3. def __init__(self):
  4. self.history=None
  5. def allow_request(self,request, view):
  6. #(1)取出访问者ip
  7. # print(request.META)
  8. ip=request.META.get('REMOTE_ADDR')
  9. import time
  10. ctime=time.time()
  11. # (2)判断当前ip不在访问字典里,添加进去,并且直接返回True,表示第一次访问
  12. if ip not in self.VISIT_RECORD:
  13. self.VISIT_RECORD[ip]=[ctime,]
  14. return True
  15. self.history=self.VISIT_RECORD.get(ip)
  16. # (3)循环判断当前ip的列表,有值,并且当前时间减去列表的最后一个时间大于60s,把这种数据pop掉,这样列表中只有60s以内的访问时间,
  17. while self.history and ctime-self.history[-1]>60:
  18. self.history.pop()
  19. # (4)判断,当列表小于3,说明一分钟以内访问不足三次,把当前时间插入到列表第一个位置,返回True,顺利通过
  20. # (5)当大于等于3,说明一分钟内访问超过三次,返回False验证失败
  21. if len(self.history)<3:
  22. self.history.insert(0,ctime)
  23. return True
  24. else:
  25. return False
  26. def wait(self):
  27. import time
  28. ctime=time.time()
  29. return 60-(ctime-self.history[-1])

内置频率类及局部使用

写一个类,继承自SimpleRateThrottle,(根据ip限制)问:要根据用户现在怎么写

  1. from rest_framework.throttling import SimpleRateThrottle
  2. class VisitThrottle(SimpleRateThrottle):
  3. scope = 'luffy'
  4. def get_cache_key(self, request, view):
  5. return self.get_ident(request)

在setting里配置:(一分钟访问三次)

  1. REST_FRAMEWORK = {
  2. 'DEFAULT_THROTTLE_RATES':{
  3. 'luffy':'3/m'
  4. }
  5. }

在视图类里使用

  1. throttle_classes = [MyThrottles,]

错误信息的中文提示

  1. class Course(APIView):
  2. authentication_classes = [TokenAuth, ]
  3. permission_classes = [UserPermission, ]
  4. throttle_classes = [MyThrottles,]
  5.  
  6. def get(self, request):
  7. return HttpResponse('get')
  8.  
  9. def post(self, request):
  10. return HttpResponse('post')
  11. def throttled(self, request, wait):
  12. from rest_framework.exceptions import Throttled
  13. class MyThrottled(Throttled):
  14. default_detail = '傻逼啊'
  15. extra_detail_singular = '还有 {wait} second.'
  16. extra_detail_plural = '出了 {wait} seconds.'
  17. raise MyThrottled(wait)

内置频率限制类:

BaseThrottle是所有类的基类:方法:def get_ident(self, request)获取标识,其实就是获取ip,自定义的需要继承它

AnonRateThrottle:未登录用户ip限制,需要配合auth模块用

SimpleRateThrottle:重写此方法,可以实现频率现在,不需要咱们手写上面自定义的逻辑

UserRateThrottle:登录用户频率限制,这个得配合auth模块来用

ScopedRateThrottle:应用在局部视图上的(忽略)

内置频率类及全局使用

  1. REST_FRAMEWORK = {
  2. 'DEFAULT_THROTTLE_CLASSES':['app01.utils.VisitThrottle',],
  3. 'DEFAULT_THROTTLE_RATES':{
  4. 'luffy':'3/m'
  5. }
  6. }

源码分析

  1. def check_throttles(self, request):
  2. for throttle in self.get_throttles():
  3. if not throttle.allow_request(request, self):
  4. self.throttled(request, throttle.wait())
  5. def throttled(self, request, wait):
  6. #抛异常,可以自定义异常,实现错误信息的中文显示
  7. raise exceptions.Throttled(wait)

SimpleRateThrottle源码分析

  1. class SimpleRateThrottle(BaseThrottle):
  2. # 咱自己写的放在了全局变量,他的在django的缓存中
  3. cache = default_cache
  4. # 获取当前时间,跟咱写的一样
  5. timer = time.time
  6. # 做了一个字符串格式化,
  7. cache_format = 'throttle_%(scope)s_%(ident)s'
  8. scope = None
  9. # 从配置文件中取DEFAULT_THROTTLE_RATES,所以咱配置文件中应该配置,否则报错
  10. THROTTLE_RATES = api_settings.DEFAULT_THROTTLE_RATES
  11.  
  12. def __init__(self):
  13. if not getattr(self, 'rate', None):
  14. # 从配置文件中找出scope配置的名字对应的值,比如咱写的‘3/m’,他取出来
  15. self.rate = self.get_rate()
  16. # 解析'3/m',解析成 3 m
  17. self.num_requests, self.duration = self.parse_rate(self.rate)
  18. # 这个方法需要重写
  19. def get_cache_key(self, request, view):
  20. """
  21. Should return a unique cache-key which can be used for throttling.
  22. Must be overridden.
  23.  
  24. May return `None` if the request should not be throttled.
  25. """
  26. raise NotImplementedError('.get_cache_key() must be overridden')
  27.  
  28. def get_rate(self):
  29. if not getattr(self, 'scope', None):
  30. msg = ("You must set either `.scope` or `.rate` for '%s' throttle" %
  31. self.__class__.__name__)
  32. raise ImproperlyConfigured(msg)
  33.  
  34. try:
  35. # 获取在setting里配置的字典中的之,self.scope是 咱写的luffy
  36. return self.THROTTLE_RATES[self.scope]
  37. except KeyError:
  38. msg = "No default throttle rate set for '%s' scope" % self.scope
  39. raise ImproperlyConfigured(msg)
  40. # 解析 3/m这种传参
  41. def parse_rate(self, rate):
  42. """
  43. Given the request rate string, return a two tuple of:
  44. <allowed number of requests>, <period of time in seconds>
  45. """
  46. if rate is None:
  47. return (None, None)
  48. num, period = rate.split('/')
  49. num_requests = int(num)
  50. # 只取了第一位,也就是 3/mimmmmmmm也是代表一分钟
  51. duration = {'s': 1, 'm': 60, 'h': 3600, 'd': 86400}[period[0]]
  52. return (num_requests, duration)
  53. # 逻辑跟咱自定义的相同
  54. def allow_request(self, request, view):
  55. """
  56. Implement the check to see if the request should be throttled.
  57.  
  58. On success calls `throttle_success`.
  59. On failure calls `throttle_failure`.
  60. """
  61. if self.rate is None:
  62. return True
  63.  
  64. self.key = self.get_cache_key(request, view)
  65. if self.key is None:
  66. return True
  67.  
  68. self.history = self.cache.get(self.key, [])
  69. self.now = self.timer()
  70.  
  71. # Drop any requests from the history which have now passed the
  72. # throttle duration
  73. while self.history and self.history[-1] <= self.now - self.duration:
  74. self.history.pop()
  75. if len(self.history) >= self.num_requests:
  76. return self.throttle_failure()
  77. return self.throttle_success()
  78. # 成功返回true,并且插入到缓存中
  79. def throttle_success(self):
  80. """
  81. Inserts the current request's timestamp along with the key
  82. into the cache.
  83. """
  84. self.history.insert(0, self.now)
  85. self.cache.set(self.key, self.history, self.duration)
  86. return True
  87. # 失败返回false
  88. def throttle_failure(self):
  89. """
  90. Called when a request to the API has failed due to throttling.
  91. """
  92. return False
  93.  
  94. def wait(self):
  95. """
  96. Returns the recommended next request time in seconds.
  97. """
  98. if self.history:
  99. remaining_duration = self.duration - (self.now - self.history[-1])
  100. else:
  101. remaining_duration = self.duration
  102.  
  103. available_requests = self.num_requests - len(self.history) + 1
  104. if available_requests <= 0:
  105. return None
  106.  
  107. return remaining_duration / float(available_requests)

  

Restful framework【第八篇】频率组件的更多相关文章

  1. Restful framework【第二篇】APIView

    安装djangorestframework 方式一:pip3 install djangorestframework 方式二:pycharm图形化界面安装 方式三:pycharm命令行下安装(装在当前 ...

  2. Restful framework【第一篇】RESTful 规范

    什么是RESTful REST与技术无关,代表的是一种软件架构风格,REST是Representational State Transfer的简称,中文翻译为“表征状态转移” REST从资源的角度类审 ...

  3. 第十八篇 admin组件

    admin组件 admin组件使用 admin源码解析 admin组件使用 Django 提供了基于 web 的管理工具. Django 自动管理工具是 django.contrib 的一部分.你可以 ...

  4. Entity Framework 第八篇 结构优化

    在之前的文章里,业务层直接调用一个包装的仓储类入口,忽略了DAL层,在业务层绕过DAL直接调用仓储类似乎也没什么大的问题,但是这样做有一个很大的弊端,就是无法做到DAL层的原子操作的复用.假如多个业务 ...

  5. Restful framework【第九篇】分页器

    基本使用 分页 -简单分页 page_size = api_settings.PAGE_SIZE :每页显示条数 page_query_param = 'page' :查询的页码数 page_size ...

  6. RESTful源码笔记之RESTful Framework的基本组件

    快速实例 Quickstart 序列化 创建一个序列化类 简单使用 开发我们的Web API的第一件事是为我们的Web API提供一种将代码片段实例序列化和反序列化为诸如json之类的表示形式的方式. ...

  7. Restful 4 -- 认证组件、权限组件、频率组件、url注册器、响应器、分页器

    一.认证组件.权限组件.频率组件总结:  只有认证通过的用户才能访问指定的url地址,比如:查询课程信息,需要登录之后才能查看,没有登录,就不能查看,这时候需要用到认证组件 1.认证组件格式 写一个认 ...

  8. RESTful-rest_framework认证组件、权限组件、频率组件-第五篇

    认证组件.权限组件.频率组件总结:  认证组件格式: 1 写一个认证类 from rest_framework.authentication import BaseAuthentication cla ...

  9. 基于Django的Rest Framework框架的频率组件

    0|1一.频率组件的作用 在我们平常浏览网站的时候会发现,一个功能你点击很多次后,系统会让你休息会在点击,这其实就是频率控制,主要作用是限制你在一定时间内提交请求的次数,减少服务器的压力. modle ...

  10. Python之路【第十八篇】:Web框架们

    Python之路[第十八篇]:Web框架们   Python的WEB框架 Bottle Bottle是一个快速.简洁.轻量级的基于WSIG的微型Web框架,此框架只由一个 .py 文件,除了Pytho ...

随机推荐

  1. VS2010和选中代码相同的代码的颜色设置,修改高亮颜色

    使用Visual Studio 2010, 发现很难看清非活动的选中代码,研究了下,发现以下方法可以设置: 1. 菜单:工具  -> 选项  ->环境  ->字体和颜色 2. 在右边 ...

  2. JavaScript 创建和浅析自定义对象

    在Js中,除了Array.Date.Number等内置对象外,开发者可以通过Js代码创建自己的对象. 目录 1. 对象特性:描述对象的特性 2. 创建对象方式:对象直接量.new 构造函数.Objec ...

  3. Python全栈-数据库介绍与基本操作

    .数据库管理软件的由来 数据库的产生是为了解决数据的永久储存.数据安全.以及对方对外服务时能够实现并发服务等效果.例如解决前面所学的Scoket编程中,在不考虑硬件问题的基础上,服务端服务多个客户端时 ...

  4. 说说html 的<!DOCTYPE>声明&标准模式与兼容模式

    我们都知道<!DOCTYPE>声明位于文档的最前面,处于<html>标签之前. <!DOCTYPE>声明不是html标签,它的作用:告知web浏览界面应该使用哪个h ...

  5. html5-article元素

    <!DOCTYPE html><html lang="en"><head>    <meta charset="UTF-8&qu ...

  6. keras如何求分类问题中的准确率和召回率

    https://www.zhihu.com/question/53294625 由于要用keras做一个多分类的问题,评价标准采用precision,recall,和f1_score:但是keras中 ...

  7. loadrunner 更新中......

    一.安装及参考说明 1.51 testing 链接:http://www.51testing.com/zhuanti/LoadRunner.html 2.官网链接:http://learnloadru ...

  8. Swift 了解(2)

    循环(Loops) 1. For条件递增语句 ; counter < ; counter++ ) { liftWeights( ) } 语法是这样的:用for作为循环的开始,告诉Xcode你要声 ...

  9. CS131&Cousera图像处理学习笔记 - L4&W2滤波和卷积

    cs131: http://vision.stanford.edu/teaching/cs131_fall1617/ coursera: https://www.coursera.org/learn/ ...

  10. python 内置函数format

    Python2.6 开始,新增了一种格式化字符串的函数 str.format(),它增强了字符串格式化的功能. 基本语法是通过 {} 和 : 来代替以前的 % . format 函数可以接受不限个参数 ...