前言

权限控制主要分为两块,认证(Authentication)与授权(Authorization)。认证之后确认了身份正确,业务系统就会进行授权,现在业界比较流行的模型就是RBAC(Role-Based Access Control)。RBAC包含为下面四个要素:用户、角色、权限、资源。用户是源头,资源是目标,用户绑定至角色,资源与权限关联,最终将角色与权限关联,就形成了比较完整灵活的权限控制模型。

资源是最终需要控制的标的物,但是我们在一个业务系统中要将哪些元素作为待控制的资源呢?我将系统中待控制的资源分为三类:

URL访问资源(接口以及网页)

界面元素资源(增删改查导入导出的按钮,重要的业务数据展示与否等)

数据资源

现在业内普遍的实现方案实际上很粗放,就是单纯的“菜单控制”,通过菜单显示与否来达到控制权限的目的。

我仔细分析过,现在大家做的平台分为To C和To B两种:

To C一般不会有太多的复杂权限控制,甚至大部分连菜单控制都不用,全部都可以访问。

To B一般都不是开放的,只要做好认证关口,能够进入系统的只有内部员工。大部分企业内部的员工互联网知识有限,而且作为内部员工不敢对系统进行破坏性的尝试。

所以针对现在的情况,考虑成本与产出,大部分设计者也不愿意在权限上进行太多的研发力量。

菜单和界面元素一般都是由前端编码配合存储数据实现,URL访问资源的控制也有一些框架比如SpringSecurity,Shiro。

目前我还没有找到过数据权限控制的框架或者方法,所以自己整理了一份。

数据权限控制原理

数据权限控制最终的效果是会要求在同一个数据请求方法中,根据不同的权限返回不同的数据集,而且无需并且不能由研发编码控制。这样大家的第一想法应该就是AOP,拦截所有的底层方法,加入过滤条件。这样的方式兼容性较强,但是复杂程度也会更高。我们这套系统中,采用的是利用Mybatis的plugin机制,在底层SQL解析时替换增加过滤条件。

这样一套控制机制存在很明显的优缺点,首先缺点:

适用性有限,基于底层的Mybatis。

方言有限,针对了某种数据库(我们使用Mysql),而且由于需要在底层解析处理条件所以有可能造成不同的数据库不能兼容。当然Redis和NoSQL也无法限制。

当然,假如你现在就用Mybatis,而且数据库使用的是Mysql,这方面就没有太大影响了。

接下来说说优点:

减少了接口数量及接口复杂度。原本针对不同的角色,可能会区分不同的接口或者在接口实现时利用流程控制逻辑来区分不同的条件。有了数据权限控制,代码中只用写基本逻辑,权限过滤由底层机制自动处理。

提高了数据权限控制的灵活性。例如原本只有主管能查本部门下组织架构/订单数据,现在新增助理角色,能够查询本部门下组织架构,不能查询订单。这样的话普通的写法就需要调整逻辑控制,使用数据权限控制的话,直接修改配置就好。

数据权限实现

上一节就提及了实现原理,是基于Mybatis的plugins)实现。

MyBatis 允许你在已映射语句执行过程中的某一点进行拦截调用。默认情况下,MyBatis 允许使用插件来拦截的方法调用包括:

Executor (update, query, flushStatements, commit, rollback, getTransaction, close, isClosed)

ParameterHandler (getParameterObject, setParameters)

ResultSetHandler (handleResultSets, handleOutputParameters)

StatementHandler (prepare, parameterize, batch, update, query)

Mybatis的插件机制目前比较出名的实现应该就是PageHelper项目了,在做这个实现的时候也参考了PageHelper项目的实现方式。所以权限控制插件的类命名为PermissionHelper。

机制是依托于Mybatis的plugins机制,实际SQL处理的时候基于jsqlparser这个包。

设计中包含两个类,一个是保存角色与权限的实体类命名为PermissionRule,一个是根据实体变更底层SQL语句的主体方法类PermissionHelper。

首先来看下PermissionRule的结构:

public class PermissionRule {

private static final Log log = LogFactory.getLog(PermissionRule.class);

/**

* codeName

* 适用角色列表

* 格式如: ,RoleA,RoleB,

*/

private String roles;

/**

* codeValue

* 主实体,多表联合

* 格式如: ,SystemCode,User,

*/

private String fromEntity;

/**

* codeDesc

* 过滤表达式字段,

* {uid}会自动替换为当前用户的userId

* {me} main entity 主实体名称

* {me.a} main entity alias 主实体别名

* 格式如:

*

*

userId = {uid}

*

(userId = {uid} AND authType > 3)

*

((userId = {uid} AND authType) > 3 OR (dept in (select dept from depts where manager.id = {uid})))

*

*/

private String exps;

/**

* codeShowName

* 规则说明

*/

private String ruleComment;

}

看完这个结构,基本能够理解设计的思路了。数据结构中保存如下几个字段:

角色列表:需要使用此规则的角色,可以多个,使用英文逗号隔开。

实体列表:对应的规则应用的实体(这里指的是表结构中的表名,可能你的实体是驼峰而数据库是蛇形,所以这里要放蛇形那个),可以多个,使用英文逗号隔开。

表达式:表达式就是数据权限控制的核心了。简单的说这里的表达式就是一段SQL语句,其中设置了一些可替换值,底层会用对应运行时的变量替换对应内容,从而达到增加条件的效果。

规则说明:单纯的一个说明字段。

核心流程

系统启动时,首先从数据库加载出所有的规则。底层利用插件机制来拦截所有的查询语句,进入查询拦截方法后,首先根据当前用户的权限列表筛选出PermissionRule列表,然后循环列表中的规则,对语句中符合实体列表的表进行条件增加,最终生成处理后的SQL语句,退出拦截器,Mybatis执行处理后SQL并返回结果。

讲完PermissionRule,再来看看PermissionHelper,首先是头:

@Intercepts({@Signature(type = Executor.class, method = "update", args = {MappedStatement.class, Object.class}),

@Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})})

public class PermissionHelper implements Interceptor {

}

头部只是标准的Mybatis拦截器写法,注解中的Signature决定了你的代码对哪些方法拦截,update实际上针对修改(Update)、删除(Delete)生效,query是对查询(Select)生效。

下面给出针对Select注入查询条件限制的完整代码:

private String processSelectSql(String sql, List rules, UserDefaultZimpl principal) {

try {

String replaceSql = null;

Select select = (Select) CCJSqlParserUtil.parse(sql);

PlainSelect selectBody = (PlainSelect) select.getSelectBody();

String mainTable = null;

if (selectBody.getFromItem() instanceof Table) {

mainTable = ((Table) selectBody.getFromItem()).getName().replace("`", "");

} else if (selectBody.getFromItem() instanceof SubSelect) {

replaceSql = processSelectSql(((SubSelect) selectBody.getFromItem()).getSelectBody().toString(), rules, principal);

}

if (!ValidUtil.isEmpty(replaceSql)) {

sql = sql.replace(((SubSelect) selectBody.getFromItem()).getSelectBody().toString(), replaceSql);

}

String mainTableAlias = mainTable;

try {

mainTableAlias = selectBody.getFromItem().getAlias().getName();

} catch (Exception e) {

log.debug("当前sql中, " + mainTable + " 没有设置别名");

}

String condExpr = null;

PermissionRule realRuls = null;

for (PermissionRule rule :

rules) {

for (Object roleStr :

principal.getRoles()) {

if (rule.getRoles().indexOf("," + roleStr + ",") != -1) {

if (rule.getFromEntity().indexOf("," + mainTable + ",") != -1) {

// 若主表匹配规则主体,则直接使用本规则

realRuls = rule;

condExpr = rule.getExps().replace("{uid}", UserDefaultUtil.getUserId().toString()).replace("{bid}", UserDefaultUtil.getBusinessId().toString()).replace("{me}", mainTable).replace("{me.a}", mainTableAlias);

if (selectBody.getWhere() == null) {

selectBody.setWhere(CCJSqlParserUtil.parseCondExpression(condExpr));

} else {

AndExpression and = new AndExpression(selectBody.getWhere(), CCJSqlParserUtil.parseCondExpression(condExpr));

selectBody.setWhere(and);

}

}

try {

String joinTable = null;

String joinTableAlias = null;

for (Join j :

selectBody.getJoins()) {

if (rule.getFromEntity().indexOf("," + ((Table) j.getRightItem()).getName() + ",") != -1) {

// 当主表不能匹配时,匹配所有join,使用符合条件的第一个表的规则。

realRuls = rule;

joinTable = ((Table) j.getRightItem()).getName();

joinTableAlias = j.getRightItem().getAlias().getName();

condExpr = rule.getExps().replace("{uid}", UserDefaultUtil.getUserId().toString()).replace("{bid}", UserDefaultUtil.getBusinessId().toString()).replace("{me}", joinTable).replace("{me.a}", joinTableAlias);

if (j.getOnExpression() == null) {

j.setOnExpression(CCJSqlParserUtil.parseCondExpression(condExpr));

} else {

AndExpression and = new AndExpression(j.getOnExpression(), CCJSqlParserUtil.parseCondExpression(condExpr));

j.setOnExpression(and);

}

}

}

} catch (Exception e) {

log.debug("当前sql没有join的部分!");

}

}

}

}

if (realRuls == null) return sql; // 没有合适规则直接退出。

if (sql.indexOf("limit ?,?") != -1 && select.toString().indexOf("LIMIT ? OFFSET ?") != -1) {

sql = select.toString().replace("LIMIT ? OFFSET ?", "limit ?,?");

} else {

sql = select.toString();

}

} catch (JSQLParserException e) {

log.error("change sql error .", e);

}

return sql;

}

重点思路

重点其实就在于Sql的解析和条件注入,使用开源项目JSqlParser。

解析出MainTable和JoinTable。from之后跟着的称为MainTable,join之后跟着的称为JoinTable。这两个就是我们PermissionRule需要匹配的表名,PermissionRule::fromEntity字段。

解析出MainTable的where和JoinTable的on后面的条件。使用and连接原本的条件和待注入的条件,PermissionRule::exps字段。

使用当前登录的用户信息(放在缓存中),替换条件表达式中的值。

某些情况需要忽略权限,可以考虑使用ThreadLocal(单机)/Redis(集群)来控制。

结束语

想要达到无感知的数据权限控制,只有机制控制这么一条路。本文选择的是通过底层拦截Sql语句,并且针对对应表注入条件语句这么一种做法。应该是非常经济的做法,只是基于文本处理,不会给系统带来太大的负担,而且能够达到理想中的效果。大家也可以提出其他的见解和思路。

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

java 数据 权限_Java如何利用Mybatis进行数据权限控制详解相关推荐

  1. java构造方法重载_Java 重载、重写、构造函数的实例详解

    Java 重载.重写.构造函数的实例详解 方法重写 1.重写只能出现在继承关系之中.当一个类继承它的父类方法时,都有机会重写该父类的方法.一个特例是父类的方法被标识为final.重写的主要优点是能够定 ...

  2. java内存 海子_Java虚拟机:JVM内存模型和volatile详解

    JVM内存模型和volatile详解 Java内存模型 随着计算机的CPU的飞速发展,CPU的运算能力已经远远超出了从主内存(运行内存)中读取的数据的能力,为了解决这个问题,CPU厂商设计出了CPU内 ...

  3. java 组合对象_Java 中组合模型之对象结构模式的详解

    Java 中组合模型之对象结构模式的详解 一.意图 将对象组合成树形结构以表示"部分-整体"的层次结构.Composite使得用户对单个对象和组合对象的使用具有一致性. 二.适用性 ...

  4. java jstack使用_JAVA语言之JVM 中jstack命令使用详解

    本文主要向大家介绍了JAVA语言之JVM 中jstack命令使用详解,通过具体的内容向大家展示,希望对大家学习JAVA语言有所帮助. Java程序问题定位时线程栈信息是一个重要线索,如下: " ...

  5. java unit包_Java接入UNIT文本对话处理源码详解

    应邀一位网友的想法,想实现调用UNIT接口,实现文字对话功能,特整理一下内容分享给大家. 此功能对于大神来说非常简单,但是对于新手理解代码处理逻辑,并且如何解析UNIT返回参数的处理,还是有一定的帮助 ...

  6. java运行原理_Java程序的加载与运行原理详解

    Java程序的运行包括两个非常重要的阶段: 一.编译阶段 第一步:程序员需要在计算机硬盘中任意位置创建一个.java扩展名的文件,该文件被称为 java源文件,源文件当中编写的是java源代码/源程序 ...

  7. java封装数组_Java封装数组之动态数组实现方法详解

    本文实例讲述了Java封装数组之动态数组实现方法.分享给大家供大家参考,具体如下: 前言:在此之前,我们封装的数组属于静态数组,也即数组空间固定长度,对于固定长度的数组当元素超过容量时会报数组空间不足 ...

  8. java游戏代码_Java与Kotlin系列文章之性能问题详解

    作者丨Jakub Anioła 译者丨姜雨生 策划丨田晓旭 随着对 Kotlin 越来越深入的了解,我发现市面上关于 Kotlin 方面,比较深入的资料几乎是 0,所以我决定,将 Kotlin 各个方 ...

  9. java 数组求和_java 实现随机数组输出及求和实例详解

    导读热词 java 实现随机数组输出及求和 问题描述: 随机生成10个数,填充一个数组,然后用消息框显示数组内容,接着计算数组元素的和,将结果也显示在消息框中. 设计思路: 使用java.util中的 ...

最新文章

  1. source命令与 .命令
  2. Gray Code LeetCode 89
  3. OPenGL模板缓冲区示例程序
  4. IISExpress Log 文件路径
  5. spring+websocket综合(springMVC+spring+MyBatis这是SSM框架和websocket集成技术)
  6. 【JavaWeb】XML和Jsoup的相关内容
  7. HTML time元素
  8. Docker上安装运行Hbase
  9. TensorFlow-JS教程 一、Node.js 设置
  10. AST语法结构树初学者完整教程
  11. PyTorch学习—16.PyTorch中hook函数
  12. mongodb 如何更改端口号_Centos7配置MongoDB以及端口修改
  13. Power Tools for TFS 11 Beta
  14. Linux修改文件编码格式
  15. java 分号作用_java枚举类型中分号的用法
  16. 谷歌浏览器手势插件-双击关闭标签等多种国人手势
  17. ANE for Android Setp by Step
  18. u-boot器件驱动模型(DeviceDrivers)之链接器的秘密
  19. 公司企业邮箱账号格式怎么填?
  20. Stockfolio 1.5 特别版 Mac 实时股票行情炒股软件

热门文章

  1. python 曲线拟合_Python实现数学模型(插值、拟合和微分方程)
  2. 局部钩子能防全局钩子吗_Vue你真的熟吗?来回答这几个问题试试
  3. 代理服务器之正向代理和反向代理
  4. 【学习笔记】操作系统之哲学原理
  5. 【过程记录 】windows和ubuntu两台电脑局域网进行socket通信收发数据和传输文件
  6. Ubuntu修改root密码以及允许证书和密码登陆
  7. spark sql 1.2.0 测试
  8. Objective-C中@property的所有属性详解
  9. IOS开发调用系统相机和打开闪光灯
  10. 求数组中元素为另外两个元素和的最大元素