Spring Security 中的身份认证

原文链接:https://blog.gaoyuexiang.cn/2020/06/07/spring-security-authentication/,内容无差别。

本文介绍 Spring Security 的身份认证的内容,研究 Spring Security 自带的身份认证方式和添加自己的身份认证方式的方法。

身份认证相关组件

上一篇文章中,我们了解到了 Spring Security 会将 DelegatingFilterProxy 插入到 Servlet Filter Chain 中,然后将要过滤的请求通过 FilterChainProxy 代理给匹配的 SecurityFilterChain;这些 SecurityFilterChain 中包含着真正做安全相关工作的 Filter

后面提到的 Filter 都是红色方框中的

这些 Filter 中的一部分,他们的职责就是进行身份验证,比如 UsernamePasswordAuthenticationFilter;而他们中的大多数都有一个共同的父类 AbstractAuthenticationProcessingFilter

AbstractAuthenticationProcessFilter

这个类是很多身份认证的 Filter 的父类,它已经实现了 doFilter 方法,流程如下:

image

本文不涉及其中的 sessionStrategy 部分

doFilter 已经帮我们搭好了这个流程,我们只需要关心其中的几个被调用的方法(红绿蓝三个颜色框)就可以了。

attemptAuthentication

这是一个抽象方法。子类实现的时候,需要从 HttpServletRequest 中获取需要的信息,构建出一个 Authentication 实例,然后调用父类中的 AuthenticationManager.authenticate() 方法,对 Authentication 对象进行认证。

unsuccessfulAuthentication

这个方法已经被实现,子类也可以选择重写。根据父类的实现,这个方法将完成一下步骤:

  1. 清理 SecurityContextHolder
  2. 清除 RememberMeService 中的信息(默认使用 NullRememberMeService
  3. 调用 AuthenticationFailureHandler.onAuthenticationFailure() 方法

默认使用的 AuthenticationFailureHandlerSimpleUrlAuthenticationFailureHandler,它的逻辑是:

  1. 如果没有配置 defaultFailureUrl (默认没有)
    1. 发送 401 响应
  2. 根据配置的布尔值 forwardToDestination (默认为 false)判断
    1. 使用 Servlet forward 到配置的 defaultFailureUrl
    2. 使用 HTTP redirect 到配置的 defaultFailureUrl

successfulAuthentication

unsuccessfulAuthentication 方法一样,这个方法也已经实现,并且可以被重写,但其中的逻辑却恰好相反:

  1. attemptAuthentication 返回的 Authentication 对象保存到 SecurityContextHolder
  2. 保存登陆信息到 RememberMeService
  3. 发布 InteractiveAuthenticationSuccessEvent 事件,这样可以被配置的 EventListener 处理
  4. 调用 AuthenticationSuccessHandler.onAuthenticationSuccess() 方法

默认使用的 AuthenticationSuccessHandlerSavedRequestAwareAuthenticationSuccessHandler,其实现就是一次重定向。我们可以看看它重定向到哪里:

  • 当配置了 alwaysUseDefaultTargetUrl 或指定了 targetUrlParameter 且此参数存在的时候
    • 如果配置了 alwaysUseDefaultTargetUrl 则重定向到 defaultTargetUrl,默认是 /
    • 如果存在 targetUrlParameter(比如 redirect_uri 之类的比较常见的参数),则重定向到这个路径
    • 如果存在 Referer,则重定向到这个地址
    • 重定向到 /
  • RequestCache 中找到了保存的请求
    • 重定向到请求中设置的重定向地址
  • 如果还是没有满足条件,则进行第一步里的逻辑

关于 RequestCache:想象你正在访问一个需要认证的资源,这个时候网站会把你重定向到登陆页面;在你登陆成功后,又会重定向回刚才的资源。RequestCache 就是为了保存登陆之前的请求而设计的。在这里,默认使用基于 session 的实现。

AuthenticationManager

AbstractAuthenticationProcessingFilter 中保存了一个 AuthenticationManager,它会在子类的 attemptAuthentication 方法中被使用。其职责是对 Filter 创建的 Authentication 对象进行身份验证,比如查询数据库匹配用户名密码、携带的 token 是否合法等。

ProviderManager 与 AuthenticationProvider

这是 AuthenticationManager 常用的实现。它没有实现任何认证逻辑,而是管理了一些 AuthenticationProvider,通过这些 provider 来实现真正的认证功能。

每个 AuthenticationProvider 实现一种特定类型的身份认证方式,比如用户名密码登陆、OIDC 登陆等。他们可以通过 Authentication 的具体类型来判断是否支持这种 Authentication 需要的认证方式。

其他的一些 Filter

除了 AbstractAuthenticationProcessFilter,还有一些进行身份验证的 Filter,它们并没有继承这个类,而是基于 OncePerRequestFilter 自己实现了一套逻辑。这些 Filter 包括 AuthenticationFilterBasicAuthenticationFilterOAuth2AuthorizationCodeGrantFilter 等等。

由于它们不再是 AbstractAuthenticationProcessFilter,所以不会再被要求使用 AuthenticationManager。尽管这样,当我们选择使用 OncePerReuqestFilter 来实现自定义的身份认证时,仍然可以考虑使用 AuthenticationManager 这种方式。

个人觉得 AuthenticationManager 还算是个不错的设计,因为做到了职责分离。

甚至还有更加放飞自我的 DigestAuthenticationFilter,直接继承 GenericFilterBean,在实现上也是我行我素,这里就不探究了。

ExceptionTranslationFilter

这个 Filter 不是用来进行身份验证的,而是用来处理认证授权过程中产生的异常的。它可以处理 AuthenticationExceptionAccessDeniedException,分别表示认证失败和授权失败。这篇文章只关心如何处理 AuthenticationException

但是这个 Filter 默认被安排在 SecurityFilterChain 的倒数第二位,所以前面的 Filter 抛出的异常并不能被它捕获。但自定义的 Filter 可以加到它后面,这样就可以利用它来处理这两种异常。

最后一位是 FilterSecurityInterceptor,可能会抛出 AccessDeniedException

处理 AuthenticationException

ExceptionTranslationFilterAuthenticationException 的处理分三步:

  1. 清理 SecurityContextHolder 中的身份信息
  2. 将当前的 request、response 保存到 RequestCache 中(用途可以回顾一下 successfulAuthentication 方法
  3. 调用 AuthenticationEntryPoint.commence() 方法

其中的 AuthenticationEntryPoint 具体实例取决于你的配置,默认会用到 BasicAuthenticationEntryPoint。这个接口的职责就是通过 WWW-Authenticate header 告诉客户端使用哪种方式进行身份验证。

处理其他异常

对于 AuthenticationExceptionAccessDeniedException 之外的异常,ExceptionTranslationFilter 会将其转换成 ServletExceptionRuntimeException 抛出。

如果想要处理这些异常,需要自己添加 Filter 实现。

Spring Security 自动配置的 FilterChainProxy

当我们启动 Spring 应用之后,会在日志里看到打印所有配置的 FilterChainProxy

默认情况下,我们会看到这样的一条链:

image

这是引入 spring-boot-starter-security 之后自动配置的 FilterChainProxy,在引入更多的 security 相关的依赖和编写了相关配置之后,这个 filter chain 也会相应变化。

几种内置的身份认证方式

接下来,我们以 UsernamePawwrodAuthenticationFilterBasicAuthenticationFilter 为例,看看他们是如何实现身份认证的。

UsernamePasswordAuthenticationFilter

UsernamePasswordAuthenticationFilter 是一个 AbstractAuthenticationProcessingFilter 的子类,实现了 attemptAuthentication 方法,没有重写其他方法。所以用户认证成功后,会被重定向到一个地址,具体逻辑参考上面的 successfulAuthentication 方法

attemptAuthentication

attemptAuthentication 方法会从 HttpServletRequest.getParameter() 方法中获取用户名密码,从而进行身份验证。具体从哪里获取用户名密码,则可以被子类通过重写 obtainUsername()obtainPassword() 方法修改。

之后,UsernamePasswordAuthenticationFilter 会构建出一个 UsernameAuthenticationToken,交给 AuthenticationManager 进行认证。

DaoAuthenticationProvider

这是 UsernamePasswordAuthenticationFilter 对应的 AuthenticationProvider,负责对 UsernameAuthenticationToken 进行认证。

它使用一个 UserDetailsService 来加载用户信息,使用 PasswordEncoder 来匹配用户的密码。

这两个接口具体使用哪一个实现,取决于具体的配置。比如 UserDetailsService 就有 in memory 和 JDBC 的实现。

UsernamePasswordAuthenticationFilter 是用于单独处理登录的 Filter,它不是用来在请求业务 API 时进行身份认证的 Filter

事实上,所有继承了 AbstractAuthenticationProcessFilter 但没有重写 successfulAuthentication 方法的 Filter 都是这样的,它们会在登陆成功后重定向到登录前的地址或默认的地址。这也符合它的语义:进行身份认证流程,而不是业务请求的一部分。

BasicAuthenticationFilter

UsernamePasswordAuthenticationFilter 不同,BasicAuthenticationFilter 没有继承 AbstractAuthenticationProcessingFilter,而是直接继承 OncePerRequestFilter。因为它是被使用在请求业务 API 的请求上,而不是进行身份认证流程。

BasicAuthenticationFilter 的实现并不复杂,无非是从 Authorization header 中取出用户名密码,然后创建出 UsernameAuthenticationToken,接着调用 AuthenticationManager.authenticate() 方法。

之所以它也会使用 AuthenticationManager,应该是出于复用的考虑。这样它就可以使用和 UsernamePasswordAuthenticationFilter 一样的 AuthenticationProvider

它与 UsernamePasswordAuthenticationFilter 的区别在于认证之后的行为。

无论认证成功与否,BasicAuthenticationFilter 都不会做出重定向的响应。

  • 如果认证失败,则通过默认的 BasicAuthenticationEntryPoint 返回 401 响应
  • 如果认证成功,则继续执行 filter chain,这样就能执行到真正的业务方法

如何添加自己的身份认证方式

前面介绍了两种不同的 Filter 实现,以及它们被使用的场景,现在我们知道了该选择哪一种方式去实现自定义的 Filter。但是,如何把它们加入到 SecurityFilterChain 中去处理身份认证呢?

配置 SecurityFilterChain

我们如果需要任何对 SecurityFilterChain 的配置,都需要扩展 WebSecurityConfigurerAdapter,实现自己的一个配置类。每创建这样的一个实现,都会创建一个 SecurityFilterChain 加入到 FilterChainProxy 中。

配置 requestMathcer

我们在前一篇文章提到过,FilterChainProxy 需要根据 url 来判断选择哪一个 SecurityFilterChain。我们需要将这个配置写到这个实现类中,比如:

@Override
protected void configure(HttpSecurity http) throws Exception {
  http.requestMatchers(matcher -> matcher.mvcMatchers("/hello"));
}

这样,FilterChainProxy 就知道了对 /hello 的请求需要使用这个 SecurityFilterChain

向 SecurityFilterChain 加入 Filter

现在有了对应的 SecurityFilterChain,我们就可以将自定义的 Filter 加入到这个 chain 中:

@Override
protected void configure(HttpSecurity http) throws Exception {
  http.addFilter(HelloFilter.class);
}

addFilter 方法也有一些变体,可以控制 Filter 在 chain 中的位置,这里就不赘述了。

添加 AuthenticationProvider

Filter 一样,AuthenticationProvider 也是被安排到单独的 FilterChainProxy 中的,并且需要自己配置。如果你的自定义 Filter 需要 AuthenticationProvider 的话,同样需要配置:

@Override
protected void configure(AuthenticationManagerBuilder auth) throws Exception {
  auth.authenticationProvider(HelloAuthenticationProvider.class);
}

总结

这篇文章比较详细的梳理了 AbstractAuthenticationProcessingFilter 及其子类 UsernamePasswordAuthenticationFilter 的实现和 BasicAuthenticationFilter 的实现,了解了需要实现自定义身份验证的 Filter 时应该选择哪种方式:

  • 只是进行身份验证,完成后进行重定向,而不调用业务方法,那么就继承 AbstractAuthenticationProcessFilter
  • 需要调用业务方法,身份验证是为了保护业务,那么就继承 OncePerRequestFilter,完全控制认证的流程

当然,这不是一个强制的限制,你仍然可以通过重写 AbstractAuthenticationProcessFilter.successfulAuthentication() 方法来修改重定向的行为。

另外,也了解到了实现完 Filter 后,需要实现 WebSecurityConfigurerAdapter,将 Filter 加入到 SecurityFilterChain 中。

👉查看系列文章

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,128评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,316评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,737评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,283评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,384评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,458评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,467评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,251评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,688评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,980评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,155评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,818评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,492评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,142评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,382评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,020评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,044评论 2 352