经过我翻来覆去的思想斗争了一个月,最后做出了一个明智的决定

开发 开发工具
最近写了几个Spring Boot组件,项目用什么功能就引入对应的依赖,配置配置就能使用,香的很!那么Spring Security能不能也弄成模块化,简单配置一下就可以用上呢?

[[395754]]

最近写了几个Spring Boot组件,项目用什么功能就引入对应的依赖,配置配置就能使用,香的很!那么Spring Security能不能也弄成模块化,简单配置一下就可以用上呢?JWT得有,RBAC动态权限更得有!花了小半天就写了个组件,用了一个月感觉还不错。是我一个人爽?还是放出来让大家一起爽?经过我翻来覆去的思想斗争了一个月,最后做出了一个明智的决定,放出来让想直接上手的同学直接使用。源码地址就在下面:

https://gitee.com/felord/security-enhance-spring-boot

用法

集成

这就是一个Spring Boot Starter,你自己打包、安装。然后引用到项目:

  1. <dependency> 
  2.             <groupId>cn.felord.security</groupId> 
  3.             <artifactId>security-enhance-spring-boot-starter</artifactId> 
  4.             <version>${version}</version> 
  5.         </dependency> 

另外你需要集成Spring Cache,比如Redis Cache:

  1. <dependency> 
  2.       <groupId>org.springframework.boot</groupId> 
  3.       <artifactId>spring-boot-starter-cache</artifactId> 
  4.   </dependency> 
  5.   <dependency> 
  6.       <groupId>org.springframework.boot</groupId> 
  7.       <artifactId>spring-boot-starter-data-redis</artifactId> 
  8.   </dependency> 
  9.   <dependency> 
  10.       <groupId>org.apache.commons</groupId> 
  11.       <artifactId>commons-pool2</artifactId> 
  12.   </dependency> 

JWT会被缓存到以usrTkn为key的缓存中,如果你想定制的话,自行实现一个JwtTokenStorage并注入Spring IoC就可以覆盖下面的配置了:

  1. @Bean 
  2.     @ConditionalOnMissingBean 
  3.     public JwtTokenStorage jwtTokenStorage() { 
  4.         return new SpringCacheJwtTokenStorage(); 
  5.     } 

你应该去了解如何自定义Spring Cache的过期时间。

数据库表设计

然后是数据库表设计,这里简单点弄个RBAC的设计,仅供参考,你可以根据你们的业务改良。

用户表:

user_id username password
1312434534 felord {noop}12345

角色表:

role_id role_name role_code
12343667867 管理员 ADMIN

用户角色关联表:

user_role_id user_id role_id
12354657777 1312434534 12343667867

一个用户可以持有多个角色,一个角色在一个用户持有的角色集合中是唯一的。

资源表:

resources_id resources_name resource_pattern method
12543667867 根据ID获取商品 /goods/{goodsId} GET

资源其实就是我们写的Spring MVC接口,这里支持ANT风格,但是尽量具体,为了灵活性考虑不推荐使用通配符。

角色资源表:

role_res_id role_id resources_id
4545466445 12343667867 12543667867

一个资源可以关联多个角色,一个角色不能重复持有一个资源。

实现UserDetailsService

实现用户加载服务接口UserDetailsService是Spring Security开发的必要步骤,跟我以前的教程差不多。

  1. @Override 
  2. public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException { 
  3.  
  4.     UserInfo userInfo = this.lambdaQuery() 
  5.             .eq(UserInfo::getUsername, username).one(); 
  6.  
  7.     if (Objects.isNull(userInfo)) { 
  8.         throw new UsernameNotFoundException("用户:" + username + " 不存在"); 
  9.     } 
  10.  
  11.     String userId = userInfo.getUserId(); 
  12.     boolean enabled = userInfo.getEnabled(); 
  13.  
  14.     Set<String> roles = iUserRoleService.getRolesByUserId(userId); 
  15.     roles.add(“"ANONYMOUS"”); 
  16.     Set<GrantedAuthority> roleSet = roles.stream() 
  17.             .map(role -> new SimpleGrantedAuthority("ROLE_" + role)) 
  18.             .collect(Collectors.toSet()); 
  19.     return new SecureUser(userId, 
  20.             username, 
  21.             userInfo.getSecret(), 
  22.             enabled, 
  23.             enabled, 
  24.             enabled, 
  25.             enabled, 
  26.             roleSet); 

这里要说一下里面为啥要内置一个ANONYMOUS角色给用户。如果希望特定的资源对用户全量开放,可配置对应的权限角色编码为ANONYMOUS。当某个资源的角色编码为ANONYMOUS时,即使不携带Token也可以访问。一般情况下匿名能访问的资源不匿名一定能访问,当然你如果不希望这样的规则存在干掉就是了。

查询用户的权限集

实现用户角色权限方法Function

配置

最后就是配置了,跟我以前教程中的配置几乎一样,application.yaml的配置为:

  1. # jwt 配置 
  2. jwt: 
  3.   cert-info: 
  4.    # keytool 密钥的 alias  
  5.     alias: felord 
  6.     # 密匙密码 
  7.     key-password: i6x123akg15v13 
  8.     # 路径 这里是在resources 包下 
  9.     cert-location: jwt.jks 
  10.   claims: 
  11.     # jwt iss 字段值 
  12.     issuer: https://felord.cn 
  13.     # sub 字段 
  14.     subject: all 
  15.     # 过期秒数 
  16.     expires-at: 604800 

最后别忘记弄个配置类并标记@EnableSpringSecurity以启用配置:

  1. @EnableSpringSecurity 
  2. @Configuration(proxyBeanMethods = false
  3. public class SecurityConfiguration { 
  4.  
  5.     /** 
  6.      * Function function
  7.      * 
  8.      * @param resourcesService the resources service 
  9.      * @return the function 
  10.      */ 
  11.     @Bean 
  12.     Function<Set<String>, Set<AntPathRequestMatcher>> function(IResourcesService resourcesService){ 
  13.         return resourcesService::matchers; 
  14.     } 
  15.      
  16.     @Bean 
  17.     UserDetailsService userDetailsService(IUserInfoService userInfoService){ 
  18.         return userInfoService::loadUserByUsername;  
  19.     } 
  20.  

记得使用@EnableCaching开启并配置缓存。

使用

登录接口

  1. POST /login?username=felord&password=12345 HTTP/1.1 
  2. Host: localhost:8080 

然后会返回一对JWT,返回包含两个token主体

  • accessToken 用来日常进行请求鉴权,有过期时间。
  • refreshToken 当accessToken过期失效时,用来刷新accessToken。

结构为:

  1.   "accessToken": { 
  2.     "tokenValue"""
  3.     "issuedAt": { 
  4.       "epochSecond": 1616827822, 
  5.       "nano": 393000000 
  6.     }, 
  7.     "expiresAt": { 
  8.       "epochSecond": 1616831422, 
  9.       "nano": 393000000 
  10.     }, 
  11.     "tokenType": { 
  12.       "value""Bearer" 
  13.     }, 
  14.     "scopes": [ 
  15.       "ROLE_ADMIN"
  16.       "ROLE_ANONYMOUS" 
  17.     ] 
  18.   }, 
  19.   "refreshToken": { 
  20.     "tokenValue"""
  21.     "issuedAt": { 
  22.       "epochSecond": 1616827822, 
  23.       "nano": 393000000 
  24.     }, 
  25.     "expiresAt"null 
  26.   }, 
  27.   "additionalParameters": {} 

调用根据ID获取商品接口时加入Token:

  1. GET /goods/234355451 HTTP/1.1 
  2. Host: localhost:8080 
  3. Authorization: Bearer eyJraWQImFsZyI6IlJTMjU2In0.eyJzdWIiOiJ1NzgsImlhdCI6MTYxNjkxODk3OCwianRpIjoiNThlOTQktNGVlYzc3MDU0ZDk3In0.ZQcN0FX7_taohqPiC1KnoF7 

本文转载自微信公众号「码农小胖哥」,可以通过以下二维码关注。转载本文请联系码农小胖哥公众号。

 

责任编辑:武晓燕 来源: 码农小胖哥
相关推荐

2021-11-18 07:29:56

AST操作ESBuild

2018-01-10 12:09:12

Android开发程序员

2018-09-04 15:15:56

程序员开发时间

2021-10-28 05:39:14

Windows 10操作系统微软

2013-05-27 09:47:33

Java开发Java跨平台

2009-11-23 08:52:02

Windows 7首月销量

2019-10-08 11:07:55

Python 开发编程语言

2016-01-11 19:38:51

七牛

2019-04-01 14:17:36

kotlin开发Java

2012-08-31 16:40:24

Mac操作系统

2021-07-20 08:57:26

滴滴上市网络安全审查

2010-09-14 16:09:49

sql日期函数

2022-12-05 18:17:06

技术

2012-12-20 10:18:10

Windows 8

2009-02-16 09:15:49

苹果乔布斯CEO

2013-08-12 16:35:22

2024-08-05 09:25:00

AI测评

2019-03-11 08:36:00

Office 应用微软

2021-01-05 15:15:23

人工智能
点赞
收藏

51CTO技术栈公众号