Spring Boot Security防重登录及在线总数

开发 前端
本篇给大家介绍Spring Boot Security防重登录及在线总数的相关知识,希望对你有所帮助!

[[395788]]

 环境:Spring Boot 2.2.11.RELEASE + JPA2

Security流程处理

Security的核心是Filter,下图是Security的执行流程

详细步骤:

1.1

UsernamePasswordAuthenticationFilter的父类是AbstractAuthenticationProcessingFilter首先执行父类中的doFilter方法。

1.2 执行

UsernamePasswordAuthenticationFilter中的attemptAuthentication方法

这里实例化

UsernamePasswordAuthenticationToken对象存入用户名及密码进行接下来的验证

1.3 进入验证

this.getAuthenticationManager().authenticate(authRequest) 这里使用的是系统提供的ProviderManager对象进行验证

关键是下面的这个for循环

 

这里先判断AuthenticationProvider是否被支持

  1. Class<? extends Authentication> toTest = authentication.getClass(); 

这里的toTest就是

UsernamePasswordAuthenticationFilter类中调用的如下对象

1.4 既然要验证用户名密码,那我们肯定地提供一个AuthenticationProvider对象同时必须还得要支持

UsernamePasswordAuthenticationToken对象类型的。所以我们提供如下一个DaoAuthenticationProvider子类,查看该类


关键在这个父类中,该父类中如下方法:

  1. public boolean supports(Class<?> authentication) { 
  2.         return (UsernamePasswordAuthenticationToken.class 
  3.                 .isAssignableFrom(authentication)); 
  4.     } 

 也就说明我们只需要提供DaoAuthenticationProvider一个子类就能对用户进行验证了。

1.5 自定义DaoAuthenticationProvider子类

  1. @Bean 
  2.     public DaoAuthenticationProvider daoAuthenticationProvider() { 
  3.         DaoAuthenticationProvider daoAuthen = new DaoAuthenticationProvider() ; 
  4.         daoAuthen.setPasswordEncoder(passwordEncoder()); 
  5.         daoAuthen.setUserDetailsService(userDetailsService()); 
  6.         daoAuthen.setHideUserNotFoundExceptions(false) ; 
  7.         return daoAuthen ; 
  8.     } 

 1.6 执行前面for中的如下代码

  1. result = provider.authenticate(authentication); 

这里进入了DaoAuthenticationProvider的父类

AbstractUserDetailsAuthenticationProvider中的authenticate方法

该方法的核心方法

retrieveUser方法在子类DaoAuthenticationProvider中实现

如果这里返回了UserDetails(查询到用户)将进入下一步

1.7 进入密码的验证

这里调用子类DaoAuthenticationProvider的方法

剩下的就是成功后的事件处理,如果有异常进行统一的异常处理

Security登录授权认证

  • 实体类
  1. @Entity 
  2. @Table(name = "T_USERS"
  3. public class Users implements UserDetails, Serializable { 
  4.   private static final long serialVersionUID = 1L; 
  5.     @Id 
  6.   @GeneratedValue(generator = "system-uuid"
  7.   @GenericGenerator(name = "system-uuid", strategy = "uuid"
  8.   private String id ; 
  9.   private String username ; 
  10.   private String password ; 
  •  DAO
  1. public interface UsersRepository extends JpaRepository<Users, String>, JpaSpecificationExecutor<Users> { 
  2.     Users findByUsernameAndPassword(String username, String password) ; 
  3.     Users findByUsername(String username) ; 
  •  Security 配置
  1. @Configuration 
  2. public class SecurityConfig extends WebSecurityConfigurerAdapter { 
  3.      
  4.     @Resource 
  5.     private UsersRepository ur ; 
  6.     @Resource 
  7.     private LogoutSuccessHandler logoutSuccessHandler ; 
  8.      
  9.     @Bean 
  10.     public UserDetailsService userDetailsService() { 
  11.         return username -> { 
  12.             Users user = ur.findByUsername(username) ; 
  13.             if (user == null) { 
  14.                 throw new UsernameNotFoundException("用户名不存在") ; 
  15.             } 
  16.             return user ; 
  17.         }; 
  18.     } 
  19.      
  20.     @Bean 
  21.     public PasswordEncoder passwordEncoder() { 
  22.         return new PasswordEncoder() { 
  23.             @Override 
  24.             public boolean matches(CharSequence rawPassword, String encodedPassword) { 
  25.                 return rawPassword.equals(encodedPassword) ; 
  26.             } 
  27.             @Override 
  28.             public String encode(CharSequence rawPassword) { 
  29.                 return rawPassword.toString() ; 
  30.             } 
  31.         }; 
  32.     } 
  33.      
  34.     @Bean 
  35.     public DaoAuthenticationProvider daoAuthenticationProvider() { 
  36.         DaoAuthenticationProvider daoAuthen = new DaoAuthenticationProvider() ; 
  37.         daoAuthen.setPasswordEncoder(passwordEncoder()); 
  38.         daoAuthen.setUserDetailsService(userDetailsService()); 
  39.         daoAuthen.setHideUserNotFoundExceptions(false) ; 
  40.         return daoAuthen ; 
  41.     } 
  42.      
  43.     @Bean 
  44.     public SessionRegistry sessionRegistry() { 
  45.         return new SessionRegistryImpl() ; 
  46.     } 
  47.      
  48.     // 这个不配置sessionRegistry中的session不失效 
  49.     @Bean 
  50.     public HttpSessionEventPublisher httpSessionEventPublisher() { 
  51.         return new HttpSessionEventPublisher(); 
  52.     } 
  53.      
  54.     @Override 
  55.     protected void configure(HttpSecurity http) throws Exception { 
  56.         http 
  57.             .csrf().disable() 
  58.             .authorizeRequests() 
  59.             .antMatchers("/pos/**"
  60.             .authenticated() 
  61.         .and() 
  62.             .formLogin() 
  63.             .loginPage("/sign/login"
  64.         .and() 
  65.             .logout() 
  66.             .logoutSuccessHandler(logoutSuccessHandler) 
  67.             .logoutUrl("/sign/logout"); 
  68.     // 这里配置最大同用户登录个数 
  69.         http.sessionManagement().maximumSessions(1).expiredUrl("/sign/login?expired").sessionRegistry(sessionRegistry()) ; 
  70.     } 
  71.      
  •  Controller相关接口
  1. @Controller 
  2. public class LoginController { 
  3.      
  4.     @RequestMapping("/sign/login"
  5.     public String login() { 
  6.         return "login" ; 
  7.     } 
  8.      
  9. @RestController 
  10. @RequestMapping("/sign"
  11. public class LogoutController { 
  12.      
  13.     @GetMapping("/logout"
  14.     public Object logout(HttpServletRequest request) { 
  15.         HttpSession session = request.getSession(false); 
  16.         if (session != null) { 
  17.             session.invalidate(); 
  18.         } 
  19.         SecurityContext context = SecurityContextHolder.getContext(); 
  20.         context.setAuthentication(null); 
  21.         SecurityContextHolder.clearContext(); 
  22.         return "success" ; 
  23.     } 
  24.      
  25. @RestController 
  26. @RequestMapping("/pos"
  27. public class PosController { 
  28.      
  29.     @GetMapping(""
  30.     public Object get() { 
  31.         return "pos success" ; 
  32.     } 
  33.      
  34. // 通过下面接口获取在线人数 
  35. @RestController 
  36. @RequestMapping("/sessions"
  37. public class SessionController { 
  38.      
  39.     @Resource 
  40.     private SessionRegistry sessionRegistry ; 
  41.      
  42.     @GetMapping(""
  43.     public Object list() { 
  44.         return sessionRegistry.getAllPrincipals() ; 
  45.     } 
  46.      

 测试:

在chrome浏览器用zs用户登录

用360浏览器也用zs登录

360登录后刷新chrome浏览器

登录已经失效了,配置的最大登录个数也生效了。

完毕!!!

 

责任编辑:姜华 来源: 今日头条
相关推荐

2022-11-26 00:00:02

2022-09-06 08:40:33

应用系统登录方式Spring

2023-12-08 12:12:21

2021-12-28 11:13:05

安全认证 Spring Boot

2021-04-28 06:26:11

Spring Secu功能实现源码分析

2021-03-04 11:50:48

微信Spring Secu登录

2021-05-12 08:32:53

Spring Secu 自定义session

2021-07-13 14:05:37

单点登录页面

2021-01-06 10:09:05

Spring Boothttps sslhttps

2021-04-23 07:33:10

SpringSecurity单元

2021-08-29 18:36:57

项目

2021-06-29 12:27:19

Spring BootCAS 登录

2022-01-26 07:01:00

开源社区项目

2022-02-15 07:35:12

服务器KeycloakOAuth2

2022-06-12 06:45:26

高并发防重

2018-06-27 14:50:06

Cloud StudiSpring Boot应用

2023-10-18 08:12:34

Spring自动配置

2023-11-26 09:10:34

WebSocketgreeting​在线用户

2021-10-14 06:51:54

SpringSecurityJWT

2022-08-30 08:50:07

Spring权限控制
点赞
收藏

51CTO技术栈公众号