成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

深入理解 SecurityConfigurer

開發 前端
SecurityConfigurer 在 Spring Security 中是一個非常重要的角色。在前面的文章中,松哥曾經多次提到過,Spring Security 過濾器鏈中的每一個過濾器,都是通過 xxxConfigurer 來進行配置的,而這些 xxxConfigurer 實際上都是 SecurityConfigurer 的實現。

[[334361]]

我們來繼續擼 Spring Security 源碼。

SecurityConfigurer 在 Spring Security 中是一個非常重要的角色。在前面的文章中,松哥曾經多次提到過,Spring Security 過濾器鏈中的每一個過濾器,都是通過 xxxConfigurer 來進行配置的,而這些 xxxConfigurer 實際上都是 SecurityConfigurer 的實現。

所以我們今天有必要來跟大家把 SecurityConfigurer 從頭到尾捋一捋。

1. SecurityConfigurerSecurityConfigurer

本身是一個接口,我們來看下:

  1. public interface SecurityConfigurer<O, B extends SecurityBuilder<O>> { 
  2.  
  3.  void init(B builder) throws Exception; 
  4.  
  5.  void configure(B builder) throws Exception; 

可以看到,SecurityConfigurer 中主要是兩個方法,init 和 configure。

init 就是一個初始化方法。而 configure 則是一個配置方法。這里只是規范了方法的定義,具體的實現則在不同的實現類中。

需要注意的是這兩個方法的參數類型都是一個泛型 B,也就是 SecurityBuilder 的子類,關于 SecurityBuilder ,它是用來構建過濾器鏈的,松哥將在下篇文章中和大家介紹。

SecurityConfigurer 有三個實現類:

  • SecurityConfigurerAdapter
  • GlobalAuthenticationConfigurerAdapter
  • WebSecurityConfigurer

我們分別來看。

1.1 SecurityConfigurerAdapter

SecurityConfigurerAdapter實現了 SecurityConfigurer 接口,我們所使用的大部分的 xxxConfigurer 也都是 SecurityConfigurerAdapter 的子類。

SecurityConfigurerAdapter 在 SecurityConfigurer 的基礎上,還擴展出來了幾個非常好用的方法,我們一起來看下:

  1. public abstract class SecurityConfigurerAdapter<O, B extends SecurityBuilder<O>> 
  2.   implements SecurityConfigurer<O, B> { 
  3.  private B securityBuilder; 
  4.  
  5.  private CompositeObjectPostProcessor objectPostProcessor = new CompositeObjectPostProcessor(); 
  6.  
  7.  public void init(B builder) throws Exception { 
  8.  } 
  9.  
  10.  public void configure(B builder) throws Exception { 
  11.  } 
  12.  
  13.  public B and() { 
  14.   return getBuilder(); 
  15.  } 
  16.  protected final B getBuilder() { 
  17.   if (securityBuilder == null) { 
  18.    throw new IllegalStateException("securityBuilder cannot be null"); 
  19.   } 
  20.   return securityBuilder; 
  21.  } 
  22.  @SuppressWarnings("unchecked"
  23.  protected <T> T postProcess(T object) { 
  24.   return (T) this.objectPostProcessor.postProcess(object); 
  25.  } 
  26.  public void addObjectPostProcessor(ObjectPostProcessor<?> objectPostProcessor) { 
  27.   this.objectPostProcessor.addObjectPostProcessor(objectPostProcessor); 
  28.  } 
  29.  public void setBuilder(B builder) { 
  30.   this.securityBuilder = builder; 
  31.  } 
  32.  private static final class CompositeObjectPostProcessor implements 
  33.    ObjectPostProcessor<Object> { 
  34.   private List<ObjectPostProcessor<?>> postProcessors = new ArrayList<>(); 
  35.  
  36.   @SuppressWarnings({ "rawtypes""unchecked" }) 
  37.   public Object postProcess(Object object) { 
  38.    for (ObjectPostProcessor opp : postProcessors) { 
  39.     Class<?> oppClass = opp.getClass(); 
  40.     Class<?> oppType = GenericTypeResolver.resolveTypeArgument(oppClass, 
  41.       ObjectPostProcessor.class); 
  42.     if (oppType == null || oppType.isAssignableFrom(object.getClass())) { 
  43.      object = opp.postProcess(object); 
  44.     } 
  45.    } 
  46.    return object; 
  47.   } 
  48.   private boolean addObjectPostProcessor( 
  49.     ObjectPostProcessor<?> objectPostProcessor) { 
  50.    boolean result = this.postProcessors.add(objectPostProcessor); 
  51.    postProcessors.sort(AnnotationAwareOrderComparator.INSTANCE); 
  52.    return result; 
  53.   } 
  54.  } 
  1. CompositeObjectPostProcessor 首先一開始聲明了一個 CompositeObjectPostProcessor 實例,CompositeObjectPostProcessor 是 ObjectPostProcessor 的一個實現,ObjectPostProcessor 本身是一個后置處理器,該后置處理器默認有兩個實現,AutowireBeanFactoryObjectPostProcessor 和 CompositeObjectPostProcessor。其中 AutowireBeanFactoryObjectPostProcessor 主要是利用了 AutowireCapableBeanFactory 對 Bean 進行手動注冊,因為在 Spring Security 中,很多對象都是手動 new 出來的,這些 new 出來的對象和容器沒有任何關系,利用 AutowireCapableBeanFactory 可以將這些手動 new 出來的對象注入到容器中,而 AutowireBeanFactoryObjectPostProcessor 的主要作用就是完成這件事;CompositeObjectPostProcessor 則是一個復合的對象處理器,里邊維護了一個 List 集合,這個 List 集合中,大部分情況下只存儲一條數據,那就是 AutowireBeanFactoryObjectPostProcessor,用來完成對象注入到容器的操作,如果用戶自己手動調用了 addObjectPostProcessor 方法,那么 CompositeObjectPostProcessor 集合中維護的數據就會多出來一條,在 CompositeObjectPostProcessor#postProcess 方法中,會遍歷集合中的所有 ObjectPostProcessor,挨個調用其 postProcess 方法對對象進行后置處理。
  2. and 方法,該方法返回值是一個 securityBuilder,securityBuilder 實際上就是 HttpSecurity,我們在 HttpSecurity 中去配置不同的過濾器時,可以使用 and 方法進行鏈式配置,就是因為這里定義了 and 方法并返回了 securityBuilder 實例。

這便是 SecurityConfigurerAdapter 的主要功能,后面大部分的 xxxConfigurer 都是基于此類來實現的。

1.2 GlobalAuthenticationConfigurerAdapter

GlobalAuthenticationConfigurerAdapter 看名字就知道是一個跟全局配置有關的東西,它本身實現了 SecurityConfigurerAdapter 接口,但是并未對方法做具體的實現,只是將泛型具體化了:

  1. @Order(100) 
  2. public abstract class GlobalAuthenticationConfigurerAdapter implements 
  3.   SecurityConfigurer<AuthenticationManager, AuthenticationManagerBuilder> { 
  4.  
  5.  public void init(AuthenticationManagerBuilder auth) throws Exception { 
  6.  } 
  7.  
  8.  public void configure(AuthenticationManagerBuilder auth) throws Exception { 
  9.  } 

可以看到,SecurityConfigurer 中的泛型,現在明確成了 AuthenticationManager 和 AuthenticationManagerBuilder。所以 GlobalAuthenticationConfigurerAdapter 的實現類將來主要是和配置 AuthenticationManager 有關。當然也包括默認的用戶名密碼也是由它的實現類來進行配置的。

我們在 Spring Security 中使用的 AuthenticationManager 其實可以分為兩種,一種是局部的,另一種是全局的,這里主要是全局的配置。

1.3 WebSecurityConfigurer

還有一個實現類就是 WebSecurityConfigurer,這個可能有的小伙伴比較陌生,其實他就是我們天天用的 WebSecurityConfigurerAdapter 的父接口。

所以 WebSecurityConfigurer 的作用就很明確了,用戶擴展用戶自定義的配置。

SecurityConfigurer 默認主要是這三個實現,考慮到大多數的過濾器配置都是通過 SecurityConfigurerAdapter 進行擴展的,因此我們今天就通過這條線進行展開。另外兩條線松哥也將擼兩篇文章和大家介紹。

2. SecurityConfigurerAdapter

SecurityConfigurerAdapter 的實現主要也是三大類:

  • UserDetailsAwareConfigurer
  • AbstractHttpConfigurer
  • LdapAuthenticationProviderConfigurer

考慮到 LDAP 現在使用很少,所以這里我來和大家重點介紹下前兩個。

2.1 UserDetailsAwareConfigurer

這個配置類看名字大概就知道這是用來配置用戶類的。

 

AbstractDaoAuthenticationConfigurer

AbstractDaoAuthenticationConfigurer 中所做的事情比較簡單,主要是構造了一個默認的 DaoAuthenticationProvider,并為其配置 PasswordEncoder 和 UserDetailsService。

UserDetailsServiceConfigurer

UserDetailsServiceConfigurer 重寫了 AbstractDaoAuthenticationConfigurer 中的 configure 方法,在 configure 方法執行之前加入了 initUserDetailsService 方法,以方便開發展按照自己的方式去初始化 UserDetailsService。不過這里的 initUserDetailsService 方法是空方法。

UserDetailsManagerConfigurer

UserDetailsManagerConfigurer 中實現了 UserDetailsServiceConfigurer 中定義的 initUserDetailsService 方法,具體的實現邏輯就是將 UserDetailsBuilder 所構建出來的 UserDetails 以及提前準備好的 UserDetails 中的用戶存儲到 UserDetailsService 中。

該類同時添加了 withUser 方法用來添加用戶,同時還增加了一個 UserDetailsBuilder 用來構建用戶,這些邏輯都比較簡單,小伙伴們可以自行查看。

JdbcUserDetailsManagerConfigurer

JdbcUserDetailsManagerConfigurer 在父類的基礎上補充了 DataSource 對象,同時還提供了相應的數據庫查詢方法。

InMemoryUserDetailsManagerConfigurer

InMemoryUserDetailsManagerConfigurer 在父類的基礎上重寫了構造方法,將父類中的 UserDetailsService 實例定義為 InMemoryUserDetailsManager。

DaoAuthenticationConfigurer

DaoAuthenticationConfigurer 繼承自 AbstractDaoAuthenticationConfigurer,只是在構造方法中修改了一下 userDetailsService 而已。

有小伙伴可能要問了,JdbcUserDetailsManagerConfigurer 或者 InMemoryUserDetailsManagerConfigurer,到底在哪里可以用到呀?

松哥給大家舉一個簡單的例子:

  1. @Configuration 
  2. public class SecurityConfig extends WebSecurityConfigurerAdapter { 
  3.  
  4.     @Override 
  5.     protected void configure(AuthenticationManagerBuilder auth) throws Exception { 
  6.         auth.inMemoryAuthentication().withUser("javaboy"
  7.                 .password("{noop}123"
  8.                 .roles("admin"); 
  9.     } 
  10.  
  11.     @Override 
  12.     protected void configure(HttpSecurity http) throws Exception { 
  13.         http.authorizeRequests() 
  14.                 .anyRequest().authenticated() 
  15.                 //省略 
  16.     } 

當你調用 auth.inMemoryAuthentication 進行配置時,實際上調用的就是 InMemoryUserDetailsManagerConfigurer。

這下明白了吧!

2.2 AbstractHttpConfigurer

AbstractHttpConfigurer 這一派中的東西非常多,我們所有的過濾器配置,都是它的子類,我們來看下都有哪些類?

 

可以看到,它的實現類還是非常多的。

這么多實現類,松哥就不一一給大家介紹了,我挑一個常用的 FormLoginConfigurer 來給大家詳細介紹,只要大家把這個理解了,其他的照貓畫虎就很好理解了。

我們一個一個來看。

2.2.1 AbstractHttpConfigurer

AbstractHttpConfigurer 繼承自 SecurityConfigurerAdapter,并增加了兩個方法,disable 和 withObjectPostProcessor:

  1. public abstract class AbstractHttpConfigurer<T extends AbstractHttpConfigurer<T, B>, B extends HttpSecurityBuilder<B>> 
  2.   extends SecurityConfigurerAdapter<DefaultSecurityFilterChain, B> { 
  3.  
  4.  /** 
  5.   * Disables the {@link AbstractHttpConfigurer} by removing it. After doing so a fresh 
  6.   * version of the configuration can be applied. 
  7.   * 
  8.   * @return the {@link HttpSecurityBuilder} for additional customizations 
  9.   */ 
  10.  @SuppressWarnings("unchecked"
  11.  public B disable() { 
  12.   getBuilder().removeConfigurer(getClass()); 
  13.   return getBuilder(); 
  14.  } 
  15.  
  16.  @SuppressWarnings("unchecked"
  17.  public T withObjectPostProcessor(ObjectPostProcessor<?> objectPostProcessor) { 
  18.   addObjectPostProcessor(objectPostProcessor); 
  19.   return (T) this; 
  20.  } 

這兩個方法松哥之前都有給大家介紹過,disable 基本上是大家的老熟人了,我們常用的 .csrf().disable() 就是出自這里,那么從這里我們也可以看到 disable 的實現原理,就是從 getBuilder 中移除相關的 xxxConfigurer,getBuilder 方法獲取到的實際上就是 HttpSecurity,所以移除掉 xxxConfigurer 實際上就是從過濾器鏈中移除掉某一個過濾器,例如 .csrf().disable() 就是移除掉處理 csrf 的過濾器。

另一個增加的方法是 withObjectPostProcessor,這是為配置類添加手動添加后置處理器的。在 AbstractHttpConfigurer 的父類中其實有一個類似的方法就是 addObjectPostProcessor,但是 addObjectPostProcessor 只是一個添加方法,返回值為 void,而 withObjectPostProcessor 的返回值是當前配置類,也就是 xxxConfigurer,所以如果使用 withObjectPostProcessor 的話,可以使用鏈式配置,事實上,在松哥之前的文章,以及 vhr(https://github.com/lenve/vhr) 項目中,使用的也都是 withObjectPostProcessor 方法(當然,你也可以使用 addObjectPostProcessor,最終效果是一樣的)。

2.2.2 AbstractAuthenticationFilter

ConfigurerAbstractAuthenticationFilterConfigurer 類的功能比較多,源碼也是相當相當長。不過我們只需要抓住兩點即可,init 方法和 configure 方法,因為這兩個方法是所有 xxxConfigurer 的靈魂。

  1. @Override 
  2. public void init(B http) throws Exception { 
  3.  updateAuthenticationDefaults(); 
  4.  updateAccessDefaults(http); 
  5.  registerDefaultAuthenticationEntryPoint(http); 

init 方法主要干了三件事:

  1. updateAuthenticationDefaults 主要是配置了登錄處理地址,失敗跳轉地址,注銷成功跳轉地址。
  2. updateAccessDefaults 方法主要是對 loginPage、loginProcessingUrl、failureUrl 進行 permitAll 設置(如果用戶配置了 permitAll 的話)。
  3. registerDefaultAuthenticationEntryPoint 則是注冊異常的處理器。

再來看 configure 方法:

  1. @Override 
  2. public void configure(B http) throws Exception { 
  3.  PortMapper portMapper = http.getSharedObject(PortMapper.class); 
  4.  if (portMapper != null) { 
  5.   authenticationEntryPoint.setPortMapper(portMapper); 
  6.  } 
  7.  RequestCache requestCache = http.getSharedObject(RequestCache.class); 
  8.  if (requestCache != null) { 
  9.   this.defaultSuccessHandler.setRequestCache(requestCache); 
  10.  } 
  11.  authFilter.setAuthenticationManager(http 
  12.    .getSharedObject(AuthenticationManager.class)); 
  13.  authFilter.setAuthenticationSuccessHandler(successHandler); 
  14.  authFilter.setAuthenticationFailureHandler(failureHandler); 
  15.  if (authenticationDetailsSource != null) { 
  16.   authFilter.setAuthenticationDetailsSource(authenticationDetailsSource); 
  17.  } 
  18.  SessionAuthenticationStrategy sessionAuthenticationStrategy = http 
  19.    .getSharedObject(SessionAuthenticationStrategy.class); 
  20.  if (sessionAuthenticationStrategy != null) { 
  21.   authFilter.setSessionAuthenticationStrategy(sessionAuthenticationStrategy); 
  22.  } 
  23.  RememberMeServices rememberMeServices = http 
  24.    .getSharedObject(RememberMeServices.class); 
  25.  if (rememberMeServices != null) { 
  26.   authFilter.setRememberMeServices(rememberMeServices); 
  27.  } 
  28.  F filter = postProcess(authFilter); 
  29.  http.addFilter(filter); 

configure 中的邏輯就很簡答了,構建各種各樣的回調函數設置給 authFilter,authFilter 再去 postProcess 中走一圈注冊到 Spring 容器中,最后再把 authFilter 添加到過濾器鏈中。

這便是 AbstractAuthenticationFilterConfigurer 的主要功能。需要提醒大家的是,我們日常配置的,如:

  • loginPage
  • loginProcessingUrl
  • permitAll
  • defaultSuccessUrl
  • failureUrl
  • ...

等方法都是在這里定義的。

最后我們再來看看 FormLoginConfigurer。

2.2.3 FormLoginConfigurer

FormLoginConfigurer 在定義是,明確了 AbstractAuthenticationFilterConfigurer 中的泛型是 UsernamePasswordAuthenticationFilter,也就是我們這里最終要配置的過濾是 UsernamePasswordAuthenticationFilter。

FormLoginConfigurer 重寫了 init 方法,配置了一下默認的登錄頁面。其他的基本上都是從父類來的,未做太多改變。

另外我們日常配置的很多東西也是來自這里:

 

好啦,這就是 FormLoginConfigurer 這個配置類,FormLoginConfigurer 對應的過濾器是 UsernamePasswordAuthenticationFilter,小伙伴們可以自行分析其他的 xxxConfigurer,每一個 xxxConfigurer 都對應了一個 不同的 Filter。

 

 

3.小結好啦,今天就主要和大家分享一下 SecurityConfigurer 的源碼,當然這里還有很多值得再次仔細討論的東西,松哥將在后面的文章中繼續和大家分享。

本文轉載自微信公眾號「江南一點雨」,可以通過以下二維碼關注。轉載本文請聯系江南一點雨公眾號。

 

責任編輯:武曉燕 來源: 江南一點雨
相關推薦

2010-06-01 15:25:27

JavaCLASSPATH

2016-12-08 15:36:59

HashMap數據結構hash函數

2023-10-19 11:12:15

Netty代碼

2021-02-17 11:25:33

前端JavaScriptthis

2009-09-25 09:14:35

Hibernate日志

2013-09-22 14:57:19

AtWood

2020-09-23 10:00:26

Redis數據庫命令

2019-06-25 10:32:19

UDP編程通信

2017-01-10 08:48:21

2024-02-21 21:14:20

編程語言開發Golang

2025-05-06 00:43:00

MySQL日志文件MIXED 3

2017-08-15 13:05:58

Serverless架構開發運維

2025-06-05 05:51:33

2015-11-04 09:57:18

JavaScript原型

2013-06-14 09:27:51

Express.jsJavaScript

2017-01-13 22:42:15

iosswift

2021-04-20 23:25:16

執行函數變量

2011-04-11 16:48:12

Solaris權限

2024-03-12 00:00:00

Sora技術數據

2022-11-04 09:43:05

Java線程
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: www.久草.com| 精品成人一区二区 | 国产欧美视频一区二区 | av在线免费看网址 | 国产激情精品视频 | 特级丰满少妇一级aaaa爱毛片 | 色视频网站在线观看 | 国产精品久久免费观看 | 久久国产一区二区三区 | 久久精品欧美视频 | 视频一区二区三区四区五区 | 一级黄色片网址 | 欧美jizzhd精品欧美巨大免费 | 中文字幕av在线 | 久久精品青青大伊人av | 中文字幕一级毛片 | 国产视频观看 | 狠狠热视频| 日本粉嫩一区二区三区视频 | 99精品视频免费观看 | 久久久www成人免费精品张筱雨 | 国产一区欧美 | 午夜精品福利视频 | www.激情.com | 精品一区二区在线观看 | 亚洲久久在线 | 国产黄色在线观看 | www.色.com| 亚洲一区二区三区在线播放 | 成人在线视频一区 | 色在线免费 | 国产一区二区免费 | 成人妇女免费播放久久久 | 精品欧美一区二区三区久久久 | 日韩在线免费看 | 亚洲中午字幕 | 夜夜骑首页 | 亚洲成人网在线 | 中文字幕av高清 | 欧美日韩在线观看一区二区三区 | 午夜精品久久久久久久久久久久 |