blog 예제 로그인(statefull)
[Spring] blog 예제 회원가입, 로그인 까지
Jan 30, 2024
기본 세팅
코드 가져와서 “화면 초기 세팅 완료” 로 reset해서 시작하기
application
application-prod.yml 수정
data:image/s3,"s3://crabby-images/8e7f2/8e7f2c3128a54c9f24b7670fe1465695af68df24" alt="notion image"
application-dev.yml 수정
data:image/s3,"s3://crabby-images/8a3ef/8a3ef193ffd21c721109f5495e15dc7e1dfd5653" alt="notion image"
application.yml
data:image/s3,"s3://crabby-images/66ad6/66ad6faf7f38bb54448dac66ddb4b6ed06376d4d" alt="notion image"
data:image/s3,"s3://crabby-images/84153/84153e2a72a5fc3421f5df166345526ea637f581" alt="notion image"
-prod이면 application-prod.yml으로 실행됨 개발 환경에서는 -dev로 할것
패키지 구성 변경
data:image/s3,"s3://crabby-images/db01d/db01d3d17f7cc36f283dd1f34c98b480caa246a7" alt="notion image"
전
data:image/s3,"s3://crabby-images/4b703/4b7031ac142aa24cea4e006920b910522eddf94f" alt="notion image"
후
회원가입 코드 작성
요청은 url, uri로 받는다
데이터는 DTO로 받는다.
UserController - join 추가
@PostMapping("/join") public String join() { return "redirect:/loginForm"; }
data:image/s3,"s3://crabby-images/80bfc/80bfcd4885ebaeb15e16abe294e613d8519049be" alt="notion image"
UserRequest 생성, DTO작성
data:image/s3,"s3://crabby-images/4977e/4977e33ec3c939c47355ccc9960289c16b724d6f" alt="notion image"
data:image/s3,"s3://crabby-images/872a0/872a0ba2a81f97c934fa85483dccbea37861b04a" alt="notion image"
UserController - join 에 파라미터를 JoinDTO로 넣어줌
H2 DataBase 사용
application-dev.yml 에 h2 DB사용을 위한 코드 작성
spring: datasource: driver-class-name: org.h2.Driver url: jdbc:h2:mem:test;MODE=MySQL username: sa password: h2: console: enabled: true
data:image/s3,"s3://crabby-images/e2df9/e2df9f936c4b51b5eb99e7e59a5271c3c8aaf11c" alt="notion image"
http://localhost:8080/h2-console 접속하여 데이터베이스 연결
DB 화면캡처
data:image/s3,"s3://crabby-images/7e20e/7e20e109be04b40ded7a4905824cc88d6b2da9b1" alt="notion image"
data:image/s3,"s3://crabby-images/e7ab9/e7ab9c50eae62095250787ece32536fbfca3a3fe" alt="notion image"
console url 설정 → url로 접속가능하게 하는 코드
data:image/s3,"s3://crabby-images/888c5/888c57f58850a42a3579118bee109578d88a39f4" alt="notion image"
User 클래스 생성, Table 생성 → Java에서 코드로 테이블 생성이 가능
data:image/s3,"s3://crabby-images/b7ef6/b7ef64b18f3e6982e982f58583f30559eb4186e0" alt="notion image"
코드
@Data @Entity @Table(name = "user_tb") public class User { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private int id; private String username; private String password; private String email; }
data:image/s3,"s3://crabby-images/9f0be/9f0becf9e2e34852908b0529fa5d8107a81bacd1" alt="notion image"
DB화면 캡처
data:image/s3,"s3://crabby-images/f6ec5/f6ec5bf5eaede8126e765c1bbd625f858a4de676" alt="notion image"
@Entity는 application-dev.yml에
data:image/s3,"s3://crabby-images/7dff2/7dff2d218ab6cac98f761661b8c6465e2ab05565" alt="notion image"
Spring 서버가 실행될때 Entity를 찾아 분석하여 create함 - 개발모드에서만 사용할 것
제약조건 설정
User
@Data @Entity @Table(name = "user_tb") public class User { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private int id; @Column(unique = true) private String username; @Column(length = 60, nullable = false) private String password; private String email; @CreationTimestamp private LocalDateTime createAt; }
data:image/s3,"s3://crabby-images/bd363/bd36364c900ca8249af982951eb3086da26ac54a" alt="notion image"
application-dev.yml → 코드 추가(Run에서 쿼리 편하게 볼 수있음)
data:image/s3,"s3://crabby-images/adac5/adac5c775eac2936c74118f9b0a30f664cb047ba" alt="notion image"
data:image/s3,"s3://crabby-images/a6a78/a6a78b8f7434c6d8781f76465eb25d7365df52bb" alt="notion image"
data:image/s3,"s3://crabby-images/d253e/d253e12cf57b30fe6f7910168d8e68fd0aa07a46" alt="notion image"
UserRepository 생성(DAO로 생각)
data:image/s3,"s3://crabby-images/cd7c9/cd7c9ddbfe2ae3f9d794f73b2cc648bc75302664" alt="notion image"
UserRepository
@Repository public class UserRepository { private EntityManager em; // 의존성 주입 // 생성자 public UserRepository(EntityManager em) { this.em = em; } public void save(UserRequest.JoinDTO requestDTO){ System.out.println("UserRequest에 save메서드 호출됨"); } }
의존성 주입
계속 필요할때 마다 new를 하게되면 메모리가 감당하지 못하기 때문에 DI(Dependency Injection)
data:image/s3,"s3://crabby-images/44b64/44b640d70daf26654cdb9f322f6c2ea33787ff2d" alt="notion image"
UserController 코드 전체
package shop.mtcoding.blog.user; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; @Controller public class UserController { private UserRepository userRepository; public UserController(UserRepository userRepository) { this.userRepository = userRepository; } @PostMapping("/join") public String join(UserRequest.JoinDTO requestDTO) { System.out.println(requestDTO); // 1. 유효성 검사 if(requestDTO.getUsername().length()<3){ return "error/400"; } // 2. Model에 위임하기 userRepository.save(requestDTO); // 3. 응답 return "redirect:/loginForm"; } @GetMapping("/joinForm") public String joinForm() { return "user/joinForm"; } @GetMapping("/loginForm") public String loginForm() { return "user/loginForm"; } @GetMapping("/user/updateForm") public String updateForm() { return "user/updateForm"; } @GetMapping("/logout") public String logout() { return "redirect:/"; } }
error패키지 만들어서 400이름으로 에러 페이지 하나 만들기
400
<!doctype html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0"> <meta http-equiv="X-UA-Compatible" content="ie=edge"> <title>Document</title> </head> <body> <h1>에러 : 잘못된 요청을 했습니다. 400</h1> </body> </html>
data:image/s3,"s3://crabby-images/5e381/5e3811824b5d778f5c4a4cc62179f5eb3dc6ce74" alt="notion image"
쿼리 작성 UserRepository
save
@Transactional public void save(UserRequest.JoinDTO requestDTO){ Query query = em.createNativeQuery("insert into user_tb(username,password,email) values(?,?,?)"); query.setParameter(1,requestDTO.getUsername()); query.setParameter(2,requestDTO.getPassword()); query.setParameter(3,requestDTO.getEmail()); query.executeUpdate(); }
data:image/s3,"s3://crabby-images/c75d7/c75d778d5155e70a449bdfc2f5c3f7eb280873d6" alt="notion image"
위에 어노테이션 붙이면됨
@Transactional을 안붙이면 쿼리를 전송하지 않는다(insert, delete, update 같은 경우는 데이터베이스를 바꾸는 위험한 쿼리로 인식하기 때문에)
data:image/s3,"s3://crabby-images/1da65/1da657df4e0608547b0b9462b6e5dd18a69f2a12" alt="notion image"
다른 기술(알아서 쿼리를 작성해준다)
saveV2
@Transactional public void saveV2(UserRequest.JoinDTO requestDTO){ User user = new User(); user.setUsername(requestDTO.getUsername()); user.setPassword(requestDTO.getPassword()); user.setEmail(requestDTO.getEmail()); em.persist(user); }
data:image/s3,"s3://crabby-images/8d908/8d908001758f01c5909430906859dd025f34ffba" alt="notion image"
현재까지의 코드 실행
data:image/s3,"s3://crabby-images/001c6/001c6e80cda18a007696653f117cf3444d94e44d" alt="notion image"
data:image/s3,"s3://crabby-images/888a6/888a6abba0a3e6ef453b94bd6b6d5a606347e964" alt="notion image"
회원가입 후 DB에 insert된 것을 확인
data:image/s3,"s3://crabby-images/7c3f9/7c3f912349ef84c663cbe852497018e0ee434388" alt="notion image"
로그인 코드
UserController-login코드 작성
@PostMapping("/login") public String login(UserRequest.LoginDTO requestDTO){ // 1. 유효성 검사 if(requestDTO.getUsername().length()<3){ return "error/400"; } // 2. Model 필요 userRepository.findByUsernameAndPassword(requestDTO); // 3. 응답 return "redirect:/"; }
data:image/s3,"s3://crabby-images/f355f/f355fd32c70a5a16eef4f2ebd6d0296762d2020e" alt="notion image"
findByUsernameAndPassword에 Alt+Enter하면 바로 생성됨
UserRepository
findByUsernameAndPassword
public User findByUsernameAndPassword(UserRequest.LoginDTO requestDTO) { Query query = em.createNativeQuery("select * from user_tb where username=? AND password=?", User.class); query.setParameter(1,requestDTO.getUsername()); query.setParameter(1,requestDTO.getPassword()); User user = (User) query.getSingleResult(); return user; }
data:image/s3,"s3://crabby-images/0aa88/0aa88a2e72eb7a8b921a5805f4226e7b149accb1" alt="notion image"
실행
회원가입
data:image/s3,"s3://crabby-images/1bb3a/1bb3a3f5ef17d718cc159f9ccc9648d731710ef9" alt="notion image"
로그인
data:image/s3,"s3://crabby-images/cb76f/cb76fcdf42e4484e4f31cc5a79a29cfaf6197bc1" alt="notion image"
로그인 완료
data:image/s3,"s3://crabby-images/524ab/524ab0c1192a88f74e6b362077e65fca215b84cf" alt="notion image"
로그인의 목적 → Statefull 만들기
session 영역에 한번이라도 접근하면 session이 만들어짐
@RequiredArgsConstructor → final들만 생성자를만들어줌
data:image/s3,"s3://crabby-images/118aa/118aaa28aa56c96252125abcbbb9e02e635cb596" alt="notion image"
data:image/s3,"s3://crabby-images/f8484/f8484d914085747e6b46e6af410f8716be88eb77" alt="notion image"
data:image/s3,"s3://crabby-images/0a4ad/0a4ada8a40db72a115c0a9ce9117f1bbb32553e6" alt="notion image"
session과 Request에 접근할 수 있음
코드
mustache: servlet: expose-session-attributes: true expose-request-attributes: true
data:image/s3,"s3://crabby-images/e45fc/e45fc903781f8530904d87dc835955208e583396" alt="notion image"
응답 유저가 null이 아니면, session 만들고, index 페이지로 이동
코드
User user = userRepository.findByUsernameAndPassword(requestDTO); if(user == null){ return "error/401"; }else{ session.setAttribute("sessionUser",user); return "redirect:/"; }
data:image/s3,"s3://crabby-images/d7810/d7810b0e3fb8a8070e48d52a72382994fa23b507" alt="notion image"
401
<!doctype html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0"> <meta http-equiv="X-UA-Compatible" content="ie=edge"> <title>Document</title> </head> <body> <h1>인증에 실패하였습니다. 401</h1> </body> </html>
로그인 전에는 회원가입과 로그인만 navbar에 보이도록
로그인이 완료되면 navbar의 회원가입과 로그인을 안보이게
layout/header.mustache
data:image/s3,"s3://crabby-images/078f1/078f129f4c73ce7bc19c898baa3118f330dca473" alt="notion image"
data:image/s3,"s3://crabby-images/b7de9/b7de96c02d56a44f4e1ed293b12ccb4d5c4bae30" alt="notion image"
실행
회원가입 후 로그인
data:image/s3,"s3://crabby-images/cdc0d/cdc0dc2d23d739c1e74eb94434689a76af5405fe" alt="notion image"
navbar 구성 달라짐
data:image/s3,"s3://crabby-images/96b5d/96b5d846bc8862b6a5a8d4e2824147dc66dd1a28" alt="notion image"
검토사항
테이블생성시 password의 nullable = false 가 안먹힘 null값이 아니라 공백이 입력되는것으로 확인된다.
유효성 검사에서 password의 길이를 어느정도 이상으로 하면 상관없을 것이다.
Share article