포트폴리오/ohouseClone

7. 게시판 소셜로그인 만들기(google, naver, kakao)-spring

가끔개발 2023. 11. 1. 17:37

0. Kanbanboard

1. 로그인 form

<a class="btn-login join_btn btn_click" id="kakao-login-btn" href="/oauth2/authorization/kakao">
                    <img alt="카카오" src="https://www.gb.go.kr/Main/Images/ko/member/certi_kakao_login.png"
                         style="max-width: 40%; height: auto; display: block; margin: 10px auto;">
                </a>
                <a href="/oauth2/authorization/google">
                    <img alt="구글" src=""
                         style="max-width: 40%; height: auto; display: block; margin: 10px auto;">
                </a>
                <a href="/oauth2/authorization/naver">
                    <img alt="네이버" src=""
                         style="max-width: 40%; height: auto; display: block; margin: 10px auto;">
                </a>

2. build.gradle

implementation 'org.springframework.boot:spring-boot-starter-oauth2-client'

3. application.yml

security:
    oauth2:
      client:
        provider:
          kakao:
            authorization-uri: https://kauth.kakao.com/oauth/authorize
            token-uri: https://kauth.kakao.com/oauth/token
            user-info-uri: https://kapi.kakao.com/v2/user/me
            user-name-attribute: 
          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
        registration:
          kakao:
              client-id: 
              client-secret: 
              client-authentication-method: POST
              redirect-uri: http://localhost:8080/login/oauth2/code/kakao
              authorization-grant-type: authorization_code
              client-name: Kakao
              scope:
                - account_email
                - profile_nickname
          naver:
            client-id: 
            client-secret: 
            redirect-uri: http://localhost:8080/login/oauth2/code/naver
            authorization-grant-type: authorization_code
            client-name: Naver
            scope:
              - email
              - nickname
              - name
          google:
            client-id: 
            client-secret:
            redirect-uri: http://localhost:8080/login/oauth2/code/google
            scope:
              - email

https://lotuus.tistory.com/104

 

[Spring Security] OAuth 카카오 로그인하기

목차 이전글 https://lotuus.tistory.com/80 [Spring Security] OAuth 네이버 로그인하기 목차 이전글 https://lotuus.tistory.com/79 [Spring Security] OAuth 구글 로그인하기 목차 [이전 게시글] 꼭! 봐주세여 [Spring Security] 동

lotuus.tistory.com

작성한 코드를 반영하여 작성하였습니다.

4. 각각의 개발자 센터


1. 카카오 

https://developers.kakao.com/

 

Kakao Developers

카카오 API를 활용하여 다양한 어플리케이션을 개발해보세요. 카카오 로그인, 메시지 보내기, 친구 API, 인공지능 API 등을 제공합니다.

developers.kakao.com

2.  Naver

https://developers.naver.com/main/

 

NAVER Developers

네이버 오픈 API들을 활용해 개발자들이 다양한 애플리케이션을 개발할 수 있도록 API 가이드와 SDK를 제공합니다. 제공중인 오픈 API에는 네이버 로그인, 검색, 단축URL, 캡차를 비롯 기계번역, 음

developers.naver.com

3. 구글

https://console.cloud.google.com/apis/dashboard?pli=1&project=tranquil-osprey-380908

 

Google 클라우드 플랫폼

로그인 Google 클라우드 플랫폼으로 이동

accounts.google.com

5. 디자인 가이드


1. kakao

https://developers.kakao.com/docs/latest/ko/kakaologin/design-guide

 

Kakao Developers

카카오 API를 활용하여 다양한 어플리케이션을 개발해보세요. 카카오 로그인, 메시지 보내기, 친구 API, 인공지능 API 등을 제공합니다.

developers.kakao.com

2.  Naver

https://developers.naver.com/docs/login/bi/bi.md

 

로그인 버튼 사용 가이드 - LOGIN

네이버 로그인은 애플리케이션에 사용할 수 있는 네이버 로그인 버튼 기본 이미지를 제공합니다. 애플리케이션의 상황에 맞게 버튼 이미지의 디자인을 변경할 수 있지만 네이버 고유의 아이덴

developers.naver.com

3. 구글

https://developers.google.com/identity/branding-guidelines?hl=ko

 

로그인 브랜드 가이드라인  |  Google ID 플랫폼  |  Google for Developers

로그인 브랜드 가이드라인 컬렉션을 사용해 정리하기 내 환경설정을 기준으로 콘텐츠를 저장하고 분류하세요. 모바일 또는 웹 앱에서 기본적인 profile 또는 email 범위로 Google 로그인을 사용하고

developers.google.com

7. Oauth2 interface

package com.portfolio.ohousev1.auth;

import java.util.Map;

public interface OAuth2UserInfo {
    Map<String, Object> getAttributes();
    String getProviderId();
    String getProvider();
    String getEmail();
    String getName();

    String getNickname();

8. Oauth2 Userinfo


이메일을 PK로 지정했기때문에 member table 이메일기준으로 저장된다.

1. kakao

package com.portfolio.ohousev1.auth;

import java.util.Map;

public class KakaoUserInfo implements OAuth2UserInfo{
    private  Map<String, Object> attributes;
    private  Map<String, Object> attributesAccount;
    private Map<String, Object> attributesProfile;

    public KakaoUserInfo(Map<String, Object> attributes) {
        this.attributes = attributes;
        this.attributesAccount = (Map<String, Object>) attributes.get("kakao_account");
        this.attributesProfile = (Map<String, Object>) attributesAccount.get("profile");
    }
    @Override
    public Map<String, Object> getAttributes() {
        return attributes;
    }

    @Override
    public String getProviderId() {
        return attributes.get("id").toString();

    }

    @Override
    public String getProvider() {
        return "kakao";
    }



    @Override
    public String getEmail() {
        return attributesAccount.get("email").toString();
    }

    @Override
    public String getName() {
        return attributesAccount.containsKey("name") ? attributesAccount.get("name").toString() : attributesProfile.get("nickname").toString();
    }


    @Override
    public String getNickname() {
        return attributesProfile.get("nickname").toString();

    }
}

2.  Naver

package com.portfolio.ohousev1.auth;

import java.util.Map;

public class NaverUserInfo implements OAuth2UserInfo{
    private final Map<String, Object> attributes; //OAuth2User.getAttributes();
    private final Map<String, Object> attributesResponse;

    public NaverUserInfo(Map<String, Object> attributes) {
        this.attributes = (Map<String, Object>) attributes.get("response");
        this.attributesResponse = (Map<String, Object>) attributes.get("response");
    }
    @Override
    public Map<String, Object> getAttributes() {
        return attributes;
    }

    @Override
    public String getProviderId() {
        return attributesResponse.get("id").toString();

    }

    @Override
    public String getProvider() {
        return  "naver";
    }



    @Override
    public String getEmail() {
        return attributes.get("email").toString();
    }

    @Override
    public String getName() {
        return attributes.get("name").toString();
    }

    @Override
    public String getNickname() {
        return attributes.get("nickname").toString();
    }




}

3. 구글

package com.portfolio.ohousev1.auth;

import java.util.Map;

public class GoogleUserInfo implements  OAuth2UserInfo{

    private final Map<String, Object> attributes;


    public GoogleUserInfo(Map<String, Object> attributes) {
        this.attributes = attributes;

    }
    @Override
    public Map<String, Object> getAttributes() {
        return attributes;
    }

    @Override
    public String getProviderId() {
        return attributes.get("sub").toString();
    }

    @Override
    public String getProvider() {
        return "google";
    }

    @Override
    public String getEmail() {
        return attributes.get("email").toString();
    }

    @Override
    public String getName() {
        return attributes.get("name").toString();
    }

    @Override
    public String getNickname() {
        return (attributes.get("nickname") != null) ? attributes.get("name").toString() : null;    }


}

9. OAuth2UserService

package com.portfolio.ohousev1.service.member;

import com.portfolio.ohousev1.auth.GoogleUserInfo;
import com.portfolio.ohousev1.auth.KakaoUserInfo;
import com.portfolio.ohousev1.auth.NaverUserInfo;
import com.portfolio.ohousev1.auth.OAuth2UserInfo;
import com.portfolio.ohousev1.dto.post.PostPrincipal;
import com.portfolio.ohousev1.entity.constant.RoleType;
import lombok.extern.slf4j.Slf4j;
import org.springframework.security.oauth2.client.userinfo.DefaultOAuth2UserService;
import org.springframework.security.oauth2.client.userinfo.OAuth2UserRequest;
import org.springframework.security.oauth2.core.OAuth2AuthenticationException;
import org.springframework.security.oauth2.core.user.OAuth2User;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import java.util.Set;
import java.util.UUID;


@Service
@Slf4j
@Transactional(readOnly = true)
public class OAuth2UserService extends DefaultOAuth2UserService {


    private final MemberService memberService;

    public OAuth2UserService(MemberService memberService) {
        this.memberService = memberService;
    }

    @Override
    @Transactional
    public OAuth2User loadUser(OAuth2UserRequest userRequest) throws OAuth2AuthenticationException {
        OAuth2User oAuth2User = super.loadUser(userRequest);
        OAuth2UserInfo oAuth2UserInfo = null;
        String provider = userRequest.getClientRegistration().getRegistrationId();
        switch (provider) {
            case "google" -> oAuth2UserInfo = new GoogleUserInfo(oAuth2User.getAttributes());
            case "naver" -> oAuth2UserInfo = new NaverUserInfo(oAuth2User.getAttributes());
            case "kakao" -> oAuth2UserInfo = new KakaoUserInfo(oAuth2User.getAttributes());
        }

        // nameAttributeKey
        String providerId = oAuth2UserInfo != null ? oAuth2UserInfo.getProviderId() : null;
        String name = provider + "_" + providerId;
        String dummyPassword = "{bcrypt}" + UUID.randomUUID();
        String email = oAuth2UserInfo != null ? oAuth2UserInfo.getEmail() : null;
        String nickname = oAuth2UserInfo.getNickname();
        Set<RoleType> roleTypes = Set.of(RoleType.USER);
        log.info("cause" + providerId, name, dummyPassword, email, nickname, roleTypes);
        return memberService.searchEmail(email).map(PostPrincipal::from)
                .orElseGet(() -> {
                    assert email != null;
                    return PostPrincipal.from(memberService.saveMember(
                            email,
                            dummyPassword,
                            roleTypes,
                            name,
                            nickname,
                            null
                    ));
                });
    }

}

유정정보를 받아오고 그 받아오는 값을  권한을 주고 DB에저장시킵니다.

1. kakao

  • email
  • 이름

2. naver

  • email
  • 이름

3. Google

  • email

비밀번호는 dummy 값으로 넣고 bcypt형태로 암호화한후 저장하고,

닉네임은 못가져올경우 앞에 가져오는 oauth2(ex naver, kakao,google)앞에 붙이고 provideID값을 넣고,

권한은 USER

생일은 null 값을 넣어준다.

10.SecurityConfig

@Configuration
@EnableWebSecurity
@RequiredArgsConstructor
@EnableMethodSecurity
public class SecurityConfig {

    private final OAuth2UserService oAuth2UserService;

    @Bean
    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
    	http
                .authorizeHttpRequests((authorizeHttpRequests) -> authorizeHttpRequests
                        .requestMatchers(PathRequest.toStaticResources().atCommonLocations()).permitAll()
                        .requestMatchers(new AntPathRequestMatcher("/**")).permitAll())
                .csrf((csrf) -> csrf
                        .csrfTokenRepository(CookieCsrfTokenRepository.withHttpOnlyFalse())
                        .ignoringRequestMatchers(new AntPathRequestMatcher("/mysql/**")))
                .headers((headers) -> headers
                        .addHeaderWriter(new XFrameOptionsHeaderWriter(
                                XFrameOptionsHeaderWriter.XFrameOptionsMode.SAMEORIGIN)));
        // login 설정
        http
                .oauth2Login(httpSecurityOAuth2LoginConfigurer -> httpSecurityOAuth2LoginConfigurer
                        .loginPage("/login")
                        .userInfoEndpoint()
                        .userService(oAuth2UserService));
                        
        http
                .logout((logout) -> logout
                        .logoutRequestMatcher(new AntPathRequestMatcher("/members/logout"))
                        .permitAll()
                        .logoutSuccessUrl("/")// logout에 성공하면 /로 redirect
                        .invalidateHttpSession(true));

        return http.build();

 

 

 

https://github.com/nodwon/OhouseV1

 

GitHub - nodwon/OhouseV1: 지금까지 공부한것을 기반으로 포트폴리오 제작

지금까지 공부한것을 기반으로 포트폴리오 제작. Contribute to nodwon/OhouseV1 development by creating an account on GitHub.

github.com