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

news2025/1/10 20:36:20

🎉🎉欢迎来到我的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.coloradmin.cn/o/1071812.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MFC扩展库BCGControlBar Pro v33.6 - 网格、报表控件功能升级

BCGControlBar库拥有500多个经过全面设计、测试和充分记录的MFC扩展类。 我们的组件可以轻松地集成到您的应用程序中&#xff0c;并为您节省数百个开发和调试时间。 BCGControlBar专业版 v33.6已正式发布了&#xff0c;此版本包含了对图表组件的改进、带隐藏标签的单类功能区栏…

Tomcat 9.0.41在IDEA中乱码问题(IntelliJ IDEA 2022.1.3版本)

1. 乱码的产生是由于编码和解码的编码表不一致引起的。 2. 排查乱码原因 2.1 在idea中启动Tomcat时控制台乱码排查 Tomcat输出日志乱码: 首先查看IDEA控制台&#xff0c;检查发现默认编码是GBK。 再查看Tomcat日志&#xff08;conf文件下logging.properties&#xff09;的默…

GaussDB数据库SQL系列-动态语句

目录 一、前言 二、构建动态SQL语句的基本步骤和注意事项 1、基本步骤 2、主要事项 三、GaussDB中执行动态查询语句&#xff08;示例&#xff09; 1、方式一&#xff1a;EXECUTE IMMEDIATE 2、方式二&#xff1a;OPEN FOR 四、GaussDB中的动态非查询语句&#xff08;示…

MathType2024苹果版数学公式编辑器

MathType苹果2024版是一款备受好评的数学公式编辑器&#xff0c;很多网友在编辑文本时对各种数学符号非常挠头&#xff0c;因为都不太好找&#xff0c;而这款软件能够在各种文档中加入复杂的数学公式和符号&#xff0c;并且与常见的文字处理软件和演示程序配合使用&#xff0c;…

华为云云耀云服务器L实例评测 | 实例使用教学之软件安装:华为云云耀云服务器环境下安装 RabbitMQ 集群

华为云云耀云服务器L实例评测 &#xff5c; 实例使用教学之软件安装&#xff1a;华为云云耀云服务器环境下安装 RabbitMQ 集群 介绍华为云云耀云服务器 华为云云耀云服务器 &#xff08;目前已经全新升级为 华为云云耀云服务器L实例&#xff09; 华为云云耀云服务器是什么华为云…

差分构造法推广:arc166_d

https://atcoder.jp/contests/arc166/tasks/arc166_d 首先肯定是这样子放&#xff1a; 考虑相邻之间的差&#xff0c;本质就是橙色区间减蓝色区间数量 区间数量越少显然越优&#xff0c;所以我们要么保留橙区间&#xff0c;要么保留紫区间&#xff0c;然后两两匹配 #include…

【linux进程(三)】进程有哪些状态?--Linux下常见的三种进程状态

&#x1f493;博主CSDN主页:杭电码农-NEO&#x1f493;   ⏩专栏分类:Linux从入门到精通⏪   &#x1f69a;代码仓库:NEO的学习日记&#x1f69a;   &#x1f339;关注我&#x1faf5;带你学更多操作系统知识   &#x1f51d;&#x1f51d; Linux进程 1. 前言2. 操作系统…

记录极致CMS非富文本标签调用不改变格式

问题 在前台如何输出这三行是换行的 前台调用{$jz[hhl]}就变成这样了“这是第一行这是第二行这是第三行” 除了富文本还有没有什么可以实现这样的呢&#xff1f; 方法 {foreach explode("\n",$jz[hhl]) as $v} {if($v)} {$v} {/if} {/foreach}

国内就能使用的chatgpt网页版,包含AIGC应用工具

Chatgpt的出现在多个领域带来了重要的影响。它能够显著提高我们的工作效率&#xff0c;无论是编写文案代码还是回答常见问题&#xff0c;都能在短时间内完成任务。通过Chatgpt&#xff0c;我们能够迅速获取所需答案。随着人工智能技术的不断发展&#xff0c;相信在未来AI能够带…

这个物业管理技巧,你一定不能错过!

在现代社会中&#xff0c;安全是至关重要的问题&#xff0c;特别是在多层建筑、住宅社区和商业物业等繁忙场所。火灾是一种常见的灾害&#xff0c;可能对人员生命和财产造成严重威胁。 为了及早检测火灾风险并采取迅速有效的应对措施&#xff0c;烟感监控系统应运而生&#xff…

Linux搭建我的世界MC服务器 【Minecraft外网联机教程】

目录 前言 1. 安装JAVA 2. MCSManager安装 3.局域网访问MCSM 4.创建我的世界服务器 5.局域网联机测试 6.安装cpolar内网穿透 7. 配置公网访问地址 8.远程联机测试 9. 配置固定远程联机端口地址 9.1 保留一个固定tcp地址 9.2 配置固定公网TCP地址 9.3 使用固定公网…

laravel9 from验证,中文提示

用的软件包官网文档地址&#xff1a;Getting Started | Laravel Lang 由于用的php8.0,所以 1.composer require --dev laravel-lang/common:v2.0.0 2.php artisan lang:add zh_CN 3. 完成&#xff0c;看下效果

Altium design 经验补充2

1. 改变现有板子的尺寸 1&#xff09;选择要更改的PCB&#xff0c;点击LS选择mechanical(机械层)&#xff0c; 2) 点击工具栏---放置---线条&#xff0c;线条最终必须是全闭合的。 3&#xff09;板子大小画完以后&#xff0c;放置原点&#xff0c;在工具栏点击 edit--origin--…

ArcGIS Pro地图可视化—双变量关系映射

原址链接ArcGIS Pro地图可视化—双变量关系映射https://mp.weixin.qq.com/s/g-pPBHPXMOEF5NHm06JcrA 这个方法很早很早以前就有了&#xff0c;可能大家早就知道了&#xff0c;可我昨天刚看到这个东西 https://en.wikipedia.org/wiki/Multivariate_map 像是上图&#xff0c;美国…

笔记本CPU温度多少正常?这些知识不可忽视!

“我的笔记本电脑使用的时候我总感觉好像很热&#xff0c;不知道正常使用电脑时&#xff0c;CPU温度应该是多少呢&#xff1f;有没有大佬可以讲解一下呀&#xff1f;” 随着电脑的使用越来越日常&#xff0c;关于电脑的相关问题也越来越受关注。今天小编将给大家分享一些关于电…

HTML+CSS跑马灯/流光字效果实例代码

简介 HTMLCSS跑马灯/流光字效果实例代码 演示 代码 css <style type"text/css">body {margin: 0;padding: 0;background-color: #2f2424;}div {margin: 400px auto;font-size: 40px;text-align: center;}p {margin: 0;background: -webkit-linear-gradient(…

面试字节跳动国际电商部,凉了。。。

大家好&#xff0c;我是程序员小灰。 最近&#xff0c;小灰的一位读者在秋招提前批的时候&#xff0c;面试了字节的国际电商部门。这场面试考察得非常全面&#xff0c;涉及到了项目经验、数据结构、设计模式、数据库、Spring等等。 为了能帮助到更多程序员朋友&#xff0c;这位…

如何恢复微信过期文件?用这4招,轻松解决!

如今&#xff0c;无论是学习还是工作我们都离不开微信&#xff0c;微信的出现改变了人们的社交方式。也正是因为微信&#xff0c;人们之间的交流变得更加便捷与多样化。 微信有很多方便的功能&#xff0c;我们可以通过微信来传输许多重要的文件。但是有时候可能由于自己的疏忽…

【表情包分享】国庆假期结束了,该上班了,来分享一下你的表情包吧!

文章目录 每日一句正能量前言正文后记 每日一句正能量 国庆结束了&#xff0c;但快乐留下了&#xff1b;长假远走了&#xff0c;但开心到来了&#xff1b;八天过去了&#xff0c;但好运来临了&#xff1b;工作开始了&#xff0c;故祝福来到了。祝&#xff1a;收假快乐&#xff…

Python 接口测试之接口请求方法封装

引言 既然我们接口测试用例写好了&#xff0c;测试数据也拿到了&#xff0c;那么就是模拟调用接口的方法了&#xff0c;方法有get,post,put,delete&#xff0c;具体是选择哪种&#xff0c;我们需要根据不同接口规定好请求方法来调用。为了方便&#xff0c;将四种方法封装到一个…