-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: 인증 객체로부터 사용자 정보를 얻을 수 있도록 한다.
- Loading branch information
Showing
19 changed files
with
206 additions
and
33 deletions.
There are no files selected for viewing
12 changes: 12 additions & 0 deletions
12
src/main/java/com/seong/shoutlink/domain/auth/LoginUser.java
This file contains 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
package com.seong.shoutlink.domain.auth; | ||
|
||
import java.lang.annotation.ElementType; | ||
import java.lang.annotation.Retention; | ||
import java.lang.annotation.RetentionPolicy; | ||
import java.lang.annotation.Target; | ||
|
||
@Target(ElementType.PARAMETER) | ||
@Retention(RetentionPolicy.RUNTIME) | ||
public @interface LoginUser { | ||
|
||
} |
This file contains 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
25 changes: 25 additions & 0 deletions
25
src/main/java/com/seong/shoutlink/domain/exception/ErrorCode.java
This file contains 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
package com.seong.shoutlink.domain.exception; | ||
|
||
import lombok.AllArgsConstructor; | ||
import lombok.Getter; | ||
|
||
@Getter | ||
@AllArgsConstructor | ||
public enum ErrorCode { | ||
ILLEGAL_ARGUMENT("SL001", Constants.BAD_REQUEST), | ||
UNAUTHENTICATED("SL101", Constants.UNAUTHORIZED), | ||
INVALID_ACCESS_TOKEN("SL102", Constants.UNAUTHORIZED), | ||
EXPIRED_ACCESS_TOKEN("SL103", Constants.UNAUTHORIZED), | ||
DUPLICATE_EMAIL("SL901", Constants.CONFLICT), | ||
DUPLICATE_NICKNAME("SL902", Constants.BAD_REQUEST); | ||
|
||
private final String errorCode; | ||
private final int status; | ||
|
||
private static class Constants { | ||
|
||
private static final int BAD_REQUEST = 400; | ||
private static final int UNAUTHORIZED = 401; | ||
private static final int CONFLICT = 409; | ||
} | ||
} |
2 changes: 1 addition & 1 deletion
2
.../global/exception/ShoutLinkException.java → .../domain/exception/ShoutLinkException.java
This file contains 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
This file contains 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
4 changes: 2 additions & 2 deletions
4
...ain/java/com/seong/shoutlink/global/auth/authentication/JwtAuthenticationInterceptor.java
This file contains 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
This file contains 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
41 changes: 41 additions & 0 deletions
41
src/main/java/com/seong/shoutlink/global/auth/resolver/LoginUserArgumentResolver.java
This file contains 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,41 @@ | ||
package com.seong.shoutlink.global.auth.resolver; | ||
|
||
import com.seong.shoutlink.domain.auth.LoginUser; | ||
import com.seong.shoutlink.global.auth.authentication.Authentication; | ||
import com.seong.shoutlink.global.auth.authentication.AuthenticationContext; | ||
import com.seong.shoutlink.domain.exception.ErrorCode; | ||
import com.seong.shoutlink.domain.exception.ShoutLinkException; | ||
import java.util.Objects; | ||
import lombok.RequiredArgsConstructor; | ||
import org.springframework.core.MethodParameter; | ||
import org.springframework.web.bind.support.WebDataBinderFactory; | ||
import org.springframework.web.context.request.NativeWebRequest; | ||
import org.springframework.web.method.support.HandlerMethodArgumentResolver; | ||
import org.springframework.web.method.support.ModelAndViewContainer; | ||
|
||
@RequiredArgsConstructor | ||
public class LoginUserArgumentResolver implements HandlerMethodArgumentResolver { | ||
|
||
private final AuthenticationContext authenticationContext; | ||
|
||
@Override | ||
public boolean supportsParameter(MethodParameter parameter) { | ||
boolean hasParameterAnnotation = parameter.hasParameterAnnotation(LoginUser.class); | ||
boolean hasLongParameterType = parameter.getParameterType().isAssignableFrom(Long.class); | ||
return hasParameterAnnotation && hasLongParameterType; | ||
} | ||
|
||
@Override | ||
public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, | ||
NativeWebRequest webRequest, WebDataBinderFactory binderFactory) { | ||
Authentication authentication = authenticationContext.getAuthentication(); | ||
checkAuthenticated(authentication); | ||
return authentication.getPrincipal(); | ||
} | ||
|
||
private void checkAuthenticated(Authentication authentication) { | ||
if(Objects.isNull(authentication)) { | ||
throw new ShoutLinkException("인증되지 않은 사용자 요청입니다.", ErrorCode.UNAUTHENTICATED); | ||
} | ||
} | ||
} |
This file contains 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
15 changes: 0 additions & 15 deletions
15
src/main/java/com/seong/shoutlink/global/exception/ErrorCode.java
This file was deleted.
Oops, something went wrong.
5 changes: 5 additions & 0 deletions
5
src/main/java/com/seong/shoutlink/global/exception/ErrorResponse.java
This file contains 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
package com.seong.shoutlink.global.exception; | ||
|
||
public record ErrorResponse(String message, String errorCode) { | ||
|
||
} |
17 changes: 17 additions & 0 deletions
17
src/main/java/com/seong/shoutlink/global/exception/globalExceptionHandler.java
This file contains 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
package com.seong.shoutlink.global.exception; | ||
|
||
import com.seong.shoutlink.domain.exception.ShoutLinkException; | ||
import org.springframework.http.ResponseEntity; | ||
import org.springframework.web.bind.annotation.ExceptionHandler; | ||
import org.springframework.web.bind.annotation.RestControllerAdvice; | ||
|
||
@RestControllerAdvice | ||
public class globalExceptionHandler { | ||
|
||
@ExceptionHandler(ShoutLinkException.class) | ||
public ResponseEntity<ErrorResponse> shoutLinkExHandle(ShoutLinkException e) { | ||
ErrorResponse errorResponse | ||
= new ErrorResponse(e.getMessage(), e.getErrorCode().getErrorCode()); | ||
return ResponseEntity.status(e.getErrorCode().getStatus()).body(errorResponse); | ||
} | ||
} |
This file contains 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
This file contains 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
This file contains 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
17 changes: 17 additions & 0 deletions
17
src/test/java/com/seong/shoutlink/global/auth/AuthTestController.java
This file contains 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
package com.seong.shoutlink.global.auth; | ||
|
||
import com.seong.shoutlink.domain.auth.LoginUser; | ||
import org.springframework.http.ResponseEntity; | ||
import org.springframework.web.bind.annotation.GetMapping; | ||
import org.springframework.web.bind.annotation.RequestMapping; | ||
import org.springframework.web.bind.annotation.RestController; | ||
|
||
@RestController | ||
@RequestMapping("/api/test") | ||
public class AuthTestController { | ||
|
||
@GetMapping("/login-user") | ||
public ResponseEntity<Long> loginUser(@LoginUser Long memberId) { | ||
return ResponseEntity.ok(memberId); | ||
} | ||
} |
This file contains 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
This file contains 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
60 changes: 60 additions & 0 deletions
60
src/test/java/com/seong/shoutlink/global/auth/resolver/LoginUserArgumentResolverTest.java
This file contains 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,60 @@ | ||
package com.seong.shoutlink.global.auth.resolver; | ||
|
||
import static org.assertj.core.api.Assertions.assertThat; | ||
import static org.springframework.restdocs.mockmvc.RestDocumentationRequestBuilders.get; | ||
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath; | ||
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; | ||
|
||
import com.seong.shoutlink.base.BaseControllerTest; | ||
import com.seong.shoutlink.domain.auth.service.response.TokenResponse; | ||
import com.seong.shoutlink.domain.member.MemberRole; | ||
import com.seong.shoutlink.fixture.AuthFixture; | ||
import com.seong.shoutlink.domain.exception.ErrorCode; | ||
import com.seong.shoutlink.domain.exception.ShoutLinkException; | ||
import org.junit.jupiter.api.DisplayName; | ||
import org.junit.jupiter.api.Nested; | ||
import org.junit.jupiter.api.Test; | ||
import org.springframework.test.web.servlet.ResultActions; | ||
|
||
class LoginUserArgumentResolverTest extends BaseControllerTest { | ||
|
||
@Nested | ||
@DisplayName("@LoginUser를 핸들러 메서드의 파라미터에 선언 시") | ||
class LoginUserTest { | ||
|
||
@Test | ||
@DisplayName("성공: 인증된 사용자의 principal을 반환") | ||
void returnAuthenticatedUserPrincipal() throws Exception { | ||
//given | ||
TokenResponse tokenResponse = AuthFixture.jwtProvider() | ||
.createToken(1L, MemberRole.ROLE_USER); | ||
String bearerAccessToken = "Bearer " + tokenResponse.accessToken(); | ||
|
||
//when | ||
ResultActions resultActions = mockMvc.perform(get("/api/test/login-user") | ||
.header(AUTHORIZATION, bearerAccessToken)); | ||
|
||
//then | ||
resultActions.andExpect(status().isOk()) | ||
.andExpect(jsonPath("$").value(1)); | ||
} | ||
|
||
@Test | ||
@DisplayName("예외(unauthenticated): 인증되지 않은 사용자 요청") | ||
void unauthenticated_WhenUnauthenticatedUserRequest() throws Exception { | ||
//given | ||
//when | ||
//then | ||
mockMvc.perform(get("/api/test/login-user")) | ||
.andExpect( | ||
result -> { | ||
Exception exception = result.getResolvedException(); | ||
assertThat(exception) | ||
.isInstanceOf(ShoutLinkException.class) | ||
.extracting(e -> ((ShoutLinkException) e).getErrorCode()) | ||
.isEqualTo(ErrorCode.UNAUTHENTICATED); | ||
} | ||
); | ||
} | ||
} | ||
} |