django认证系统实现自定义权限管理的方法


Posted in Python onJuly 16, 2018

本文记录使用django自带的认证系统实现自定义的权限管理系统,包含组权限、用户权限等实现。

0x01. django认证系统

django自带的认证系统能够很好的实现如登录、登出、创建用户、创建超级用户、修改密码等复杂操作,并且实现了用户组、组权限、用户权限等复杂结构,使用自带的认证系统就能帮助我们实现自定义的权限系统达到权限控制的目的。

0x02. 认证系统User对象

User对象顾名思义即为表示用户的对象,里面的属性包括:

  1. username
  2. password
  3. email
  4. first_name
  5. last_name
  6. is_superuser
  7. is_active

创建好对象后,django会自动生成表,表名为auth_user,包含以上字段。具体的api文档如下所示:

class models.User

User 对象具有如下字段:

username
必选。少于等于30个字符。 用户名可以包含字母、数字、_、@、+、.和- 字符。

first_name
可选。 少于等于30个字符。

last_name
可选。少于30个字符。

email
可选。邮箱地址。

password
必选。 密码的哈希及元数据。(Django 不保存原始密码)。原始密码可以无限长而且可以包含任意字符。参见密码相关的文档。

groups
与Group 之间的多对多关系。

user_permissions
与Permission 之间的多对多关系。

is_staff
布尔值。指示用户是否可以访问Admin 站点。

is_active
布尔值。指示用户的账号是否激活。

is_superuser
布尔值。只是这个用户拥有所有的权限而不需要给他们分配明确的权限。

last_login
用户最后一次登录的时间。

date_joined
账户创建的时间。当账号创建时,默认设置为当前的date/time。

一般在注册操作中会用到该方法,实现注册一个用户,用到的函数是User.objects.create_user(),在新建用户的时候需要判断用户是否存在,我的实现方式是,User.objects.get(username=xxx)去获取一个用户User对象,用try except实现,如果用户不存在则抛出User.DoesNotExist异常,在这个异常中进行创建用户的操作。具体代码如下:

# 注册操作
from django.contrib.auth.models import User

try:
 User.objects.get(username=username)
 data = {'code': '-7', 'info': u'用户已存在'}
except User.DoesNotExist:
 user = User.objects.create_user(username, email, password)
 if user is not None:
  user.is_active = False
  user.save()

该过程中密码字段会自动加密存储。无需关注过多细节。

0x03. 登录登出用户

创建好用户后,就是登录及登出了,django认证系统提供了login()logout()函数,能够自动登录登出,并且修改session值,非常方便。验证用户身份使用authenticate函数能自动进行password字段的hash比对。

具体实现代码如下:

from django.contrib.auth import authenticate, login, logout

# 认证操作
ca = Captcha(request)
if ca.validate(captcha_code):
 user = authenticate(username=username, password=password)
 if user is not None:
  if user.is_active:
   # 登录成功
   login(request, user) # 登录用户
   data = {'code': '1', 'info': u'登录成功', 'url': 'index'}
  else:
   data = {'code': '-5', 'info': u'用户未激活'}
 else:
   data = {'code': '-4', 'info': u'用户名或密码错误'}
else:
 data = {'code': '-6', 'info': u'验证码错误'}

登出操作如下:

from django.contrib.auth import authenticate, login, logout

def logout_system(request):
  """
  退出登录
  :param request:
  :return:
  """
  logout(request)
  return HttpResponseRedirect('/')

0x04. login_required装饰器

通过该装饰器能够使视图函数首先判断用户是否登录,如果未登录会跳到默认在settings.py设置的LOGIN_URL参数对应的url,如:LOGIN_URL = '/'。使用方法如下所示:

from django.contrib.auth.decorators import login_required

@login_required
def user_index(request):
  """
  用户管理首页
  :param request:
  :return:
  """
  if request.method == "GET":
    # 用户视图实现

0x05. 用户组及权限分配

组对象包含的字段只有name,但是外键了几张表,能够与user、permissions,产生多对多的关系,我在自定义权限实现中,采用的是权限写死的方法,添加用户组权限。

创建组的函数采用Group.objects.create(name=xxx),就能实现了。当然也跟创建用户一样,需要先判断是否组名已经存在。

创建好组名后,下一步就需要为每个组分配权限了,从前端提交过来的权限列表,然后后端采用groups.permissions.add(permission)的方式依次将权限添加进组。

添加完组权限后,最后一步是将组名添加进用户属性,区分用户属于哪个组。

具体实现代码如下:

# 创建组
try:
  Group.objects.get(name=role_name)
  data = {'code': -7, 'info': u'组名已存在'}
except Group.DoesNotExist:
  groups = Group.objects.create(name=role_name)
  if log_manage == 'true':
    permission = Permission.objects.get(codename='access_log')
    groups.permissions.add(permission)
  if role_manage == 'true':
    permission = Permission.objects.get(codename='access_role_manage')
    groups.permissions.add(permission)
  if user_manage == 'true':
    permission = Permission.objects.get(codename='access_user_manage')
    groups.permissions.add(permission)
  if get_users is not None:
    for user in get_users:
      # 每个user添加组属性
      db_user = get_object_or_404(User, username=user)
      db_user.groups.add(groups)
      data = {'code': 1, 'info': u'添加成功'}
  return HttpResponse(json.dumps(data))

0x06. 权限模型及权限控制

在上一点中用到的Permission.objects.get(codename='access_user_manage')是通过权限模型创建,需要在models中创建一个权限类,然后在meta中进行定义codename

class AccessControl(models.Model):
  """
  自定义权限控制
  """
  class Meta:
    permissions = (
      ('access_dashboard', u'控制面板'),
      ('access_log', u'日志管理'),
      ('access_role_manage', u'角色管理'),
      ('access_user_manage', u'用户管理'),
    )

运行后,会自动在数据库中创建相应的表,并且插入数据。

在创建好权限之后,下一步就是在各个视图中插入权限控制代码了。permission_required(),参数为当前应用名.codename。这样就能控制用户访问,如果用户非法访问则会清空session退出登录。

@permission_required('webcenter.access_role_manage')
@login_required
def role_index(request):
  """
  角色管理首页
  :param request:
  :return:
  """

同时在前端模板页面中也需要进行权限控制,前端要获取request对象的话,后端返回就需要使用render函数,render(request,xxx,xxx) ,具体代码就如下:

{% if request.user.is_superuser or 'webcenter.access_user_manage' in request.user.get_group_permissions or 'webcenter.access_role_manage' in request.user.get_group_permissions or 'webcenter.access_log' in request.user.get_group_permissions %}
<li class="treeview">
 <a href="#" rel="external nofollow" rel="external nofollow" >
   <i class="fa fa-fw fa-skyatlas"></i>
   <span>站点管理</span> <i class="fa fa-angle-left pull-right"></i>
 </a>
 <ul class="treeview-menu">
 {% if request.user.is_superuser or 'webcenter.access_log' in request.user.get_group_permissions %}
   <li><a href="#" rel="external nofollow" rel="external nofollow" id="log_view">日志管理</a></li>
 {% endif %}
 {% if request.user.is_superuser or 'webcenter.access_role_manage' in request.user.get_group_permissions %}
   <li><a href="/role/index/" rel="external nofollow" >角色管理</a></li>
 {% endif %}
 {% if request.user.is_superuser or 'webcenter.access_user_manage' in request.user.get_group_permissions %}
   <li><a href="/user/index/" rel="external nofollow" >用户管理</a></li>
 {% endif %}
</ul>
</li>
{% endif %}

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

Python 相关文章推荐
Python的面向对象思想分析
Jan 14 Python
python实现在IDLE中输入多行的方法
Apr 19 Python
Tesserocr库的正确安装方式
Oct 19 Python
详解分布式任务队列Celery使用说明
Nov 29 Python
Python多线程处理实例详解【单进程/多进程】
Jan 30 Python
python如何解析配置文件并应用到项目中
Jun 27 Python
pyqt5 使用cv2 显示图片,摄像头的实例
Jun 27 Python
Python3简单爬虫抓取网页图片代码实例
Aug 26 Python
Python3.x+迅雷x 自动下载高分电影的实现方法
Jan 12 Python
Python异常继承关系和自定义异常实现代码实例
Feb 20 Python
python GUI库图形界面开发之PyQt5信号与槽事件处理机制详细介绍与实例解析
Mar 08 Python
浅谈Python项目的服务器部署
Apr 25 Python
Sanic框架路由用法实例分析
Jul 16 #Python
Sanic框架安装与简单入门示例
Jul 16 #Python
python 除法保留两位小数点的方法
Jul 16 #Python
Python自定义装饰器原理与用法实例分析
Jul 16 #Python
python 正确保留多位小数的实例
Jul 16 #Python
浅谈Python里面小数点精度的控制
Jul 16 #Python
详解Django中间件执行顺序
Jul 16 #Python
You might like
详解PHP显示MySQL数据的三种方法
2008/06/05 PHP
PHP提示Notice: Undefined variable的解决办法
2012/11/24 PHP
解析PHP中intval()等int转换时的意外异常情况
2013/06/21 PHP
基于php导出到Excel或CSV的详解(附utf8、gbk 编码转换)
2013/06/25 PHP
PHP安装memcached扩展笔记
2015/05/28 PHP
PHP实现连接设备、通讯和发送命令的方法
2015/10/13 PHP
PHP 数组基本操作小结(推荐)
2016/06/13 PHP
thinkPHP商城公告功能开发问题分析
2016/12/01 PHP
PHP进阶学习之垃圾回收机制详解
2019/06/18 PHP
基于jquery的气泡提示效果
2010/05/31 Javascript
jqeury eval将字符串转换json的方法
2011/01/20 Javascript
js的onload事件及初始化按钮事件示例代码
2013/09/25 Javascript
浅谈javascript的调试
2015/01/28 Javascript
jQuery制作可自定义大小的拼图游戏
2015/03/30 Javascript
jQuery简单实现仿京东分类导航层效果
2016/06/07 Javascript
归纳下js面向对象的几种常见写法总结
2016/08/24 Javascript
AngularJS ng-style中使用filter
2016/09/21 Javascript
jQuery滑动到底部加载下一页数据的实例代码
2017/05/22 jQuery
详解RequireJs官方使用教程
2017/10/31 Javascript
微信小程序ibeacon三点定位详解
2018/10/31 Javascript
详解ng-alain动态表单SF表单项设置必填和正则校验
2019/06/11 Javascript
[48:39]Ti4主赛事胜者组第一天 EG vs NEWBEE 2
2014/07/19 DOTA
python模拟登陆Tom邮箱示例分享
2014/01/13 Python
Python 2与Python 3版本和编码的对比
2017/02/14 Python
python爬虫基础之urllib的使用
2020/12/31 Python
Pyside2中嵌入Matplotlib的绘图的实现
2021/02/22 Python
CSS3 Backgrounds属性相关介绍
2011/05/11 HTML / CSS
Kate Spade美国官网:纽约新兴时尚品牌,以包包闻名于世
2017/11/09 全球购物
Ego Shoes官网:英国时髦鞋类品牌
2020/10/19 全球购物
心理健康心得体会
2014/01/02 职场文书
学校三节实施方案
2014/06/09 职场文书
法人代表资格证明书
2015/06/18 职场文书
公司回复函格式
2015/07/14 职场文书
2016年班主任培训心得体会
2016/01/07 职场文书
node快速搭建后台的实现步骤
2022/02/18 NodeJs
使用JS前端技术实现静态图片局部流动效果
2022/08/05 Javascript