๐คREST API) ์นด์นด์ค ๋ก๊ทธ์ธ API ์๋น์ค ํ๊ฒฝ์ค์
1. ์นด์นด์ค ๊ฐ๋ฐ์์ผํฐ์ ์ ์ ํ ๋ก๊ทธ์ธํ๋ค.
2. ์ ํ๋ฆฌ์ผ์ด์ ์ ์ถ๊ฐํ๊ธฐ
3. ์๋์ ๊ฐ์ด ์ด๋ฆ์ ์ค์ ํด์ ์ถ๊ฐํด์ค๋ค.
4. WEB ํ๋ซํผ ๋ฑ๋ก
5. ๋๋ฉ์ธ ๋ฑ๋ก
6. ์นด์นด์ค ๋ก๊ทธ์ธ ํ์ฑํ


7. Redirect URI ์ค์
8. ๋์ ํญ๋ชฉ ์ค์


9. ์ด๊ณณ์์ ์นด์นด์ค๋ก๊ทธ์ธ ๋ฒํผ ์์ค๋ฅผ ๊ฐ์ ธ์จ๋ค. (์ถ์ฝํ, Middle)
10. ๋ก๊ทธ์ธ ํ๋ฉด์ ์นด์นด์ค ๋ก๊ทธ์ธ ๋ฒํผ์ ์ถ๊ฐํ๋ค.
<a href="#"><img height="38px" src="/image/kakao_login_button.png"/></a>
11. ์นด์นด์ค ๋ก๊ทธ์ธ ์์ฒญ์ ๋ณด๋ด๋ Request URI๋ ์๋์ ๊ฐ์ด ๋ง๋ค๋ฉด ๋๋ค.
GET /oauth/authorize?client_id=${REST_API_KEY}&redirect_uri=${REDIRECT_URI}&response_type=code HTTP/1.1
Host: kauth.kakao.com
REST_API_KEY๋ ์ ํ๋ฆฌ์ผ์ด์ ์์ฑ ํ ๋ฐ๊ธ๋๋ ํค๋ฅผ ๋ฃ์ด์ฃผ๊ณ REDIRECT_URI๋ ์์์ ์ค์ ํ Redirect URI๋ฅผ ๋ฃ์ด์ฃผ๋ฉด ๋๋ค.
12. ๋ฒํผ ์ถ๊ฐ ํ ์คํ์ ํด๋ณด๋ฉด ์๋์ ๊ฐ์ ํ๋ฉด์ด ๋์ค๊ณ
13. #์ผ๋ก ๋น์๋ <a>ํ๊ทธ์ ์ Request URI๋ฅผ ๋ด์ ๋ค์ ๋ฒํผ์ ํด๋ฆญํด๋ณด๋ฉด ์๋์ ๊ฐ์ด ์นด์นด์ค ๋ก๊ทธ์ธ ํ๋ฉด์ด ๋์จ๋ค.
๐คREST API) ์นด์นด์ค ๋ก๊ทธ์ธ API ์๋น์ค ๊ตฌํ ์๋ฃ
14. UserController์ ์นด์นด์ค๋ก๊ทธ์ธ ์ฝ๋ฐฑ ๋ฉ์๋๋ฅผ ์ถ๊ฐํ๋ค.
@GetMapping("/auth/kakao/callback")
public String kakaoCallback(String code) {
}
- ์นด์นด์ค ๋ก๊ทธ์ธ์ ์๋ฃํ๋ ค๋ฉด ์ธ๊ฐ ์ฝ๋๋ก ํ ํฐ ๋ฐ๊ธ ์์ฒญ์ ํด์ผ ํจ.
- ์ธ๊ฐ ์ฝ๋ ๋ฐ๊ธฐ๋ง์ผ๋ก๋ ์นด์นด์ค๋ก๊ทธ์ธ์ด ์๋ฃ๋์ง ์์ผ๋ฉฐ ํ ํฐ ๋ฐ๊ธฐ๊น์ง ๋ง์ณ์ผ ์นด์นด์ค ๋ก๊ทธ์ธ์ ์ ์์ ์ผ๋ก ์๋ฃํ ์ ์์.
- ํ์ ํ๋ผ๋ฏธํฐ๋ฅผ ํฌํจํ์ฌ POST๋ก ์์ฒญํ๊ณ ์์ฒญ ์ฑ๊ณต ์ ์ก์ธ์ค ํ ํฐ, ๋ฆฌํ๋ ์ ํ ํฐ๊ณผ ํ ํฐ ์ ๋ณด๋ฅผ ํฌํจํ JSON๊ฐ์ฒด๋ฅผ ๋ฐ์.
- ํ ํฐ ๋ฐ๊ธฐ๋ฅผ ํตํด ๋ฐ๊ธ๋ฐ์ ์ก์ธ์ค ํ ํฐ์ ์ฌ์ฉ์ ์ ๋ณด ๊ฐ์ ธ์ค๊ธฐ์ ๊ฐ์ ์นด์นด์ค ๋ก๊ทธ์ธ์ด ํ์ํ API๋ฅผ ํธ์ถํ ๋ ์ฌ์ฉํจ.
- ํด๋น ์ฌ์ฉ์ ์ ๋ณด๋ฅผ ๋ฐ์ ์๋น์ค ํ์๊ฐ์ ๋ฐ ๋ก๊ทธ์ธ ๋ฑ์ ์ฒ๋ฆฌํ๊ฒ ๋จ.
15. ์๋์ ๊ฐ์ ์ฝ๋๋ ๋นํจ์จ์ ์ด์ง๋ง ์ข๋ ๋ช ํํ๊ฒ ์ดํดํ๊ธฐ ์ํด ์ด๋ ๊ฒ ์์ฑ.
16. ์ปจํธ๋กค๋ฌ๋จ ์ ์ฒด ์ฝ๋
17. ์ด์ ๋ก๊ทธ์ธํ ์ฌ์ฉ์ ์ ๋ณด๋ฅผ ๊ฐ์ ธ์ค๊ธฐ ์ํด ์ ์ฝ๋๋ฅผ ๊ทธ๋๋ก ๋ณต์ฌํด์ ์กฐ๊ธ ์์ ํด์ ์ฌ์ฉํ๋ค.
18. ๋ก๊ทธ์ธ ํ JSON์ผ๋ก ๋ฐํ๋๋ ์ฌ์ฉ์ ์ ๋ณด๋ฅผ JSONPOJO ์ฌ์ดํธ์ ์ ์ํด์ ๋ถ์ฌ๋ฃ๊ธฐ ํ ๋ค์
19. ์๋์ ๊ฐ์ด KakaoProfile ํด๋์ค๋ฅผ ์์ฑํ๋ค.
@Data
public class KakaoProfile {
private Long id;
private String connected_at;
private Properties properties;
private KakaoAccount kakao_account;
@Data
public class Properties {
private String nickname;
public String profile_image;
public String thumbnail_image;
}
@Data
public class KakaoAccount {
private Boolean profile_nickname_needs_agreement;
private Profile profile;
private Boolean has_email;
private Boolean email_needs_agreement;
public Boolean is_email_valid;
public Boolean is_email_verified;
public String email;
@Data
class Profile {
private String nickname;
public String thumbnail_image_url;
public String profile_image_url;
}
}
}
20. ๋ค์ ์ปจํธ๋กค๋ฌ๋ก ์์
- ์นด์นด์ค๋ก๊ทธ์ธ์ ํ ์ฌ์ฉ์ ์ ๋ณด๊ฐ ๊ธฐ์กด ํ์์ธ์ง, ์๋์ง ์กฐํ ํ
- ๊ธฐ์กด ํ์์ด ์๋ ๊ฒฝ์ฐ ์๋ ํ์ ๊ฐ์ ์ฒ๋ฆฌ๋๋๋ก ์ฝ๋๋ฅผ ์์ฑํ๋ค.
- ๊ธฐ์กด ํ์์ผ ๊ฒฝ์ฐ ์๋ ๋ก๊ทธ์ธ ์ฒ๋ฆฌํ๊ณ
- ๋ก๊ทธ์ธ ํ์๋ / ์ฃผ์๋ก ๋ฆฌ๋ค์ด๋ ํธ ์ํจ๋ค.
application.yml
cos:
key: cos****
UserController
@Value("${cos.key")
private String cosKey;
user.java
private String oauth; //kakao, google
updateForm.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ include file="../layout/header.jsp"%>
<div class="container">
<!-- <form action="/user/join" method="post"> ์๋ ๋ฐฉ์ ์ฌ์ฉ ์ํจ -->
<form>
<input type="hidden" id="id" value="${principal.user.id}"/>
<div class="form-group">
<label for="username">Username</label>
<input type="text" value="${principal.user.username }" class="form-control" placeholder="Enter Username" id="username" readonly>
</div>
<c:if test="${empty principal.user.oauth }"> <!-- oauth ๊ฐ์ด ๋น์ด์์ง ์์๋๋ง ๋น๋ฐ๋ฒํธ ๋ณ๊ฒฝ๊ฐ๋ฅ -->
<div class="form-group">
<label for="pwd">Password</label>
<input type="password" class="form-control" placeholder="Enter password" id="password">
</div>
</c:if>
<div class="form-group">
<label for="email">Email</label> <input type="email" value="${principal.user.email }" class="form-control" placeholder="Enter Email" id="email" readonly>
</div>
</form>
<button id="btn-update" class="btn btn-primary">ํ์์์ ์๋ฃ</button>
</div>
<script src="/js/user.js"></script>
<%@ include file="../layout/footer.jsp"%>
UserService
package com.cos.blog.service;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.security.authentication.AuthenticationManager;
import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder;
import org.springframework.stereotype.Service;
// ์คํ๋ง์ด ์ปดํฌ๋ํธ ์ค์บ์ ํตํด์ Bean์ ๋ฑ๋ก์ ํด์ค . IOC๋ฅผ ํด์ค๋ค
import org.springframework.transaction.annotation.Transactional;
import com.cos.blog.model.RoleType;
import com.cos.blog.model.User;
import com.cos.blog.repository.UserRepository;
@Service
public class UserService {
@Autowired
private UserRepository userRepository;
@Autowired
private BCryptPasswordEncoder encoder;
@Transactional
public void ํ์๊ฐ์
(User user) {
String rawPassword = user.getPassword(); // 1234 ์๋ฌธ
String encPassword = encoder.encode(rawPassword); //ํด์ฌ
user.setPassword(encPassword);
user.setRole(RoleType.USER);
userRepository.save(user);
}
@Transactional
public void ํ์์์ (User user) {
// ์์ ์์๋ ์์์ฑ ์ปจํ
์คํธ User์ค๋ธ์ ํธ๋ฅผ ์์ํ ์ํค๊ณ , ์์ํ๋ User ์ค๋ธ์ ํธ ์์
// select๋ฅผ ํด์ User์ค๋ธ์ ํธ๋ฅผDB๋ก ๋ถํฐ ๊ฐ์ ธ์ค๋ ์ด์ ๋ ์์ํ๋ฅผ ํ๊ธฐ ์ํด์!!
// ์์ํ๋ ์ค๋ธ์ ํธ๋ฅผ ๋ณ๊ฒฝํ๋ฉด ์๋์ผ๋กDB์ update๋ฅผ ๋ ๋ ค์ค๋ค
User persistance = userRepository.findById(user.getId()).orElseThrow(()->{
return new IllegalArgumentException("ํ์ ์ฐพ๊ธฐ ์คํจ" );
});
// Validate ์ฒดํฌ => oauth ํ๋์ ๊ฐ์ด ์์ผ๋ฉด ์์ ๊ฐ๋ฅ
if(persistance.getOauth() == null || persistance.getOauth().equals("")){
String rawPassword = user.getPassword();
String encPassword = encoder.encode(rawPassword);
persistance.setPassword(encPassword);
persistance.setEmail(user.getEmail());
}
// ํ์ ์์ ํจ์ ์ข
๋ฃ์ = ์๋น์ค ์ข
๋ฃ = ํธ๋์ญ์
์ข
๋ฃ = commit์ด ์๋์ผ๋ก ๋ฉ๋๋ค.
// ์์ํ๋ persistance๊ฐ์ฒด์ ๋ณํ๊ฐ ๊ฐ์ง๋๋ฉด ๋ํฐ์ฒดํน์ด ๋์ด update๋ฌธ์ ๋ ๋ ค์ค
}
@Transactional
public User ํ์์ฐพ๊ธฐ(String username) {
User user = userRepository.findByUsername(username).orElseGet(()->{
return new User();
});
return user;
}
}
// @Transactional(readOnly = true) // Select ํ ๋ ํธ๋์ญ์
์์, ์๋น์ค ์ข
๋ฃ์์ ํธ๋์ญ์
์ข
๋ฃ (์ ํฉ์ฑ)
// public User ๋ก๊ทธ์ธ(User user) {
// return userRepository.findByUsernameAndPassword(user.getUsername(), user.getPassword());
// }
// @Transactional
// public int ํ์๊ฐ์
(User user) {
// try {
// userRepository.save(user);
// return 1;
// } catch (Exception e) {
// e.printStackTrace();
// System.out.println("UserService : ํ์๊ฐ์
() : " + e.getMessage());
// }
// return -1;
// }
// User ์ค๋ธ์ ํธ : username, password, email
System.out.println("์นด์นด์ค ์์ด๋(๋ฒํธ) : " + kakaoProfile.getId());
System.out.println("์นด์นด์ค ์ด๋ฉ์ผ : " + kakaoProfile.getKakao_account().getEmail());
System.out.println("๋ธ๋ก๊ทธ์๋ฒ ์ ์ ๋ค์ : "+kakaoProfile.getKakao_account().getEmail()+"_"+kakaoProfile.getId());
System.out.println("๋ธ๋ก๊ทธ์๋ฒ ์ด๋ฉ์ผ : "+kakaoProfile.getKakao_account().getEmail());
// UUID๋ -> ์ค๋ณต๋์ง ์๋ ์ด๋ค ํน์ ๊ฐ์ ๋ง๋ค์ด๋ด๋ ์๊ณ ๋ฆฌ์ฆ
System.out.println("๋ธ๋ก๊ทธ์๋ฒ ํจ์ค์๋ : "+cosKey);
User kakaoUser = User.builder()
.username(kakaoProfile.getKakao_account().getEmail()+"_"+kakaoProfile.getId())
.password(cosKey)
.email(kakaoProfile.getKakao_account().getEmail())
.oauth("kakao")
.build();
// ๊ฐ์
์ ํน์ ๋น๊ฐ์
์ ์ฒดํฌ ํด์ ์ฒ๋ฆฌ
User originUser = userService.ํ์์ฐพ๊ธฐ(kakaoUser.getUsername());
if(originUser.getUsername() == null) {
System.out.println("๊ธฐ์กด ํ์์ด ์๋๊ธฐ์ ์๋ ํ์๊ฐ์
์ ์งํํฉ๋๋ค");
userService.ํ์๊ฐ์
(kakaoUser);
}
System.out.println("์๋ ๋ก๊ทธ์ธ์ ์งํํฉ๋๋ค.");
// ๋ก๊ทธ์ธ ์ฒ๋ฆฌ
Authentication authentication = authenticationManager.authenticate(new UsernamePasswordAuthenticationToken(kakaoUser.getUsername(), cosKey));
SecurityContextHolder.getContext().setAuthentication(authentication);
return "redirect:/";โ
์ฒ์ Kakao API๋ฅผ ํด๋ดค๋ค.. ์ ๋ง ํท๊ฐ๋ฆฌ๊ณ ๊ณ ์น ๊ป ๋ง์์ ํ๋ค์๋ค.
์ฝ๋๋ฅผ ๋ค ๊ณ ์ณค๋์ง๋ ๋ชจ๋ฅด์ง๋ง ํ๋ฒ ํ์ํ์๋ฉด ๋ค๊ณ ๊ฐ์ ํด๋ณด์ ์
'Springboot > Springboot Blog Project' ์นดํ ๊ณ ๋ฆฌ์ ๋ค๋ฅธ ๊ธ
ํ์๊ฐ์ ๋ฌธ์ ์ ๊ฒ์๊ธ ์ญ์ ๋ฌธ์ ํด๊ฒฐ (0) | 2022.09.27 |
---|---|
ํ์ ์ ๋ณด ์์ ๊ตฌํํ๊ธฐ (0) | 2022.09.25 |
์คํ๋ง ์๋ ์๋ฆฌ ๋ณต์ต! ์ค์โโ (0) | 2022.09.25 |
๊ธ ์์ธ๋ณด๊ธฐ, ๊ธ ์ญ์ ํ๊ธฐ, ๊ธ ์์ ํ๊ธฐ ๊ตฌํํ๊ธฐ (1) | 2022.09.25 |
๊ธ์ฐ๊ธฐ, ๊ธ ๋ชฉ๋ก๋ณด๊ธฐ, ํ์ด์ง ๊ตฌํํ๊ธฐ (0) | 2022.09.25 |