当前位置: 首页 > 编程日记 > 正文

浅谈权限设计(来自深空老大)

2019独角兽企业重金招聘Python工程师标准>>> hot3.png

By 深空, 2009-09-13 21:45:07

PHPChina的专家版在谈权限设计,苦于没有权限回帖,特发此博文谈谈简单的权限设计。讨论在这里
最简单的权限验证,应该是登录态的验证,如果登录,则可以怎样,没有登录,则不能怎样:

1
2
3
4
5
if ( $isLogin === true) {
     //do something
} else {
     //do nothing
}

一般使用会话或者Cookie来保存登录态,具体实现不在此文讨论范围。一般权限都和人挂勾,首先识别你是谁,然后看你有能力做什么,然后再确认你的能力在这个地方是否可以使,一个权限验证算是基本上完成。我们围绕这几点来看权限如何去设计。
首先要能识别操作者是何许人,我们需要一张保存操作者信息的表,也就是通常所说的用户表。简单的用户表如下:

1
2
3
4
5
CREATE TABLE user (
     userId int (10) unsigned NOT NULL ,
     username varchar (255) NOT NULL ,
     PRIMARY KEY (userId)
)

一般使用一个用户ID来标识一个唯一的用户,可以使用数字,或者直接使用用户名作为主键(如果用户名不重复)。这里我们使用userId来唯一标识一个用户。
有了用户以后,接下来需要确认这个用户所具有的能力,也就是权限,那么首先我们需要列一下我们的系统总共需要几个权限,比如增、删、改、查等。增加一张权限表:

1
2
3
4
5
CREATE TABLE permission (
     permissionId int (10) unsigned NOT NULL ,
     permissionName varchar (255) NOT NULL ,
     PRIMARY KEY (permissionId)
)

同样的我们以permissionId作为主键来唯一标识一个权限,当然也可以使用permissionName来标识(如果你能确定唯一的话)。我们新增几条记录在这张表里:

1
2
3
4
5
6
7
8
+--------------+----------------+
| permissionId | permissionName |
+--------------+----------------+
|            1 | add            |
|            2 | del            |
|            3 | modify         |
|            4 | select         |
+--------------+----------------+

这里列举了4个权限,简单的表示我们的用户在系统里可能具有的增、删、改、查4种不同的能力。
接下来把这些能力赋给用户,需要一张对应表来保存:

1
2
3
4
5
CREATE TABLE userPermission (
     userId int (10) unsigned NOT NULL ,
     permissionId int (10) unsigned NOT NULL ,
     PRIMARY KEY (userId, permissionId)
)

其中将userId和permissionId设置为主键,表示某个用户具有某种权限。表内容可能如下:

1
2
3
4
5
6
7
8
9
10
+--------+--------------+
| userId | permissionId |
+--------+--------------+
|      1 |            1 |
|      1 |            4 |
|      2 |            1 |
|      2 |            2 |
|      2 |            3 |
|      2 |            4 |
+--------+--------------+

以上权限配置表明用户1具有增、查权限,用户2具有增、删、改、查权限(嗯可以猜想用户1是个普通用户,用户2是个管理员)。
写到这里,我发现基本的用户权限系统雏型已经完成了。这么简单?看起来好像确实就是这么简单。一个用户拥有哪些权限,那么只需要勾选相应的权限分配给这个用户。在验证权限的时候,取出用户所拥有的所有权限,然后判断是否存在该权限即可。
实际上的权限设计要比这个复杂一些,到底复杂在哪里呢?我们接下来分析。当用户比较多而且权限数量比较多的时候,你是不是要每个用户都去勾选一堆权限呢?如何简化这个操作?OK,用户组的概念推出。所谓用户组,就是具有某些权限的一类人的集合。我们赋予用户组某些权限,然后把这个用户加到这个用户组里即可,来看看用户组长什么样子:

1
2
3
4
5
CREATE TABLE group (
     groupId int (10) unsigned NOT NULL ,
     groupName varchar (255) NOT NULL ,
     PRIMARY KEY (groupId)
)

和用户表类似,我们需要标识一个唯一的组,这里分配一个组ID作为主键来标识。内容可能如下:

1
2
3
4
5
6
+---------+-----------+
| groupId | groupName |
+---------+-----------+
|       1 | user      |
|       2 | admin     |
+---------+-----------+

有了用户组表后,我们需要把一些权限赋给用户组,就需要一张用户组权限表:

1
2
3
4
5
CREATE TABLE groupPermission (
     groupId int (10) unsigned NOT NULL ,
     permissionId int (10) unsigned NOT NULL ,
     PRIMARY KEY (groupId, permissionId)
)

我们分配增、查权限给user组,分配增、删、改、查权限给admin组:

1
2
3
4
5
6
7
8
9
10
+---------+--------------+
| groupId | permissionId |
+---------+--------------+
|       1 |            1 |
|       1 |            4 |
|       2 |            1 |
|       2 |            2 |
|       2 |            3 |
|       2 |            4 |
+---------+--------------+

用户组表和用户组所对应的权限表有了,那么要把用户分配给一个用户组,就需要一张用户和组的对应关系表:

1
2
3
4
5
CREATE TABLE userGroup (
     userId int (10) unsigned NOT NULL ,
     groupId int (10) unsigned NOT NULL ,
     PRIMARY KEY (userId, groupId)
)

把用户1赋给user组,把用户2赋给admin组,和一开始我们直接分配权限一样:

1
2
3
4
5
6
+--------+---------+
| userId | groupId |
+--------+---------+
|      1 |       1 |
|      2 |       2 |
+--------+---------+

很明显这里的配置信息相对比userPermission表少很多,这里只需要记录用户属于什么组就可以了,那么检测用户权限,就需要查出用户所在的组,然后再查出这个组(或者这些组)所拥有的权限,就可以得出用户所具备的权限,再进行验证即可。当然会比直接查询userPermission表绕一点点,不过相对比维护成本,这点点消耗不算什么。更何况我们其实仍然可以保存userPermission表,在分配用户组的时候,同时更新userPermission表即可。
这里可以看到,除了在分配用户权限方便以外,当你需要更改某类用户权限的时候,你只需要更改其所在组的权限,那么这个组下所有成员的权限也会随之更改,非常方便。
到这里用户、用户组的权限构成基本完成。它能解决大部分问题,可是我发现它仍然有一些小的问题。比如如果某个用户只有查权限,我不得不再新增一个用户组,搞得用户组也很多,怎么办呢?如果这个用户属于普通用户组,其实可以考虑也分配普通用户组给这个用户,然后再从普通用户组里“扣掉”增权限。要达到这样的效果,怎么处理呢?
其实很简单,我们刚才没有去掉的userPermission表派上用场,这个表存储了用户的实际单个权限,我们只需要增加一个字段标识用户是拥有这个权限,还是没有这个权限即可,这样可以解决两个问题:一是从现有用户组中扣掉某些权限,二是在现有用户组中,再给这个用户增加用户组以外的权限。来看一下userPermission表:

1
2
3
4
5
6
CREATE TABLE userPermission (
   userId int (10) unsigned NOT NULL ,
   permissionId int (10) unsigned NOT NULL ,
   has enum( 'yes' , 'no' ) NOT NULL ,
   PRIMARY KEY (userId, permissionId)
)

把用户1的增权限去掉,那么内容可能像这样:

1
2
3
4
5
6
7
8
9
10
+--------+--------------+-----+
| userId | permissionId | has |
+--------+--------------+-----+
|      1 |            1 | no  |
|      1 |            4 | yes |
|      2 |            1 | yes |
|      2 |            2 | yes |
|      2 |            4 | yes |
|      2 |            3 | yes |
+--------+--------------+-----+

这个用户依然在user组里,只不过user组所拥有的2个权限(add, select),他少了个add(ID为1被标记为no)权限而已。
嗯,这样做有解决了这个小问题,不过这个功能增强会让分配权限代码更复杂一些,不仅要给用户分配组,还可能需要操作具体权限,让他有或者让他没有相应的权限。
OK,简单的权限设计全部完成,只不过,细心的读者,你是否意识到,还少点什么呢?没错,即使到这里,整个权限验证还少了一块很重要的部分,那就是用户拥有这些权限,那么他能在哪里使用这些权限。考虑一个例子,一个论坛版主在他所管理的论坛版块里拥有删、改帖子的权限,他在其他非他所管理的版块就没有这些权限,可能在其他论坛版块他像是一个普通用户一样,我们上面讨论的权限设计如何做到这个验证呢?那么我个人觉得,权限设计到上面已经完成了,接下去这种情况,属于业务逻辑层的验证,我们从权限系统中已经获得用户的权限,那么在具体业务逻辑中,和权限进行绑定即可,以上例子可以用一个版块用户关系表来解决这个问题:

1
2
3
4
5
CREATE TABLE userBoard (
   userId int (10) unsigned NOT NULL ,
   boardId int (10) unsigned NOT NULL ,
   PRIMARY KEY (userId, boardId)
)

标记用户拥有哪些版块的管理权限,那么在严重用户拥有管理权限的时候,还要看当前版块是否是用户管辖内的版块,最终确定用户是否有操作权限。那么我将这类和业务逻辑相关的权限分配归到用户角色里,同样可以创建一系列角色,来管理用户所管辖的范围,比如超级版主,他也是具有管理删、改权限,只不过他的权限作用于全论坛,那么普通版主就需要指定论坛,这样来区分用户组和角色组我想会使整个权限系统更加清晰。
到这里,权限验证全部完成,以上没有具体实现代码,我相信这样已经足够了,具体的实现代码和业务逻辑由具体的应用实现吧。

http://www.phpfans.org/53.html

转载于:https://my.oschina.net/tenking/blog/28571

相关文章:

5年Python功力,总结了10个开发技巧

作者 | 写代码的明哥来源 |Python编程时光(ID: Cool-Python)如何在运行状态查看源代码?查看函数的源代码,我们通常会使用 IDE 来完成。比如在 PyCharm 中,你可以 Ctrl 鼠标点击 进入函数的源代码。那如果没有 IDE 呢&…

怎样给目录加权限0777

# chmod -R 0777 /var/www/html/子目录

php学习,一个简单的Calendar(2) 一个简单的活动页面

有了前面的基础&#xff0c;后面就是将页面展示出来。 预览图如下&#xff1a;1号和31号分别有活动&#xff0c;会一并显示出来 这里需要搞定几个问题&#xff0c;一个就是数据库的连接&#xff0c;我们用\sys\class\class.db_connect.inc.php <?php /* * 数据库操作&#…

涨见识了,在终端执行 Python 代码的 6 种方式

作者 | BRETT CANNON译者 | 豌豆花下猫Python猫为了我们推出的 VS Code 的 Python 插件[1]&#xff0c;我写了一个简单的脚本来生成变更日志[2]&#xff08;类似于Towncrier[3]&#xff0c;但简单些&#xff0c;支持 Markdown&#xff0c;符合我们的需求&#xff09;。在发布过…

ASP.NET中DataGrid鼠标经过感知以及点击行弹出窗口

选择自 xujh 的 Blog 作者Blog&#xff1a;http://blog.csdn.net/xujh/ 很多人说很难&#xff0c;其实就这几行代码。只要在DataGrid1的ItemDataBound中写入下代码即可 private void DataGrid1_ItemDataBound(object sender, System.Web.UI.WebControls.DataGridItemEventAr…

python中的module

Python中的Module是比较重要的概念。常见的情况是&#xff0c;事先写好一个.py文件&#xff0c;在另一个文件中需要import时&#xff0c;将事先写好的.py文件拷贝到当前目录&#xff0c;或者是在sys.path中增加事先写好的.py文件所在的目录&#xff0c;然后import。这样的做法&…

找子串替换(kmp)poj1572

题目链接&#xff1a;http://poj.org/problem?id1572 输入数据时要注意&#xff0c;这里是string型 用getline(cin,origin[i]); #include <string> #include <iostream> #include <algorithm> #include <stdio.h>using namespace std;const int maxn …

dll的概念、dll导出类(转)

1、 DLL的概念DLL(Dynamic Linkable Library)&#xff0c;动态链接库&#xff0c;可以向程序提供一些函数、变量或类。这些可以直接拿来使用。静态链接库与动态链接库的区别&#xff1a;&#xff08;1&#xff09;静态链接库与动态链接库都是共享代码的方式。静态链接库把最后的…

墨奇科技汤林鹏:如何用 AI 技术颠覆指纹识别?

受访者 | 墨奇科技联合创始人& CTO 汤林鹏 记者 | Aholiab&#xff0c;编辑 | Carol 出品 | AI科技大本营&#xff08;ID&#xff1a;rgznai100&#xff09; 随着深度学习等AI技术的成熟&#xff0c;生物识别成为了关注度较高的领域&#xff0c;指纹、人脸、虹膜等识别技术…

ASP.Net ViewState的实现

选择自 timmy3310 的 Blog ViewState是.Net中提出的状态保存的一种新途径&#xff08;实际上也是老瓶装新酒&#xff09;&#xff1b;我们知道&#xff0c;传统的Web程序保存状态的方式有这样几种&#xff1a;1、Application 这是Web应用程序生命期中的全局保存区&#xff0c…

【51CTO学院三周年】遇到

作为一名二流学校的大学生&#xff0c;因为学校的一门嵌入式Linux应用程序开发而喜欢上了嵌入式&#xff0c;但是学校却是只上了一学期的课&#xff0c;无奈只能自己找教程继续学习。在3个月前&#xff0c;无意中找到了朱有鹏老师的嵌入式学习基础视频&#xff0c;通过老师视频…

ASP.NET图象处理详解

作者&#xff1a;未知 请与本人联系在使用ASP的时候&#xff0c;我们时常要借助第三方控件来实现一些图象功能。而现在&#xff0c;ASP.NET的推出&#xff0c;我们已经没有必要再使用第三方控件来实现&#xff0c;因为ASP.NET 已经具有强大的功能来实现一些图象处理。现在&…

IT工作者,你们的爱情是这样的吗?

今天在博客里看到了这篇文章&#xff0c;看完这个视频我随笔写了点自己的感受和看法&#xff0c; 视频链接在下方&#xff1a; http://leidu.blog.51cto.com/3245712/622534 很感谢90 男孩提供&#xff0c;建议IT人员看一下&#xff0c;看完写一下你们的感受吧&#xff01;…

平头哥玄铁处理器Linux新版本,5大亮点速览

来源 | 芯片开放社区为了便于 CPU 评估&#xff0c;系统集成&#xff0c;快速上手玄铁处理器 Linux 操作系统&#xff0c;平头哥更新了玄铁处理器 linux 版本&#xff0c;结合 gitlab 开源 CI/CD 系统&#xff0c;对已发布到开源社区的玄铁架构 CPU 相关的生态软件形成持续保障…

费用保险单,如何失焦时自动补零

费用&#xff0c;如何失焦时自动补零转载于:https://www.cnblogs.com/maojiayan/p/5606247.html

腾讯AI种番茄双丰收:参赛AI全胜专家,辽宁试点净利增千元

6月9日&#xff0c;腾讯宣布了两项AI农业领域进展。在研究侧&#xff0c;腾讯 AI Lab 与荷兰瓦赫宁根大学&#xff08;下称WUR&#xff09;联办的“第二届国际智慧温室种植挑战赛”&#xff08;下称比赛&#xff09;落幕。在全球疫情肆虐之时&#xff0c;复赛的五支队伍挑战用 …

在ASP.NET中值得注意的两个地方

在ASP.NET中ASPX页面的Page_Load事件有两个让人奇怪的地方&#xff0c;你应该记住它们&#xff1a; a.有时Page_Load事件在你的ASP.NET页面里会发生多次。这种情况发生的一个可能的原因是你把ASPX页面的AutoEvenWireup值设置成了True。如果是这样&#xff0c;那么在“Sub Page…

yii 级联删除

alter table 外键表 add constraint 级联删除名 foreign key (外键字段) references 主表名(主表字段)ON delete CASCADE go 转载于:https://blog.51cto.com/dasangshu/624605

iOS--优秀博客记录

感谢唐巧整理&#xff0c;我又加了一些备注。原地址&#xff1a;https://github.com/tangqiaoboy/iOSBlogCN 博客地址RSS地址psOneVs Denhttp://onevcat.com/atom.xml 一只魔法师的工坊http://blog.ibireme.com/feed/图片处理、YYKit破船之家http://beyondvincent.com/atom.xml…

linux环境下和网络服务相关的配置文件含义及如何配置

要建立一个安全Linux服务器就首先要了解Linux环境下和网络服务相关的配置文件的含义及如何进行安全的配置。那天查看服务器的eth0地址&#xff0c;后来想了一些问题&#xff0c;到家里就翻了翻以前的文档&#xff0c;无意中看到了这个&#xff0c;现在发布出来&#xff0c;希望…

ASP.NET保持用户状态的九种选择

2003-06-10 ■陶刚编译 ■yesky摘要&#xff1a;ASP.NET为保持用户请求之间的数据提供了多种不同的途径。你可以使用Application对象、cookie、hidden fields、Sessions或Cache对象&#xff0c;以及它们的大量的方法。决定什么时候使用它们有时很困难。本文将介绍了上述的技术&…

Python 图像处理 | 图像平滑之均值滤波、方框滤波、高斯滤波及中值滤波

作者 | 杨秀璋&#xff0c;责编 | 夕颜 题图 | 视觉中国出品 | CSDN博客本篇文章主要讲解Python调用OpenCV实现图像平滑&#xff0c;包括四个算法&#xff1a;均值滤波、方框滤波、高斯滤波和中值滤波。全文均是基础知识&#xff0c;希望对您有所帮助。知识点如下&#xff1a;…

Laravel Lumen之Eloquent ORM使用速查-基础部分

使用Eloquent [eləkwənt] 时&#xff0c;数据库查询构造器的方法对模型类也是也用的&#xff0c;使用上只是省略了DB::table(表名)部分。 在模型中使用protected成员变量$table指定绑定的表名。<?php namespace App;use Illuminate\Database\Eloquent\Model;class Flight…

数据库连接字在Web.config里的用法

作者&#xff1a;未知 请速与本人联系在asp.net中的WEB程序的设置中我们必须用到Web.config来存储数据库连接字.事实上这是个很好的做法,因为可以省去我们很多的麻烦还可以帮助我们避免不必要的错位,是的很多情况下我就是这样做.它通过XML来记录这些信息.具体的是在....这个标…

重构ncnn,腾讯优图开源新一代移动端推理框架TNN

来源 | 腾讯优图从学界到工业界&#xff0c;“开源”已经成为AI领域的一个关键词。一方面&#xff0c;它以“授人以渔”的方式为AI构建了一个开放共进的生态环境&#xff0c;帮助行业加速AI应用落地&#xff1b;另一方面&#xff0c;在解决行业实际问题时持续更新和迭代&#x…

java读取文件

1 java8读取文本文件2 3 4 public static void java8ReadFileLines(String fileName) throws IOException {5 List lineList Files.readAllLines(Paths.get(fileName), StandardCharsets.UTF_8);6 7 for(String line:lineList){8 Sys…

常见 Datagrid 错误

Marcie Robillard DatagridGirl.comDatagrid 控件是 Microsoft ASP.NET 中功能最强、用途最广的 Web 控件之一&#xff0c;这一点已经得到了 ASP.NET 权威人士的认同。虽然 Datagrid 控件易于使用&#xff0c;但同样易于给使用者带来麻烦。以下是许多人所犯的一些错误&#xff…

干货!3 个重要因素,带你看透 AI 技术架构方案的可行性!

作者 | 房磊责编 | Carol出品 | AI 科技大本营&#xff08;ID&#xff1a;rgznai100&#xff09;人工智能这几年发展的如火如荼&#xff0c;不仅在计算机视觉和自然语言处理领域发生了翻天覆地的变革&#xff0c;在其他领域也掀起了技术革新的浪潮。无论是在新业务上的尝试&…

mysql从另一张获取数据的方法

方法一 CREATE TABLE tmp AS SELECT a.id FROM t_user t JOIN temp a ON t.email a.email; 方法二 INSERT INTO t_user (id,username,PASSWORD,email,user_type,STATUS) SELECT id,REPLACE(email,,_),PASSWORD,email,0,0 FROM temp; 这两个sql都是从另外一张中获取的数据插入…

手动创建Spring项目 Spring framework

之前学习框架一直是看的视频教程&#xff0c;并且在都配套有项目源码&#xff0c;跟着视频敲代码总是很简单&#xff0c;现在想深入了解&#xff0c;自己从官网下载文件手动搭建&#xff0c;就遇到了很多问题记载如下。 首先熟悉一下spring的官方网站&#xff1a;http://spring…