[김성경] Sprint9#277
Open
conradrado wants to merge 43 commits into
Open
Conversation
- build.gradle 에 Spring Security 의존성을 추가하였습니다. - application-dev.yaml 파일에서 Spring Security 모듈 로깅 레벨을 Trace로 설정하였습니다. - SecurityConfig에서 기초적인 필터를 등록하였습니다.
- SecurityConfig 필터 체인을 설정 - CsrfTokenRepository 구현체를 CookieCsrfTokenRepository로 설정 - CsrfTokenRequestHandler를 SpaCsrfTokenRequestHandler로 설정 - SpaCsrfTokenRequestHandler 작성. - CsrfController 작성 - /csrf-token 엔드포인트로 GET 전달 시 203 상태 반환 및 CSRF 발급
- DiscodeitUserDetailsService는 UserDetailsService를 대체합니다. - DiscodeitUserDetails는 UserDetails를 대체합니다.
- SecurityConfig - PasswordEncoder를 Bean 등록 - RoleHierarchy를 Bean 등록 - 필터 체인 내에 권한 허용 - AdminAccountInitializer - ApplicationRunner의 구현체 - admin 계정이 없을 때 새로 만들고 초기화 함.
- UserStatus 관련 코드 삭제하고 SessionRegistry를 통해 로그인 여부 판단 - schema.sql의 유저 테이블에 role 추가 및 user 엔티티에도 ROLE 추가 - SessionRegistry를 활용한 단 한개의 세션만 허용 - 커스텀한 UserDetails를 SessionRegistry와 같이 사용하기 위해 hashcode, equals를 오버라이딩
- PreAuthorize 어노테이션을 사용하여 사용자 정보 수정, 삭제 시 자기 자신의 정보만 수정하도록 함 - PreAuthorize 어노테이션을 사용하여 자기 자신이 만든 메시지만 수정 및 삭제하도록 함
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Add this suggestion to a batch that can be applied as a single commit.This suggestion is invalid because no changes were made to the code.Suggestions cannot be applied while the pull request is closed.Suggestions cannot be applied while viewing a subset of changes.Only one suggestion per line can be applied in a batch.Add this suggestion to a batch that can be applied as a single commit.Applying suggestions on deleted lines is not supported.You must change the existing code in this line in order to create a valid suggestion.Outdated suggestions cannot be applied.This suggestion has been applied or marked resolved.Suggestions cannot be applied from pending reviews.Suggestions cannot be applied on multi-line comments.Suggestions cannot be applied while the pull request is queued to merge.Suggestion cannot be applied right now. Please check back later.
Spring Security 환경설정
[x] 프로젝트에 Spring Security 의존성을 추가하세요.
Security 설정 클래스를 생성하세요.
SecurityFilterChain Bean을 선언하세요.
가장 기본적인 SecurityFilterChain을 등록하고, 이때 등록되는 필터 목록을 디버깅해보세요. 필터 목록은 PR에 첨부하세요.
개발 환경에서 Spring Security 모듈의 로깅 레벨을 trace로 설정하세요.
CSRF 보호 설정하기
CsrfTokenRepository 구현체를 CookieCsrfTokenRepository로 설정하세요.
CsrfTokenRequestHandler 컴포넌트를 대체하세요.
디폴트 구현체는 XorCsrfTokenRequestAttributeHandler 입니다.
Spring 공식문서에서 권장하는 CSR+SPA(Single Page Application) 환경에 적합한 구현체를 정의하세요.
CSRF 토큰을 발급하는 API를 구현하세요.
CsrfToken 파라미터를 메서드 인자로 선언하면, HandlerMethodArgumentResolver를 통해 자동으로 주입됩니다. (공식문서)
GET 요청에는 CSRF 인증이 이루어지지 않기 때문에 토큰이 초기화되지 않습니다. 따라서 명시적으로 메소드에서 토큰을 호출합니다.
회원가입
인증 - 로그인
이번 미션에서는 보라색 음영 처리된 5가지 컴포넌트를 대체합니다.
각 컴포넌트의 기본 구현체가 무엇인지 디버깅해보세요.
로그인을 처리할 url을 /api/auth/login로 설정하세요.
UserDetailsService 컴포넌트를 대체하세요.
디폴트 구현체는 InMemoryUserDetailsManager입니다.
DiscodeitUserDetailsService를 정의하세요.
UserDetails 컴포넌트를 대체하세요.
디폴트 구현체는 org.springframework.security.core.userdetails.User입니다.
DiscodeitUserDetails를 정의하세요.
앞서 정의한 DiscodeitUserDetailsService에서 DiscodeitUserDetails를 생성 후 반환하세요.
AuthenticationSuccessHandler 컴포넌트를 대체하세요.
디폴트 구현체는 SavedRequestAwareAuthenticationSuccessHandler입니다.
LoginSuccessHandler를 정의하고 대체하세요.
설정에 추가하세요.
AuthenticiationFailureHandler 컴포넌트를 대체하세요.
디폴트 구현체는 SimpleUrlAuthenticationFailureHandler입니다.
LoginFailureHandler를 정의하고 대체하세요.
설정에 추가하세요.
이제 로그인 처리는 SecurityFilterChain에서 모두 처리되기 때문에 기존에 구현했던 로그인 관련 코드는 제거하세요.
인증 - 세션을 활용한 현재 사용자 정보 조회
GET /api/auth/meHeader(자동 포함) Cookie: JSESSIONID=…200 UserDto-
SecurityFilterChain의 필터를 통해 인증에 성공하면Controller에서@AuthenticationPrincipal를 통해 인증 정보에 접근할 수 있습니다.인증 - 로그아웃
Spring Security의 logout 흐름은 그대로 유지하면서 필요한 부분만 대체합니다.
이번 미션에서는 2가지 요소를 대체합니다.
LogoutSuccessHandler로그아웃을 처리할 url을 /api/auth/logout로 설정하세요.
LogoutSuccessHandler 컴포넌트를 대체하세요.
디폴트 구현체는
SimpleUrlLogoutSuccessHandler입니다.HttpStatusReturningLogoutSuccessHandler로 대체하세요.인가 - 권한 정의
관리자:
ADMIN채널 매니저:
CHANNEL_MANAGER일반 사용자:
USER데이터베이스 스키마를 변경하세요.
회원 가입 시 모든 사용자는 USER 권한을 기본 권한으로 설정하세요.
사용자 권한을 수정하는 API를 구현하세요.
애플리케이션 실행 시 ADMIN 권한을 가진 어드민 계정이 초기화되도록 구현하세요.
DiscodietUserDetails.getAuthorities를 수정하세요.
인가 - 권한 적용
authorizeHttpRequests를 활성화하고, 모든 요청을 인증하도록 설정하세요.
다음의 요청은 인증하지 않도록 설정하세요.
Method Security를 활성화하세요.
Service의 메소드 별로 아래의 조건에 맞게 권한을 수정하세요.
적절한 권한이 없는 경우 403 응답을 반환하세요.
SecurityFilterChainRoleHierarchy를 활용해 권한의 계층 구조를 정의하세요.
세션 관리 고도화
동일한 계정으로 동시 로그인할 수 없도록 설정하세요.
sessionConcurrency설정을 활용하세요.세션의 동일성을 보장하기 위해 DiscodeitUserDetails의 equals(), hashcode() 메소드를 오버라이딩하세요.
권한이 변경된 사용자가 로그인 상태라면 세션을 무효화하세요.
sessionRegistry를 활용하세요.UserStatus 엔티티 대신 SessionRegistry를 활용해 사용자의 로그인 여부를 판단하도록 리팩토링하세요.