SpringSecurity中WebSecurity和HttpSecurity的关系-SpringSecurity中利用JWT退出登录-《Java笔记》

admin 2025-10-19 04:35:09 编程 来源:ZONE.CI 全球网 0 阅读模式

Java Spring Spring Security在Spring Security中用JWT做退出登录的时无法获取当前用户,导致无法证明“我就是要退出的那个我”,业务失败!经过一番排查找到了原因。

Session会话

之所以要说Session会话,是因为Spring Security默认配置就是有会话的,所以当登录以后Session就会由服务端保持直到退出登录。只要Session保持住,请求只要进入服务器就可以从ServletRequest中获取到当前的HttpSession,然后会根据HttpSession来加载当前的SecurityContext。相关的逻辑在Spring Security默认的过滤器SecurityContextPersistenceFilter中,有兴趣可以看相关的源码。而且默认情况下SecurityContextPersistenceFilter的优先级是高于退出过滤器LogoutFilter的,所以能够保证有Session会话的情况下退出一定能够获取当前用户。

无Session会话

使用了JWT后,每次请求都要携带Bearer Token并且被专门的过滤器拦截解析之后才能将用户认证信息保存到SecurityContext中去。Token认证实现JwtAuthenticationFilter,相关逻辑为:

  1. // 当token匹配
  2. if (jwtToken.equals(accessToken)) {
  3. // 解析 权限集合 这里
  4. JSONArray jsonArray = jsonObject.getJSONArray("roles");
  5. List<String> roles = jsonArray.toList(String.class);
  6. String[] roleArr = roles.toArray(new String[0]);
  7. List<GrantedAuthority> authorities = AuthorityUtils.createAuthorityList(roleArr);
  8. User user = new User(username, "[PROTECTED]", authorities);
  9. // 构建用户认证token
  10. UsernamePasswordAuthenticationToken usernamePasswordAuthenticationToken = new UsernamePasswordAuthenticationToken(user, null, authorities);
  11. usernamePasswordAuthenticationToken.setDetails(new WebAuthenticationDetailsSource().buildDetails(request));
  12. // 放入安全上下文中
  13. SecurityContextHolder.getContext().setAuthentication(usernamePasswordAuthenticationToken);
  14. } else {
  15. // token 不匹配
  16. if (log.isDebugEnabled()){
  17. log.debug("token : {} is not in matched", jwtToken);
  18. }
  19. throw new BadCredentialsException("token is not matched");
  20. }

为什么退出登录无法获取当前用户

分析了两种情况下用户认证信息的安全上下文配置后,回到问题的本身。来看看为什么用JWT会出现无法获取当前认证信息的原因。在HttpSecurity中,配置JwtAuthenticationFilter的顺序的:

  1. httpSecurity.addFilterBefore(jwtAuthenticationFilter, UsernamePasswordAuthenticationFilter.class)

再看看Spring Security过滤器排序图:image.pngSpring Security过滤器排序也就说LogoutFilter执行退出的时候,JWT还没有被JwtAuthenticationFilter拦截,当然无法获取当前认证上下文SecurityContext

解决方法

解决方法就是必须在LogoutFilter执行前去解析JWT并将成功认证的信息存到SecurityContext。可以这样配置:

  1. httpSecurity.addFilterBefore(jwtAuthenticationFilter, LogoutFilter.class)

这样问题就解决了,只要实现把当前JWT作废掉就退出登录了。

以太坊cppgolang区别 编程

以太坊cppgolang区别

以太坊是一种去中心化的开源平台,它采用智能合约技术,旨在构建和运行不受干扰的分布式应用程序。作为目前最受欢迎的区块链平台之一,以太坊提供了多种编程语言的支持,其
progolang 编程

progolang

Go语言(Golang)是由Google开发的一门静态类型编程语言。作为一名专业的Golang开发者,我深知这门语言的优势和特点。在本文中,我将介绍Golang
golangn个发送者 编程

golangn个发送者

Golang是一种开源的编程语言,由Google团队开发,旨在提高程序的并发性和简化软件开发过程。在Go语言中,有时需要向多个接收者发送信息。本文将介绍如何在G
golang技能图谱 编程

golang技能图谱

从互联网行业的快速发展到人工智能技术的日益成熟,各种编程语言也应运而生。而在这众多的编程语言中,Golang(即Go)作为一门强大且高效的开发语言备受关注。Go
评论:0   参与:  6