04,认证、权限、频率

 

认证组件

Django原生的authentic组件为我们的用户注册与登录提供了认证功能,十分的简介与强大。同样DRF也为我们提供了认证组件,一起来看看DRF里面的认证组件是怎么为我们工作的!
models.py
# 定义一个用户表和一个保存用户Token的表 class UserInfo(models.Model): username = models.CharField(max_length=16) password = models.CharField(max_length=32) type = models.SmallIntegerField( choices=((1, '普通用户'), (2, 'VIP用户')), default=1 ) class Token(models.Model): user = models.OneToOneField(to='UserInfo') token_code = models.CharField(max_length=128)
15
1
# 定义一个用户表和一个保存用户Token的表
2
 
3
 
4
class UserInfo(models.Model):
5
    username = models.CharField(max_length=16)
6
    password = models.CharField(max_length=32)
7
    type = models.SmallIntegerField(
8
        choices=((1, '普通用户'), (2, 'VIP用户')),
9
        default=1
10
    )
11
 
12
 
13
class Token(models.Model):
14
    user = models.OneToOneField(to='UserInfo')
15
    token_code = models.CharField(max_length=128)
url
path('login/', views.LoginView.as_view()),
1
1
    path('login/', views.LoginView.as_view()),
views.py
# 视图主要处理用户名、密码是否正确,用户每一次请求都要带着专有token来! import hashlib, time from rest_framework.response import Response from rest_framework.views import APIView def get_random_token(username): """ 根据用户名和时间戳生成随机token :param username: :return: """ timestamp = str(time.time()) m = hashlib.md5(bytes(username, encoding="utf8")) m.update(bytes(timestamp, encoding="utf8")) return m.hexdigest() class LoginView(APIView): """ 校验用户名密码是否正确从而生成token的视图 """ def post(self, request): res = {"code": 0} print(request.data) username = request.data.get("username") password = request.data.get("password") user = models.UserInfo.objects.filter(username=username, password=password).first() if user: # 如果用户名密码正确 token = get_random_token(username) models.Token.objects.update_or_create(defaults={"token_code": token}, user=user) res["token"] = token else: res["code"] = 1 res["error"] = "用户名或密码错误" return Response(res)
38
1
# 视图主要处理用户名、密码是否正确,用户每一次请求都要带着专有token来!
2
import hashlib, time
3
from rest_framework.response import Response
4
from rest_framework.views import APIView
5
 
6
 
7
def get_random_token(username):
8
    """
9
    根据用户名和时间戳生成随机token
10
    :param username:
11
    :return:
12
    """
13
    timestamp = str(time.time())
14
    m = hashlib.md5(bytes(username, encoding="utf8"))
15
    m.update(bytes(timestamp, encoding="utf8"))
16
    return m.hexdigest()
17
 
18
 
19
class LoginView(APIView):
20
    """
21
    校验用户名密码是否正确从而生成token的视图
22
    """
23
    def post(self, request):
24
        res = {"code": 0}
25
        print(request.data)
26
        username = request.data.get("username")
27
        password = request.data.get("password")
28
 
29
        user = models.UserInfo.objects.filter(username=username, password=password).first()
30
        if user:
31
            # 如果用户名密码正确
32
            token = get_random_token(username)
33
            models.Token.objects.update_or_create(defaults={"token_code": token}, user=user)
34
            res["token"] = token
35
        else:
36
            res["code"] = 1
37
            res["error"] = "用户名或密码错误"
38
        return Response(res)
定义认证类model_serializer.py
# 这一步是要对着源码才能写出来 from rest_framework.authentication import BaseAuthentication from rest_framework.exceptions import AuthenticationFailed class MyAuth(BaseAuthentication): def authenticate(self, request): if request.method in ["POST", "PUT", "DELETE"]: request_token = request.data.get("token", None) if not request_token: raise AuthenticationFailed('缺少token') token_obj = models.Token.objects.filter(token_code=request_token).first() if not token_obj: raise AuthenticationFailed('无效的token') return token_obj.user.username, None else: return None, None
18
1
# 这一步是要对着源码才能写出来
2
from rest_framework.authentication import BaseAuthentication
3
from rest_framework.exceptions import AuthenticationFailed
4
 
5
 
6
class MyAuth(BaseAuthentication):
7
    def authenticate(self, request):
8
        if request.method in ["POST", "PUT", "DELETE"]:
9
            request_token = request.data.get("token", None)
10
            if not request_token:
11
                raise AuthenticationFailed('缺少token')
12
            token_obj = models.Token.objects.filter(token_code=request_token).first()
13
            if not token_obj:
14
                raise AuthenticationFailed('无效的token')
15
            return token_obj.user.username, None
16
        else:
17
            return None, None
18
 
全局配置
# 在settings.py中配置 REST_FRAMEWORK = { "DEFAULT_AUTHENTICATION_CLASSES": ["app01.utils.MyAuth", ] }
4
1
# 在settings.py中配置
2
REST_FRAMEWORK = {
3
    "DEFAULT_AUTHENTICATION_CLASSES": ["app01.utils.MyAuth", ]
4
}

权限组件

只有vip才能看的内容
自定义权限类
# 自定义权限类 from rest_framework.permissions import BasePermission class MyPermission(BasePermission): message = 'VIP用户才能访问' def has_permission(self, request, view): """ 自定义权限只有VIP用户才能访问 """ # 因为在进行权限判断之前已经做了认证判断,所以这里可以直接拿到request.user if request.user and request.user.type == 2: # 如果是VIP用户 return True else: return False
15
1
# 自定义权限类
2
from rest_framework.permissions import BasePermission
3
 
4
class MyPermission(BasePermission):
5
    message = 'VIP用户才能访问'
6
 
7
    def has_permission(self, request, view):
8
        """
9
        自定义权限只有VIP用户才能访问
10
        """
11
        # 因为在进行权限判断之前已经做了认证判断,所以这里可以直接拿到request.user
12
        if request.user and request.user.type == 2:  # 如果是VIP用户
13
            return True
14
        else:
15
            return False
视图级别配置
class CommentViewSet(ModelViewSet): queryset = models.Comment.objects.all() serializer_class = app01_serializers.CommentSerializer authentication_classes = [MyAuth, ] permission_classes = [MyPermission, ]
6
1
class CommentViewSet(ModelViewSet):
2
 
3
    queryset = models.Comment.objects.all()
4
    serializer_class = app01_serializers.CommentSerializer
5
    authentication_classes = [MyAuth, ]
6
    permission_classes = [MyPermission, ]
全局配置
REST_FRAMEWORK = { "DEFAULT_AUTHENTICATION_CLASSES": ["app01.utils.MyAuth", ], "DEFAULT_PERMISSION_CLASSES": ["app01.utils.MyPermission", ] }
4
1
REST_FRAMEWORK = {
2
    "DEFAULT_AUTHENTICATION_CLASSES": ["app01.utils.MyAuth", ],
3
    "DEFAULT_PERMISSION_CLASSES": ["app01.utils.MyPermission", ]
4
}

频率组件

频率:限制用户访问网站的频率
自定义限制类
VISIT_RECORD = {} # 自定义限制 class MyThrottle(object): def __init__(self): self.history = None def allow_request(self, request, view): """ 自定义频率限制60秒内只能访问三次 """ # 获取用户IP ip = request.META.get("REMOTE_ADDR") timestamp = time.time() if ip not in VISIT_RECORD: VISIT_RECORD[ip] = [timestamp, ] return True history = VISIT_RECORD[ip] self.history = history history.insert(0, timestamp) while history and history[-1] < timestamp - 60: history.pop() if len(history) > 3: return False else: return True def wait(self): """ 限制时间还剩多少 """ timestamp = time.time() return 60 - (timestamp - self.history[-1])
33
1
VISIT_RECORD = {}
2
# 自定义限制
3
class MyThrottle(object):
4
 
5
    def __init__(self):
6
        self.history = None
7
 
8
    def allow_request(self, request, view):
9
        """
10
        自定义频率限制60秒内只能访问三次
11
        """
12
        # 获取用户IP
13
        ip = request.META.get("REMOTE_ADDR")
14
        timestamp = time.time()
15
        if ip not in VISIT_RECORD:
16
            VISIT_RECORD[ip] = [timestamp, ]
17
            return True
18
        history = VISIT_RECORD[ip]
19
        self.history = history
20
        history.insert(0, timestamp)
21
        while history and history[-1] < timestamp - 60:
22
            history.pop()
23
        if len(history) > 3:
24
            return False
25
        else:
26
            return True
27
 
28
    def wait(self):
29
        """
30
        限制时间还剩多少
31
        """
32
        timestamp = time.time()
33
        return 60 - (timestamp - self.history[-1])
视图级别配置
class CommentViewSet(ModelViewSet): queryset = models.Comment.objects.all() serializer_class = app01_serializers.CommentSerializer throttle_classes = [MyThrottle, ]
x
1
class CommentViewSet(ModelViewSet):
2
 
3
    queryset = models.Comment.objects.all()
4
    serializer_class = app01_serializers.CommentSerializer
5
    throttle_classes = [MyThrottle, ]
全局配置
# 在settings.py中设置rest framework相关配置项 REST_FRAMEWORK = { "DEFAULT_AUTHENTICATION_CLASSES": ["app01.utils.MyAuth", ], "DEFAULT_PERMISSION_CLASSES": ["app01.utils.MyPermission", ] "DEFAULT_THROTTLE_CLASSES": ["app01.utils.MyThrottle", ] }
1
# 在settings.py中设置rest framework相关配置项
2
REST_FRAMEWORK = {
3
    "DEFAULT_AUTHENTICATION_CLASSES": ["app01.utils.MyAuth", ],
4
    "DEFAULT_PERMISSION_CLASSES": ["app01.utils.MyPermission", ]
5
    "DEFAULT_THROTTLE_CLASSES": ["app01.utils.MyThrottle", ]
6
}

认证组件的源码阅读

 

新构建的request里面的源码的user方法

 

源码走到了这里其实就已经需要我们自己来进行认证了。就可以对前端的请求进行认证,到底该怎么认证,还得继续往下走!

 

权限组件的源码阅读

相比较与认证组件,权限组件就更加的简洁了

频率组件的注意部分

转载于:https://www.cnblogs.com/pontoon/p/10217414.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/366018.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

sun.misc.Unsafe和堆外内存

sun.misc.Unsafe类允许您执行许多Java中不应该做的事情&#xff0c;但是在非常特殊的情况下仍然有用。 必须在99&#xff05;的时间避免这种情况&#xff0c;但是在极少数情况下&#xff0c;这是唯一有意义的解决方案。 这篇文章考虑了它在OpenHFT中的使用方式以及我希望在Jav…

jq获取input选取的文件名_tushare获取交易数据并可视化分析

获取数据是金融量化分析的第一步&#xff0c;找不到可靠、准确的数据&#xff0c;量化分析就无从谈起。随着信息技术的不断发展&#xff0c;数据获取渠道也越来越多&#xff0c;尤其是Python网络爬虫&#xff0c;近几年愈来愈火。然而&#xff0c;很多人毕竟精力有限&#xff0…

linux电脑做笔记软件,知识管理工具, 自由格式数据库, 笔记软件以及个人信息管理...

myBase Desktop 7.3.5 for Linux/MacOSX/Windows 2020-2-20新增 [数学公式 MathJax for Markdown]、[自动锁屏保护]、[粘贴图片为附件]、[完整复制带图片内容]&#xff0c;以及部分问题修订&#xff0c;详见&#xff1a;更新日志&#xff1b;数学公式插件&#xff1a;缺省安装后…

js模板引擎

模板引擎 function tmpl(str, o) {return str.replace(/\\?\{([^{}] )\}/g, function (match, name) {return (o[name] undefined) ? : o[name];});}调用示例 var html tmpl(html, { user_id: user_id, user_name: user_name });更多专业前端知识&#xff0c;请上 【猿204…

原来游戏技术行业最大的秘密竟然是...

欢迎大家前往腾讯云社区&#xff0c;获取更多腾讯海量技术实践干货哦~ 本文由腾讯游戏云发表于云社区专栏 本篇文章主要是分享游戏业务面临的安全风险场景&#xff0c;以及基于这些场景的特点&#xff0c;我们应该如何做好对应的防护。 【一、背景&#xff1a;游戏行业DDoS攻击…

Leetcode 565. Array Nesting

题目 链接&#xff1a;https://leetcode.com/problems/array-nesting/ Level: Medium Discription: A zero-indexed array A of length N contains all integers from 0 to N-1. Find and return the longest length of set S, where S[i] {A[i], A[A[i]], A[A[A[i]]], ... }…

指定Gradle构建属性

属性是用于轻松自定义Gradle构建和Gradle环境的宝贵工具。 我将在本文中演示一些用于指定Gradle构建中使用的属性的方法。 Gradle支持项目属性和系统属性 。 这篇文章中有趣的是两者之间的主要区别是如何访问它们。 通过常规Java / Groovy系统属性访问方法访问系统属性时&…

python数字转中文字符_Python实现中文数字转换为阿拉伯数字的方法示例

本文实例讲述了Python实现中文数字转换为阿拉伯数字的方法。分享给大家供大家参考&#xff0c;具体如下&#xff1a; 一、需求 今天写了三千二百行代码。 今天写了3200行代码。 两行意思相同&#xff0c;只是表达方式不太能够&#xff0c;统一掉。 二、原理 数字的特征是 数字 …

调程序时的小错误

1.未能找到类型或命名空间名称“Form1”(是否缺少 using 指令或程序集引用?) 应该是复制代码的时候&#xff0c;把程序入口处的命名空间和Form1的命名空间不一致 2.检索 COM 类工厂中 CLSID 为 {XXXX-XXXX-XXX-XXXXX-XXX} 的组件时失败&#xff0c;原因是出现以下错误: 80040…

eclipse中设置Linux编码环境,更改eclipse中console编码

1.在eclipse.ini的-vmargs下面加上「-Dfile.encodingUTF-8」。2.eclipse-->Window-->Preferences-->General-->Workspace-->Text file encoding-->UTF-83.Installed JREs-->Default VM Arguments:-->-Dfile.encodingUTF-84.eclipse-->Run-->Run …

[记录] JavaScript 中的BOM操作

转载于:https://www.cnblogs.com/yuxi2018/p/9520059.html

GC对吞吐量和延迟的影响

每个Java应用程序都必须解决的一种类型的问题与垃圾回收有关。 当垃圾收集器工作时&#xff0c;它代表了一项奇妙的发明。 如果不是这样&#xff0c;或者GC做家务的方式变得不可预测&#xff0c;那么您就有一个朋友变成了敌人。 这篇文章是关于垃圾收集暂停时间的。 或更准确地…

linux odbc连接数据库失败,在Linux中使用RODBC连接到SQL Server时出错

我在linux服务器上运行不同的r程序。程序因数据库连接失败而出现以下错误&#xff1a; -1: In odbcDriverConnect(paste0("DRIVER{SQL Server}; server", server,: [RODBC] ERROR: state 08001, code 0, message[unixODBC][FreeTDS][SQL Server]Unable to connect to…

高级cmd攻击命令_一步一步学习DVWA渗透测试(Command Injection命令行注入)-第七次课...

各位小伙伴&#xff0c;今天我们继续学习Command Injection&#xff0c;翻译为中文就是命令行注入。是指通过提交恶意构造的参数破坏命令语句结构&#xff0c;从而达到执行恶意命令的目的。在OWASP TOP 10中一种存在注入漏洞&#xff0c;最常见的就是SQL和命令行注入。PHP开发的…

IDEA插件之 CodeGlance

在编辑代码最右侧&#xff0c;显示一块代码小地图 这款插件使用效果图如下&#xff0c;个人感觉还是有点用处&#xff0c;滚动条太小&#xff0c;有这个地图&#xff0c;拖动起来更加方便一点 原文地址:http://tengj.top/2017/02/22/idea1-1/转载于:https://www.cnblogs.com/al…

移动端图片上传方法

移动端图片上传方法 实现效果 文件下载 http://files.cnblogs.com/files/sntetwt/移动端图片上传.rar 实现步骤 一、隐藏<input type"file" id"file" name"Filedata" style"display:none;" accept"image/*" /> 二、…

SQL UPDATE with INNER JOIN

mysql - SQL UPDATE with INNER JOIN - Stack Overflowhttps://stackoverflow.com/questions/14491042/sql-update-with-inner-join MySQL UPDATE JOIN | Cross-Table Update in MySQLhttp://www.mysqltutorial.org/mysql-update-join/ 转载于:https://www.cnblogs.com/rgqancy…

linux 29900端口,USB2.0接口100M以太网芯片SR9900(A)的应用

1&#xff1a;SR9900A和SR9900有啥区别&#xff1a;(就是同一颗料)SR9900A默认随机MAC地址&#xff1a;00E0XXXXXXXXXXSR9900默认纯0MAC地址&#xff1a;000000000000都可以二次烧录MAC地址的… 如果SR9900没货用SR9900A没问题的&#xff0c;但是SR9900A没货&#xff0c;用SR99…

easyui常见问题

1.EasyUI的combobox可以看到选项但是不能选择怎么办 检查一下绑定的字段是否名称一样&#xff0c;valueField:id,textField:name&#xff0c;主要检查valueField $(#contentsName).combotree({panelHeight: auto,panelMaxHeight: 200,multiple: false,required: false,editable…

c语言最大公约数和最小公倍数_五年级奥数课堂之七:公因数和公倍数

乘积尾0的个数公因数和公倍数的基本概念公因数的释义给定若干个整数&#xff0c;如果有一个(些)数是它们共同的因数&#xff0c;那么这个(些)数就叫做它们的公因数。而全部公因数中最大的那个&#xff0c;称为这些整数的最大公因数。公约数与公倍数相反&#xff0c;就是既是A的…