当前位置: 首页 > news >正文

MyBatis-Plus演绎:数据权限控制,优雅至极!

🎉🎉欢迎来到我的CSDN主页!🎉🎉
🏅我是尘缘,一个在CSDN分享笔记的博主。📚📚
👉点击这里,就可以查看我的主页啦!👇👇
尘缘的个人主页
🎁如果感觉还不错的话请给我点赞吧!🎁🎁
💖期待你的加入,一起学习,一起进步!💖💖

在这里插入图片描述

目录

    • 前言
    • 1 数据范围
    • 2 修改SQL

前言

项目使用mybaits-plus,所以在mybaits-plus的基础上增加数据权限的过滤

mybaits-plus自带数据权限支持,但由于系统数据权限相对复杂,通过查看文档发现好像并不适用,且原项目版本低,所以最终还是通过自己的方式实现

1 数据范围

我们系统相对复杂,比如可以按机构/用户等多种维度过滤,并且可以指定全局和某个特定接口的过滤方式

其实数据范围过滤落地也不过是:数据表的某字段限制在一个范围内,即sql中添加column in (1,2,3...)

不管怎么说第一步都是要获取用户的数据范围,比如某用户的数据范围为机构id为(1,2,3)下的数据,那么先要获取(1,2,3)

首先建立一个类来存储用户的数据范围,由于数据权限是多维度的,所以存储的是一个Map<String, List<String>>结构

public class GerneralScope extends HashMap<String,  List<String>> {
}

存储的数据类似如下

{"org_id": [1,2,3], // 机构id"user_id": [], // 为空代表不过滤用户id"xxx_id": [4,8] // 其它为敌
}

使用ThreadLocal进行暂存,并在拼接sql时使用,这样可以避免代码侵入

public class ScopeDataHolder {public final static ThreadLocal<GerneralScope> SCOPE_DATA = new ThreadLocal<>();public static GerneralScope get() {GerneralScope gerneralScope = SCOPE_DATA.get();SCOPE_DATA.remove(); // 获取一次就删除return gerneralScope;}public static void set(GerneralScope data) {SCOPE_DATA.set(data);}
}

数据结构准备好了,接下来就是获取当前用户数据范围存入ScopeDataHolder,采用注解+AOP的方式避免代码侵入

新增注解@Scope

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Scope {ApiType value() default ApiType.COMMON;
}

其中加一个参数value用来区分不同接口,即可实现特定接口单独过滤方式

AOP获取并设置数据范围

@Component
public class ScopeAspect {@Pointcut("@annotation(com.xxx.Scope)")public void injectScope() {}/*** 注入数据权限* @param joinPoint* @return*/@Before("injectScope()")public void around(JoinPoint joinPoint) {Scope annotation = ((MethodSignature) joinPoint.getSignature()).getMethod().getAnnotation(Scope.class);GerneralScope userScopeData = getCurrentUserScopeData(annotation.value()); // 数据库获取当前用户+当前接口的数据范围ScopeDataHolder.set(userScopeData); // 存入ThreadLocal}
}

到此,零侵入代码情况下,通过ThreadLocal暂存了用户所配的数据范围

2 修改SQL

获取到了用户的数据范围,下一步就是在查询中加入数据范围的过滤,即修改sql

刚开始本来打算用mybaits-plus的自定义拦截器实现sql的修改,后来发现有很多坑,主要是当sql中存在left join且分页时,mybaits-plus的分页器在count查询时自动把没有查询条件的left join表去掉,如果限定数据范围的字段刚好在join表上,就会导致错误

所以最终没有采用拦截器,而是采取重写mybaits-plus的QueryWrapper类来实现,代码如下

public class ScopeQueryWrapper<T> extends QueryWrapper<T> {private final GerneralScope queryScope;public ScopeQueryWrapper() {this.queryScope = ScopeDataHolder.get(); // 从ThreadLocal获取数据范围if (this.queryScope==null) {throw new IllegalStateException();}}/*** 过滤需要筛选的字段* @param column*/@SuppressWarnings("unchecked")public void scope(ScopeEnum type, SFunction<T, ?> column) {List<String> els = queryScope.get(type.getValue());if (els!=null && els.size()!=0) {lambda().in(column, els);}}/*** 过滤需要筛选的字段* @param fieldName*/@SuppressWarnings("unchecked")public void scope(ScopeEnum type, String fieldName) {List<String> els = queryScope.get(type.getValue());if (els!=null && els.size()!=0) {in(fieldName, els);}}
}

这样只需在查询层把QueryWrapper替换为ScopeQueryWrapper,并使用scopeFilter方法来指定界限字段即可,写法如下

public Page<User> page(UserQuery query) {Page<User> page = new Page<>(query.getPageNum(), query.getPageSize());ScopeQueryWrapper<User> wrapper = new ScopeQueryWrapper<>();if (query.getName()!=null) {wrapper.lambda().like(User::getName,query.getName());}/** 数据权限 start **/wrapper.scope(ScopeEnum.orgId, User:getOrgId); // 指定机构id字段wrapper.scope(ScopeEnum.userId, "user.id"); // 指定用户id字段,字符串方式可以防止join字段重名...省略其它过滤条件/** 数据权限 end**/wrapper.lambda().orderByDesc(User::getId);Page<User> result = page(page, wrapper);return result;
}

如上,需要指定具体需要过滤的字段,由于是多维度,可能会指定很多,ScopeEnum即各维度的枚举,scope方法中的getValue获取到的即用户设置范围数据的key
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
ScopeEnum

scope接受字符串形式,可以避免join时字段有歧义

以上代码出现了代码的侵入,但自认为可以接受,如果不需要多维度可以进一步简略

最终,执行的sql大体如下

select * from user where name like "%pq%" and org_id in (1,2,3) and user.id in (4,8,10)

在这里插入图片描述

到这里我的分享就结束了,欢迎到评论区探讨交流!!
💖如果觉得有用的话还请点个赞吧 💖

http://www.lryc.cn/news/186056.html

相关文章:

  • 医学专题--多组学在药物治疗靶点筛选中的研究思路
  • 搜索与图论总结
  • lv8 嵌入式开发-网络编程开发 15I/O多路复用及select函数
  • 阿里云 linux tomcat 无法访问方法
  • 公园视频监控系统如何改造?人工智能又能提供哪些帮助?
  • 面试算法19:最多删除一个字符得到回文
  • H5+Css3文本溢出添加省略号(包括插件)
  • 将休眠镜像文件hiberfil.sys移动到D盘,可以减少C盘好几个G的空间占用
  • YTM32的模数转换器ADC外设模块详解
  • 前端vue学习笔记——Vuex
  • 7个在Github上的flutter开源程序
  • 计算机基础
  • Oracle-ASM实例communication error问题处理
  • gin路由相关方法
  • vue项目 Editor.md使用示例
  • 12.3 实现模拟鼠标录制回放
  • 【计算机网络-自顶向下方法】应用层(SMTP、POP3、DNS)
  • 【Pm4py第八讲】关于Statistics
  • 【Azure 架构师学习笔记】-Azure Data Factory (5) --Data Flow
  • uniapp之ios开发及支付整体流程爬坑记录
  • AutoDL百川大模型体验
  • 蓝桥杯每日一题2023.10.8
  • jmeter,性能测试,Locust
  • opencv图像的直方图,二维直方图,直方图均衡化
  • c++中的map和set
  • Swagger使用详解
  • ToBeWritten之车联网安全中常见的TOP 10漏洞
  • 软考-密码学概述
  • windows 2003、2008远程直接关闭远程后设置自动注销会话
  • iOS BUG UIView转UIImage模糊失真