5 팔로우
팔로우 service
import com.post_show_blues.vine.domain.follow.FollowRepository;
import com.post_show_blues.vine.domain.member.MemberRepository;
import com.post_show_blues.vine.domain.notice.NoticeRepository;
import com.post_show_blues.vine.dto.NoticeResultDTO;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
@RequiredArgsConstructor
@Service
public class FollowService {
private final FollowRepository followRepository;
private final NoticeRepository noticeRepository;
private final MemberRepository memberRepository;
/**
* 팔로우
*/
@Transactional
public void isFollow(long fromMemberId, long toMemberId){
followRepository.rFollow(fromMemberId, toMemberId);
noticeSave(fromMemberId, toMemberId);
}
/**
* 언팔로우
*/
@Transactional
public void isUnFollow(long fromMemberId, long toMemberId){
followRepository.rUnFollow(fromMemberId, toMemberId);
}
private void noticeSave(long fromMemberId, long toMemberId){
String fromMemberNickname = memberRepository.findById(fromMemberId).get().getNickname();
NoticeResultDTO noticeResultDTO = NoticeResultDTO.builder()
.memberId(toMemberId)
.text(fromMemberNickname + "님이 회원님을 팔로잉 했습니다.")
.link("/member/"+fromMemberId)
.build();
noticeRepository.save(noticeResultDTO.toEntity());
}
}
사실 follow service의 메소드도 이름을 follow로 하고 싶었는데 그러면 엔티티 이름이랑 repository의 메소드 이름이랑 겹칠 것 같아서 차선책으로 isFollow로 지었다. 이런 경우 어떻게 해야될 지는 계속 고민해야 될 듯 하다
우리 서비스는 인스타그램처럼 팔로우하면 알람테이블에 저장해놓는 구조이므로 follow 함과 동시에 테이블에 저장하도록 했다. link 에는 해당 메시지를 넣었을때 어디로 갈 지 주소값을 넣어봤다.
알림 dto
import com.post_show_blues.vine.domain.notice.Notice;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
@Data
@Builder
@AllArgsConstructor
@NoArgsConstructor
public class NoticeResultDTO {
private Long memberId;
private String text;
private String link;
public Notice toEntity() {
return Notice.builder()
.memberId(memberId)
.text(text)
.link(link)
.build();
}
}
entity랑 모양 똑같은데 entity는 언제든지 변경될 수 있으니 dto를 따로 만들어주는게 좋을 듯 하다
팔로우 repository
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Modifying;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.query.Param;
public interface FollowRepository extends JpaRepository<Follow, Long> {
@Modifying
@Query(value="INSERT INTO follow (from_member_id, to_member_id) VALUES (:fromMemberId, :toMemberId)", nativeQuery = true)
void rFollow(@Param("fromMemberId") long fromMemberId,@Param("toMemberId") long toMemberId);
@Modifying
@Query(value="DELETE FROM follow WHERE from_member_id=:fromMemberId AND to_member_id=:toMemberId", nativeQuery = true)
void rUnFollow(@Param("fromMemberId") long fromMemberId,@Param("toMemberId") long toMemberId);
}
insert를 할 때 Spring data JPA의 save()를 사용하면 되기 때문에 JPQL에는 insert 문이 없다. 그러므로 nativeQuery를 쓰자.
어차피 각각 받으므로 묶어줬다가 다시 나누지 말고 각각 나눠서 매핑해주자
팔로우 엔티티
import com.post_show_blues.vine.domain.member.Member;
import lombok.*;
import javax.persistence.*;
@Entity
@Builder
@AllArgsConstructor
@NoArgsConstructor
@Getter
@ToString
@Table(
uniqueConstraints = {
@UniqueConstraint(
name="follow_uk",
columnNames={"from_member_id", "to_member_id"}
)
}
)
public class Follow {
@Id @GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name="follow_id")
private Long id;
@ManyToOne(fetch=FetchType.LAZY)
@JoinColumn(name="from_member_id",nullable = false)
private Member fromMemberId; //팔로우 하는 멤버
@ManyToOne(fetch=FetchType.LAZY)
@JoinColumn(name="to_member_id",nullable = false)
private Member toMemberId; //팔로우 받는 멤버
}
XToOne은 fetch=FetchType.Lazy를 붙여주는 게 좋다고 한다. 안 그러면 연관관계에 있는 테이블 다 끌어와서 검색한다고 함.
팔로우 서비스 테스트
import com.post_show_blues.vine.domain.follow.Follow;
import com.post_show_blues.vine.domain.follow.FollowRepository;
import com.post_show_blues.vine.domain.member.Member;
import com.post_show_blues.vine.domain.member.MemberRepository;
import com.post_show_blues.vine.domain.notice.Notice;
import com.post_show_blues.vine.domain.notice.NoticeRepository;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.transaction.annotation.Transactional;
import java.util.List;
import static org.assertj.core.api.Assertions.*;
@SpringBootTest
@Transactional
public class FollowServiceTest {
@Autowired
FollowService followService;
@Autowired
FollowRepository followRepository;
@Autowired
MemberRepository memberRepository;
@Autowired
NoticeRepository noticeRepository;
Member memberA;
Member memberB;
@BeforeEach
public void setup(){
memberA = memberA();
memberB = memberB();
followService.isFollow(memberA.getId(), memberB.getId());
}
@Test
public void 팔로우() throws Exception {
//given
//setup()에서 함
System.out.println("memberA = " + memberA);
//when
List<Follow> followList = followRepository.findAll();
Follow follow = followList.get(followList.size()-1);
System.out.println("follow = " + follow);
System.out.println(followList.contains(follow));
//then
assertThat(follow.getFromMemberId().getId()).isEqualTo(memberA.getId());
assertThat(follow.getToMemberId().getId()).isEqualTo(memberB.getId());
}
@Test
public void 알람테이블() throws Exception {
//given
//setup()에서 함
//when
List<Notice> noticeList = noticeRepository.findAll();
Notice notice = noticeList.get(noticeList.size() - 1);
//then
assertThat(notice.getMemberId()).isEqualTo(memberB.getId());
}
@Test
public void 언팔로우() throws Exception {
//given
System.out.println("memberA = " + memberA);
List<Follow> followList = followRepository.findAll();
Follow follow = followList.get(followList.size()-1);
//when
followService.isUnFollow(memberA.getId(), memberB.getId());
List<Follow> unFollow = followRepository.findAll();
//then
assertThat(unFollow.contains(follow)).isEqualTo(false);
}
Member memberA() {
Member memberA = Member.builder()
.name("memberA")
.email("member@duksung.ac.kr")
.nickname("memberNickname")
.password("1111")
.phone("010-0000-0000")
.university("덕성대학교")
.build();
memberRepository.save(memberA);
return memberA;
}
Member memberB() {
Member memberB = Member.builder()
.name("memberB")
.email("member@kookmin.ac.kr")
.nickname("Nickname")
.password("1111")
.phone("010-0000-0000")
.university("국민대학교")
.build();
memberRepository.save(memberB);
return memberB;
}
}
@BeforeEach
각각의 테스트 메소드 실행 전에 실행
List<Follow> followList = followRepository.findAll();
Follow follow = followList.get(followList.size()-1);
assertThat(follow.getFromMemberId().getId()).isEqualTo(memberA.getId());
사실 이 부분에서 이렇게 비교하는게 아니라 followRepository에 jpa쿼리메소드 형식으로 findByFromMemberIdAndToMemberId로 메소드를 만들려고 했었다.
그런데 테스트에서만 사용할 메소드를 실제 우리 서비스에 올라갈 코드에 넣는 건 아니라고 생각해서 테스트 클래스 안에서 extends FollowRepository로 클래스 하나 만들어서 메소드 집어넣었는데 안 돌아간다. 뭐지...? 배움의 부족인 것 같다 더 공부하는 걸로...
코드 깃헙 주소 https://github.com/JMine97/vine
참고 강의 https://www.easyupclass.com/course/218/about