Spring Security自定义登录原理及实现详解

作者:码农小胖哥 时间:2022-11-20 21:57:39 

1. 前言

前面的关于 Spring Security 相关的文章只是一个预热。为了接下来更好的实战,如果你错过了请从 Spring Security 实战系列 开始。安全访问的第一步就是认证(Authentication),认证的第一步就是登录。今天我们要通过对 Spring Security 的自定义,来设计一个可扩展,可伸缩的 form 登录功能。

2. form 登录的流程

下面是 form 登录的基本流程:

Spring Security自定义登录原理及实现详解

只要是 form 登录基本都能转化为上面的流程。接下来我们看看 Spring Security 是如何处理的。

3. Spring Security 中的登录

昨天 Spring Security 实战干货:自定义配置类入口WebSecurityConfigurerAdapter 中已经讲到了我们通常的自定义访问控制主要是通过 HttpSecurity 来构建的。默认它提供了三种登录方式:

  • formLogin() 普通表单登录

  • oauth2Login() 基于 OAuth2.0 认证/授权协议

  • openidLogin() 基于 OpenID 身份认证规范

以上三种方式统统是 AbstractAuthenticationFilterConfigurer 实现的,

4. HttpSecurity 中的 form 表单登录

启用表单登录通过两种方式一种是通过 HttpSecurity 的 apply(C configurer) 方法自己构造一个 AbstractAuthenticationFilterConfigurer 的实现,这种是比较高级的玩法。 另一种是我们常见的使用 HttpSecurity 的 formLogin() 方法来自定义 FormLoginConfigurer 。我们先搞一下比较常规的第二种。

4.1 FormLoginConfigurer

该类是 form 表单登录的配置类。它提供了一些我们常用的配置方法:

  • loginPage(String loginPage) : 登录 页面而并不是接口,对于前后分离模式需要我们进行改造 默认为 /login。

  • loginProcessingUrl(String loginProcessingUrl) 实际表单向后台提交用户信息的 Action,再由过滤器UsernamePasswordAuthenticationFilter 拦截处理,该 Action 其实不会处理任何逻辑。

  • usernameParameter(String usernameParameter) 用来自定义用户参数名,默认 username 。

  • passwordParameter(String passwordParameter) 用来自定义用户密码名,默认 password

  • failureUrl(String authenticationFailureUrl) 登录失败后会重定向到此路径, 一般前后分离不会使用它。

  • failureForwardUrl(String forwardUrl) 登录失败会转发到此, 一般前后分离用到它。 可定义一个 Controller (控制器)来处理返回值,但是要注意 RequestMethod。

  • defaultSuccessUrl(String defaultSuccessUrl, boolean alwaysUse) 默认登陆成功后跳转到此 ,如果 alwaysUse 为 true 只要进行认证流程而且成功,会一直跳转到此。一般推荐默认值 false

  • successForwardUrl(String forwardUrl) 效果等同于上面 defaultSuccessUrl 的 alwaysUse 为 true 但是要注意 RequestMethod。

  • successHandler(AuthenticationSuccessHandler successHandler) 自定义认证成功处理器,可替代上面所有的 success 方式

  • failureHandler(AuthenticationFailureHandler authenticationFailureHandler) 自定义失败成功处理器,可替代上面所有的 success 方式

  • permitAll(boolean permitAll) form 表单登录是否放开

知道了这些我们就能来搞个定制化的登录了。

5. Spring Security 聚合登录 实战

接下来是我们最激动人心的实战登录操作。 有疑问的可认真阅读 Spring 实战 的一系列预热文章。

5.1 简单需求

我们的接口访问都要通过认证,登陆错误后返回错误信息(json),成功后前台可以获取到对应数据库用户信息(json)(实战中记得脱敏)。

我们定义处理成功失败的控制器:


@RestController
@RequestMapping("/login")
public class LoginController {
  @Resource
  private SysUserService sysUserService;

/**
   * 登录失败返回 401 以及提示信息.
   *
   * @return the rest
   */
  @PostMapping("/failure")
  public Rest loginFailure() {

return RestBody.failure(HttpStatus.UNAUTHORIZED.value(), "登录失败了,老哥");
  }

/**
   * 登录成功后拿到个人信息.
   *
   * @return the rest
   */
  @PostMapping("/success")
  public Rest loginSuccess() {
     // 登录成功后用户的认证信息 UserDetails会存在 安全上下文寄存器 SecurityContextHolder 中
    User principal = (User) SecurityContextHolder.getContext().getAuthentication().getPrincipal();
    String username = principal.getUsername();
    SysUser sysUser = sysUserService.queryByUsername(username);
    // 脱敏
    sysUser.setEncodePassword("[PROTECT]");
    return RestBody.okData(sysUser,"登录成功");
  }
}

然后 我们自定义配置覆写 void configure(HttpSecurity http) 方法进行如下配置(这里需要禁用crsf):


@Configuration
@ConditionalOnClass(WebSecurityConfigurerAdapter.class)
@ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.SERVLET)
public class CustomSpringBootWebSecurityConfiguration {

@Configuration
  @Order(SecurityProperties.BASIC_AUTH_ORDER)
  static class DefaultConfigurerAdapter extends WebSecurityConfigurerAdapter {
    @Override
    protected void configure(AuthenticationManagerBuilder auth) throws Exception {
      super.configure(auth);
    }

@Override
    public void configure(WebSecurity web) throws Exception {
      super.configure(web);
    }

@Override
    protected void configure(HttpSecurity http) throws Exception {
      http.csrf().disable()
          .cors()
          .and()
          .authorizeRequests().anyRequest().authenticated()
          .and()
          .formLogin()
          .loginProcessingUrl("/process")
          .successForwardUrl("/login/success").
          failureForwardUrl("/login/failure");

}
  }
}

使用 Postman 或者其它工具进行 Post 方式的表单提交 http://localhost:8080/process?username=Felordcn&password=12345 会返回用户信息:


{
  "httpStatus": 200,
  "data": {
    "userId": 1,
    "username": "Felordcn",
    "encodePassword": "[PROTECT]",
    "age": 18
  },
  "msg": "登录成功",
  "identifier": ""
}

把密码修改为其它值再次请求认证失败后 :


 {
   "httpStatus": 401,
   "data": null,
   "msg": "登录失败了,老哥",
   "identifier": "-9999"
 }

6. 多种登录方式的简单实现

就这么完了么?现在登录的花样繁多。常规的就有短信、邮箱、扫码 ,第三方是以后我要讲的不在今天范围之内。 如何应对想法多的产品经理? 我们来搞一个可扩展各种姿势的登录方式。我们在上面 2. form 登录的流程 中的 用户 和 判定 之间增加一个适配器来适配即可。 我们知道这个所谓的 判定就是 UsernamePasswordAuthenticationFilter 。

我们只需要保证 uri 为上面配置的/process 并且能够通过 getParameter(String name) 获取用户名和密码即可 。

我突然觉得可以模仿 DelegatingPasswordEncoder 的搞法, 维护一个注册表执行不同的处理策略。当然我们要实现一个 GenericFilterBean 在 UsernamePasswordAuthenticationFilter 之前执行。同时制定登录的策略。

6.1 登录方式定义

定义登录方式枚举 ``。


 public enum LoginTypeEnum {

/**
   * 原始登录方式.
   */
   FORM,
   /**
   * Json 提交.
   */
   JSON,
   /**
   * 验证码.
   */
   CAPTCHA
 }

6.2 定义前置处理器接口


 public interface LoginPostProcessor {

/**
   * 获取 登录类型
   *
   * @return the type
   */
   LoginTypeEnum getLoginTypeEnum();

/**
   * 获取用户名
   *
   * @param request the request
   * @return the string
   */
   String obtainUsername(ServletRequest request);

/**
   * 获取密码
   *
   * @param request the request
   * @return the string
   */
   String obtainPassword(ServletRequest request);

}

6.3 实现登录前置处理过滤器

该过滤器维护了 LoginPostProcessor 映射表。 通过前端来判定登录方式进行策略上的预处理,最终还是会交给


package cn.felord.spring.security.filter;

import cn.felord.spring.security.enumation.LoginTypeEnum;
import org.springframework.security.web.util.matcher.AntPathRequestMatcher;
import org.springframework.security.web.util.matcher.RequestMatcher;
import org.springframework.util.Assert;
import org.springframework.util.CollectionUtils;
import org.springframework.web.filter.GenericFilterBean;

import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;

import static org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter.SPRING_SECURITY_FORM_PASSWORD_KEY;
import static org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter.SPRING_SECURITY_FORM_USERNAME_KEY;

/**
 * 预登录控制器
 *
 * @author Felordcn
 * @since 16 :21 2019/10/17
 */
public class PreLoginFilter extends GenericFilterBean {

private static final String LOGIN_TYPE_KEY = "login_type";

private RequestMatcher requiresAuthenticationRequestMatcher;
  private Map<LoginTypeEnum, LoginPostProcessor> processors = new HashMap<>();

public PreLoginFilter(String loginProcessingUrl, Collection<LoginPostProcessor> loginPostProcessors) {
    Assert.notNull(loginProcessingUrl, "loginProcessingUrl must not be null");
    requiresAuthenticationRequestMatcher = new AntPathRequestMatcher(loginProcessingUrl, "POST");
    LoginPostProcessor loginPostProcessor = defaultLoginPostProcessor();
    processors.put(loginPostProcessor.getLoginTypeEnum(), loginPostProcessor);

if (!CollectionUtils.isEmpty(loginPostProcessors)) {
      loginPostProcessors.forEach(element -> processors.put(element.getLoginTypeEnum(), element));
    }

}

private LoginTypeEnum getTypeFromReq(ServletRequest request) {
    String parameter = request.getParameter(LOGIN_TYPE_KEY);

int i = Integer.parseInt(parameter);
    LoginTypeEnum[] values = LoginTypeEnum.values();
    return values[i];
  }

/**
   * 默认还是Form .
   *
   * @return the login post processor
   */
  private LoginPostProcessor defaultLoginPostProcessor() {
    return new LoginPostProcessor() {

@Override
      public LoginTypeEnum getLoginTypeEnum() {

return LoginTypeEnum.FORM;
      }

@Override
      public String obtainUsername(ServletRequest request) {
        return request.getParameter(SPRING_SECURITY_FORM_USERNAME_KEY);
      }

@Override
      public String obtainPassword(ServletRequest request) {
        return request.getParameter(SPRING_SECURITY_FORM_PASSWORD_KEY);
      }
    };
  }

@Override
  public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
    ParameterRequestWrapper parameterRequestWrapper = new ParameterRequestWrapper((HttpServletRequest) request);
    if (requiresAuthenticationRequestMatcher.matches((HttpServletRequest) request)) {

LoginTypeEnum typeFromReq = getTypeFromReq(request);

LoginPostProcessor loginPostProcessor = processors.get(typeFromReq);

String username = loginPostProcessor.obtainUsername(request);

String password = loginPostProcessor.obtainPassword(request);

parameterRequestWrapper.setAttribute(SPRING_SECURITY_FORM_USERNAME_KEY, username);
      parameterRequestWrapper.setAttribute(SPRING_SECURITY_FORM_PASSWORD_KEY, password);

}

chain.doFilter(parameterRequestWrapper, response);

}
}

6.4 验证

通过 POST 表单提交方式 http://localhost:8080/process?username=Felordcn&password=12345&login_type=0 可以请求成功。或者以下列方式也可以提交成功:

Spring Security自定义登录原理及实现详解

更多的登录方式 只需要实现接口 LoginPostProcessor 注入 PreLoginFilter

7. 总结

今天我们通过各种技术的运用实现了从简单登录到可动态扩展的多种方式并存的实战运用。

来源:https://blog.51cto.com/14901317/2528852

标签:Spring,Security,自定义,登录
0
投稿

猜你喜欢

  • java实现给图片加铺满的网格式文字水印

    2023-07-30 05:22:12
  • 简单了解JAVA public class与class区别

    2023-11-15 23:59:26
  • Android使用SmsManager实现短信发送功能

    2023-08-24 17:54:20
  • Android 设置颜色的方法总结

    2023-12-14 16:41:57
  • 函数指针的一些概念详解

    2023-11-22 09:47:18
  • 关于Java从本地文件复制到网络文件上传

    2022-05-10 16:49:02
  • android 调用系统的照相机和图库实例详解

    2022-01-02 11:46:47
  • iOS WebView中使用webp格式图片的方法

    2023-06-17 22:06:03
  • Springboot+AOP实现返回数据提示语国际化的示例代码

    2021-08-18 19:49:12
  • Springboot集成Spring Security实现JWT认证的步骤详解

    2021-09-18 13:18:08
  • java实现简单斗地主(看牌排序)

    2023-09-12 14:42:40
  • springboot集成mybatisplus的方法

    2022-08-02 16:35:12
  • 详解Spring MVC CORS 跨域

    2023-11-25 08:04:37
  • Java常用锁synchronized和ReentrantLock的区别

    2023-06-01 04:42:21
  • Android实现左侧滑动菜单

    2022-10-10 14:58:41
  • @Async导致controller 404及失效原因解决分析

    2021-12-17 01:51:44
  • 详细图解Java中字符串的初始化

    2023-11-20 19:34:24
  • Android App开发中ViewPager组件的入门使用教程

    2023-04-30 21:45:34
  • Android实现简易计步器功能隔天步数清零查看历史运动纪录

    2021-07-24 00:37:02
  • C语言中结构体与内存对齐实例解析

    2022-05-16 12:25:18
  • asp之家 软件编程 m.aspxhome.com