当前位置: 首页 > 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)

在这里插入图片描述

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

相关文章:

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

&#x1f389;&#x1f389;欢迎来到我的CSDN主页&#xff01;&#x1f389;&#x1f389; &#x1f3c5;我是尘缘&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; &#x1f449;点击这里&#xff0c;就可以查看我的主页啦&#xff01;&#x1f447;&#x…...

医学专题--多组学在药物治疗靶点筛选中的研究思路

研究背景 药物靶点&#xff08;drug target&#xff09;&#xff1a;是指药物在体内的作用结合位点&#xff0c;包括基因位点、受体、酶、离子通路、核酸等生物大分子&#xff0c;靶向药物从单靶点药物到多靶点药物都在不断进行临床探索。小分子多靶点药物因其多靶向性、高效率…...

搜索与图论总结

算法博文DFSDFS&#xff08;深度优先&#xff09;-CSDN博客BFSBFS(宽度优先)&#xff08;最短路&#xff09;-CSDN博客有向图与无向图有向图与无向图&#xff08;邻接表&#xff09;_无向图有向图邻接表_人生导师yxc的博客-CSDN博客拓扑排序topsort(拓扑排序)-CSDN博客最短路最…...

lv8 嵌入式开发-网络编程开发 15I/O多路复用及select函数

目录 1 I/O多路复用 1.1 select函数及其他接口相关介绍 1.2 原TCP—socket示例&#xff1a; 1.3 实现select函数TCP—socket示例&#xff1a; 2 练习 1 I/O多路复用 多路复用的实现方式 1.1 select函数及其他接口相关介绍 int select(int nfds, fd_set *readfds, fd_set…...

阿里云 linux tomcat 无法访问方法

1、阿里云放行tomcat端口 例如7077端口号 2、linux 命令行防火墙 设置端口打开 以下命令查看是否开启指定端口 firewall-cmd --list-ports以下命令添加指定端口让防火墙放行 firewall-cmd --zonepublic --add-port3306/tcp --permanent以下命令重新启动防火墙 systemctl re…...

公园视频监控系统如何改造?人工智能又能提供哪些帮助?

近日合肥市骆岗公园宣布正式开园&#xff0c;作为目前世界最大的城市公园&#xff0c;占地12.7万平方公里&#xff0c;如此壮观宏伟的建设&#xff0c;也吸引到了不少市民进行参观打卡。不管大型小型&#xff0c;城市里的公园都是随处可见的&#xff0c;那么&#xff0c;公园安…...

面试算法19:最多删除一个字符得到回文

题目 给定一个字符串&#xff0c;请判断如果最多从字符串中删除一个字符能不能得到一个回文字符串。例如&#xff0c;如果输入字符串"abca"&#xff0c;由于删除字符’b’或’c’就能得到一个回文字符串&#xff0c;因此输出为true。 分析 本题还是从字符串的两端…...

H5+Css3文本溢出添加省略号(包括插件)

一、单行 溢出隐藏 添加省略号 p{overflow: hidden;text-overflow:ellipsis;white-space: nowrap; }二、多行 溢出隐藏 省略号 p{display: -webkit-box;-webkit-box-orient: vertical;/*设置省略号在容器第四行文本后*/-webkit-line-clamp: 4; overflow: hidden; }局限性&…...

将休眠镜像文件hiberfil.sys移动到D盘,可以减少C盘好几个G的空间占用

hiberfil.sys是什么文件&#xff1f; 该文件是开启休眠功能后&#xff0c;系统自动生成的内存镜像文件&#xff0c;以便我们唤醒电脑之后可以快速开启程序。 1、首先打开电脑&#xff0c;使用“windowsR”组合键进入运行&#xff0c;输入“regedit”命令。 2、在注册表编辑器中…...

YTM32的模数转换器ADC外设模块详解

文章目录 简介原理与机制ADC转换器的上下电和省电模式ADC转换结果和FIFOADC转换队列的工作模式ADC转换器的触发信号ADC转换器的看门狗中断事件和DMA 应用要点&#xff08;软件&#xff09;总结参考文献 简介 YTM32的ADC转换器外设最多可以集成32个输入通道&#xff0c;最高12b…...

前端vue学习笔记——Vuex

1.概念 在Vue中实现集中式状态&#xff08;数据&#xff09;管理的一个Vue插件&#xff0c;对vue应用中多个组件的共享状态进行集中式的管理&#xff08;读/写&#xff09;&#xff0c;也是一种组件间通信的方式&#xff0c;且适用于任意组件间通信。 2.何时使用&#xff1f;…...

7个在Github上的flutter开源程序

阅读大量代码是提高开发技能的最佳方法之一。该开源项目是了解最佳实践、编码风格和许多其他主题的最佳场所。 软件开发最受欢迎的领域之一是跨平台移动应用程序开发。Flutter 是您可以使用的最流行的跨平台移动应用程序开发工具之一。今天&#xff0c;我们将了解 7 个开源 Flu…...

计算机基础

分值&#xff1a;3-7 1. 计算机系统概述 2. 计算机组成结构 3. 存储结构 3.1. 层次化存储结构 一般用什么调什么&#xff0c;局部性原理 内存和外存可以统称为虚拟存储器 我们可以操作哪些&#xff1a;操作外存、内存、CPU寄存器。Cache具有透明性。 3.2. Cache Cache的功…...

Oracle-ASM实例communication error问题处理

问题背景&#xff1a; Oracle数据库日志出现大量的WARNING: ASM communication error: op 0 state 0x0 (15055)错误 问题分析: 首先检查ASM实例的状态,尝试通过sqlplus / as sysasm连接asm实例&#xff0c;出现Connected to an idle instance连接asm实例失败 检查ASM实例的后台…...

gin路由相关方法

c.Request.URL.Path 拿到请求的路径 package mainimport ( "fmt" "github.com/gin-gonic/gin" "net/http")//路由重定向&#xff0c;请求转发&#xff0c;ANY &#xff0c;NoRoute&#xff0c;路由组func main() { r : gin.Default() // -------…...

vue项目 Editor.md使用示例

简介 Editor.md 支持“标准” Markdown / CommonMark 和 Github 风格的语法&#xff0c;也可变身为代码编辑器&#xff1b; 支持实时预览、图片&#xff08;跨域&#xff09;上传、预格式文本/代码/表格插入、代码折叠、搜索替换、只读模式、自定义样式主题和多语言语法高亮等…...

12.3 实现模拟鼠标录制回放

本节将向读者介绍如何使用键盘鼠标操控模拟技术&#xff0c;键盘鼠标操控模拟技术是一种非常实用的技术&#xff0c;可以自动化执行一些重复性的任务&#xff0c;提高工作效率&#xff0c;在Windows系统下&#xff0c;通过使用各种键盘鼠标控制函数实现动态捕捉和模拟特定功能的…...

【计算机网络-自顶向下方法】应用层(SMTP、POP3、DNS)

目录 1. Electronic Mail电子邮件应用画像1.1 电子邮件系统1.2 邮件报文格式1.3 邮件访问 2. DNS&#xff08;Domain Name System&#xff09;2.1 DNS提供的服务2.2 DNS工作机理2.3 DNS资源记录2.4 DNS协议&#xff0c;报文2.5 小结 1. Electronic Mail 电子邮件应用画像 应用…...

【Pm4py第八讲】关于Statistics

本节用于介绍pm4py中的统计函数&#xff0c;包括统计轨迹变体、案例持续时间、案例到达时间等。 1.函数概述 本次主要介绍Pm4py中一些常见的统计函数&#xff0c;总览如下表&#xff1a; 函数名说明pm4py.stats.get_start_activities()从事件日志中获取开始活动。pm4py.stats.…...

【Azure 架构师学习笔记】-Azure Data Factory (5) --Data Flow

本文属于【Azure 架构师学习笔记】系列。 本文属于【Azure Data Factory】系列。 接上文【Azure 架构师学习笔记】-Azure Data Factory (4)-触发器详解-事件触发器 前言 Azure Data Factory&#xff0c; ADF 是微软Azure 的ETL 首选服务之一&#xff0c; 是Azure data platfor…...

《Qt C++ 与 OpenCV:解锁视频播放程序设计的奥秘》

引言:探索视频播放程序设计之旅 在当今数字化时代,多媒体应用已渗透到我们生活的方方面面,从日常的视频娱乐到专业的视频监控、视频会议系统,视频播放程序作为多媒体应用的核心组成部分,扮演着至关重要的角色。无论是在个人电脑、移动设备还是智能电视等平台上,用户都期望…...

uni-app学习笔记二十二---使用vite.config.js全局导入常用依赖

在前面的练习中&#xff0c;每个页面需要使用ref&#xff0c;onShow等生命周期钩子函数时都需要像下面这样导入 import {onMounted, ref} from "vue" 如果不想每个页面都导入&#xff0c;需要使用node.js命令npm安装unplugin-auto-import npm install unplugin-au…...

镜像里切换为普通用户

如果你登录远程虚拟机默认就是 root 用户&#xff0c;但你不希望用 root 权限运行 ns-3&#xff08;这是对的&#xff0c;ns3 工具会拒绝 root&#xff09;&#xff0c;你可以按以下方法创建一个 非 root 用户账号 并切换到它运行 ns-3。 一次性解决方案&#xff1a;创建非 roo…...

视频字幕质量评估的大规模细粒度基准

大家读完觉得有帮助记得关注和点赞&#xff01;&#xff01;&#xff01; 摘要 视频字幕在文本到视频生成任务中起着至关重要的作用&#xff0c;因为它们的质量直接影响所生成视频的语义连贯性和视觉保真度。尽管大型视觉-语言模型&#xff08;VLMs&#xff09;在字幕生成方面…...

什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南

文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果![在这里插入图片描述](https://i-blog.csdnimg.cn/direct/55aefaea8a9f477e86d065227851fe3d.pn…...

企业如何增强终端安全?

在数字化转型加速的今天&#xff0c;企业的业务运行越来越依赖于终端设备。从员工的笔记本电脑、智能手机&#xff0c;到工厂里的物联网设备、智能传感器&#xff0c;这些终端构成了企业与外部世界连接的 “神经末梢”。然而&#xff0c;随着远程办公的常态化和设备接入的爆炸式…...

《C++ 模板》

目录 函数模板 类模板 非类型模板参数 模板特化 函数模板特化 类模板的特化 模板&#xff0c;就像一个模具&#xff0c;里面可以将不同类型的材料做成一个形状&#xff0c;其分为函数模板和类模板。 函数模板 函数模板可以简化函数重载的代码。格式&#xff1a;templa…...

如何应对敏捷转型中的团队阻力

应对敏捷转型中的团队阻力需要明确沟通敏捷转型目的、提升团队参与感、提供充分的培训与支持、逐步推进敏捷实践、建立清晰的奖励和反馈机制。其中&#xff0c;明确沟通敏捷转型目的尤为关键&#xff0c;团队成员只有清晰理解转型背后的原因和利益&#xff0c;才能降低对变化的…...

DBLP数据库是什么?

DBLP&#xff08;Digital Bibliography & Library Project&#xff09;Computer Science Bibliography是全球著名的计算机科学出版物的开放书目数据库。DBLP所收录的期刊和会议论文质量较高&#xff0c;数据库文献更新速度很快&#xff0c;很好地反映了国际计算机科学学术研…...

c# 局部函数 定义、功能与示例

C# 局部函数&#xff1a;定义、功能与示例 1. 定义与功能 局部函数&#xff08;Local Function&#xff09;是嵌套在另一个方法内部的私有方法&#xff0c;仅在包含它的方法内可见。 • 作用&#xff1a;封装仅用于当前方法的逻辑&#xff0c;避免污染类作用域&#xff0c;提升…...