저는 오픈소스 컨트리뷰션 아카데미 및 전자정부 프레임워크 기여에서 자신감을 얻고 Spring Security에 대해 기여를 해보려 합니다!
저는 이전 전자정부 표준 프레임워크에 기여했던 경험이 있지만 이는 docs에 대한 기여라서 그렇게 코드를 읽고 본격적인 기여는 아니었습니다.
하지만 본격적인 기여는 처음이니 열심히 진행해보고 이를 기반으로 제 개발 생활에 스며들도록 해보려 합니다!
저는 다음의 포스팅을 참고하여 이슈 선정을 하였습니다!
https://medium.com/opensource-contributors/오픈소스-멘토링-기여-가이드-오픈소스-멘토링에서-10명-넘는-오픈소스-컨트리뷰터가-첫-기여를-성공할-수-있었던-방법-3ff09c9b6f83
이렇게 하여 선택된 것은 Spring Security의 #15817 이슈입니다!
해당 이슈의 링크입니다!
이제 해당 이슈에 대한 기여를 시작해보겠습니다!
이슈 파악하기
제가 이 이슈를 선택한 것은 Spring Security를 자주 사용하고 해당 기술에 더 익숙해지기 위함도 있습니다.
해당 이슈는 다음과 같은 내용을 가지고 있습니다.
To active OIDC Back-Channel Logout support in the DSL, an application does this:
http
.oidcLogout((oidc) -> oidc.backChannel(Customizer.withDefaults())
)
This could be simplified to:
http
.oidcBackChannelLogout(Customizer.withDefaults())
This would be place the logout DSL at the same level as other logout DSLs:
http
.logout((logout) -> logout ...)
.saml2Logout((saml2) -> saml2 ...
.oidcBackChannelLogout((oidc) -> oidc ...)
Also, it’s less nesting which often makes the DSL more navigable.
This would mean deprecating the existing backChannel
DSL method with the intent to remove in the next major version.
우선 해당 이슈를 해결하기 위해 기반 지식들을 학습하였습니다.
학습 내용은 해당 링크에 작성되어 있습니다! 이에 대해 읽고 오시면 이해가 빠르실 것 같습니다!
해당 이슈를 살펴보면 다음과 같습니다.
이슈 설명
- Spring Security는 OIDC 백채널 로그아웃을 지원하여 애플리케이션이 인증 서버로부터 로그아웃 요청을 받아 세션을 종료할 수 있도록 합니다.
- 현재는 oidcLogout 메소드 안에 backChannel 메소드를 호출하여 다음과 같이 백채널 로그아웃을 구성합니다.
http
.oidcLogout((oidc) -> oidc.backChannel(Customizer.withDefaults())
)
하지만 이는 DSL 구문이 중첩되어 있어 사용자가 설정하기 헷갈릴 수 있고 다른 로그아웃 DSL과 비교하여 일관성이 부족하다는 문제가 있습니다.
이슈 목적
해당 이슈의 목적은 위의 문제를 해결하기 위해 DSL을 단순화하여 다음과 같이 oidcBackChannelLogout 메소드를 직접 호출할 수 있도록 합니다.
http
.oidcBackChannelLogout(Customizer.withDefaults());
이와 같이 단순화한다면 oidcBackChannelLogout을 다른 로그아웃 DSL과 동일한 수준에 배치할 수 있습니다.
http
.logout((logout) -> logout ...)
.saml2Logout((saml2) -> saml2 ...)
.oidcBackChannelLogout((oidc) -> oidc ...);
이슈 할당받기
저는 이슈를 해결하기에 앞서 제가 해당 이슈를 먼저 해결하기 전 다른 사람이 이슈를 해결하거나 이슈가 닫힐 수도 있어서 다음과 같이 해당 이슈를 해결해도 되는지 물어본 후 해당 이슈를 할당받았습니다.
data:image/s3,"s3://crabby-images/60047/60047b5647f9b868f588d5663f3f384e01847b4c" alt="image"
10월 1일에 코멘트를 달았는데 11월 초에 답글을 달아주었습니다.
오픈 소스 컨트리뷰션 아카데미에서 이런거 일 처리가 굉장히 느긋하게 되니까 천천히 마음먹고 하라는 얘기를 들었는데 이정도일 줄은 몰랐습니다..
그러면 이에 대한 구현을 해보겠습니다.
기여 시작
브랜치 생성하기
저는 단순히 Github Repository를 fork하여 제 Main Branch에서 작업하려 했습니다.
하지만 다른 사람들이 올린 PR을 살펴봤을 때 다음과 같이 작업한 것을 확인할 수 있습니다.
data:image/s3,"s3://crabby-images/3d246/3d246d60b5c35920fce0e52ff6f197d5c9f22d90" alt="image"
data:image/s3,"s3://crabby-images/9d896/9d8962011399fdb56f64ccdd9fa4222c8500f3ae" alt="image"
다음과 같이 진행되는 관습을 따라 저도 브랜치를 생성하여 진행하려 합니다.
곁눈질로 배운거지만 어떤 이슈 번호에 대한 브랜치인지 파악할 수 있어서 좋은 것 같습니다!
코드 구성 확인하기
스프링 시큐리티의 폴더 구조는 굉장히 복잡합니다..
data:image/s3,"s3://crabby-images/11a1b/11a1b38c3ad2b0adaa491691cb6aec5c55233245" alt="image"
이에 대한 모든 구조를 파악하기는 힘들 것 같아 해당 oidcBackChannelLogout을 다른 로그아웃과 동일한 DSL 수준에 두는 것이 목표이니 다른 로그아웃의 디렉토리를 확인해주었습니다.
data:image/s3,"s3://crabby-images/c7c2e/c7c2e56fc43257d88c126a228b795a0da50fdb97" alt="image"
제가 사용하는 Security Config에서 logout으로 들어가보면 다음과 같습니다.
data:image/s3,"s3://crabby-images/72bd6/72bd609e8793a4ce1157eb337f5eef9bdf71828d" alt="image"
가장 위로 올라가서 패키지 구조를 보면 다음과 같이 되어 있습니다.
package org.springframework.security.config.annotation.web.builders;
해당 디렉토리의 HttpSecurity 클래스에 존재하는 것을 확인할 수 있습니다.
Spring Security로 들어가 해당 폴더를 확인해보겠습니다.
data:image/s3,"s3://crabby-images/5e32e/5e32ea94fafdbf65bf7c136b08255391f6eeaae8" alt="image"
해당 HttpSecurity 내에 기존 코드인 oidcLogout이 있는 것을 확인할 수 있습니다.
data:image/s3,"s3://crabby-images/52f48/52f484d2890f9a062e2f40806e1b54e3f2ed2e0b" alt="image"
저의 목표는 이와 같은 레벨에 oidcBackChannelLogout 메서드를 만들고 메서드 체이닝으로 사용 가능하도록 만들어야 할 것 같습니다.
그렇다면 기존 코드를 분석해보도록 하겠습니다.
기존 코드 분석
기존 코드는 다음과 같습니다.
http
.oidcLogout((oidc) -> oidc.backChannel(Customizer.withDefaults())
)
HttpSecurity
우선 SecurityConfig의 filterChain에서 자주 쓰는 http는 저희가 위에서 봤던 HttpSecurity의 객체입니다.
data:image/s3,"s3://crabby-images/f92fb/f92fb4edefd8270e3ae705908dd6b9d2a59e33b3" alt="image"
그렇다면 HttpSecurity에 대해서 알아보겠습니다.
Spring Docs의 HttpSecurity의 공식 문서에서는 HttpSecurity를 다음과 같이 설명하고 있습니다.
A HttpSecurity
is similar to Spring Security’s XML element in the namespace configuration. It allows configuring web based security for specific http requests. By default it will be applied to all requests, but can be restricted using `#requestMatcher(RequestMatcher)` or other similar methods.
공식 문서에 대한 내용을 정리해보면 다음과 같습니다.
- HttpSecurity 클래스는 특정 HTTP 요청에 대해 웹 보안을 구성할 수 있는 기능을 제공한다.
- HttpSecurity 클래스는 Spring Security에서 XML 설정의 요소와 유사한 역할을 한다.
HttpSecurity 클래스는 보통 다음과 같이 사용됩니다.
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception {
http
.authorizeHttpRequests()
.requestMatchers("/**")
.hasRole("USER")
.and()
.formLogin();
return http.build();
}
Builder 패턴을 사용하고 있습니다.
그래서 저는 이 build() 메서드가 어떻게 설정되어 있는지 궁금해서 코드 안쪽으로 들어가보았습니다.
HttpSecurity의 build() 메서드
우선 위에서 securityFilterChain이 http.build();
를 통해 반환하는 객체를 보면 SecurityFilterChain 타입을 가지고 있습니다.
또한 이를 찾기 위해 HttpSecurity에서 build() 메서드를 찾아봤는데 해당 메서드가 없어서 HttpSecurity의 부모 클래스인 AbstractConfiguredSecurityBuilder 클래스를 보았습니다.
data:image/s3,"s3://crabby-images/d0e9c/d0e9c9838a7b1e06cd13642246ddc0015f32471d" alt="image"
그런데 또 AbstractConfiguredSecurityBuilder 클래스에도 build 메서드가 존재하지 않아 이의 부모 클래스인 AbstractSecurityBuilder를 찾아보았습니다.
data:image/s3,"s3://crabby-images/306b4/306b4bb639dc3cd5da9f8653a1ec668a3fc12e3c" alt="image"
그리하여 결국 AbstractSecurityBuilder 클래스에서 build() 메서드를 찾을 수 있었습니다.
data:image/s3,"s3://crabby-images/eb16d/eb16dfe7169daa04846c4f700adde230795e9cb1" alt="image"
우선 AbstractSecurityBuilder 클래스는 SecurityBuilder 인터페이스를 구현한 추상 클래스로, 객체를 빌드하는 로직을 제공하고 일부 구현을 하위 클래스에 위임하고 있습니다.
object 변수는 빌드된 최종 객체를 저장하는 필드이고 build 메서드에서 최종 객체를 할당하고 반환하는 데에 사용하고 있습니다.
또한 build() 메서드는 final 키워드를 통해 하위 클래스에서 오버라이드하지 못하도록 하고 있습니다.
if 문을 통해 이전에 빌드가 시작되지 않은 경우에만 빌드를 시작할 수 있도록 하고 이 조건이 충족되면 빌드가 시작됩니다.
빌드가 시작되면 doBuild() 메서드를 호출하여 실제 빌드 작업을 수행하고 결과를 object에 저장하고 반환됩니다.
여기서 주목해야 할 것은 doBuild() 메서드입니다.
doBuild()
data:image/s3,"s3://crabby-images/700b3/700b309d68bced125e9191fd2e3c5337592a47d1" alt="image"
doBuilder는 protected를 통해 같은 패키지나 하위 클래스에서만 접근할 수 있도록 하였고 abstract를 통해 하위 클래스에서 반드시 이 메서드를 구현하도록 하였습니다.
결국 build() 메서드에서 doBuild()를 호출하여 객체를 생성하고 빌드 과정에서 수행할 특정한 작업은 doBuilder 메서드에서 정의한다는 것을 알 수 있습니다.
그러면 AbstractSecurityBuilder의 하위 클래스였던 AbstractConfiguredSecurityBuilder에서 doBuild() 메서드를 찾아보겠습니다.
doBuild를 소개하는 공식 문서 링크입니다.
data:image/s3,"s3://crabby-images/5c3ee/5c3ee429e637d0ef1f63444a6f2d1d265e70cb1a" alt="image"
AbstractConfiguredSecurityBuilder에서 doBuild() 코드는 다음과 같이 구성되어 있습니다.
@Override
protected final O doBuild() throws Exception {
synchronized (this.configurers) {
this.buildState = BuildState.INITIALIZING;
beforeInit();
init();
this.buildState = BuildState.CONFIGURING;
beforeConfigure();
configure();
this.buildState = BuildState.BUILDING;
O result = performBuild();
this.buildState = BuildState.BUILT;
return result;
}
}
doBuild() 메서드는 보안 설정을 빌드하기 위한 여러 단계의 작업을 순차적으로 수행하는 메서드입니다.
우선 이 메서드를 더 잘 이해하기 위해 BuildState를 알아보겠습니다.
BuildState
BuildState는 애플리케이션의 빌드 상태를 나타내기 위한 enum입니다.
BuildState는 다음과 같이 구성되어 있습니다.
private enum BuildState {
UNBUILT(0),
INITIALIZING(1),
CONFIGURING(2),
BUILDING(3),
BUILT(4);
private final int order;
BuildState(int order) {
this.order = order;
}
public boolean isInitializing() {
return INITIALIZING.order == this.order;
}
public boolean isConfigured() {
return this.order >= CONFIGURING.order;
}
}
가독성을 위해 주석은 제거하였고 각 상태에 대한 설명은 다음과 같습니다.
- UNBUILT (0)
빌드가 시작되기 전의 초기 상태입니다. Builder의 build() 메서드가 호출되기 전의 상태입니다.
- INITIALIZING (1)
빌드가 시작된 후 SecurityConfigurer의 init() 메서드들이 호출되는 동안의 상태입니다.
이 상태에서는 SecurityConfigurer를 초기화하여 빌드에 필요한 준비 작업을 수행합니다.
- CONFIGURING (2)
SecurityConfigurer의 모든 init() 메서드가 완료된 후, configure() 메서드들이 호출되는 동안의 상태입니다.
빌드 구성을 설정하는 단계로, 보안 설정이 이 상태에서 적용됩니다.
- BUILDING (3)
SecurityConfigurer의 모든 configure() 메서드가 완료된 후부터, performBuild() 메서드를 통해 실제 객체가 빌드되기 전까지의 상태입니다.
빌드가 진행되는 상태로, 필요한 설정이 완료되었고 최종 객체 생성 단계로 진입하는 시점입니다.
- BUILT (4)
객체가 완전히 빌드된 상태를 나타냅니다.
performBuild() 메서드 호출이 완료되고, 최종 객체가 반환된 후의 상태입니다.
이렇게 빌드의 단계를 명확히 구분하여 상태를 관리하면 현재 빌드 단계를 알고 빌드 프로세스를 제어할 수 있을 것 같습니다.
또한 에러 방지에도 중요한 역할을 할 수 있을 것 같습니다!
그러면 다시 AbstractConfiguredSecurityBuilder에서 doBuild()로 돌아가 코드를 살펴보겠습니다.
다시 doBuild()
@Override
protected final O doBuild() throws Exception {
synchronized (this.configurers) {
this.buildState = BuildState.INITIALIZING;
beforeInit();
init();
this.buildState = BuildState.CONFIGURING;
beforeConfigure();
configure();
this.buildState = BuildState.BUILDING;
O result = performBuild();
this.buildState = BuildState.BUILT;
return result;
}
}
우선 doBuild() 메서드가 실행되면 synchronized 블록을 사용하여 다중 스레드 환경에서 동시 접근을 제어하고 코드를 수행합니다.
첫 번째로 BuildState를 INITIALIZING로 설정하고 beforeInit() 메서드와 init() 메서드를 실행합니다.
beforeInit()과 Init() 메서드는 다음과 같습니다.
data:image/s3,"s3://crabby-images/9b0df/9b0df65355238ef16538af06b0ada6ee93e33c5f" alt="image"
data:image/s3,"s3://crabby-images/6431a/6431acc3523bd36a93893ec7ed64cf7ff788264b" alt="image"
우선 beforeInit() 메서드는 init() 메서드가 실행되기 전 하위 클래스가 오버라이드하여 추가적인 초기화 작업을 수행할 수 있도록 해주는 메서드인 것 같습니다.
그렇다면 init을 살펴보겠습니다.
init()
init() 메서드는 처음에 getConfigurers()를 통해 SecurityConfigurer 객체들의 컬렉션을 가져오고 있습니다.
getConfigurers() 메서드는 다음과 같습니다.
data:image/s3,"s3://crabby-images/8ad80/8ad805fab5e955a0c5ae98d565f07fa4df6b52f3" alt="image"
getConfigurers()는 우선 SecurityConfigurer에 대한 리스트를 만듭니다.
이후 configurers.values()를 차례차례 가져오며 result에 저장하고 이를 반환합니다.
configurers는 다음과 같습니다.
private final LinkedHashMap<Class<? extends SecurityConfigurer<O, B>>, List<SecurityConfigurer<O, B>>> configurers = new LinkedHashMap<>();
configurers는 LinkedHashMap이며 Key로 Class<? extends SecurityConfigurer<O, B»를 가지고 Value로 List<SecurityConfigurer<O, B»를 가집니다.
Class<? extends SecurityConfigurer<O, B»는 SecurityConfigurer의 클래스 타입을 나타내는 Class 객체인 것 같은데 구조가 조금 어렵습니다.
하나씩 뜯어보겠습니다.
이는 제네릭 타입의 Class 객체를 나타내는 것입니다!
제네릭 타입을 가지는 SecurityConfigurer 클래스 또는 인터페이스를 의미하는 것 같습니다. 제네릭 타입을 통해 이 클래스에서 사용할 매개변수를 나타내고 있습니다.
- ? extends SecurityConfigurer<O, B>
이는 와일드카드(?)를 사용해 SecurityConfigurer의 하위 타입을 받는다는 의미를 가지고 있습니다.
이를 종합해보면 제네릭 타입을 가지는 SecurityConfigurer 자체나 하위 클래스들의 Class 객체를 나타낸다!로 정리해볼 수 있을 것 같습니다.
그렇다면 이 configurers의 Key는 위와 같은 의미를 가지고 Value는 SecurityConfigurer의 List를 가지는 구조인 것 같습니다.
다시 init()으로 돌아가보겠습니다.
private Collection<SecurityConfigurer<O, B>> getConfigurers() {
List<SecurityConfigurer<O, B>> result = new ArrayList<>();
for (List<SecurityConfigurer<O, B>> configs : this.configurers.values()) {
result.addAll(configs);
}
return result;
}
이제 getConfigurers()의 의미를 이해할 수 있을 것 같습니다.
configurers에서 모든 SecurityConfigurer 객체들을 모아서 List에 저장하고 반환한다!
그렇다면 다시 Init() 메서드로 가보겠습니다.
다시 init()
private void init() throws Exception {
Collection<SecurityConfigurer<O, B>> configurers = getConfigurers();
for (SecurityConfigurer<O, B> configurer : configurers) {
configurer.init((B) this);
}
for (SecurityConfigurer<O, B> configurer : this.configurersAddedInInitializing) {
configurer.init((B) this);
}
}
이제 조금 이해가 되는 것 같습니다.
configurers에 SecurityConfigurer 객체들을 모두 담고 이를 한 개씩 반복하여 configurer.init((B) this);
를 실행한다!
여기서 실행되는 SecurityConfigurer의 init()은 다음과 같습니다.
data:image/s3,"s3://crabby-images/0428a/0428af010923f45b44bf4e89c990801a281ed4f5" alt="image"
SecurityBuilder의 초기화 작업을 수행하기 위한 메서드로 설정이 올바르게 작동할 수 있도록 준비하는 역할이라고 합니다!
그리고 하위의 반복문은 configurersAddedInInitializing을 반복하며 SecurityConfigurer 객체들에 대한 초기화를 진행해주고 있습니다.
configurersAddedInInitializing 필드는 다음과 같습니다.
private final List<SecurityConfigurer<O, B>> configurersAddedInInitializing = new ArrayList<>();
data:image/s3,"s3://crabby-images/3fe3a/3fe3a197bc574f1035026953c78d90c7f4310288" alt="image"
이 필드가 사용되는 곳을 확인해보면 add, remove, 현재 메서드가 있는데 add() 메서드는 주어진 SecurityConfigurer 객체를 검증하고 configurersAddedInInitializing 필드에 추가하는 역할을 하고 있습니다.
종합해보면 init() 메서드는 SecurityConfigurer 객체들을 모두 가져와 초기화한다고 생각하면 될 것 같습니다!!
다시다시 doBuild()
@Override
protected final O doBuild() throws Exception {
synchronized (this.configurers) {
this.buildState = BuildState.INITIALIZING;
beforeInit();
init();
this.buildState = BuildState.CONFIGURING;
beforeConfigure();
configure();
this.buildState = BuildState.BUILDING;
O result = performBuild();
this.buildState = BuildState.BUILT;
return result;
}
}
그러면 이제 doBuild()의 상위 3줄까지를 정리해보면 다음과 같을 것 같습니다.
buildState를 INITIALIZING로 변경하고 하위 클래스에서 추가한 초기화 작업을 실행한 뒤 SecurityConfigurer 객체들에 대한 초기화를 진행한다!
이제 아래로 내려가 보겠습니다.
buildState를 INITIALIZING로 바꿔주고 있습니다.
위에서 INITIALIZING는 빌드가 시작된 후 SecurityConfigurer의 init() 메서드들이 호출되는 동안의 상태라고 했습니다.
그 이후의 단계는 위에서 init()을 해줬던 것과 유사하게 진행되고 있습니다.
beforeConfigure()과 configure() 메서드를 보겠습니다.
data:image/s3,"s3://crabby-images/c48b6/c48b656fb65190ab29eefffcfbf9df9c4367d5cc" alt="image"
data:image/s3,"s3://crabby-images/eaa63/eaa6361bb47405edbd8269149ee71c5754ae4549" alt="image"
우선 beforeConfigure() 메서드는 beforeInit()과 유사하게 SecurityConfigurer 객체의 configure() 메서드가 호출되기 전에 실행되며, 설정 작업이 시작되기 전에 추가 작업을 수행할 수 있도록 하는 메서드입니다.
configure() 메서드는 init()과 동일하게 getConfigurers()를 통해 SecurityConfigurer 객체를 가져오고 해당 객체의 configure() 작업을 실행한다고 이해할 수 있을 것 같습니다.
data:image/s3,"s3://crabby-images/34228/34228a9fd467f33707afb163a492b940564a4b75" alt="image"
또한 이 곳에서의 configure()는 SecurityBuilder 객체의 속성을 설정하여 보안 구성을 적용하는 역할을 한다고 합니다.
정리해보면 configure() 메서드는 보안 설정의 구체적인 작업을 수행하는 단계입니다.
다시다시다시 doBuild()
@Override
protected final O doBuild() throws Exception {
synchronized (this.configurers) {
this.buildState = BuildState.INITIALIZING;
beforeInit();
init();
this.buildState = BuildState.CONFIGURING;
beforeConfigure();
configure();
this.buildState = BuildState.BUILDING;
O result = performBuild();
this.buildState = BuildState.BUILT;
return result;
}
}
그렇다면 configure에 관한 코드도 이해가 됩니다.
이에 대한 로직을 정리해보면 다음과 같이 정리해볼 수 있을 것 같습니다.
buildState를 CONFIGURING로 변경하고 하위 클래스에서 추가한 보안 설정을 수행한 뒤 SecurityConfigurer 객체들에 대한 보안 설정을 진행한다!
그 후로는 BuildState를 BUILDING으로 변경해주고 performBuild() 메서드의 결과를 result에 넣어주고 있습니다.
perfomeBuild()는 다음과 같습니다.
data:image/s3,"s3://crabby-images/3626a/3626a8c1102241fa858850727e30b71aa47181c8" alt="image"
perfomeBuild()는 abstract로 하위 클래스에서 구현해야하고 최종적인 보안 객체인 SecurityFilterChain 객체를 빌드하고 반환하는 메서드입니다.
그 다음 BuildState를 BUILT로 변환해주고 최종 보안 객체를 반환해줍니다.
다시 build()
data:image/s3,"s3://crabby-images/0f1f6/0f1f6d3edc7cd02e1749f7acf154ab80a06084b4" alt="image"
다시 AbstractSecurityBuilder의 build()를 보면 또 이해가 되는 것 같습니다.
if 문을 통해 이전에 빌드가 시작되지 않은 경우에 object에 최종 보안 객체를 doBuild() 메서드에서 반환받아 넣고 이를 반환하는 메서드라고 설명할 수 있을 것 같습니다.
정리
저희는 SecurityConfig에서 다음과 같이 사용한다고 했었습니다.
@Bean
public SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception {
http
.authorizeHttpRequests()
.requestMatchers("/**")
.hasRole("USER")
.and()
.formLogin();
return http.build();
}
이는 메서드 체이닝을 통해 HttpSecurity 객체에서 원하는 필드를 설정해주고 위의 복잡한 빌드 과정을 통해 SecurityFilterChain 클래스의 최종 보안 객체를 빌드합니다.
어쩌다 이렇게 봤는지 모르겠지만 저희가 할 일은 “HttpSecurity에 oidcBackChannelLogout 메서드를 만들어 다른 로그아웃들과 동일한 DSL 수준에서 OIDC 백채널 로그아웃을 지원하도록 하자!”가 될 것 같습니다.
디버깅으로 실행 과정 알아보기
그러면 디버깅으로 기존 코드의 실행 과정을 알아보겠습니다.
data:image/s3,"s3://crabby-images/73954/73954cc73cf398e0cc23113ad069e04fba94bbb0" alt="image"
위처럼 코드를 구성하고 아래처럼 break point를 두어 애플리케이션을 실행해보겠습니다.
data:image/s3,"s3://crabby-images/9f855/9f8552dcead45d6e860935313d6489e348dd5ede" alt="image"
실행
data:image/s3,"s3://crabby-images/57135/571357ba9a142ac3dab83d766e1dbc933ccb66fb" alt="image"
우선 실행해보니 해당 break point에 잘 걸렸습니다.
이제 step into로 들어가보겠습니다.
1. backChannel() 메서드 진입
data:image/s3,"s3://crabby-images/8070e/8070eacdd2ef3676ccdd26b65643e7278ac7e087" alt="image"
OidcLogoutConfigurer 클래스의 backChannel() 메서드로 진입하였습니다.
data:image/s3,"s3://crabby-images/6c4fc/6c4fcd32866c198380d875e9264d32c7b43acea6" alt="image"
해당 시점의 backChannel 필드는 null이기 떄문에 새로운 BackChannelLogoutConfigurer 객체를 만들어 bakcChannel 필드에 넣어줍니다.
data:image/s3,"s3://crabby-images/46b62/46b6227d2882de927a52c99e35699bf4e6c7b472" alt="image"
이후 파라미터로 전달받은 backChannelLogoutConfigurer의 customize() 메서드에 backChannel 필드를 넣어주고 실행합니다.
data:image/s3,"s3://crabby-images/73927/73927c425bcd1e7cb01bc32f7c143f5170cba529" alt="image"
이후 다시 빠져나옵니다.
2. oidcLogout() 메서드 진입
다음으로 oidcLogout() 메서드에 진입합니다.
data:image/s3,"s3://crabby-images/2aea9/2aea9ce7952f2f69536e0135f59384b205d183f3" alt="image"
해당 메서드에 대해 알아보기 위해 공식 문서를 찾아봤는데 설명이 없는 것을 확인할 수 있었습니다.
data:image/s3,"s3://crabby-images/14931/1493197fc399e6f11aa53472fa616df42fa69899" alt="image"
차근차근 알아보겠습니다.
매개변수로 Custimizer를 받고 있습니다.
data:image/s3,"s3://crabby-images/e8e36/e8e36c9631d34344de55a68322b711ad7ebe90f9" alt="image"
Custimizer는 주어진 객체의 설정을 수정할 수 있도록 돕습니다.
@FunctionalInterface는 하나의 추상 메서드를 가지는 기능적 인터페이스를 의미합니다.
이 customize() 추상메서드를 통해 객체의 설정을 수정합니다.
다음으로 oauth2LoginCustomizer.customize(getOrApply(new OAuth2LoginConfigurer<>()));
를 알아보겠습니다.
oauth2LoginCustomizer.customize
는 위에서 봤듯이 해당 매개변수에 대한 설정을 하는 메서드입니다.
getOrApply(new OidcLogoutConfigurer<>())는 다음과 같습니다.
data:image/s3,"s3://crabby-images/c2874/c2874266996c38e1c1e69250c581f6158982c26a" alt="image"
기존의 SecurityConfigurer 객체를 반환하거나 해당 객체가 없으면 새로운 SecurityConfigurerAdapter를 적용하여 새로운 SecurityConfigurer 객체를 반환한다고 합니다.
이에 대한 파라미터로 OidcLogoutConfigurer를 넣었으니 이전에 생성된 OidcLogoutConfigurer 객체가 있다면 이를 반환하고 없다면 새로운 OidcLogoutConfigurer 객체를 생성하여 반환합니다.
data:image/s3,"s3://crabby-images/d8eec/d8eecb4748397610b65b0a43547bd8aef661aa6f" alt="image"
새로운 객체를 만드는 apply 메서드는 다음과 같이 구성되어 있습니다.
내부의 add 메서드는 다음과 같습니다.
data:image/s3,"s3://crabby-images/57e52/57e526454c4b0291dfc20f46ac2203aefb579ce4" alt="image"
이는 새로운 SecurityConfigurer 객체를 configurers 맵에 추가하는 역할을 합니다.
위에서 init()이나 configurer()을 학습할 때 봤던 configurers라서 좀 익숙해진 것 같습니다!!
Assert.notNull(configurer, "configurer cannot be null");
Class<? extends SecurityConfigurer<O, B>> clazz = (Class<? extends SecurityConfigurer<O, B>>) configurer
.getClass();
우선 configure가 nouNull인 것을 확인하고 null이라면 예외 처리를 진행하고 null이 아니라면 configurer의 클래스 타입을 가져와서 clazz에 저장합니다.
이는 추후 configurers 맵의 키로 사용됩니다.
synchronized (this.configurers) {
if (this.buildState.isConfigured()) {
throw new IllegalStateException("Cannot apply " + configurer + " to already built object");
}
이후 synchronized를 사용하여 configurers 맵이 다중 스레드 환경에서도 안전하게 추가 작업을 수행할 수 있도록 하고 빌드가 완료된 상태인지 확인합니다.
BuildState가 CONFIGURING 상태라면 이미 설정이 완료된 상태이기 때문에 configurer를 추가할 수 없기 때문에 예외를 던져줍니다.
List<SecurityConfigurer<O, B>> configs = null;
if (this.allowConfigurersOfSameType) {
configs = this.configurers.get(clazz);
}
configs = (configs != null) ? configs : new ArrayList<>(1);
configs.add(configurer);
this.configurers.put(clazz, configs);
allowConfigurersOfSameType이 false인 경우, 동일한 타입의 SecurityConfigurer를 여러 개 추가할 수 없습니다.
allowConfigurersOfSameType이 true이면 configurers 맵에서 clazz 타입의 기존 SecurityConfigurer 리스트를 가져옵니다.
그리고 Configs가 null이라면 새로운 리스트를 만들어 줍니다.
다음으로 해당 configurer를 configs 리스트에 추가해준 뒤 configurers 맵에 Put 해줍니다.
if (this.buildState.isInitializing()) {
this.configurersAddedInInitializing.add(configurer);
}
buildState가 INITIALIZING 상태라면 configurersAddedInInitializing 리스트에도 configurer를 추가합니다.
초기화 단계에서 추가된 SecurityConfigurer 객체를 추적하여 필요한 경우 추가 작업을 수행할 수 있게 하기 위함입니다!
add() 메서드의 동작은 SecurityConfigurer 객체를 configurers 맵에 타입별로 추가한다!라고 정리해볼 수 있을 것 같습니다.
apply() 정리
public <C extends SecurityConfigurer<O, B>> C apply(C configurer) throws Exception {
add(configurer);
return configurer;
}
그러면 apply()는 configurer를 configurers 맵에 추가하고 다시 반환해주는 메서드라고 설명할 수 있을 것 같습니다.
getOrApply 정리
private <C extends SecurityConfigurerAdapter<DefaultSecurityFilterChain, HttpSecurity>> C getOrApply(C configurer)
throws Exception {
C existingConfig = (C) getConfigurer(configurer.getClass());
if (existingConfig != null) {
return existingConfig;
}
return apply(configurer);
}
그렇다면 getOrApply()는 configurer의 클래스 타입을 확인하고 없으면 있으면 기존의 configurer를 반환하고 없다면 apply() 메서드를 통해 configurer를 등록해준 뒤 반환해주는 메서드라고 정리할 수 있을 것 같습니다!
oidcLogout() 정리
public HttpSecurity oidcLogout(Customizer<OidcLogoutConfigurer<HttpSecurity>> oidcLogoutCustomizer)
throws Exception {
oidcLogoutCustomizer.customize(getOrApply(new OidcLogoutConfigurer<>()));
return HttpSecurity.this;
}
그렇다면 oidcLogout()은 OidcLogoutConfigurer의 새로운 객체를 생성하여 configurer를 등록하고 init()과 configure() 과정에서 포함될 수 있도록 하고 객체 설정을 커스터마이즈하여 반환하는 메서드라고 설명할 수 있을 것 같습니다!
다시 filterchain
data:image/s3,"s3://crabby-images/e2688/e2688cf2964751b0e3efe019bef09fc74a06c104" alt="image"
data:image/s3,"s3://crabby-images/4399d/4399d88a880bc911af9a2a28696783b0c5666289" alt="image"
이 과정 후 다시 filterChain으로 나오고 최종 객체가 빌드됩니다.
정리
@Bean
public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
http
.oidcLogout((oidc) -> oidc.backChannel(Customizer.withDefaults()));
return http.build();
}
따라서 해당 기능의 동작은 다음과 같습니다.
- HttpSecurity 객체로, Spring Security에서 보안 설정 시작, 메서드 체이닝 방식으로 설정
- oidcLogout(…) 메서드 호출하여 OIDC 로그아웃 관련 설정을 시작
- 내부적으로 getOrApply(new OidcLogoutConfigurer<>())를 호출하여 OidcLogoutConfigurer 객체를 가져오거나 생성
- oidcLogoutCustomizer.customize(…)를 통해 OidcLogoutConfigurer 객체를 설정
- oidc.backChannel(Customizer.withDefaults())를 호출하여 백채널 로그아웃 설정
- OidcBackChannelLogoutConfigurer 객체를 생성하거나 가져와서 customize를 통해 백채널 로그아웃 설정 적용
- http.build()를 통해 보안 구성 초기화 및 적용 후 빌드
- 각 SecurityConfigurer의 init() 메서드를 호출하여 필요한 초기 설정을 수행
- 각 SecurityConfigurer의 configure() 메서드를 호출하여 보안 설정을 적용
- performBuild()를 통해 최종적으로 SecurityFilterChain 객체를 생성