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

本文涉及的产品
应用实时监控服务-用户体验监控,每月100OCU免费额度
应用实时监控服务-可观测链路OpenTelemetry版,每月50GB免费额度
应用实时监控服务-应用监控,每月50GB免费额度
简介: 项目使用mybaits-plus,所以在mybaits-plus的基础上增加数据权限的过滤mybaits-plus自带数据权限支持,但由于系统数据权限相对复杂,通过查看文档发现好像并不适用,且原项目版本低,所以最终还是通过自己的方式实现

前言

项目使用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

Snipaste_2023-10-09_12-42-27.png

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)
目录
相关文章
|
7月前
|
Java 数据库连接 mybatis
Mybatis Plus保存数据返回主键id
Mybatis Plus保存数据返回主键id
294 1
|
3月前
|
Java 数据库连接 数据库
mybatis查询数据,返回的对象少了一个字段
mybatis查询数据,返回的对象少了一个字段
226 8
|
2月前
|
SQL JSON Java
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
这篇文章介绍了如何在Spring Boot项目中整合MyBatis和PageHelper进行分页操作,并且集成Swagger2来生成API文档,同时定义了统一的数据返回格式和请求模块。
65 1
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
|
4月前
|
Java 数据库连接 测试技术
SpringBoot 3.3.2 + ShardingSphere 5.5 + Mybatis-plus:轻松搞定数据加解密,支持字段级!
【8月更文挑战第30天】在数据驱动的时代,数据的安全性显得尤为重要。特别是在涉及用户隐私或敏感信息的应用中,如何确保数据在存储和传输过程中的安全性成为了开发者必须面对的问题。今天,我们将围绕SpringBoot 3.3.2、ShardingSphere 5.5以及Mybatis-plus的组合,探讨如何轻松实现数据的字段级加解密,为数据安全保驾护航。
318 1
|
4月前
|
SQL 关系型数据库 MySQL
解决:Mybatis-plus向数据库插入数据的时候 报You have an error in your SQL syntax
该博客文章讨论了在使用Mybatis-Plus向数据库插入数据时遇到的一个常见问题:SQL语法错误。作者发现错误是由于数据库字段中使用了MySQL的关键字,导致SQL语句执行失败。解决方法是将这些关键字替换为其他字段名称,以避免语法错误。文章通过截图展示了具体的操作步骤。
|
4月前
|
SQL Java 关系型数据库
MyBatis-Plus 分页魅力绽放!紧跟技术热点,带你领略数据分页的高效与便捷
【8月更文挑战第29天】在 Java 开发中,数据处理至关重要,尤其在大量数据查询与展示时,分页功能尤为重要。MyBatis-Plus 作为一款强大的持久层框架,提供了便捷高效的分页解决方案。通过封装数据库分页查询语句,开发者能轻松实现分页功能。在实际应用中,只需创建 `Page` 对象并设置页码和每页条数,再通过 `QueryWrapper` 构建查询条件,调用 `selectPage` 方法即可完成分页查询。MyBatis-Plus 不仅生成分页 SQL 语句,还自动处理参数合法性检查,并支持条件查询和排序等功能,极大地提升了系统性能和稳定性。
65 0
|
4月前
|
存储 SQL Java
MyBatis batchInsert 批量插入数据
MyBatis batchInsert 批量插入数据
93 0
|
4月前
|
前端开发 JavaScript Java
解决springboot+vue+mybatis中,将后台数据分页显示在前台,并且根据页码自动跳转对应页码信息
该博客文章讲述了如何在Spring Boot + Vue + MyBatis的项目中实现后台数据的分页查询,并在前端进行显示和页码跳转,包括后端的分页查询实现、前端与后端的交互以及使用Element UI进行分页展示的方法。
|
5月前
|
Java 数据库连接 Maven
文本,使用SpringBoot工程创建一个Mybatis-plus项目,Mybatis-plus在编写数据层接口,用extends BaseMapper<User>继承实体类
文本,使用SpringBoot工程创建一个Mybatis-plus项目,Mybatis-plus在编写数据层接口,用extends BaseMapper<User>继承实体类
|
6月前
|
Java 关系型数据库 MySQL
Mybatis入门之在基于Springboot的框架下拿到MySQL中数据
Mybatis入门之在基于Springboot的框架下拿到MySQL中数据
52 4