在SpringSecurity认证中使用数据库数据

本文介绍了如何在SpringBoot项目中使用SpringSecurity,并替换默认的内存查询,改为从数据库获取用户信息。首先,通过分析SpringSecurity的过滤器链原理,特别是UsernamePasswordAuthenticationFilter的工作方式,理解了需要自定义UserDetailsService。然后,展示了项目依赖环境,包括SpringBoot、MybatisPlus和相关库。接着,编写了UserDetailsServiceImpl实现UserDetailsService接口,从数据库查询用户并处理认证。最后,给出了数据库表结构和配置,强调了在测试时密码需加{noop}

在SpringSecurity中使用数据库代替原本的内存查询

1.实现原理

​ SpringSecurity的原理其实就是一个过滤器链,内部包含了提供各种功能的过滤器。这里是入门案例中的过滤器。

image-20220515224217283

UsernamePasswordAuthenticationFilter:负责处理我们在登陆页面填写了用户名密码后的登陆请求。入门案例的认证工作主要有它负责。

image-20220515224323120

然后这个过滤器会调用这些方法 这些方法默认写好了 我们要做的就是自己写一个UserDetailsService的实现类 来代替它默认的
InMemeryUserDetilsManager。因为我们想到去数据库中查找数据。

我们怎么样可以覆盖这个类呢?因为当前是springboot项目 我猜想autoConfig类应该是写了 如果我们自定义了UserDetailsService的实现类,springboot 就不会容器注入默认的InMemeryUserDetilsManager

接下来我们查看springbooot源码查看(扩展)

  1. 因为它sping自己家的东西 我们在这里查看源码image-20220515225355854

image-20220515225411849

判断spring容器中不存在AuthenticationManager、AuthenticationProvider、UserDetailsService这三个类的实例,默认我们没有自定义这三个类,满足条件综上,满足所有条件,将UserDetailsServiceAutoConfiguration实例注册到spring容器中最终将InMemoryUserDetailsManager注册到spring容器中

我们要自己写UserDetailsService类的实例 所以这个自动配置实效了(符合我的猜想)

image-20220515230251037

2.怎么使用

demo结构 (许多东西在这个当前demo都没用到)

image-20220515224921893

(1)环境

springboot mybatisPuls

<properties>
    <maven.compiler.source>11</maven.compiler.source>
    <maven.compiler.target>11</maven.compiler.target>
</properties>
<parent>
    <artifactId>spring-boot-starter-parent</artifactId>
    <groupId>org.springframework.boot</groupId>
    <version>2.6.7</version>
</parent>
<dependencies>
    <dependency>
        <groupId>com.baomidou</groupId>
        <artifactId>mybatis-plus-boot-starter</artifactId>
        <version>3.5.1</version>
    </dependency>
    <dependency>
        <groupId>mysql</groupId>
        <artifactId>mysql-connector-java</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
        <version>2.6.7</version>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-security</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-test</artifactId>
    </dependency>
    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <optional>true</optional>
    </dependency>
</dependencies>

(2)测试代码

写一个UserDetailsServiceImpl代替原本的UserDetailsService

@Service
public class UserDetailsServiceImpl implements UserDetailsService {

    @Autowired
    private UserMapper userMapper;

    @Override
    public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
        // 查询用户信息
        LambdaQueryWrapper<User> queryWrapper = new LambdaQueryWrapper<>();
        queryWrapper.eq(User::getUserName,username);
        User user = userMapper.selectOne(queryWrapper);
        //没有查询到用户
        if(Objects.isNull(user)){
            throw new RuntimeException("用户名或者密码错误");
            //这里的  “用户名或者密码错误” 会错误后回显在默认页面 搭建完可以自己试一下
             // 对这个异常的处理 security有默认的过滤器处理 这个以后再说 
        }

        // TODO 查询对应权限信息

        //查到了 返回
        return new LoginUser(user);
    }
}

里面用到的mapper 以及实体类(许多属性没用到)

@Data
@AllArgsConstructor
@NoArgsConstructor
@TableName(value = "sys_user")
public class User implements Serializable {
    private static final long serialVersionUID = -40356785423868312L;
    
    /**
    * 主键
    */
    @TableId
    private Long id;
    /**
    * 用户名
    */
    private String userName;
    /**
    * 昵称
    */
    private String nickName;
    /**
    * 密码
    */
    private String password;
    /**
    * 账号状态(0正常 1停用)
    */
    private String status;
    /**
    * 邮箱
    */
    private String email;
    /**
    * 手机号
    */
    private String phonenumber;
    /**
    * 用户性别(0男,1女,2未知)
    */
    private String sex;
    /**
    * 头像
    */
    private String avatar;
    /**
    * 用户类型(0管理员,1普通用户)
    */
    private String userType;
    /**
    * 创建人的用户id
    */
    private Long createBy;
    /**
    * 创建时间
    */
    private Date createTime;
    /**
    * 更新人
    */
    private Long updateBy;
    /**
    * 更新时间
    */
    private Date updateTime;
    /**
    * 删除标志(0代表未删除,1代表已删除)
    */
    private Integer delFlag;
}

记得加注解扫描

@SpringBootApplication
@MapperScan("com.dong.jwt.mapper")
public interface UserMapper extends BaseMapper<User> {
}

因为UserDetailsServiceImpl 的接口UserDetailsService中的方法让我们返回一个UserDetails

然后UserDetails 是一个接口 所以我们就自己写一个类去实现一下它

@Data
@AllArgsConstructor
@NoArgsConstructor
public class LoginUser implements UserDetails {

    private User user;
    @Override
    public Collection<? extends GrantedAuthority> getAuthorities() {
        return null;
    }

    @Override
    public String getPassword() {
        return user.getPassword();
    }

    @Override
    public String getUsername() {
        return user.getUserName();
    }

    @Override
    public boolean isAccountNonExpired() {
        return true;
    }

    @Override
    public boolean isAccountNonLocked() {
        return true;
    }

    @Override
    public boolean isCredentialsNonExpired() {
        return true;
    }

    @Override
    public boolean isEnabled() {
        return true;
    }
}
spring:
  datasource:
    url: jdbc:mysql://localhost:3306/dzzs_t?characterEncoding=utf-8&serverTimezone=UTC
    username: root
    password: 123456
    driver-class-name: com.mysql.cj.jdbc.Driver
CREATE TABLE `sys_user` (
  `id` BIGINT(20) NOT NULL AUTO_INCREMENT COMMENT '主键',
  `user_name` VARCHAR(64) NOT NULL DEFAULT 'NULL' COMMENT '用户名',
  `nick_name` VARCHAR(64) NOT NULL DEFAULT 'NULL' COMMENT '昵称',
  `password` VARCHAR(64) NOT NULL DEFAULT 'NULL' COMMENT '密码',
  `status` CHAR(1) DEFAULT '0' COMMENT '账号状态(0正常 1停用)',
  `email` VARCHAR(64) DEFAULT NULL COMMENT '邮箱',
  `phonenumber` VARCHAR(32) DEFAULT NULL COMMENT '手机号',
  `sex` CHAR(1) DEFAULT NULL COMMENT '用户性别(0男,1女,2未知)',
  `avatar` VARCHAR(128) DEFAULT NULL COMMENT '头像',
  `user_type` CHAR(1) NOT NULL DEFAULT '1' COMMENT '用户类型(0管理员,1普通用户)',
  `create_by` BIGINT(20) DEFAULT NULL COMMENT '创建人的用户id',
  `create_time` DATETIME DEFAULT NULL COMMENT '创建时间',
  `update_by` BIGINT(20) DEFAULT NULL COMMENT '更新人',
  `update_time` DATETIME DEFAULT NULL COMMENT '更新时间',
  `del_flag` INT(11) DEFAULT '0' COMMENT '删除标志(0代表未删除,1代表已删除)',
  PRIMARY KEY (`id`)
) ENGINE=INNODB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4 COMMENT='用户表'

image-20220515223859074

密码这个在目前测试时 记得在前面加一个{noop} !! 表示明文密码 因为默认情况是有加密的 这个以后再说,不是当前讨论的。
测试的话 写一个hello的controller去访问即可 会先让你登录

参考博主:三更草堂

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值