乐趣区

企业管理系统前后端分离架构设计 系列一 权限模型篇

前段时间分别用 vue 和 react 写了两个后台管理系统的模板 vue-quasar-admin 和 3YAdmin。两个项目中都实现了基于 RBAC 的权限控制。因为本职工作是后端开发,比较清楚权限控制一个管理系统应该必须具备的核心功能,而且是可以做到通用的。打算写写关于管理系统前后端分离方面的文章,也是做一个知识的总结,其中会涉及到 vue,react,node,.net core 等方面的知识。
术语描述

用户(Subject):发起操作的主体
对象(Object):指操作所针对的客体对象,比如文章或评论
权限(Permission):用来指代对某种对象的某一种操作,例如“添加文章的操作”
权限码:权限的代号,例如用“ARTICLE_ADD”来指代“添加文章的操作”权限

权限有时候也可以称为动作或者功能。比如“添加文章”,既可以认为它是一个动作,也可以认为它是一个功能。对象也可以称为资源。
常用的权限模型

ACL(Access Control List)(访问控制列表)
DAC(Discretionary Access Control)(自主访问控制)
MAC(Mandatory Access Control)(强制访问控制)
RBAC(Role-Based Access Control)(基于角色的访问控制)
ABAC(Attribute-Based Access Control)(基于属性的访问控制)

ACL(Access Control List)(访问控制列表)
ACL 是最早也是最基本的一种访问控制机制,它是用来描述用户和权限之间关系的数据列表。它的原理非常简单:每一项资源,都配有一个列表,这个列表记录的就是哪些用户可以对这项资源执行 CRUD 等操作。当试图访问这项资源时,会首先检查这个列表中是否有关于当前用户的访问权限,从而确定当前用户可否执行相应的操作。
例如一个文件对象的 ACL 为 Alice: read,write; Bob: read,这代表 Alice 对该文件既能读又能写,而 Bob 只能读取。
由于 ACL 的简单性,使得它几乎不需要任何基础设施就可以完成访问控制。但同时它的缺点也是很明显的,由于需要维护大量的访问权限列表,ACL 在性能上有明显的缺陷。另外,对于拥有大量用户与众多资源的应用,管理访问控制列表本身就变成非常繁重的工作。

最开始的 ACL 定义中,用户直接和权限挂钩,数据存储的是用户与权限的关联关系。如果两个用户的权限是一样的,那么就需要分别存储这两个用户与权限的关联关系,也是上面所提到的 ACL 的缺陷。为了解决这些问题,便有了对 ACL 设计的改进,相同权限的用户放到同一个分组里,分组与权限挂钩,不再是用户直接与权限挂钩。以及后来出现的 RBAC(基于角色的访问控制),角色与分组也是差不多的概念,角色直接与权限挂钩,用户再与角色进行关联。
所以,现在一般说 ACL, 不再是用户直接和权限挂钩的一种权限控制模型,把它看做一个单纯的访问控制列表即可。列表里维护的可能是用户与权限的关系,也可以是用户组与权限的关系,也可以是角色与权限的关系,甚至是部门,职位等等于权限的关系。ACL 是权限体系中的业务规则。RBAC 等权限模型要用到 ACL 才能工作,ACL 服务于 RBAC 等权限模型,其它权限控制体系里的权限规则也叫 ACL。

DAC(Discretionary Access Control)(自主访问控制)
系统会识别用户,然后根据被操作对象(Subject)的权限控制列表(ACL: Access Control List)或者权限控制矩阵(ACL: Access Control Matrix)的信息来决定用户的是否能对其进行哪些操作,例如读取或修改。而拥有对象权限的用户,又可以将该对象的权限分配给其他用户,所以称之为“自主(Discretionary)”控制。
因为用户能自主地将自己拥有的权限授予其他用户,所以 DAC 模型可以任意传递权限,用户能间接获得本不具有的访问权限,因此 DAC 模型的安全性较低,不能给系统充分的数据保护。

DAC 可以直接使用 ACL 的物理模型,区别在于,DAC 模型中用户可以将自己具备的权限分配给其它用户(程序里的操作就是根据用户 ID 筛选出权限列表,根据列表为要分配权限的用户构造出新的权限列表并保存)
DAC 是传统的 UNIX 访问控制模型,也是 Windows 文件系统的访问控制模型。

Windows 的文件访问权限的设置中,除了用户,还有组。这个组与后面要说到的 RABC 模型的角色有什么区别呢?
https://stackoverflow.com/questions/7770728/group-vs-role-any-real-difference
我认为没必要去划分的太清楚,不管是组还是角色,都是为了更好的管理和分配权限在最原始的 ACL 模型上做的改进。如果有需要,甚至可以把权限分配到部门,职位上。
MAC(Mandatory Access Control)(强制访问控制)
MAC 是为了弥补 DAC 权限控制过于分散的问题而诞生的。在 MAC 的设计中,每一个对象都有一些权限标识,每个用户同样也会有一些权限标识,而用户能否对该对象进行操作取决于双方的权限标识的关系,这个限制判断通常是由系统硬性限制的。访问时,系统先对用户的访问许可级别和资源对象的密级进行比较,再决定用户是否可以访问资源对象。用户不能改变自身和资源对象的安全级别,只有系统管理员或管理程序才能 控制资源对象和用户的级别。比如在影视作品中我们经常能看到特工在查询机密文件时,屏幕提示需要“无法访问,需要一级安全许可”,这个例子中,文件上就有“一级安全许可”的权限标识,而用户并不具有。
MAC 非常适合机密机构或者其他等级观念强烈的行业,但对于类似商业服务系统,则因为不够灵活而不能适用。
MAC 可以继续使用 DAC 的模型,但是要对用户进行等级划分,比如一级,二级,三级。。。,对对象资源也要做划分,比如机密,秘密和最高机密。用户访问的资源的时候,根据用户等级与资源访问级别来做判断,比如一级用户只能访问机密文件,如果访问的是最高机密文件,系统就会拒绝。这一系列规则是优先于 DAC 的,如果 MAC 与 DAC 混用,要先校验 MAC 再校验 DAC。
RBAC(Role-Based Access Control)(基于角色的访问控制)
ACL 的访问控制机制中,直接维护的是用户与功能的关系,这一系列的关系就是一个权限列表。当很多的用户具有相同功能权限的时候,就要进行繁琐的关联操作。RBAC 就是在用户与权限之间引入了角色的概念。用户与角色之间做关联,权限列表维护的是角色与功能的关系。

RBAC 是目前使用最普遍的权限控制模型。当某些用户具备相同的权限的时候,只需要为这些用户建一个角色,把相应的功能关联到这个角色上,生成角色的权限列表。当有新的用户需要相同权限的时候,把用户关联到这个角色上即可。而当用检查或校验用户的操作权限的时候,查询用户所属角色的权限列表即可。
当然,RBAC 也不是完美的,比如想要为某个用户单独设置某个功能权限,可能需要为这个功能权限单独创建一个角色,然后把特定的用户关联到这个角色上。当想要移除某个用户的特定功能权限的时候,可能需要重新设置角色的功能权限,把特定功能权限从当前角色中移除,建立新的角色并关联特定的功能权限,然后再把新角色与相关的用户做关联(也可以直接在特定功能的程序里校验操作用户)
这里说一个比较常见的 RBAC 的错误的用法:那就是直接使用角色做权限判断。比如只有角色 A 才能做文章的删除操作。
function delPost(postId){
if(!isRole(‘A’)){
return false;
}
}
如果需求该为角色 B 也可以删除文章。那就必须修改代码
function delPost(postId){
if(!isRole(‘A’)&&!isRole(‘B’)){
return false;
}
}
正确的做法应该是添加 ” 删除文章 ” 这个功能,把这个功能关联到相应的角色上。判断的时候是根据功能去判断而不是角色。
function delPost(postId){
if(!hasPermission(‘POST_DEL’)){
return false;
}
}
针对“只有角色 A 才能做文章的删除操作”这一需求,把这个删除功能关联到角色 A 上,然后把需要这个操作权限的用户加入到角色 A 中即可。当别的角色也需要这个操作权限,把功能关联到对应角色上即可,不需要再修改代码。
在 RBAC 的核心基础上,还可以做相应的扩展,比如角色继承,角色分组之类的,这些扩展都是为了在一定程度简化权限管理工作。
ABAC(Attribute-Based Access Control)(基于属性的权限控制)
RBAC 虽然是目前最普遍的权限控制模型。但是某些情况下,RBAC 是无法满足并且也实现不了的。比如业务员 1 和业务员 2 都属于业务员角色,都有查看客户订单的权限。当有一个需求,要求业务员 1 只能查看北京地区的客户的订单,业务员 2 只能查看上海的客户的订单。这单单使用 RBAC 是无法实现。借助 RBAC,可行的做法是,分地区创建角色,然后程序中根据角色做数据的过滤,这种做法缺点之前也提到过,需求变更的时候可能需要每次都修改代码。
上面业务员查看订单的例子,地区是订单的一个属性,需求就是针对这个地区属性来做订单的查询范围的权限控制。这种权限控制方式就是 ABAC(Attribute-Based Access Control)(基于属性的权限控制),也被一些人称为是权限系统设计的未来。
不同于常见的将用户通过某种方式关联到权限的方式,ABAC 则是通过动态计算一个或一组属性是否满足某种条件来进行授权判断的(可以编写简单的逻辑)。属性通常来说分为四类:用户属性(如用户年龄),环境属性(如当前时间),操作属性(如读取)和对象属性(如一篇文章,又称资源属性),所以理论上能够实现非常灵活的权限控制,几乎能满足所有类型的需求。
例如规则:“允许所有班主任在上课时间自由进出校门”这条规则,其中,“班主任”是用户的角色属性,“上课时间”是环境属性,“进出”是操作属性,而“校门”就是对象属性了。
ABAC 非常的灵活,但是实现也是非常的难。这其中涉及到逻辑的动态执行,数据动态过滤等,更加具体就是动态拼接 SQL 语句(使用 ORM 的话就是动态组装对应 ORM 的查询语句)。
感兴趣的可以在 Github 上搜索 ABAC,看看不同语言是否已经有现成的解决方案。下面说说我学习到的一种实现方式:
还是业务员查看订单的例子,在 RBAC 的基础上,扩展一个实体规则,订单就是实体,也就是针对订单设置一系列的规则。规则存储格式可以是 json 也可以是 xml, 甚至是 Sql 语句,能解析即可。比如北京地区这个规则:
{
“regionId”:1
}
上海地区:
{
“regionId”:3
}
regionId 就是系统里对应区域的 Id, 也是订单或订单相关表的某个字段。
保存这个规则的时候,规则内容 (就是上面的 json), 规则实体(也就是订单,表明这个规则是针对订单的) 是必须的。也可以加上这个规则是适用增删改查中的一种或多种。
创建好实体的规则,将规则与角色做关联,也就是将北京地区的规则关联到北京地区角色上,上海地区的规则关联到上海地区角色上。
后端做权限校验的时候,还是先按 RBAC 模型的控制方式进行校验(是否具备订单查看权限),然后根据当前操作对象(也就是实体),取出用户所属角色关联的对应实体的规则。然后解析规则,动态拼接 Sql 或者 ORM 语句。
没做地区限制 (或没配置规则) 的时候,Sql 可能是
select userId,orderNo,createdDate from T_Order
配置了规则,解析拼接后可能就是
select userId,orderNo,createdDate from T_Order where regionId=1
这里是针对地区这个属性实现了动态的权限控制。实际开发过程中,要控制的东西是非常多了,查看字段的控制,数据范围的控制。要满足这些复杂的控制,需要制定一套完整的规则,以及针对规则编写相应的解析程序。比如根据配置的规则,最后解析出来可能是各种 Sql 语句:<,>,=,like,in,not in 等等。
可以看出,要真正的落地实现 ABAC 是多么的复杂。每次都要解析规则,对程序的性能也造成的影响,就算使用缓存,命中的概率也是非常的小,因为很多因素都是动态的。
所以,如果需要根据属性做权限判断的场景不是很多的话,还是建议使用 RBAC,然后程序中做判断比较省事省力。
总结
ACL 早期定义中是一种权限控制机制,这种机制直接维护的是用户与功能的关系,功能就是针对对象定义的一些操作,比如增删改查的等。用户与功能的关系列表也称为权限列表或访问控制列表,现在说 ACL,一般就是指这个权限列表或访问控制列表,但是里面维护的关系不一定是用户与功能的关系,在 RBAC 中维护的就是角色与功能的关系。
RBAC 在 ACL 的基础上加入了角色的概念,权限列表或访问控制列表里维护的不再是用户与功能的关系,而是角色与功能的关系。ACL 可以和 RBAC 混着用,既可以在角色上设置权限,也可以直接给用户设置权限,更加灵活。借助角色的思想,可以在用户组,组织,职位等等上设置权限,以便更好的做好权限管理,也就是将权限设置从单一个体转移到某一类组合上。
ABAC 非常的灵活,也非常的难实现。
参考文章
权限系统设计模型分析
Authorization Models: ACL, DAC, MAC, RBAC, ABAC

退出移动版