news 2026/4/15 3:20:42

Ruoyi框架 | 扩展部门数据权限实现

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Ruoyi框架 | 扩展部门数据权限实现

一、背景与目标

在若依框架原有DataScope的基础上,实现一套独立的、基于部门层级的数据权限过滤机制,用于按组织结构灵活控制数据可见范围。

设计目标

  • 不依赖角色、不判断是否管理员

  • 通过注解参数动态控制数据范围

  • 支持:

    • 是否包含本部门
    • 向上查询 N 级部门
    • 向下查询 N 级部门 / 所有子部门
  • 与若依原有BaseEntity + params + MyBatis XML机制完全兼容


二、核心设计思路

1. 技术方案

  • 使用AOP + 自定义注解拦截查询方法

  • 在方法执行前:

    • 根据当前用户部门 ID
    • 动态拼接部门过滤 SQL
    • 注入到BaseEntity.params.dataScope
  • Mapper XML 中通过${params.dataScope}拼接 WHERE 条件

2. 依赖表结构(sys_dept)

/* by 01130.hk - online tools website : 01130.hk/zh/endecodejs.html */ dept_id 部门ID parent_id 父部门ID ancestors 祖先路径,如:0,1,3,10

三、自定义注解:ExtendedDataScope

/* by 01130.hk - online tools website : 01130.hk/zh/endecodejs.html */ @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface ExtendedDataScope { // 部门表别名(必填) String deptAlias() default ""; // 用户表别名(预留扩展) String userAlias() default ""; // 权限类型(当前主要使用 dept) String type() default "dept"; // 向上级部门层数(0 = 不包含) int upLevel() default 0; // 向下级部门层数(0 = 不包含,999 = 所有子级) int downLevel() default 0; // 是否包含本部门 boolean includeSelf() default true; }

四、AOP 实现要点

1. 切面职责

  • 拦截所有标注@ExtendedDataScope的方法
  • 清空历史 dataScope,防止 SQL 注入
  • 生成部门层级 SQL
  • 写入BaseEntity.params.dataScope

2. 核心处理流程

@Before ├─ clearDataScope() ├─ 获取当前用户 ├─ 读取注解参数 ├─ 构建部门范围 SQL └─ 写入 params.dataScope
点击查看代码
@Aspect @Component public class ExtendedDataScopeAspect { /** * 参数 key(和若依一致) */ public static final String DATA_SCOPE = "dataScope"; @Before("@annotation(dataScope)") public void doBefore(JoinPoint joinPoint, ExtendedDataScope dataScope) { clearDataScope(joinPoint); handleDataScope(joinPoint, dataScope); } private void handleDataScope(JoinPoint joinPoint, ExtendedDataScope dataScope) { LoginUser loginUser = SecurityUtils.getLoginUser(); if (loginUser == null) { return; } String deptAlias = dataScope.deptAlias(); if (StringUtils.isBlank(deptAlias)) { return; } String sql = buildDeptScopeSql(loginUser.getDeptId(), deptAlias, dataScope); if (StringUtils.isBlank(sql)) { return; } Object params = joinPoint.getArgs()[0]; if (params instanceof BaseEntity) { BaseEntity baseEntity = (BaseEntity) params; baseEntity.getParams().put(DATA_SCOPE, " AND (" + sql + ")"); } } /** * 构建部门层级 SQL */ private String buildDeptScopeSql(Long deptId, String deptAlias, ExtendedDataScope scope) { List<String> conditions = new ArrayList<>(); /* ========== 本部门 ========== */ if (scope.includeSelf()) { conditions.add(deptAlias + ".dept_id = " + deptId); } /* ========== 向上 ========== */ if (scope.upLevel() > 0) { // ancestors 形如:0,1,3,10 // 向上 N 级:取 ancestors 中倒数 N 位 conditions.add(buildUpDeptSql(deptAlias, deptId, scope.upLevel())); } /* ========== 向下 ========== */ if (scope.downLevel() > 0) { if (scope.downLevel() >= 999) { // 所有子级 conditions.add( deptAlias + ".dept_id IN (" + "SELECT dept_id FROM sys_dept " + "WHERE find_in_set(" + deptId + ", ancestors)" + ")" ); } else { conditions.add(buildDownDeptSql(deptAlias, deptId, scope.downLevel())); } } return String.join(" OR ", conditions); } /** * 向上 N 级部门 */ private String buildUpDeptSql(String deptAlias, Long deptId, int upLevel) { // 使用子查询,取 ancestors 中的上级 return deptAlias + ".dept_id IN (" + " SELECT t.dept_id FROM sys_dept t " + " WHERE t.dept_id IN ( " + " SELECT SUBSTRING_INDEX(" + " SUBSTRING_INDEX(d.ancestors, ',', -( " + upLevel + " + 1)), ',', 1" + " ) FROM sys_dept d WHERE d.dept_id = " + deptId + " )" + ")"; } /** * 向下 N 级部门 */ private String buildDownDeptSql(String deptAlias, Long deptId, int downLevel) { // ancestors 深度控制(当前 depth + N) return deptAlias + ".dept_id IN (" + " SELECT d.dept_id FROM sys_dept d " + " WHERE find_in_set(" + deptId + ", d.ancestors) " + " AND (LENGTH(d.ancestors) - LENGTH(REPLACE(d.ancestors, ',', ''))) <= " + " ( " + " SELECT (LENGTH(ancestors) - LENGTH(REPLACE(ancestors, ',', ''))) + " + downLevel + " FROM sys_dept WHERE dept_id = " + deptId + " )" + ")"; } /** * 清空 dataScope,防止 SQL 注入 */ private void clearDataScope(JoinPoint joinPoint) { Object params = joinPoint.getArgs()[0]; if (params instanceof BaseEntity) { ((BaseEntity) params).getParams().put(DATA_SCOPE, ""); } } }

五、部门层级 SQL 构建规则

1. 本部门

d.dept_id = {currentDeptId}

includeSelf = true控制


2. 向上 N 级部门(upLevel)

原理:

  • 利用ancestors字段
  • 从 ancestors 中向前截取 N 个父级

示意 SQL:

d.dept_id IN ( SELECT SUBSTRING_INDEX( SUBSTRING_INDEX(ancestors, ',', -(N + 1)), ',', 1 ) FROM sys_dept WHERE dept_id = 当前部门ID )

3. 向下 N 级部门(downLevel)

3.1 所有子级(downLevel = 999)
d.dept_id IN ( SELECT dept_id FROM sys_dept WHERE find_in_set(当前部门ID, ancestors) )
3.2 限定层级子部门

思路:

  • 计算 ancestors 的层级深度(逗号个数)
  • 控制最大深度 = 当前深度 + N
d.dept_id IN ( SELECT d.dept_id FROM sys_dept d WHERE find_in_set(当前部门ID, d.ancestors) AND 层级深度(d) <= 当前层级 + N )

六、Mapper XML 使用方式

<select id="selectList" resultType="xxx"> SELECT * FROM biz_table t LEFT JOIN sys_dept d ON t.dept_id = d.dept_id <where> 1 = 1 ${params.dataScope} </where> </select>

说明:

  • ${params.dataScope}必须保留
  • AOP 动态注入AND ( ... )

七、使用示例

@ExtendedDataScope( deptAlias = "d", includeSelf = true, upLevel = 1, downLevel = 2 ) public List<SysDept> selectDeptList(SysDept dept) { return deptMapper.selectDeptList(dept); }

含义说明

参数含义
includeSelf包含本部门
upLevel=1包含上一级部门
downLevel=2包含下两级部门
downLevel=999包含所有子部门

八、方案特点总结

  • ✅ 与若依原生 DataScope 解耦
  • ✅ 仅依赖部门层级,不依赖角色权限
  • ✅ 控制粒度细,适合复杂组织结构
  • ✅ 非侵入式,Mapper 无需改动
  • ✅ 特别适合安全监管 / GIS / 组织树场景

九、可扩展方向(后续优化)

  • 使用 MySQL 8 / PostgreSQL 的WITH RECURSIVE优化层级查询
  • 部门层级缓存(Redis)减少子查询
  • 扩展到:部门 + 用户混合数据权限
  • 支持多部门归属(兼职部门)
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/10 13:41:21

YOLOFuse pycharm模板代码配置提升编码效率

YOLOFuse&#xff1a;PyCharm 模板配置驱动下的高效多模态目标检测实践 在智能安防、自动驾驶和夜间监控等现实场景中&#xff0c;单一可见光摄像头在低光照、雾霾或遮挡环境下常常“力不从心”——行人模糊、车辆轮廓消失、关键目标漏检频发。这时候&#xff0c;红外&#xff…

作者头像 李华
网站建设 2026/4/12 20:22:17

YOLOFuse html meta标签优化SEO搜索引擎收录

YOLOFuse&#xff1a;轻量级多模态目标检测的工程实践与部署优化 在智能安防、夜间巡检和自动驾驶等实际场景中&#xff0c;单一可见光摄像头在低光照、雾霾或伪装干扰下常常“力不从心”。行人可能隐匿于黑暗角落&#xff0c;车辆轮廓在浓雾中模糊不清——这些挑战暴露了传统R…

作者头像 李华
网站建设 2026/4/12 11:53:45

仁怀商家轻松出圈!触福 SR 视频 AI 助力本地生意宣传

在仁怀做本地生意&#xff0c;不管是开酒坊、餐馆&#xff0c;还是做特产店、民宿&#xff0c;都想靠短视频吸引客户 —— 毕竟短视频能直观展示产品和服务&#xff0c;传播力极强。可很多商家都面临同一个难题&#xff1a;没专业团队、没拍摄技术&#xff0c;想做短视频却无从…

作者头像 李华
网站建设 2026/4/14 20:27:06

YOLOFuse ultraiso注册码最新版安全性评估

YOLOFuse 多模态检测系统安全与部署实践深度解析 在夜间监控、边境安防或消防救援等关键场景中&#xff0c;单一可见光摄像头常常“失明”——低光照、烟雾遮挡让传统目标检测算法频频漏检。为突破这一瓶颈&#xff0c;RGB-红外双模态融合检测正成为新一代智能感知的核心技术。…

作者头像 李华
网站建设 2026/4/6 3:06:42

处理音视频业务

目录前言一、音视频业务的本质&#xff08;先把“是什么”讲清楚&#xff09;二、音视频业务的完整生命周期&#xff08;核心主线&#xff09;1、采集&#xff08;Capture&#xff09;2、预处理&#xff08;Processing&#xff09;3、编码&#xff08;Encoding&#xff09;4、传…

作者头像 李华
网站建设 2026/4/12 8:03:53

RBAC角色权限控制系统:多用户协作场景下的必要配置

RBAC角色权限控制系统&#xff1a;多用户协作场景下的必要配置 在当今的AI开发环境中&#xff0c;一个团队共享同一套大模型训练与部署平台已是常态。设想这样一个场景&#xff1a;一名实习生误点了“全量微调”按钮&#xff0c;瞬间占用了整个H100集群&#xff1b;或者某位研究…

作者头像 李华