Post

OAuth2 Deepdive

OAuth2 Deepdive

πŸ”” Oauth2 적용 μ „
πŸ”” Oauth2 μ‹€μŠ΅

κ°œμš”

졜근 개인 ν”„λ‘œμ νŠΈμ—μ„œ 넀이버 λ‘œκ·ΈμΈμ„ μœ„ν•΄ ν•„μš”ν•œ 뢀뢄을 μ„€μ •ν•˜κ³  λ‘œμ§μ„ κ΅¬ν˜„ν–ˆλ‹€. 개발 κ°€μ΄λ“œμ— 맞게 각 λ‹¨κ³„λ³„λ‘œ 진행을 ν–ˆμœΌλ‚˜β€¦

아무리 봐도 κ³Όμ • ν•˜λ‚˜ ν•˜λ‚˜λ₯Ό λ‚΄ μ†μœΌλ‘œ 직접 κ΅¬ν˜„ν•˜λŠ”κ²Œ λ§žλŠ”κ±΄κ°€? λΌλŠ” μ˜λ¬Έμ„ μ‹œμž‘μœΌλ‘œ μ’€ 더 효율적으둜 κ΅¬ν˜„ν•˜λŠ” 방법을 μ°Ύμ•„λ΄€κ³ , κ·Έλ ‡κ²Œ Oauth 2.0으둜 λ‘œκ·ΈμΈν•˜λŠ” 방식을 찾게 λ˜μ—ˆλ‹€.

Oauth2 적용 μ „

μ½”λ“œ μŠ€νƒ€μΌμ„ λ– λ‚˜μ„œ μ§€κΈˆ μ™€μ„œ λ³΄λ‹ˆ 마치 Oauth 1.0처럼 κ΅¬ν˜„μ„ ν•˜κ³  μžˆμ—ˆλ‹€. λ‚˜ λ˜ν•œ, 각 ν”Œλ‘œμš°λ§ˆλ‹€ ν•˜λ“œ 코딩은 쀄이고 λ³€μˆ˜λ‚˜ ν•¨μˆ˜λ₯Ό μž¬ν™œμš©ν•˜κΈ° μœ„ν•΄, λ³΄μ•ˆ μš”μ†Œλ₯Ό μƒκ°ν•˜λ©΄μ„œ κ΅¬ν˜„ν–ˆμ—ˆλ‹€.

μ„œλ“œ νŒŒν‹° κ΄€λ ¨ APIλ₯Ό ν˜ΈμΆœν•  λ•Œ, 응닡 값을 Json으둜 κ°€μ Έμ˜€κΈ° μœ„ν•΄ JsonNodeλ₯Ό 많이 μ‚¬μš©ν–ˆλŠ”λ° λ™μΌν•˜κ²Œ μ μš©ν•˜λ €κ³  λ³΄λ‹ˆ μ†ŒμŠ€κ°€ λ„ˆλ¬΄ 길어진 것 κ°™λ‹€.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
private final NaverApiService naverApiService;
private final SnsInfoService snsInfoService;
private final UserService userService;

public String getNaverOauth2LoginUrl(HttpServletRequest request) {
    String authorizeUrl = BASE_URL+"/authorize";
    String callbackUrl = getServiceUrl()+"/login/naver/callback";
    String state = NakjiUtil.generateTokenState();
    request.getSession().setAttribute("state", state);

    return UriComponentsBuilder
            .fromHttpUrl(authorizeUrl)
            .queryParam("response_type", "code")
            .queryParam("client_id", secrets.naver().naverId())
            .queryParam("redirect_uri", callbackUrl)
            .queryParam("state", state)
            .toUriString();
}

public String processNaverLogin(LoginProfile profile, Oauth2AccessToken tokenInfo) {
    SnsInfo checkSnsInfo = snsInfoService.getOrCreateUser(profile, tokenInfo);
    Optional<User> user = userService.getUserByUserId(checkSnsInfo.getSnsId());
    if (user.isPresent()) {
        // Update user info

    } else {
        // Create user
    }
    return "redirect:/";
}

public String processNaverLoginCallback(String state, String code, HttpSession session) {
    Oauth2AccessToken token = getToken(state, code, session);
    if (token == null) {
        throw new BadRequestException(ErrorCode.INVALID_PARAMETER);
    }

    try (Response response = naverApiService.naverProfileConnection(token.tokenType(), token.accessToken())) {
        if (response.status() == 200) {
            JsonNode resultJson = NakjiUtil.readBody(response.body().asInputStream());
            if ("00".equals(resultJson.get("resultcode")) && "success".equals(resultJson.get("message"))) {
                JsonNode profileInfo = resultJson.get("response");
                LoginProfile profile = new LoginProfile("naver",
                        Optional.ofNullable(profileInfo.get("id")).map(JsonNode::asText).orElse(""),
                        Optional.ofNullable(profileInfo.get("nickname")).map(JsonNode::asText).orElse(""),
                        Optional.ofNullable(profileInfo.get("gender")).map(JsonNode::asText).orElse(""),
                        Optional.ofNullable(profileInfo.get("age")).map(JsonNode::asText).orElse("")
                );

                return processNaverLogin(profile, token);
            } else {
                throw new Exception();
            }
        } else {
            log.info("NaverLoginService.getNaverProfileUrl Request Status: {}, Body: {}", response.status(), response.body().toString());
            //throw new BadRequestException("Bad request with status: " + response.status());
        }
    } catch (Exception e) {
        log.error("NaverLoginService.getNaverProfileUrl Error: ", e);
    }

    return "";
}

public Oauth2AccessToken getToken(String state, String code, HttpSession session) {
    try (Response response = getGenerateToken(state, code, session)) {
        if (response.status() == 200) {
            JsonNode resultJson = NakjiUtil.readBody(response.body().asInputStream());
            Oauth2AccessToken resultToken = new Oauth2AccessToken(
                    Optional.ofNullable(resultJson.get("access_token")).map(JsonNode::asText).orElse(""),
                    Optional.ofNullable(resultJson.get("refresh_token")).map(JsonNode::asText).orElse(""),
                    Optional.ofNullable(resultJson.get("token_type")).map(JsonNode::asText).orElse(""),
                    Optional.ofNullable(resultJson.get("expires_in")).map(JsonNode::asInt).orElse(0),
                    Optional.ofNullable(resultJson.get("error")).map(JsonNode::asText).orElse(""),
                    Optional.ofNullable(resultJson.get("error_description")).map(JsonNode::asText).orElse("")
            );

            if ("invalid_request".equals(resultToken.errorCode())) {
                throw new BadRequestException(ErrorCode.INVALID_PARAMETER);
            } else if ("unauthorized_client".equals(resultToken.errorCode())) {
                throw new UnauthorizedException(ErrorCode.INVALID_AUTH_CODE);
            }

            return resultToken;
        } else {
            log.info("NaverLoginService.getToken Request Status: {}, Body: {}", response.status(), response.body().toString());
            //throw new BadRequestException("Bad request with status: " + response.status());
        }
    } catch (Exception e) {
        log.error("NaverLoginService.getToken Error: ", e);
    }

    return null;
}

private Response getGenerateToken(String state, String code, HttpSession session) {
    String storedState = (String)session.getAttribute("state");
    if (!state.equals(storedState)) {
        throw new UnauthorizedException(ErrorCode.INVALID_AUTH_PATH);
    }

    return Feign.builder()
            .encoder(new GsonEncoder())
            .decoder(new GsonDecoder())
            .target(NaverGenerateTokenClient.class, BASE_URL)
            .generateToken(secrets.naver().naverId(), secrets.naver().naverKey(), state, code);
}

Oauth2 적용 ν›„

gpt와 일뢀 λΈ”λ‘œκ·Έλ₯Ό μ°Ύμ•„λ³΄λ©΄μ„œ λ‚΄ λ°©μ‹λŒ€λ‘œ μ μš©ν•΄λ΄€λ‹€. μš°μ„  λ‘œκ·ΈμΈμ€ 잘 되고, 초기 μ„ΈνŒ…λ§Œ 해두면 κ·Έ μ΄ν›„μ—λŠ” μ†ŒμŠ€λ₯Ό μž¬ν™œμš©ν•˜κ±°λ‚˜ 쑰금만 더 μΆ”κ°€ν•˜λŠ” κ²ƒμœΌλ‘œ 해결될 것 κ°™λ‹€.

λ‹€μŒμ—λŠ” νŽ˜μ΄μ§€λ§ˆλ‹€ μ—­ν•  λΆ€μ—¬ν•˜λŠ” 것과 jwt을 μ΄μš©ν•œ λ‘œκ·ΈμΈμ„ μΆ”κ°€ν•  μ˜ˆμ •μ΄λ‹€.

security.yml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
spring:
  security:
    oauth2:
      client:
        registration:
          naver:
            client-id: ${NAVER_CLIENT_ID}
            client-secret: ${NAVER_CLIENT_SECRET}
            authorization-grant-type: authorization_code
            redirect-uri: ${SERVICE_URL}${NAVER_REDIRECT_URI}
            scope: name, nickname, email, gender, birthyear, profile_image
        provider:
          naver:
            authorization-uri: https://nid.naver.com/oauth2.0/authorize
            token-uri: https://nid.naver.com/oauth2.0/token
            user-info-uri: https://openapi.naver.com/v1/nid/me
            user-name-attribute: response

SecurityConfig

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
@Configuration
@EnableWebSecurity
@RequiredArgsConstructor
public class SecurityConfig {
    private final CustomOAuth2UserService customOAuth2UserService;

    @Bean
    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
        return http
                .csrf(csrf -> csrf.disable())
                .authorizeHttpRequests(authorize -> authorize
                        ...
                )
                .oauth2Login(oauth2 -> oauth2
                        .loginPage("/login")
                        .defaultSuccessUrl("/")
                        .failureUrl("/login?error")
                        .userInfoEndpoint(userInfo -> userInfo
                                .userService(customOAuth2UserService)
                        )
                )
                .logout(logout -> logout
                        .logoutSuccessUrl("/")
                        .invalidateHttpSession(true)
                        .deleteCookies("JSESSIONID")
                )
                .build();
    }
}

OAuthAttributes

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
@Builder
public record OAuthAttributes(Map<String, Object> attributes, String nameAttributeKey, User user) {
    public static OAuthAttributes of(String registrationId, String userNameAttributeName, Map<String, Object> attributes) {
        if ("naver".equals(registrationId)) {
            return ofNaver(userNameAttributeName, attributes);
        } else {
            return null;
        }
    }

    private static OAuthAttributes ofNaver(String userNameAttributeName, Map<String, Object> attributes) {
        Map<String, Object> response = (Map<String, Object>) attributes.get("response");

        return OAuthAttributes.builder()
                .attributes(attributes)
                .nameAttributeKey(userNameAttributeName)
                .user(User.builder()
                        .oauthId((String) response.get("id"))
                        .provider("naver")
                        .name((String) response.get("name"))
                        .nickname((String) response.get("nickname"))
                        .profileImage((String) response.get("profile_image"))
                        .email((String) response.get("email"))
                        .gender((String) response.get("gender"))
                        .birthYear((String) response.get("birthYear"))
                        .role(Role.USER)
                        .build())
                .build();
    }

    public User toEntity() {
        return user;
    }
}

OAuth2UserService

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
@Service
@RequiredArgsConstructor
public class CustomOAuth2UserService implements OAuth2UserService<OAuth2UserRequest, OAuth2User> {
    private final UserRepository userRepository;
    private final HttpSession httpSession;

    @Override
    public OAuth2User loadUser(OAuth2UserRequest userRequest) throws OAuth2AuthenticationException {
        OAuth2UserService<OAuth2UserRequest, OAuth2User> delegate = new DefaultOAuth2UserService();
        OAuth2User oAuth2User = delegate.loadUser(userRequest);
        String registrationId = userRequest.getClientRegistration().getRegistrationId();
        String userNameAttributeName = userRequest.getClientRegistration()
                                        .getProviderDetails()
                                        .getUserInfoEndpoint()
                                        .getUserNameAttributeName();

        OAuthAttributes attributes = OAuthAttributes.of(registrationId, userNameAttributeName, oAuth2User.getAttributes());
        User user = saveOrUpdate(attributes);
        httpSession.setAttribute("user", new SessionUser(user));

        return new DefaultOAuth2User(
                Collections.singleton(new SimpleGrantedAuthority(user.getRoleKey())),
                attributes.getAttributes(),
                attributes.getNameAttributeKey());
    }

    private User saveOrUpdate(OAuthAttributes attributes) {
        User userInfo = attributes.getUser();
        User target = userRepository.findByOauthId(userInfo.getOauthId())
                .map(entity -> entity.update(userInfo.getNickname(), userInfo.getProfileImage(), userInfo.getEmail()))
                .orElse(attributes.toEntity());

        return userRepository.save(target);
    }
}

ν›„κΈ°

μ—¬λ‹΄μ΄μ§€λ§Œ 킀값을 관리할 λ•Œ, ν•΄λ‹Ή μ„€μ • νŒŒμΌμ„ μ €μž₯μ†Œμ—λŠ” μ»€λ°‹ν•˜μ§€ μ•Šμ•˜λ‹€. λ‹€λ₯Έ PCμ—μ„œ 접속할 λ•ŒλŠ” 개인 ν΄λΌμš°λ“œμ—μ„œ 킀값이 μžˆλŠ” νŒŒμΌμ„ λΆˆλŸ¬μ™€μ„œ μ‚¬μš©ν–ˆλŠ”λ°, μ΄λ²ˆμ— oauthλ₯Ό κ²½ν—˜ν•˜λ©΄μ„œ 점점 λΆˆνŽΈν•΄μ§ˆ 수 μžˆμ„ 것 κ°™λ‹€.

κ·Έλž˜μ„œ dotenvλ₯Ό μ‚¬μš©ν•˜κ³  κΈ°μ‘΄ ν™˜κ²½μ„€μ • 값듀은 env νŒŒμΌμ—μ„œ κ°€μ Έμ˜€λŠ” λ°©μ‹μœΌλ‘œ μˆ˜μ •ν–ˆλ‹€.
큰 μ°¨μ΄λŠ” 없을 수 μžˆμ§€λ§Œ, λ‚΄κ°€ ν™˜κ²½μ„€μ •ν•  λ•Œ μ–΄λ–€ ꡬ쑰둜 μ„€μ •ν–ˆλŠ”μ§€λŠ” λ³Ό 수 있게 된 것이 κ°œμ„ μ μ΄λ‹€.

참고 자료

Spring io
Velog
ChatGPT

This post is licensed under CC BY 4.0 by the author.