Board의 Entity 안에는 많은 정보가 들어간다. PK인 id값을 포함해 제목, 내용, 작성자 등도 들어가지만, 가장 중요한 게시물을 지울 수 있는 password값이 들어가있다. password값까지 response로 리턴해줄 필요도 없고, 또한 누가 접근했을 때 한단계 더 숨겨줄 수 있기 때문이다.(Encapsulism과 비슷한 맥락이라고 보면 된다.)
이 Letter를 list로 만들어서 반환하는 것이 아니라,
이렇게 ResponseDto에 담아 주는것이 현명하다.
그럼 Letter를 ResponseDto로 바꿔 주어야 하는데, 어떻게, 또 어느 Layer에서 변환을 해 주어야 할까?
Controller Layer에서 변환해주어도 괜찮지만, 보통은 Service Layer에서 변환작업을 해 주는게 좋기도 하고, flow에 맞다.
그럼 Entity를 DTO로 어떻게 바꿔줄까?
내가 알아본 방법은 두 가지였다.
1. Stream을 이용해 바꿔주기
2. Array로 만들고 List로 바꿔주기
그 중 Stream을 이용하기로 했다.
이렇게 자바8의 collection을 사용해서 바꾸어 주었다.
LetterController
@RestController
@RequiredArgsConstructor
@RequestMapping("/api")
public class MainController {
private final LetterService letterService;
@GetMapping("/posts")
public List<LetterResponseDto> viewAll(){
return letterService.findAll();
}
@PostMapping("/post")
public Long postLetter(@RequestBody LetterRequestDto letterRequestDto){
return letterService.postLetter(letterRequestDto);
}
}
여기서 @RestController란 무엇일까?
@RestController = @Controller + @ResponseBody이다.
@Controller 어노테이션은 기존의 컨트롤러 어노테이션과 똑같고,
@ResponseBody 어노테이션은 return해주는 객체를 MessageConverter를 통해 매핑해서 본문에 담아 클라이언트에 전달한다.
처음에는 RequestBody와 헷갈려서 그대로 전달했다가 에러가 나버렸다.
그래서 Controller에 @RequestBody를 다 붙여주었다.
그랬더니 또 문제가 났다.
api 명세서에는 password만 들고 들어와야 하는 것이다.
보통 우리가 게시판에서 글을 지우려면 pop-up창에 password 하나만 치면 삭제가 되는 구조를 많이 이용한다.
그래서 이 password값만 받아오고 싶은데, 도저히 RequestDto에 있는 password값을 가져와서 다시 확인하기는 싫었다.
나중에 상세 페이지에 들어가서 삭제를 누르려면 RequestDto를 이용하면 되겠지만, 상세 페이지 밖에서 글을 삭제하려면
RequestDto에 있는 타이틀, 제목같은 not null값을 다 채워줘야 하기 때문이다.
한참을 500에러와 씨름하다가 다음과 같은 깨달음을 얻었다.
@RequestBody
RequestBody는 요청값을 json type로 받겠다고 선언하는 어노테이션으로, 이 어노테이션 덕분에 우리는 DTO 형태로 값을 가져올 수 있다. Spring은 Controller로 요청을 받을 때 jackson이라는 내장 라이브러리에 있는 MappingJackon2HttpMessageConverter를 이용해 요청값으로 들어온 json형태의 데이터를 DTO에 맞는 형태로 파싱해주게 된다. 그래서 우리는 스프링을 통해 json 형태를 편리하게 DTO라는 객체로 받아올 수 있는 것이다.
그래서 이 JSON 형태의 글이 @Responsebody 어노테이션을 통해
RequestDto requestDto = new RequestDto(title,content,author,password)
형태로 자동 바인딩되어 저장된다.
이를 "역직렬화 / DeSerialization"이라고 부른다! (Singleton패턴에서 왜 이런 개사기스킬을 만들어놨는지 궁금했는데 여기서 깨달았다.)
역직렬화는 생성자를 거치지 않고, Java의 Reflection을 통해 객체를 정의하는 매커니즘이다. 직렬화 가능한 클래스들은 기본 생성자가 항상 있어야 하기 때문에 RequestDto에 기본 생성자가 정의되어있지 않으면 데이터 바인딩에 실패한다.
아무튼 나는
@DeleteMapping("/post/{id}") //RequestBody 방식으로 구현, body : raw - text
public String deleteLetter(@PathVariable Long id, @RequestBody String password) throws JSONException {
return letterService.deleteLetter(id, password);
}
이렇게 RequestBody 어노테이션 뒤에 String password를 선언하였기 때문에 JSON타입의 password가 내가 생각한 JSON객체로 들어오지않고
String password = “{\“password\“:\“1234\“}”;
와 같은 형태로 들어왔던 것이다.
그럼 어떻게 하는데!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
방법은 3개가 있다.
1. 죽어도 RequestBody를 쓰고 싶으면 @RequestBody JSONObject password로 받기
2. @ModelAttribute를 쓰기 : ModelAttribute는 RequestBody나 RequestParam을 선언하지 않으면 디폴트로 들어가기 때문에
@DeleteMapping("/post/{id}") //ModelAttribute 방식으로 구현, body : x-www.form-urlencoded
public String deleteLetter(@PathVariable Long id, String password) throws JSONException {
return letterService.deleteLetter(id, password);
}
이렇게 선언해서 나중에 body타입으로 받아주기
3. @RequestParam 사용하기 : RequestParam은 1개의 parameter를 받기 위해 사용되고있다. null 들어가면 안된다.
@DeleteMapping("/post/{id}") //REQUESTPARAM 방식으로 구현, body : form-data
public String deleteLetter(@PathVariable Long id, @RequestParam String password, Model m) throws JSONException {
m.addAttribute("password",password);
return letterService.deleteLetter(id, password);
}
느낀점
정말 맛있는 과제였다. 덕분에 정말 HTTP request, response, 그리고 MVC패턴과 각종 어노테이션에 대해서 학습할수 있는 좋은 기회였다....
그런데 내가 만들어야 하는것은 FE에서 준 query를 가지고 youtube에 get 방식으로 요청을 보내서 받은 response를 파싱해서 다시 FE로 보내주어야 프론트에서 그것을 렌더링할 수 있고, 그 기능을 실제로 구현해야 한다!
youtube api를 쓰면 쉽게 파싱할 수 있지만, 화이트카드를 계속 발급받으면서 Auth를 하는데 번거로움이 있어 그냥 대가리 박고 파싱하기로 했다.
문제 발생
근데 이거 어케함?
8080에 요청이 들어올때 response를 쥐어주는 법, request로 파싱하는 법은 알았다.
근데 내가 클라이언트 입장에서 다른 서버에 요청을 보내고 받아오는 것은 한번도 해보지 않은 기술이라 벌벌 헤멨다.
어찌저찌 유튜브에서 받아온 html(유튜브는 SSR 방식을 이용하기 때문에, 쉽게 말하면 빈 템플릿을 보내고 그 안에 이런저런 요청받은 정보들을 가져와서 렌더링한다). 은 또 어떻게 파싱할꼬...
시도해본 것들
일단 처음으로 youtube에 get 메서드로 통신하기 위해서는 여러 방법이 있었다.
HttpUrlConnection 사용하기 -> 너무 옛날 기술이라 요즘엔 잘 사용하지 않는다고 함
RestTemplate -> Spring 3 이후에 나온 기술, 현재는 또 Depricated당했다.
WebClient -> Spring 5 이후에 나온 기술
그중에 RestTemplate를 사용해보기로 했다.
UriComponentsBuilder를 이용해 Uri에 쿼리를 담아 보냈는데,
텅 빈 ResponseBody만 온다..
다음 코드의 문제점은 뭘까?
8080포트에서는 200번대로 다 떴는데, status code를 로그로 찍어보니 300번대가 떴다.
왜 301이 떴지? 하면서 온갖 생각을 다 하다가 하나를 생각해냈다.
HTTPS
HTTPS 프록시 기술을 사용하기 때문에 리다이렉팅을 하는 과정에서 불러오지 못했던 것이었다.
이걸가지고 3~4시간을 헤메다니...
아무튼 성공했고, 다음과 같은 responseBody를 받았다.
이거를~ 어떻게~ 파싱할까요~?
이 긴 String 사이에 JSON이 숨어있다. 그 숨겨진 JSON을 꺼내서 우리가 원하는 타이틀로 바꿀것이다.
jsoup은 html parser라 html이 변동되면 사용할 수 없기 때문에
youtube api를 쓰지 않으려면 일일히 파싱을 해 주어야 한다. (CSR은 깡통 html만 주기 때문에 jsoup을 쓸 수 없다.)
파싱은 JSONParser를 이용해서 파싱하려고 했다.Json.simple 라이브러리를 이용해서 바꾸어 주려고 하니
이 안에 파싱을 할 때 JSONObject로 파싱되는지, JSONArray로 파싱되는지에 따라 계속 확인해주어야 하고, 계속 형변환을 해 주었다.
JSONParser를 쓰는 레퍼런스들은 많아서 찾아볼만했지만 보통은
[{곤충1 : 잠자리}, {곤충2 : 매미}] 등의 단순화된 스트링에서만 파싱을 했고,
{asdfasdfdsfa:{asdfsdafsdfa{asdfdsaffdsafds:gdfadsaffdsafsadfdsa:sdaffsadfsadfdas:···}···}···}등의 구조를 알아보기 힘든 긴 JSON을 파싱하는 레퍼런스들은 없었기에, 일일히 해 주어야 했는데,
이 문장을 1억줄 쓰려니 생각만 해도 아찔했다.
바로 ObjectMapper의 등장이다.
ObjectMapper에서 readTree로 데이터를 읽으면 JsonNode로 반환한다. 더욱이나 좋은것은, get 메서드 안에 index와 fieldname을 둘 다 혼용해서 쓸 수 있기에 형변환을 안해주어도 된다는 점이 매우매우 마음에 들었다.
알게 된 것
머리를 박으면서 코딩을 하다보니 알게된것들이 있었다.
1. BE api의 flow
MusicParsercontroller에서 MusicParserService를 const injection을 통해서 주입하고 GetMapping으로 /search에서 FE에서 날려준 request를 가지고 MusicparserService.method(request)한다는 것
객체 지향적 언어란 말 그대로 '지향'일 뿐이다. 객체 지향적 언어에 정답은 없다고 생각한다.
객체 지향에 다가가고 있어도, '이것이 객체 지향이다'는 말은 섣불리 할 수가 없었다.
SOLID 원칙을 배우면서 객체를 어떻게 구성해야 하는지에 대해서도 조금은 알 수 있었지만,
내가 앞으로 짜는 코드에 대해서 어떻게 객체지향적으로 짜야하는지는 아직 너무 이르기도 하고 디자인 패턴들을 많이 배워야 할 것 같다. JVM도 마찬가지다. CS지식이 전무한 내게 JVM의 시스템 구조를 조금 본 것은
혀에 잠깐 한 방울 닿은 정도로만 다가왔다.
그래서 이번주는 솔직히 점수를 많이 줄 수 없다고 생각한다.
조금 더 과장을 보태면 0점을 주고 싶다.
느낀 점
아는 것을 다시 공부하기는 굉장히 힘들다. 자바의 정석을 몇번이나 봐 온 나로써는 솔직히 조금 지루했다.
문제는 지루하면 안 된다는 것에 있다. 볼때마다 새로운 지식이 나오는데도 불구하고 무엇에 우선순위를 두어야 하는지에 대해 계속 뺑뺑 돌았다. 알고리즘 공부하다가 자바보다가 새로운 지식 보다가 하니 결국 남은게 없었다.
아는 형에게 정말 좋은 제안이 왔다. 스프링 개발을 부로, swift를 주로 하는 곳인데, 과장님이 나를 좋게 보신다고 이력서를 한 번 내보라고 했었는데, 그것에 대해 정말 고민했다.
그리고 나서 며칠 뒤 밤에, 같이 개발자를 목표로 했던 동료에게도 전화가 왔다. 면접을 봤었는데 스프링은 왜 쓰는지, 또 다른 개념에 대해선 무엇을 알고 있는지를 물어봤다고 했다. 곰곰히 생각해 봤는데 내가 거기에 대해서 답변할 수 있는게 너무 적었다. 그래서 부끄러워서 아는 형에게 조금 더 공부해야 할 것 같다고 연락을 남겼다.
이 상태로 취업을 했다간 나보다 나를 소개시켜준 형에게 부끄러워서 회사를 다니지 못할것 같았다. 취업시켜준다는 말은 안했지만... 정말 많은 것을 보고 듣고 알고 가야겠다고 생각했다.
다음 주 목표
하나에 몰두하지 않고 여러개를 돌린 시도 자체는 좋으나 결과가 너무 참담했다. 조금 더 개념에 치중해서 공부하고, 스프링 주간에 가면 할 게 많으니 빠트리지 말고 모조리 챙기고 가도록 하자.
pymongo를 이용해 로그인 페이지와 회원가입 페이지를 만들고, ID와 PW는 몽고DB에서 관리, 로그인 정보는 session에서 관리하게 만들었다.
클라이언트단은 완벽하지만, 서버단의 구조는 매우 빈약한것을 볼 수 있다. 조금만 더 알았더라면, 조금만 더 공부했더라면 다른 팀원들이 구현하고자 했던 기능들을 더 구현할 수 있었는데, 내 능력의 부족으로 이루지 못해 너무 속상했고, 각오를 다졌다.
느낀 점
컴퓨터는 앞으로만 가지만, 사람은 양방향성을 띈다.
앞으로만 가는 컴퓨터의 활동을 제어하는것은 쉽지만, 때로는 전진하고, 때로는 후퇴하는 사람을 제어하는것은 훨씬 힘든 일이다. 첫 협업에서부터 삐끗했던 활동을 생각하며, 옛날에 떠올렸던 '이런 사람과도 협업을 못하는데, 내가 과연 smooth한 협업을 진행할 수 있을까?'라는 생각을 다시금 했다.
내 코드는 컴퓨터만 읽는 게 아니라, 사람도 읽는다.
컴퓨터가 읽기 좋은 코드로 개발하는것도 좋지만, 사람이 읽기 좋은 코드로 개발하는것도 꼭 필요한 프로세스다.
읽기 쉽게 코딩하지 않으면 남들이 보는 데 거부감이 든다.
제대로 알지 못하면, 안 하느니만 못하다.
인강을 다 들었어도 실제 구현을 하려고 하면 막막하기도 하고, 도대체 내가 무엇을 하고 있는지, 어디에 있는지조차 헷갈린다.
다음 주 목표
- 위의 것들을 기반으로 한 원활한 소통과 협업이 이루어지는, '케미스트리'가 가득한 오케스트라 같은 팀 구성하기.
- RESTful한 API 설계로 모르는 사람이 보아도 한 눈에 무슨 api인지 알게 하기
- code-convention과 commit-convention으로 명확한 의미를 지닌 코드/소통 설계하기
첫 번째 명령어는 python3 ~~~~ 라고 해야하는 명령어를 python ~~~~로 바꿔주는 명령어다. 프로그래머는 귀찮아해야한다. 그래야 더 나은 방식을 생각할 수 있으니까.
두 번째 명령어는 flask 환경에서 구동할 때 쓰이는 패키지들을 관리해주는 패키지 툴인 pip3를 설치하고, 이름을 pip로 바꿔주는 것이다.
이게 가장 중요한데 우리가 계속 진행해왔던 localhost:5000에서 이 포트번호 5000을 포트번호 80으로 리다이렉팅 해준다는 것이다. 리다이렉팅과 포워딩의 차이는 추후에 다시 설명하겠다.
그리고 Filezila에서 다시 연동하고 아마존의 인스턴스에 실행할 app.py나 index 등을 넣어준 후, ls(list : 데이터 목록 보여주기)
cd(change directory : 디렉토리 변경)
mkdir(make directory : 디렉토리 생성)
pip install packagename(파이썬의 packang
등의 명령어를 입력해 '알아서 잘' 하고 python app.py 명령어로 가동시키면 된다.
인바운드 규칙 추가
인바운드 규칙 추가는 우리가 사용할 인스턴스의 포트번호를 추가해준다.
이렇게 TCP/5000 포트와 HTTP/80포트를 열어주어야 비로소 우리가 만들었던 홈페이지에 접속 가능하다.
80포트의 비밀
우리가 접속하는 대부분의 인터넷 사이트(네이버, 다음 등)는 사실 #http://www.naver.com:80 인데, 80포트가 생략된 것이다. 80은 생략이 가능하기 때문에. 그래서 아까 우리가 포트포워딩으로 5000번을 80에 연결해놓은 것이다. 그러면 접속할때마다 사이트 뒤에 5000포트를 생략해도 된다.
이 포트포워딩은 리눅스에서 자동 지원된다.
nohup 설정하기
nohup은 우리가 인스턴스에서 구동한 app.py를, git bash를 종료해도 구동될 수 있도록 하는것이다. 방법은 매우 쉽다.
그냥 git bash에 nohup python app.py를 써주면 계속 돌아간다.
나는 왜 그런진 모르겠지만 습관적으로 nohub를 써주는 경향이 있어 조금 더 주의를 요해야 한다. 왜 그럴까?