Spring Security驗(yàn)證流程剖析及自定義驗(yàn)證方法
Spring Security的本質(zhì)
Spring Security 本質(zhì)上是一連串的 Filter , 然后又以一個(gè)獨(dú)立的 Filter 的形式插入到 Filter Chain 里,其名為 FilterChainProxy 。 如圖所示。
實(shí)際上 FilterChainProxy 下面可以有多條 Filter Chain ,來(lái)針對(duì)不同的URL做驗(yàn)證,而 Filter Chain 中所擁有的 Filter 則會(huì)根據(jù)定義的服務(wù)自動(dòng)增減。所以無(wú)需要顯示再定義這些 Filter ,除非想要實(shí)現(xiàn)自己的邏輯。
關(guān)鍵類(lèi)
Authentication
Authentication 是一個(gè)接口,用來(lái)表示用戶(hù)認(rèn)證信息,在用戶(hù)登錄認(rèn)證之前相關(guān)信息會(huì)封裝為一個(gè) Authentication 具體實(shí)現(xiàn)類(lèi)的對(duì)象,在登錄認(rèn)證成功之后又會(huì)生成一個(gè)信息更全面,包含用戶(hù)權(quán)限等信息的 Authentication 對(duì)象,然后把它保存在 SecurityContextHolder 所持有的 SecurityContext 中,供后續(xù)的程序進(jìn)行調(diào)用,如訪(fǎng)問(wèn)權(quán)限的鑒定等。
AuthenticationManager
用來(lái)做驗(yàn)證的最主要的接口為 AuthenticationManager ,這個(gè)接口只有一個(gè)方法:
public interface AuthenticationManager {
Authentication authenticate(Authentication authentication)
throws AuthenticationException;
}
其中 authenticate() 方法運(yùn)行后可能會(huì)有三種情況:
驗(yàn)證成功,返回一個(gè)帶有用戶(hù)信息的 Authentication 。
驗(yàn)證失敗,拋出一個(gè) AuthenticationException 異常。
無(wú)法判斷,返回 null 。
ProviderManager
ProviderManager 是上面的 AuthenticationManager 最常見(jiàn)的實(shí)現(xiàn),它不自己處理驗(yàn)證,而是將驗(yàn)證委托給其所配置的 AuthenticationProvider 列表,然后會(huì)依次調(diào)用每一個(gè) AuthenticationProvider 進(jìn)行認(rèn)證,這個(gè)過(guò)程中只要有一個(gè) AuthenticationProvider 驗(yàn)證成功,就不會(huì)再繼續(xù)做更多驗(yàn)證,會(huì)直接以該認(rèn)證結(jié)果作為 ProviderManager 的認(rèn)證結(jié)果。
認(rèn)證過(guò)程
用戶(hù)使用用戶(hù)名和密碼進(jìn)行登錄。
Spring Security 將獲取到的用戶(hù)名和密碼封裝成一個(gè) Authentication 接口的實(shí)現(xiàn)類(lèi),比如常用的 UsernamePasswordAuthenticationToken 。
將上述產(chǎn)生的 Authentication 對(duì)象傳遞給 AuthenticationManager 的實(shí)現(xiàn)類(lèi) ProviderManager 進(jìn)行認(rèn)證。
ProviderManager 依次調(diào)用各個(gè) AuthenticationProvider 進(jìn)行認(rèn)證,認(rèn)證成功后返回一個(gè)封裝了用戶(hù)權(quán)限等信息的 Authentication 對(duì)象。
將 AuthenticationManager 返回的 Authentication 對(duì)象賦予給當(dāng)前的 SecurityContext 。
自定義驗(yàn)證
有了以上的知識(shí)儲(chǔ)備后就可以來(lái)自定義驗(yàn)證方法了。通過(guò)上面可以看出,實(shí)際上真正來(lái)做驗(yàn)證操作的是一個(gè)個(gè)的 AuthenticationProvider ,所以如果要自定義驗(yàn)證方法,只需要實(shí)現(xiàn)一個(gè)自己的 AuthenticationProvider 然后再將其添加進(jìn) ProviderManager 里就行了。
自定義AuthenticationProvider
@Component
public class CustomAuthenticationProvider
implements AuthenticationProvider {
@Override
public Authentication authenticate(Authentication authentication)
throws AuthenticationException {
String name = authentication.getName();
String password = authentication.getCredentials().toString();
if (shouldAuthenticateAgainstThirdPartySystem()) {
// use the credentials
// and authenticate against the third-party system
return new UsernamePasswordAuthenticationToken(
name, password, new ArrayList<>());
} else {
return null;
}
}
@Override
public boolean supports(Class<?> authentication) {
return authentication.equals(
UsernamePasswordAuthenticationToken.class);
}
}
其中的 supports() 方法接受一個(gè) authentication 參數(shù),用來(lái)判斷傳進(jìn)來(lái)的 authentication 是不是該 AuthenticationProvider 能夠處理的類(lèi)型。
注冊(cè)AuthenticationProvider
現(xiàn)在再將剛創(chuàng)建的 AuthenticationProvider 在 與ProviderManager 里注冊(cè),所有操作就完成了。
@Configuration
@EnableWebSecurity
@ComponentScan("org.baeldung.security")
public class SecurityConfig extends WebSecurityConfigurerAdapter {
@Autowired
private CustomAuthenticationProvider authProvider;
@Override
protected void configure(
AuthenticationManagerBuilder auth) throws Exception {
auth.authenticationProvider(authProvider);
}
@Override
protected void configure(HttpSecurity http) throws Exception {
http.authorizeRequests().anyRequest().authenticated()
.and()
.httpBasic();
}
}
總結(jié)
以上所述是小編給大家介紹的Spring Security驗(yàn)證流程剖析及自定義驗(yàn)證方法,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
Springboot如何使用Aspectj實(shí)現(xiàn)AOP面向切面編程
這篇文章主要介紹了Springboot如何使用Aspectj實(shí)現(xiàn)AOP面向切面編程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01
詳解java.lang.reflect.Modifier.isInterface()方法
這篇文章主要介紹了詳解java.lang.reflect.Modifier.isInterface()方法的相關(guān)資料,這里提供實(shí)例幫助大家理解這個(gè)方法的使用,需要的朋友可以參考下2017-09-09
Java創(chuàng)建二叉搜索樹(shù),實(shí)現(xiàn)搜索,插入,刪除的操作實(shí)例
下面小編就為大家分享一篇Java創(chuàng)建二叉搜索樹(shù),實(shí)現(xiàn)搜索,插入,刪除的操作實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助2017-12-12
Java中利用BitMap位圖實(shí)現(xiàn)海量級(jí)數(shù)據(jù)去重
有許多方法可以用來(lái)去重,比如使用列表、集合等等,但這些方法通常只適用于一般情況,然而,當(dāng)涉及到大量數(shù)據(jù)去重時(shí),常見(jiàn)的 Java Set、List,甚至是 Java 8 的新特性 Stream 流等方式就顯得不太合適了,本文給大家介紹了Java中利用BitMap位圖實(shí)現(xiàn)海量級(jí)數(shù)據(jù)去重2024-04-04
基于Java編寫(xiě)一個(gè)粽子大作戰(zhàn)小游戲
端午節(jié),又稱(chēng)龍舟節(jié)、重午節(jié),是中國(guó)的傳統(tǒng)節(jié)日之一,每年農(nóng)歷五月初五慶祝,雖然端午假期已經(jīng)過(guò)去了,小編還是用Java編寫(xiě)了一個(gè)粽子大作戰(zhàn)小游戲,感興趣的可以了解一下2023-06-06
Java實(shí)戰(zhàn)之藥品管理系統(tǒng)的實(shí)現(xiàn)
這篇文章主要介紹了利用Java實(shí)現(xiàn)的藥品管理系統(tǒng),本項(xiàng)目屬于前后端分離的項(xiàng)目,分為兩個(gè)角色藥品管理員和取藥處人員,感興趣的小伙伴可以學(xué)習(xí)一下2022-04-04
JDK9為何要將String的底層實(shí)現(xiàn)由char[]改成了byte[]
String 類(lèi)的源碼已經(jīng)由?char[]?優(yōu)化為了?byte[]?來(lái)存儲(chǔ)字符串內(nèi)容,為什么要這樣做呢?本文就詳細(xì)的介紹一下,感興趣的可以了解一下2022-03-03

