-
Notifications
You must be signed in to change notification settings - Fork 203
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[4기 강병곤, 최준혁] Spring Boot JPA 게시판 구현 미션 제출합니다. #223
base: hyukgon
Are you sure you want to change the base?
Conversation
List<PostResponseDTO> postList = Arrays.asList( | ||
PostResponseDTO.builder() | ||
.title("title3") | ||
.content("content3") | ||
.createdAt(LocalDateTime.of(2000, 1, 1, 1, 1)) | ||
.modifiedAt(LocalDateTime.of(2000, 1, 1, 1, 1)) | ||
.build(), | ||
|
||
PostResponseDTO.builder() | ||
.title("title4") | ||
.content("content4") | ||
.createdAt(LocalDateTime.of(2000, 1, 1, 1, 1)) | ||
.modifiedAt(LocalDateTime.of(2000, 1, 1, 1, 1)) | ||
.build() | ||
); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
반복해서 사용하는 내용들이라면 beforeEach를 만들어서 넣고 사용해도 좋을 것 같아요~
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
PostResponseDTO를 공통적으로 생성해서 그 부분을 setUP메서드에 넣었습니다~! a9824bc
Long postId = 1L; | ||
|
||
when(postService.findPost(postId)).thenReturn(responseDTO); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
PostId 를 직접 지정해서 사용하는 것보다 anyLong을 사용해보는 건 어떤가요??
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
고민 해보았는데 입력한 postId를 컨트롤러에서 동일하게 찾는지 테스트 해보기 위해서 postId를 계속 사용할려고 합니다!
import lombok.Getter; | ||
|
||
@Getter | ||
@AllArgsConstructor |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
저는 최근부터 AllargsConstructor를 지양하는 방향으로 생각해서 사용하고 있는데요, 그 이유에 대해서도 한번 생각해보면 좋을 것 같아요!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
저도 AllargsConstructor가 일으키는 버그에 대해 검색해 보고 AllargsConstructor를 사용하지 않도록 전체적으로 수정 하였습니다! f74b8d9
@NoArgsConstructor | ||
public class Post extends BaseEntity { | ||
@Id | ||
@GeneratedValue(strategy = GenerationType.AUTO) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
(strategy = GenerationType.AUTO) default라 생략해도 될 것 같아요 의도하신거라면 패쓰
private String title; | ||
private String content; | ||
private Long userId; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
개인적으로 생각했을때, 필드를 final로 선언하지 않으면 데이터의 조작가능성이 생길 수도 있어서, Dto객체는 final로 선언하는 것이 좋다고 생각합니다 ! 어떻게 생각하실까요 ?!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
음 저도 다시 생각 해보니 그렇게 생각해서 Dto 객체를 record 클래스로 변경 하였습니다! f74b8d9
@Lob | ||
private String content; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Lob 어노테이션은 처음 알아가네요 ! 배워갑니다 👍
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p2;
@Lob
는 내부에서 추가로 별도의 조회를 하여 느릴 수도 있다고 합니다. 컬럼 타입을 text 로 직접 명시해서 사용하는 것은 어떤가요~?
this.content = content; | ||
} | ||
|
||
public void setUser(User user) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p3
요즘 setter의 사용에 관심이 많은데요, 한 번에 메소드의 사용의도를 파악하기 힘든 setUser보다는,
협업에서도 사용 가능한 메소드이니 allocate / attach User같은 의미있는 메소드명은 어떠실까요 ?!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
set 대신 의도를 쉽게 파악할 수 있도록 수정하였고 사용하지 않는 setter도 전체적으로 삭제하였습니다~! c7107c4
if (Objects.nonNull(this.user)) { | ||
this.user.getPosts().remove(this); | ||
} | ||
|
||
this.user = user; | ||
user.getPosts().add(this); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p1
만약 user가 갖고있는 post의 자료형이 list가 아닌 hashMap 등으로 변경되면 setUser 메소드도 수정이 추가되어야 할 것 같아요!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
내부 자료형을 모르더라도 추가, 삭제할 수 있도록 수정하였습니다! aac52b4
public ApiResponse<Slice<PostResponseDTO>> getAllPost(@RequestParam int page, @RequestParam int size) { | ||
Slice<PostResponseDTO> posts = postService.getPosts(page, size); | ||
return ApiResponse.ok(posts); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p3
slice를 그냥 반환하면 프론트에서 식별하기 힘든 정보들 (ex : Pageable type : "INSTANCE")같은 정보들도 반환되는 것으로 알고 있는데, 프론트단에서 좀 더 알아보기 쉬운 필드만 갖고 있는 Wrapper 클래스를 하나 만들어 주는 건 어떨까요 ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
slice를 그대로 반환하지 않고 클라이언트에 필요한 정보만 반환될 수 있도록 수정하고 restdocs도 수정하였습니다~! 0db39ee
User user = userRepository.findById(createPostRequestDto.getUserId()) | ||
.orElseThrow(() -> new RuntimeException("User Not Found!")); | ||
|
||
post.setUser(user); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p3
저는 Post객체에 User는 필수적인 존재라고 생각하여 Post생성자에 user를 넣어주었습니다.
만약 post.setUser(user)메소드를, post를 생성하는 다른 메소드에서 호출하는 것을 깜빡하는 것을 방지하기 위해 생성자 혹은 user세팅을 강제하는 다른 방법을 고안해 보는건 어떨까요 ?!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
이 부분은 준혁님과 의논하고 반영하도록 하겠습니다!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Post의 생성자에 User를 넣어도 매핑 메서드를 실행하지 않으면 동일한 문제가 발생한다고 생각합니다.
@Builder | ||
public Post(String title, String content) { | ||
this.title = title; | ||
this.content = content; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p1
Builder로 Post객체를 생성할 시에, nullable하지 않은 필드에 대한 검증(Notnull)이 이루어지지 않은 것 같습니다.
이는 DB에 영속화 시키기 이전까지 잡기 힘든 에러인것 같은데, 유효성 검증로직을 추가하는 것은 어떨까요 ?!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
생성자에서 유효성 검사를 하도록 수정하였습니다! 3f95c3c
@EnableJpaAuditing | ||
@Configuration | ||
public class JpaAuditingConfiguration { | ||
|
||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
저는 main 어플리케이션 클래스에 EnableJpaAuditing 어노테이션을 붙여주었는데, Config클래스에 붙이는게 좀 더 맞다는 생각이 드네요 ! 배워갑니다 👍
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
성공, 실패에 대한 메소드를 따로 만든 점 좋은 것 같습니다 ! 배워갑니다 👍
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
병곤님, 준혁님 과제 하느라 수고 많으셨습니다!! 👏👏
@Lob | ||
private String content; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p2;
@Lob
는 내부에서 추가로 별도의 조회를 하여 느릴 수도 있다고 합니다. 컬럼 타입을 text 로 직접 명시해서 사용하는 것은 어떤가요~?
public record PostResponseDto(String title, String content, LocalDateTime createdAt, LocalDateTime modifiedAt) { | ||
@Builder | ||
public PostResponseDto { | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p5;
@Builder
를 record 단에 붙일 수 있는 것으로 압니다! 어노테이션을 저기에 달아주신 이유가 있나여??
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
이 부분은 몰랐네요 가독성을 위해 수정하였습니다
.orElseThrow(() -> new RuntimeException("Post Not Found!")); | ||
|
||
return PostResponseDto.toResponse(retrievedPost); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p1;
조회 같은 경우에는 성능을 위해 트랜잭션을 읽기 전용 모드로 사용하시는 것이 어떤가요??
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
오호 꿀팁 감사합니다~ 수정하였습니다~! abdc8c8
.orElseThrow(() -> new RuntimeException("Post Not Found!")); | ||
|
||
retrievedPost.changeTitle(updatePostRequestDto.title()); | ||
retrievedPost.changeContents(updatePostRequestDto.content()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p2;
서비스에서 changeTitle
, changeContents
를 각각 호출하기 보다 Post
내부에서 처리해주도록 하는 것은 어떤가여? 만약 메소드명이 변경될 경우 서비스까지 영향을 받을 것 같네요~
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
post 내부에서 처리하도록 수정하였습니다~! 54c57af
private Long id; | ||
|
||
@Column | ||
private String name; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p5;
컬럼에 따로 nullable 같은 세팅도 없고, 컬럼이라는 것을 명시적으로 표현하기 위해 @Column
을 붙여주신 건가요??
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
전체적으로 유효성 검사를 하도록 수정하였습니다~! 3f95c3c
public static <T> ApiResponse<T> fail(int statusCode, T data) { | ||
return new ApiResponse<>(statusCode, data); | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
P4;
ApiSliceResponse
와 공통적인 부분이 있는 것 같은데 상속을 활용해보는 것은 어떤가요??
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
컨트롤러에서 전체적으로 ApiResponse를 응답하도록 수정하였습니다! 5171145
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
준혁님 병곤님! 과제 진행하시느라 수고하셨어요!
다른 팀원분들이 리뷰를 정성스레 달아주셔서 공감하는 부분에 이모지 표시해두었습니다! 👍
private final UserService userService; | ||
|
||
@PostMapping | ||
public ApiResponse<UserResponseDto> createUser(@RequestBody CreateUserRequestDto createUserRequestDto) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@Valid
를 사용해서 dto 검증을 진행하는것도 좋을거 같습니다!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@Transactional | ||
public Slice<PostResponseDto> getPosts(int page, int size) { | ||
PageRequest pageRequest = PageRequest.of(page, size); | ||
|
||
return postRepository.findSliceBy(pageRequest) | ||
.map(PostResponseDto::toResponse); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p1;
offset으로 구현한 걸로 알고 있었는데 이야기해보니 cursor 방식으로 구현하신 건가요?? 설명을 한번 해주시면 감사하겠습니다!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
offset 방식으로 구현했는데 그 때 제가 잘 못 말했습니다.
라이브 리뷰 완료 |
📌 과제 설명
강병곤, 최준혁 spring boot jpa 게시판 구현하여 제출 합니다.
👩💻 요구 사항과 구현 내용
✅ PR 포인트 & 궁금한 점
post 테스트에 getPosts 메서드에서 페이지네이션 응답 결과를 문서화하기 위해 slice 객체를 다음과 같이 작성하였는데 이런 식으로 하는게 맞는지 궁금합니다! (767a062)-> 팀 미팅 때 답변 해주셔서 반영하고 있습니다!