Spring Security實(shí)現(xiàn)不同接口安全策略方法詳解
1. 前言
歡迎閱讀 Spring Security 實(shí)戰(zhàn)干貨 系列文章 。最近有開(kāi)發(fā)小伙伴提了一個(gè)有趣的問(wèn)題。他正在做一個(gè)項(xiàng)目,涉及兩種風(fēng)格,一種是給小程序出接口,安全上使用無(wú)狀態(tài)的JWT Token;另一種是管理后臺(tái)使用的是Freemarker,也就是前后端不分離的Session機(jī)制。用Spring Security該怎么辦?
2. 解決方案
我們可以通過(guò)多次繼承WebSecurityConfigurerAdapter構(gòu)建多個(gè)HttpSecurity。HttpSecurity 對(duì)象會(huì)告訴我們?nèi)绾悟?yàn)證用戶的身份,如何進(jìn)行訪問(wèn)控制,采取的何種策略等等。
我們是這么配置的:
/**
* 單策略配置
*
* @author felord.cn
* @see org.springframework.boot.autoconfigure.security.servlet.SpringBootWebSecurityConfiguration
* @since 14 :58 2019/10/15
*/
@Configuration
@EnableGlobalMethodSecurity(prePostEnabled = true, jsr250Enabled = true, securedEnabled = true)
@EnableWebSecurity
@ConditionalOnClass(WebSecurityConfigurerAdapter.class)
@ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.SERVLET)
public class CustomSpringBootWebSecurityConfiguration {
/**
* The type Default configurer adapter.
*/
@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) {
super.configure(web);
}
@Override
protected void configure(HttpSecurity http) throws Exception {
// 配置 httpSecurity
}
}
}
上面的配置了一個(gè)HttpSecurity,我們?nèi)绶ㄅ谥圃僭黾右粋€(gè)WebSecurityConfigurerAdapter的子類(lèi)來(lái)配置另一個(gè)HttpSecurity。伴隨而來(lái)的還有不少的問(wèn)題要解決。
2.1 如何路由不同的安全配置
我們配置了兩個(gè)HttpSecurity之后,程序如何讓小程序接口和后臺(tái)接口走對(duì)應(yīng)的HttpSecurity?
HttpSecurity.antMatcher(String antPattern)可以提供過(guò)濾機(jī)制。比如我們配置:
@Override
protected void configure(HttpSecurity http) throws Exception {
// 配置 httpSecurity
http.antMatcher("/admin/v1");
}
那么該HttpSecurity將只提供給以/admin/v1開(kāi)頭的所有URL。這要求我們針對(duì)不同的客戶端指定統(tǒng)一的URL前綴。
舉一反三只要HttpSecurity提供的功能都可以進(jìn)行個(gè)性化定制。比如登錄方式,角色體系等。
2.2 如何指定默認(rèn)的 HttpSecurity
我們可以通過(guò)在WebSecurityConfigurerAdapter實(shí)現(xiàn)上使用@Order注解來(lái)指定優(yōu)先級(jí),數(shù)值越大優(yōu)先級(jí)越低,沒(méi)有@Order注解將優(yōu)先級(jí)最低。
2.3 如何配置不同的 UserDetailsService
很多情況下我們希望普通用戶和管理用戶完全隔離,我們就需要多個(gè)UserDetailsService,你可以在下面的方法中對(duì)AuthenticationManagerBuilder進(jìn)行具體的設(shè)置來(lái)配置UserDetailsService,同時(shí)也可以配置不同的密碼策略。
@Override
protected void configure(AuthenticationManagerBuilder auth) throws Exception {
DaoAuthenticationProvider daoAuthenticationProvider = new DaoAuthenticationProvider();
daoAuthenticationProvider.setUserDetailsService(new UserDetailsService() {
@Override
public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
// 自行實(shí)現(xiàn)
return null ;
}
});
// 也可以設(shè)計(jì)特定的密碼策略
BCryptPasswordEncoder bCryptPasswordEncoder = new BCryptPasswordEncoder();
daoAuthenticationProvider.setPasswordEncoder(bCryptPasswordEncoder);
auth.authenticationProvider(daoAuthenticationProvider);
}
2.4 最終的配置模板
上面的幾個(gè)問(wèn)題解決之后,我們基本上掌握了在一個(gè)應(yīng)用中執(zhí)行多種安全策略。配置模板如下:
/**
* 多個(gè)策略配置
*
* @author felord.cn
* @see org.springframework.boot.autoconfigure.security.servlet.SpringBootWebSecurityConfiguration
* @since 14 :58 2019/10/15
*/
@Configuration
@EnableGlobalMethodSecurity(prePostEnabled = true, jsr250Enabled = true, securedEnabled = true)
@EnableWebSecurity
@ConditionalOnClass(WebSecurityConfigurerAdapter.class)
@ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.SERVLET)
public class CustomSpringBootWebSecurityConfiguration {
/**
* 后臺(tái)接口安全策略. 默認(rèn)配置
*/
@Configuration
@Order(1)
static class AdminConfigurerAdapter extends WebSecurityConfigurerAdapter {
@Override
protected void configure(AuthenticationManagerBuilder auth) throws Exception {
DaoAuthenticationProvider daoAuthenticationProvider = new DaoAuthenticationProvider();
//用戶詳情服務(wù)個(gè)性化
daoAuthenticationProvider.setUserDetailsService(new UserDetailsService() {
@Override
public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
// 自行實(shí)現(xiàn)
return null;
}
});
// 也可以設(shè)計(jì)特定的密碼策略
BCryptPasswordEncoder bCryptPasswordEncoder = new BCryptPasswordEncoder();
daoAuthenticationProvider.setPasswordEncoder(bCryptPasswordEncoder);
auth.authenticationProvider(daoAuthenticationProvider);
}
@Override
public void configure(WebSecurity web) {
super.configure(web);
}
@Override
protected void configure(HttpSecurity http) throws Exception {
// 根據(jù)需求自行定制
http.antMatcher("/admin/v1")
.sessionManagement(Customizer.withDefaults())
.formLogin(Customizer.withDefaults());
}
}
/**
* app接口安全策略. 沒(méi)有{@link Order}注解優(yōu)先級(jí)比上面低
*/
@Configuration
static class AppConfigurerAdapter extends WebSecurityConfigurerAdapter {
@Override
protected void configure(AuthenticationManagerBuilder auth) throws Exception {
DaoAuthenticationProvider daoAuthenticationProvider = new DaoAuthenticationProvider();
//用戶詳情服務(wù)個(gè)性化
daoAuthenticationProvider.setUserDetailsService(new UserDetailsService() {
@Override
public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
// 自行實(shí)現(xiàn)
return null;
}
});
// 也可以設(shè)計(jì)特定的密碼策略
BCryptPasswordEncoder bCryptPasswordEncoder = new BCryptPasswordEncoder();
daoAuthenticationProvider.setPasswordEncoder(bCryptPasswordEncoder);
auth.authenticationProvider(daoAuthenticationProvider);
}
@Override
public void configure(WebSecurity web) {
super.configure(web);
}
@Override
protected void configure(HttpSecurity http) throws Exception {
// 根據(jù)需求自行定制
http.antMatcher("/app/v1")
.sessionManagement(Customizer.withDefaults())
.formLogin(Customizer.withDefaults());
}
}
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- SpringSecurity 自定義表單登錄的實(shí)現(xiàn)
- 解析SpringSecurity自定義登錄驗(yàn)證成功與失敗的結(jié)果處理問(wèn)題
- spring security自定義認(rèn)證登錄的全過(guò)程記錄
- spring security自定義登錄頁(yè)面
- Spring Security保護(hù)用戶密碼常用方法詳解
- Spring Security UserDetails實(shí)現(xiàn)原理詳解
- SpringSecurity自定義成功失敗處理器的示例代碼
- Spring Security如何優(yōu)雅的增加OAuth2協(xié)議授權(quán)模式
- Spring Security自定義登錄原理及實(shí)現(xiàn)詳解
相關(guān)文章
MyBatis-Flex實(shí)現(xiàn)分頁(yè)查詢的示例代碼
在MyBatis-Flex中實(shí)現(xiàn)分頁(yè)查詢時(shí),需要注意維護(hù)一個(gè)獲取數(shù)據(jù)庫(kù)總數(shù)的方法,詳細(xì)介紹了UserService、UserServiceImpl類(lèi)以及Mapper.xml配置,感興趣的可以了解一下2024-10-10
詳解Spring?中?Bean?對(duì)象的存儲(chǔ)和取出
由于?Spring?擁有對(duì)象的管理權(quán),所以我們也需要擁有較為高效的對(duì)象存儲(chǔ)和取出的手段,下面我們來(lái)分別總結(jié)一下,對(duì)Spring?中?Bean?對(duì)象的存儲(chǔ)和取出知識(shí)感興趣的朋友跟隨小編一起看看吧2022-11-11
搭建簡(jiǎn)單的Spring-Data JPA項(xiàng)目
本文主要介紹了搭建簡(jiǎn)單的Spring-Data JPA項(xiàng)目,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02
SpringBoot整合FastDFS方法過(guò)程詳解
這篇文章主要介紹了SpringBoot整合FastDFS方法過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05
java實(shí)現(xiàn)voctor按指定方式排序示例分享
這篇文章主要介紹了java實(shí)現(xiàn)voctor按指定方式排序示例,需要的朋友可以參考下2014-03-03
springboot項(xiàng)目中常用的工具類(lèi)和api詳解
在Spring Boot項(xiàng)目中,開(kāi)發(fā)者通常會(huì)依賴一些工具類(lèi)和API來(lái)簡(jiǎn)化開(kāi)發(fā)、提高效率,以下是一些常用的工具類(lèi)及其典型應(yīng)用場(chǎng)景,涵蓋 Spring 原生工具、第三方庫(kù)(如Hutool、Guava) 和 Java 自帶工具,本文給大家介紹springboot項(xiàng)目中常用的工具類(lèi)和api,感興趣的朋友一起看看吧2025-04-04
詳解Spring Cloud Hystrix斷路器實(shí)現(xiàn)容錯(cuò)和降級(jí)
本篇文章主要介紹了詳解Spring Cloud Hystrix斷路器實(shí)現(xiàn)容錯(cuò)和降級(jí),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05
java中map與實(shí)體類(lèi)的相互轉(zhuǎn)換操作
這篇文章主要介紹了java中map與實(shí)體類(lèi)的相互轉(zhuǎn)換操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07

