data:image/s3,"s3://crabby-images/8b68f/8b68f39b78790bb16fbb458f94f5f57acd091d6e" alt="notion image"
제목 폼 데이터, 내용 폼 데이터를 클라이언트에게 받아서 submit 해서 날릴거다. 이 화면을 봤을 때, PostMapping, x-form 타입이란 걸 알 수 있다
1. saveForm.mustache
처음엔 view에 가서 체크를 하자! 화면이 완벽한지 확인하고 가자!
data:image/s3,"s3://crabby-images/1e664/1e664b2953e781368017abd23d0483f23aad746c" alt="notion image"
boardController에... PostMapping을 /board/save로 받는 save 메소드가 있나? 없다!!!! 만들어 주자
[ BoardController ]
data:image/s3,"s3://crabby-images/9bdc3/9bdc3e886ae5e3b176f86d30be22227305ecc796" alt="notion image"
다 만들어줬으면 DTO 만들어서 BODY 데이터 받자
[ BoardRequest 클래스 생성 ]
data:image/s3,"s3://crabby-images/1b1a3/1b1a34854853e3159f71ae4cd745cc7cab7bcbe4" alt="notion image"
아까 메서드 이름이 save 였으니까 SaveDTO로 만들어준다
package shop.mtcoding.blog.board; import lombok.Data; public class BoardRequest { @Data public static class SaveDTO { private String title; private String content; } }
[ Http Body 데이터 받기 테스트 ]
data:image/s3,"s3://crabby-images/5962d/5962dd37b7f87bfda36589e2c0f7a5c8cfb1239f" alt="notion image"
BoardController 에서 save 메소드에 DTO를 request 로 받는다
3. 유효성 검사
[ Board ]
data:image/s3,"s3://crabby-images/5d95b/5d95b137e0d6a03f995cd591cc5960b82c9e1208" alt="notion image"
[ BoardController ]
data:image/s3,"s3://crabby-images/9460e/9460e23fe11069e3c600dc499f7db080e47c0d5d" alt="notion image"
만약, 회사에서 일하는데 30 글자 이상이 들어왔다?
그럼 postman 같은 걸로 우회해서 지속적으로 요청하는 거니까…
우리에 대한 공격으로 간주하고 ip 블락하면 된다
[ 글쓰기 유효성 검사 테스트 ]
data:image/s3,"s3://crabby-images/e2679/e267938368ad6362f86fc3fd51b6ce2d84f38b8d" alt="notion image"
제목을 한 100자 정도로 넣어봤다.
data:image/s3,"s3://crabby-images/c66ab/c66abd868c53538775860f51e31919908c2932fb" alt="notion image"
에러 페이지 뜸
5. 에러 페이지를 동적페이지로 만들기
data:image/s3,"s3://crabby-images/10892/10892aa58ea240a81d4c7552cb1547cf9c5fef17" alt="notion image"
[ 40x.mustache ]
<!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>{{msg}} : {{status}}</h1> </body> </html>
가방에 넣었다가 꺼내쓰게 request를 매개변수로 받아줬다.
data:image/s3,"s3://crabby-images/2ab80/2ab804c149c8f5205c46b7ac49342410bde2be27" alt="notion image"
내가 메세지랑 상태코드만 담으면
String 오브젝트로 만들어진 해쉬맵?
data:image/s3,"s3://crabby-images/8403d/8403dd9313943d3e5034fae79e89662a405013b3" alt="notion image"
error/40x 페이지가 뜬다
지금까지 검증해오면서 코드를 짰으면, 앞으로 오류가 나면 이후에 코드짠게 잘못된 거란걸 바로 알 수 있다.
꼭 검증하면서 코드를 짜자
6. DB insert 하기
data:image/s3,"s3://crabby-images/5883e/5883e405ec6d3a79af86fb50420f2e14ab6a902b" alt="notion image"
title, content는 클라이언트로부터 (DTO) 받기 userId 는 세션에서 가져온다. **세션의 user객체의 아이디**를 들고 와야 한다 createAt 자동생성
순서
1. 인증체크 : 인증이 안되어있으면 바로 내보내야함.
→ 처음에 세션에 접근해야함. 그 이후로 null인지 null 아닌지
data:image/s3,"s3://crabby-images/307de/307de12d21b6b400262b2ad16929cd7e88af6280" alt="notion image"
2. 바디 데이터 확인 및 유효성 검사
data:image/s3,"s3://crabby-images/6243f/6243f247d36c6819232fdd6b85247a9536e24a56" alt="notion image"
3. 모델 위임
data:image/s3,"s3://crabby-images/675a7/675a7bf118be0d2309448aa2b46af8c74d84d982" alt="notion image"
BoardRepository에 save 메소드 만들어주자 alt + enter
[ BoardController ]
@PostMapping("/board/save") public String save(BoardRequest.SaveDTO requestDTO, HttpServletRequest request) { // 1. 인증 체크 User sessionUser = (User) session.getAttribute("sessionUser"); if (sessionUser == null) { return "redirect:/loginForm"; } // 2. 바디 데이터 확인 및 유효성 검사 System.out.println(requestDTO); if (requestDTO.getTitle().length() > 30) { request.setAttribute("status", 400); request.setAttribute("msg", "title의 길이가 30자를 초과해서는 안되요"); return "error/40x"; // BadRequest } // 3. 모델 위임 // insert into board_tb(title, content, user_id, created_at) values(?,?,?, now()); boardRepository.save(requestDTO, sessionUser.getId()); return "redirect:/"; }
[ BoardRepository ]
@Transactional public void save(BoardRequest.SaveDTO requestDTO, int userId) { Query query = em.createNativeQuery("insert into board_tb(title, content, user_id, created_at) values(?,?,?, now())"); query.setParameter(1, requestDTO.getTitle()); query.setParameter(2, requestDTO.getContent()); query.setParameter(3, userId); query.executeUpdate(); }
write하는 건 무조건 @Transactional을 걸어주자. 고립에 걸리니까!
user_id는 게시글을 작성한 사용자의 ID를 나타내며, 현재 로그인한 사용자의 ID를 전달받아서 저장한다. 이렇게 하면 게시글 작성자의 ID를 게시글과 함께 저장할 수 있다. 그래서 user_id를 받는 것!
7. 게시글 쓰기 최종 테스트
1. 로그인 없이 게시글 쓰기 테스트
data:image/s3,"s3://crabby-images/bda13/bda1317d3674242b98c899029747f4cc18960b6b" alt="notion image"
[ 로그인 함 ]
data:image/s3,"s3://crabby-images/abb7a/abb7a6d3ab0919bf7a690779899a05ce33c5460f" alt="notion image"
세션키 나왔죠?
data:image/s3,"s3://crabby-images/a9f0c/a9f0c2cb050f0c200b7113163a31477d4c209eb6" alt="notion image"
data:image/s3,"s3://crabby-images/caf43/caf43a16eab6f809a667fa80f642fc0be75525c2" alt="notion image"
정상 작동!
[ NullPointException ]
data:image/s3,"s3://crabby-images/6bedc/6bedc1f9f205e4e94441a2fab7975ed704610919" alt="notion image"
만약, title을 안 쓰면 유효성 검사 통과가 안돼서 NullPointException이 뜬다
Share article