IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> security认证流程理解 -> 正文阅读

[Java知识库]security认证流程理解

前言:
spring security的认证流程非常复杂,我也只是简单学习,通过security web项目和security在微服务中的使用,有一些理解,首先需要理解的一点就是我们可以把security看成一个过滤器链,后面我们会看到一些过滤器。

一、用户登录
(一)、UsernamePasswordAuthenticationFilter
用户登录时这个过滤器会起作用,其中有一个方法

@Override
    public Authentication attemptAuthentication(HttpServletRequest request, HttpServletResponse response)

这个方法会根据用户登录时输入的username和password封装成UsernamePasswordAuthenticationToken并生成Authentication并返回,需要注意的是,这个时候Authentication并没有通过认证,也就没有放到security的上下文中,接下来会调用IOC容器中实现UserDetailsService接口的类,也就是我们的自定义类,例如UserDetailsServiceImpl

(二)、UserDetailsService

@Service
public class UserDetailsServiceImpl implements UserDetailsService {

    @Autowired
    private PasswordEncoder passwordEncoder;

    @Override
    public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
        if(!"zhangsan".equals(username)){
            throw new UsernameNotFoundException("用户名不存在");
        }

        String password = passwordEncoder.encode("123456");


        return new User(username,password, AuthorityUtils.commaSeparatedStringToAuthorityList("salary,ROLE_abc,bbb"));
    }
}

这个类里会调用loadUserByUsername方法,通过username查询数据库中是否存在User,如果存在,接下来查询User的权限列表,之后封装成UserDetails并返回,之后security会比对Authentication和UserDetails中的密码是否一致,如果不一致,直接认证失败,如果一致,会把UserDetails中的权限给Authentication,之后把Authentication放入security的上下文,由此就完成了认证和授权。

二、用户登录之后访问
如果是单体web项目,security会把用户的登录信息存储在session中,用户再次登录的时候,直接从session中获取即可,然后进行权限校验即可;
如果是微服务项目,由于采用的是前后端分离,用session存储用户的登录信息非常的不方便,所以我们这里采用基于jwt的方式进行权限认证,登录的逻辑基本不变,唯一的区别就是认证后需要返回给前端token,同时还需要在redis中存储用户的权限信息,之后每次用户访问都需要带上token,用户带着token访问会经过下面的过滤器(需要自己实现并配置到security中生效)。

public class TokenAuthFilter extends BasicAuthenticationFilter {

     private TokenManager tokenManager;
     private RedisTemplate redisTemplate;

     public TokenAuthFilter(AuthenticationManager authenticationManager, TokenManager tokenManager, RedisTemplate redisTemplate) {
         super(authenticationManager);
         this.tokenManager = tokenManager;
         this.redisTemplate = redisTemplate;
     }

     public TokenAuthFilter(AuthenticationManager authenticationManager) {
         super(authenticationManager);
     }

     @Override
     protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain chain) throws IOException, ServletException {
         //获取当前认证成功的用户权限信息
         UsernamePasswordAuthenticationToken authRequest = getAuthention(request);
         //判断如果有权限授权信息,就放到权限上下文中
         if (authRequest!=null){
             SecurityContextHolder.getContext().setAuthentication(authRequest);
         }
         chain.doFilter(request,response);
     }

     public UsernamePasswordAuthenticationToken getAuthention(HttpServletRequest request){
         //从header中获取token
         String token = request.getHeader("token");
         if (token!=null){
             String username = tokenManager.getUserInfoFromToken(token);

             //从redis中获取权限列表信息
             List<String> permissionsValueList = (List<String>) redisTemplate.opsForValue().get(username);
             Collection<GrantedAuthority> authority =  new ArrayList<>();
             for (String permissionsValue : permissionsValueList) {
                 SimpleGrantedAuthority auth = new SimpleGrantedAuthority(permissionsValue);
                 authority.add(auth);
             }
             return new UsernamePasswordAuthenticationToken(username,token,authority);
         }
         return null;
     }
 }

这里核心doFilterInternal方法,根据token获取用户信息,并校验token的合法性,如果合法,再在redis中查询用户的权限信息,封装成UsernamePasswordAuthenticationToken(也就是Authentication的子类)直接放入security的上下文,由此就完成了认证授权。

以上只是个人理解,理解的比较肤浅,如果有错误,肯定评论指正,多多学习!

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-04-28 11:39:48  更:2022-04-28 11:42:46 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 2:55:00-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码