设为首页收藏本站
网站公告 | 这是第一条公告
     

 找回密码
 立即注册
缓存时间00 现在时间00 缓存数据 对自己狠一点,逼自己努力,再过几年你将会感谢今天发狠的自己、恨透今天懒惰自卑的自己。晚安!

对自己狠一点,逼自己努力,再过几年你将会感谢今天发狠的自己、恨透今天懒惰自卑的自己。晚安!

查看: 114|回复: 0

基于Spring Security的动态权限系统设计与实现

[复制链接]

  离线 

TA的专栏

  • 打卡等级:热心大叔
  • 打卡总天数:227
  • 打卡月天数:0
  • 打卡总奖励:3346
  • 最近打卡:2025-11-12 20:18:24
等级头衔

等級:晓枫资讯-上等兵

在线时间
0 小时

积分成就
威望
0
贡献
398
主题
372
精华
0
金钱
4550
积分
830
注册时间
2023-1-8
最后登录
2025-11-12

发表于 2025-8-27 20:31:46 | 显示全部楼层 |阅读模式

本文介绍一个基于 Spring Boot 2.7.18 和 Spring Security 实现的权限系统,支持接口级权限控制,支持权限点的动态配置与加载。

技术栈

  • Spring Boot 2.7.18
  • Spring Security
  • MyBatis Plus(用于持久化)
  • MySQL

核心表结构设计

权限点表auth_permission_point

用于定义所有权限点(如

  1. user:create
复制代码
,
  1. user:update
复制代码
):

字段名类型说明
idbigint主键
codevarchar权限点编码(唯一)
namevarchar权限点名称
typevarchar类型(操作、页面、字段等)
resourcevarchar资源模块标识
actionvarchar操作标识
remarkvarchar备注说明

角色表auth_role

字段名类型说明
idbigint主键
role_codevarchar角色编码
namevarchar角色名称
is_builtinboolean是否为系统内置角色
enabledboolean是否启用

用户角色关联表auth_user_role

字段名类型说明
idbigint主键
user_idvarchar用户唯一 ID
role_codevarchar关联角色编码

角色权限点关联表auth_role_permission_point

字段名类型说明
idbigint主键
role_codevarchar角色编码
permission_codevarchar权限点编码

接口权限映射表auth_url_permission_point

字段名类型说明
idbigint主键
urlvarchar接口路径
methodvarchar请求方法(GET/POST/PUT/DELETE)
permission_codevarchar所需权限点编码

✅ 每个接口可以绑定多个权限点,满足任意一个即视为拥有权限。

权限系统运行机制

1. 动态加载权限点

实现自定义

  1. FilterInvocationSecurityMetadataSource
复制代码
,在系统启动和权限点发生变更时,自动扫描
  1. auth_url_permission_point
复制代码
表,将 URL、METHOD -> 权限点集合 的映射加载至内存。

  1. @Component
  2. @RequiredArgsConstructor
  3. public class CustomSecurityMetadataSource implements FilterInvocationSecurityMetadataSource {
  4. private final AntPathMatcher pathMatcher = new AntPathMatcher();
  5. @Resource
  6. private UrlPermissionMappingService urlPermissionMappingService;
  7. // TODO: 后期可替换为 Redis 或数据库缓存
  8. private static final Map<String, List<PermissionExpressionConfigAttribute>> URL_PERMISSION_MAP = new ConcurrentHashMap<>();
  9. private volatile Map<String, List<PermissionExpressionConfigAttribute>> permissionMap = new ConcurrentHashMap<>();
  10. static {
  11. // 示例数据,正式请从数据库加载
  12. URL_PERMISSION_MAP.put("/api/user/**", List.of(new PermissionExpressionConfigAttribute("user:query")));
  13. URL_PERMISSION_MAP.put("/api/user/updatePassword", List.of(new PermissionExpressionConfigAttribute("user:updatePassword")));
  14. }
  15. @PostConstruct
  16. public void init() {
  17. // 启动时加载一次
  18. reload();
  19. }
  20. public void reload() {
  21. Map<String, List<PermissionExpressionConfigAttribute>> newMap = new HashMap<>();
  22. for (UrlPermissionMapping mapping : urlPermissionMappingService.loadAllUrlPermissionMappings()) {
  23. newMap.computeIfAbsent(mapping.getUrlPattern(), k -> new ArrayList<>())
  24. .add(new PermissionExpressionConfigAttribute(mapping.getPermissionCode()));
  25. }
  26. this.permissionMap = newMap;
  27. }
  28. @Override
  29. public Collection<ConfigAttribute> getAttributes(Object object) {
  30. String requestPath = ((FilterInvocation) object).getRequest().getRequestURI();
  31. // 先尝试精确匹配
  32. List<PermissionExpressionConfigAttribute> exact = permissionMap.get(requestPath);
  33. if (exact != null) {
  34. return new HashSet<>(exact);
  35. }
  36. // 再尝试通配匹配
  37. for (Map.Entry<String, List<PermissionExpressionConfigAttribute>> entry : permissionMap.entrySet()) {
  38. if (pathMatcher.match(entry.getKey(), requestPath)) {
  39. return new HashSet<>(entry.getValue());
  40. }
  41. }
  42. return null;
  43. }
  44. @Override
  45. public Collection<ConfigAttribute> getAllConfigAttributes() {
  46. return URL_PERMISSION_MAP.values().stream()
  47. .flatMap(List::stream)
  48. .collect(Collectors.toSet());
  49. }
  50. @Override
  51. public boolean supports(Class<?> clazz) {
  52. return FilterInvocation.class.isAssignableFrom(clazz);
  53. }
  54. }
复制代码

2. 动态权限校验

实现

  1. AccessDecisionVoter<FilterInvocation>
复制代码
,对每个请求:

    1. SecurityMetadataSource
    复制代码
    拿到该接口需要的权限点
    1. Authentication#getAuthorities()
    复制代码
    拿到用户权限点集合
  • 判断是否命中
  1. public class PermissionExpressionVoter implements AccessDecisionVoter<FilterInvocation> {
  2. @Override
  3. public int vote(Authentication authentication, FilterInvocation filterInvocation,
  4. Collection<ConfigAttribute> attributes) {
  5. Assert.notNull(authentication, "authentication must not be null");
  6. Assert.notNull(filterInvocation, "filterInvocation must not be null");
  7. Assert.notNull(attributes, "attributes must not be null");
  8. Set<String> requiredExpressions = findConfigAttribute(attributes);
  9. // 获取当前登录用户拥有的权限点表达式
  10. Set<String> userPermissions = authentication.getAuthorities().stream()
  11. .map(GrantedAuthority::getAuthority)
  12. .collect(Collectors.toSet());
  13. if (CollectionUtils.isEmpty(requiredExpressions)) {
  14. // 如果没有定义表达式,弃权,交给下一个 voter
  15. log.trace("Abstained since did not find a config attribute of instance WebExpressionConfigAttribute");
  16. return ACCESS_ABSTAIN;
  17. }
  18. for (String required : requiredExpressions) {
  19. if (userPermissions.contains(required)) {
  20. return ACCESS_GRANTED;
  21. }
  22. }
  23. log.warn("权限校验失败: 当前用户权限 = {}, 资源需要权限 = {}", userPermissions, requiredExpressions);
  24. return ACCESS_DENIED;
  25. }
  26. private Set<String> findConfigAttribute(Collection<ConfigAttribute> attributes) {
  27. // 取出当前资源对应的权限表达式
  28. return attributes.stream()
  29. .filter(attribute -> attribute instanceof PermissionExpressionConfigAttribute)
  30. .map(ConfigAttribute::getAttribute)
  31. .collect(Collectors.toSet());
  32. }
  33. @Override
  34. public boolean supports(ConfigAttribute attribute) {
  35. return attribute instanceof PermissionExpressionConfigAttribute;
  36. }
  37. @Override
  38. public boolean supports(Class<?> clazz) {
  39. return FilterInvocation.class.isAssignableFrom(clazz);
  40. }
  41. }
复制代码
  1. ☑️ 未配置权限点的接口可设置默认放行,也可以走 fallback 权限点逻辑。
复制代码

总结

该系统实现了:

  • 权限点粒度统一、接口权限与角色权限解耦
  • 接口权限点支持动态注册与配置
  • 权限控制基于 Spring Security 标准扩展机制,具备良好扩展性

TODO(可选增强)

  • 支持权限表达式解析(如
    1. @hasAny('user:create', 'admin')
    复制代码
  • 支持字段级、按钮级权限点
  • 权限点变更自动刷新缓存
  • 提供权限控制台(前端联动)

到此这篇关于基于Spring Security的动态权限系统设计与实现的文章就介绍到这了,更多相关SpringSecurity动态权限内容请搜索晓枫资讯以前的文章或继续浏览下面的相关文章希望大家以后多多支持晓枫资讯!


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
晓枫资讯-科技资讯社区-免责声明
免责声明:以上内容为本网站转自其它媒体,相关信息仅为传递更多信息之目的,不代表本网观点,亦不代表本网站赞同其观点或证实其内容的真实性。
      1、注册用户在本社区发表、转载的任何作品仅代表其个人观点,不代表本社区认同其观点。
      2、管理员及版主有权在不事先通知或不经作者准许的情况下删除其在本社区所发表的文章。
      3、本社区的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,举报反馈:点击这里给我发消息进行删除处理。
      4、本社区一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
      5、以上声明内容的最终解释权归《晓枫资讯-科技资讯社区》所有。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~
严禁发布广告,淫秽、色情、赌博、暴力、凶杀、恐怖、间谍及其他违反国家法律法规的内容。!晓枫资讯-社区
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|晓枫资讯--科技资讯社区 本站已运行

CopyRight © 2022-2025 晓枫资讯--科技资讯社区 ( BBS.yzwlo.com ) . All Rights Reserved .

晓枫资讯--科技资讯社区

本站内容由用户自主分享和转载自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。

如有侵权、违反国家法律政策行为,请联系我们,我们会第一时间及时清除和处理! 举报反馈邮箱:点击这里给我发消息

Powered by Discuz! X3.5

快速回复 返回顶部 返回列表