SpringSecurity登录失败返回错误信息

现在有这么一个需求,客户端登录验证失败之后显示两种错误信息--"用户名不存在"或者"账号密码错误",但是spring Security好像并没有返回错误信息的类,所以需要我们自己从request域中去获取错误信息,首先我们要设置验证失败后的返回地址。

http.formLogin()   //自定义自己编写的登陆页面
                .loginPage("/login.html")  //登陆页面设置
                .loginProcessingUrl("/user/login")  //登陆访问路径
                .defaultSuccessUrl("/") //登陆成功后跳转路径
                .failureUrl("/user/fail")//注意这里用请求转发,不要重定向
                .permitAll()
                .and().authorizeRequests()
                .antMatchers("/user/**",
                        "/swagger-ui.html", "/v2/**", "/swagger-resources/**", "/webjars/springfox-swagger-ui/**")
                .permitAll()//不拦截的路径
                .antMatchers("/school/**").hasAuthority("admin")
//                .antMatchers("/test/findAll").hasAnyAuthority("addUser,findAll")
//                .antMatchers("/test/hello").hasRole("admin")
//                .antMatchers("/test/hello").hasAnyRole("admin")
                .anyRequest().authenticated()
                .and().csrf().disable();  //关闭csrf的保护

然后我们在失败请求中利用request 来获取错误信息,所有的错误信息都会保存至"SPRING_SECURITY_LAST_EXCEPTION"中。

@ResponseBody
    @RequestMapping("fail")
    public ResultInfo loginFail(HttpServletRequest request , HttpServletResponse response){
        //获取security的异常错误信息
        AuthenticationException exception =
                (AuthenticationException)request.getSession().getAttribute("SPRING_SECURITY_LAST_EXCEPTION");
        String msg = exception.getMessage();
        System.out.println(msg);
        
        return new ResultInfo("error!" , msg);
    }

但是我们发现无论怎么抛出异常,最终都会显示“Bad credentials”,异常处理代码如下:

@Override
    public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {

        //调用usersMapper方法,根据用户名查询数据库
        QueryWrapper<User> wrapper = new QueryWrapper<>();
        wrapper.eq("username",username);
        User user = userMapper.selectOne(wrapper);

        //判断
        if(user==null){//数据库没有数据,认证失败
            throw new UsernameNotFoundException("用户名不存在!");
        }

		//手动设置了role,也可以通过数据库查询获取
        List<GrantedAuthority> auths = AuthorityUtils.commaSeparatedStringToAuthorityList(user.getRole());  //配置角色

        return new org.springframework.security.core.userdetails.User(user.getUsername(),
                new BCryptPasswordEncoder().encode(user.getPassword()),auths);
    }

此处详细处理方法参考http://t.csdn.cn/pnZhx

我这里用的是方法一

@Override
    public UserDetails loadUserByUsername(String username) throws BadCredentialsException {

        //调用usersMapper方法,根据用户名查询数据库
        QueryWrapper<User> wrapper = new QueryWrapper<>();
        wrapper.eq("username",username);
        User user = userMapper.selectOne(wrapper);

        //判断
        if(user==null){//数据库没有数据,认证失败
            throw new BadCredentialsException("用户名不存在!");
        }

		//手动设置了role,也可以通过数据库查询获取
        List<GrantedAuthority> auths = AuthorityUtils.commaSeparatedStringToAuthorityList(user.getRole());  //配置角色

        return new org.springframework.security.core.userdetails.User(user.getUsername(),
                new BCryptPasswordEncoder().encode(user.getPassword()),auths);
    }

问题解决。

注:有时候会出现返回值乱码的问题,原因是因为security的拦截器会默认将自己的拦截器放在第一位,而在SpringMVC中编码拦截器必须放在第一位。解决方法:在security的拦截器前加上编码拦截器

@Override
    protected void configure(HttpSecurity http) throws Exception {
        //添加转码
        CharacterEncodingFilter encodingFilter = new CharacterEncodingFilter();
        encodingFilter.setEncoding("UTF-8");
        encodingFilter.setForceEncoding(true);
        http.addFilterBefore(encodingFilter, CsrfFilter.class);

        //配置没有权限访问跳转自定义页面
        http.exceptionHandling().accessDeniedPage("/403.html");
        http.logout()
                .logoutUrl("/logout")
                .logoutSuccessUrl("/login.html")
                .permitAll();

        http.formLogin()   //自定义自己编写的登陆页面
                .loginPage("/login.html")  //登陆页面设置
                .loginProcessingUrl("/user/login")  //登陆访问路径
                .defaultSuccessUrl("/") //登陆成功后跳转路径
                .failureUrl("/user/fail")
                .permitAll()
                .and().authorizeRequests()
                .antMatchers("/user/**",
                        "/swagger-ui.html", "/v2/**", "/swagger-resources/**", "/webjars/springfox-swagger-ui/**")
                .permitAll()//不拦截的路径
                .antMatchers("/school/**").hasAuthority("admin")
//                .antMatchers("/test/findAll").hasAnyAuthority("addUser,findAll")
//                .antMatchers("/test/hello").hasRole("admin")
//                .antMatchers("/test/hello").hasAnyRole("admin")
                .anyRequest().authenticated()
                .and().csrf().disable();  //关闭csrf的保护
    }

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
抱歉,我是一个AI语言模型,无法提供源代码。但我可以给出一个大致的思路供您参考。 首先,在Spring Boot项目中,需要在Maven或Gradle配置文件中添加JSR303依赖,例如: ```xml <!-- JSR303校验依赖 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-validation</artifactId> </dependency> ``` 其次,在需要校验的实体类中,使用JSR303提供的注解进行字段校验,如下所示: ```java public class Student { @NotNull(message = "姓名不能为空") private String name; @Max(value = 150, message = "年龄不能超过150岁") @Min(value = 0, message = "年龄不能小于0岁") private int age; // 省略其它字段 // ... } ``` 以上代码中,@NotNull注解表示该字段不允许为空,@Max和@Min注解表示该字段的值必须在指定范围内。 接下来,在需要校验的Controller中,使用@Valid注解标注待校验的实体类对象,并将BindingResult对象作为方法参数,用于存储校验结果: ```java @RestController @RequestMapping("/student") public class StudentController { @PostMapping("/add") public String addStudent(@Valid @RequestBody Student student, BindingResult bindingResult) { if (bindingResult.hasErrors()) { // 校验失败返回错误信息 return bindingResult.getFieldError().getDefaultMessage(); } else { // 校验成功,执行添加操作 // ... return "success"; } } } ``` 最后,实现校验信息国际化需要在Spring Boot项目的配置文件中添加相关配置,并在资源目录下创建不同语言的properties文件来存放各种校验错误信息的key-value对,如下所示: ```yaml # 配置国际化 spring.messages.basename=messages # 指定默认语言 spring.messages.fallback-to-system-locale=false spring.messages.encoding=UTF-8 ``` 其中,messages.properties文件中存放默认语言(如中文)下的校验错误信息,messages_en_US.properties文件中存放英文下的校验错误信息。例如: messages.properties ```properties NotNull.student.name=姓名不能为空 Max.student.age=年龄不能超过150岁 Min.student.age=年龄不能小于0岁 ``` messages_en_US.properties ```properties NotNull.student.name=Name cannot be null. Max.student.age=Age cannot be greater than 150. Min.student.age=Age cannot be less than 0. ``` 这样,在校验失败时,只需返回对应的key值即可,在前端页面通过i18n框架自动加载对应语言的国际化信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值