diff --git a/backend/src/main/java/corea/DataInitializer.java b/backend/src/main/java/corea/DataInitializer.java new file mode 100644 index 000000000..2d1291d51 --- /dev/null +++ b/backend/src/main/java/corea/DataInitializer.java @@ -0,0 +1,64 @@ +package corea; + +import corea.domain.Member; +import corea.domain.Participation; +import corea.domain.Room; +import corea.repository.MemberRepository; +import corea.repository.ParticipationRepository; +import corea.repository.RoomRepository; +import lombok.RequiredArgsConstructor; +import org.springframework.boot.CommandLineRunner; +import org.springframework.stereotype.Component; + +import java.time.LocalDateTime; +import java.util.List; + +@Component +@RequiredArgsConstructor +public class DataInitializer implements CommandLineRunner { + private final MemberRepository memberRepository; + private final RoomRepository roomRepository; + private final ParticipationRepository participationRepository; + + @Override + public void run(final String... args) throws Exception { + final var members = memberRepository.saveAll(List.of( + new Member("pororo@gmail.com", "뽀로로"), + new Member("ashty@gmal.com", "애쉬"), + new Member("moving@gmail.com", "무빙"), + new Member("joyson@gmail.com", "조이썬썬"), + new Member("tenten@gmail.com", "팔팔"), + new Member("choco@gmail.com", "초코"), + new Member("dal@gmail.com", "다르"), + new Member("pobi@gmail.com", "포비") + )); + + final Room room = new Room( + "레이싱 카와 함께하는 TDD", + members.get(0) + .getId(), + "https://github.com/jcoding-play/java-racingcar", + "", + 3, + "TDD", + LocalDateTime.now() + .plusDays(7), + LocalDateTime.now() + .plusDays(14) + ); + roomRepository.save(room); + + participationRepository.saveAll( + List.of( + new Participation(room.getId(),members.get(0).getId()), + new Participation(room.getId(),members.get(1).getId()), + new Participation(room.getId(),members.get(2).getId()), + new Participation(room.getId(),members.get(3).getId()), + new Participation(room.getId(),members.get(4).getId()), + new Participation(room.getId(),members.get(5).getId()), + new Participation(room.getId(),members.get(6).getId()), + new Participation(room.getId(),members.get(7).getId()) + + )); + } +} diff --git a/backend/src/main/java/corea/domain/Member.java b/backend/src/main/java/corea/domain/Member.java index 22487ce1f..3bb8a8e87 100644 --- a/backend/src/main/java/corea/domain/Member.java +++ b/backend/src/main/java/corea/domain/Member.java @@ -21,8 +21,9 @@ public class Member { private Long id; private String email; + private String name; - public Member(final String email) { - this(null, email); + public Member(final String email, final String name) { + this(null,email,name); } } diff --git a/backend/src/main/java/corea/dto/ApiResponse.java b/backend/src/main/java/corea/dto/ApiResponse.java new file mode 100644 index 000000000..e79360ae4 --- /dev/null +++ b/backend/src/main/java/corea/dto/ApiResponse.java @@ -0,0 +1,4 @@ +package corea.dto; + +public record ApiResponse(String message) { +} diff --git a/backend/src/main/java/corea/dto/MatchedGroupResponse.java b/backend/src/main/java/corea/dto/MatchedGroupResponse.java new file mode 100644 index 000000000..6085c0f2c --- /dev/null +++ b/backend/src/main/java/corea/dto/MatchedGroupResponse.java @@ -0,0 +1,6 @@ +package corea.dto; + +import java.util.List; + +public record MatchedGroupResponse(List emails) { +} diff --git a/backend/src/main/java/corea/member/controller/MemberController.java b/backend/src/main/java/corea/member/controller/MemberController.java new file mode 100644 index 000000000..27251db17 --- /dev/null +++ b/backend/src/main/java/corea/member/controller/MemberController.java @@ -0,0 +1,29 @@ +package corea.member.controller; + +import corea.dto.ApiResponse; +import corea.dto.MatchedGroupResponse; +import corea.member.service.MatchService; +import lombok.RequiredArgsConstructor; +import org.springframework.web.bind.annotation.*; + +@RestController +@RequiredArgsConstructor +public class MemberController { + private final MatchService matchService; + + @GetMapping("/my/reviewer") + public MatchedGroupResponse getMyReviewer(@RequestHeader("Authorization") final String email) { + return matchService.getMatchedGroup(email); + } + + @GetMapping("/my/reviewee") + public MatchedGroupResponse getMyReviewee(@RequestHeader("Authorization") final String email) { + return matchService.getMatchedGroup(email); + } + + @PostMapping("/match/{roomId}") + public ApiResponse match(@PathVariable final long roomId) { + matchService.match(roomId); + return new ApiResponse("GOOD"); + } +} diff --git a/backend/src/main/java/corea/member/domain/Matching.java b/backend/src/main/java/corea/member/domain/Matching.java index 8f050e23c..489140795 100644 --- a/backend/src/main/java/corea/member/domain/Matching.java +++ b/backend/src/main/java/corea/member/domain/Matching.java @@ -1,6 +1,5 @@ package corea.member.domain; -import corea.domain.Member; import lombok.NoArgsConstructor; import org.springframework.stereotype.Component; @@ -13,18 +12,17 @@ @NoArgsConstructor(access = PROTECTED) public class Matching { - public Map> matchGroup(final List members, final int matchingSize) { + public Map> matchGroup(final List memberIds, final int matchingSize) { final Map> matchedGroup = new HashMap<>(); - final List memberIds = new ArrayList<>(members.stream().map(Member::getId).toList()); Collections.shuffle(memberIds); - final List> groupedMemberIds = IntStream.range(0, (members.size() + matchingSize - 1) / matchingSize) + final List> groupedMemberIds = IntStream.range(0, (memberIds.size() + matchingSize - 1) / matchingSize) .mapToObj(i -> memberIds.subList(i * matchingSize, Math.min(i * matchingSize + matchingSize, memberIds.size()))) .toList(); long groupId = 1L; - for (List l : groupedMemberIds) { + for (final List l : groupedMemberIds) { matchedGroup.put(groupId++, l); } diff --git a/backend/src/main/java/corea/member/entity/MatchedGroup.java b/backend/src/main/java/corea/member/entity/MatchedGroup.java index bba05e210..be1de8e26 100644 --- a/backend/src/main/java/corea/member/entity/MatchedGroup.java +++ b/backend/src/main/java/corea/member/entity/MatchedGroup.java @@ -4,6 +4,7 @@ import jakarta.persistence.GeneratedValue; import jakarta.persistence.Id; import lombok.AllArgsConstructor; +import lombok.Getter; import lombok.NoArgsConstructor; import static jakarta.persistence.GenerationType.IDENTITY; @@ -12,6 +13,7 @@ @Entity @NoArgsConstructor(access = PROTECTED) @AllArgsConstructor +@Getter public class MatchedGroup { @Id @@ -25,4 +27,7 @@ public class MatchedGroup { public MatchedGroup(final Long groupId, final Long memberId) { this(null, groupId, memberId); } + public boolean isEqualMember(final long memberId){ + return this.memberId.equals(memberId); + } } diff --git a/backend/src/main/java/corea/member/repository/MatchedGroupRepository.java b/backend/src/main/java/corea/member/repository/MatchedGroupRepository.java index 6dad5156e..270ac67ed 100644 --- a/backend/src/main/java/corea/member/repository/MatchedGroupRepository.java +++ b/backend/src/main/java/corea/member/repository/MatchedGroupRepository.java @@ -4,10 +4,11 @@ import org.springframework.data.jpa.repository.JpaRepository; import java.util.List; +import java.util.Optional; public interface MatchedGroupRepository extends JpaRepository { - Long findGroupIdByMemberId(final Long memberId); + Optional findOneByMemberId(final long memberId); - List findMemberIdsByGroupId(final Long groupId); + List findByGroupId(final long groupId); } diff --git a/backend/src/main/java/corea/member/service/MatchService.java b/backend/src/main/java/corea/member/service/MatchService.java new file mode 100644 index 000000000..b89a5a981 --- /dev/null +++ b/backend/src/main/java/corea/member/service/MatchService.java @@ -0,0 +1,53 @@ +package corea.member.service; + +import corea.domain.Member; +import corea.domain.Participation; +import corea.dto.MatchedGroupResponse; +import corea.member.entity.MatchedGroup; +import corea.member.repository.MatchedGroupRepository; +import corea.repository.MemberRepository; +import corea.repository.ParticipationRepository; +import lombok.RequiredArgsConstructor; +import org.springframework.stereotype.Service; + +import java.util.List; + +@Service +@RequiredArgsConstructor +public class MatchService { + private final MatchingService matchingService; + private final ParticipationRepository participationRepository; + private final MatchedGroupRepository matchedGroupRepository; + private final MemberRepository memberRepository; + + public void match(final long roomId) { + final List participations = participationRepository.findAllByRoomId(roomId); + + matchingService.matchMaking(participations, 2); + } + + public MatchedGroupResponse getMatchedGroup(final String email) { + final Member member = memberRepository.findByEmail(email) + .orElseThrow(() -> new IllegalArgumentException(String.format("%s는 없는 이메일입니다.", email))); + + final MatchedGroup matchedGroup = matchedGroupRepository.findOneByMemberId(member.getId()) + .orElseThrow(IllegalArgumentException::new); + + final List memberIds = matchedGroupRepository.findByGroupId(matchedGroup.getGroupId()) + .stream() + .filter(matching -> !matching.isEqualMember(member.getId())) + .map(MatchedGroup::getMemberId) + .toList(); + + + + final List memberEmails = + memberIds.stream() + .map(id -> memberRepository.findById(id) + .orElse(null)) + .map(Member::getEmail) + .toList(); + + return new MatchedGroupResponse(memberEmails); + } +} diff --git a/backend/src/main/java/corea/member/service/MatchingService.java b/backend/src/main/java/corea/member/service/MatchingService.java index 5bcd82492..9347ef2a2 100644 --- a/backend/src/main/java/corea/member/service/MatchingService.java +++ b/backend/src/main/java/corea/member/service/MatchingService.java @@ -1,12 +1,13 @@ package corea.member.service; -import corea.domain.Member; +import corea.domain.Participation; import corea.member.domain.Matching; import corea.member.entity.MatchedGroup; import corea.member.repository.MatchedGroupRepository; import lombok.AllArgsConstructor; import org.springframework.stereotype.Service; +import java.util.ArrayList; import java.util.List; import java.util.Map; @@ -17,10 +18,17 @@ public class MatchingService { private final MatchedGroupRepository matchedGroupRepository; private final Matching matching; - public void matchMaking(final List members, final int matchingSize) { - final Map> results = matching.matchGroup(members, matchingSize); - results.entrySet().stream() - .flatMap(entry -> entry.getValue().stream() + public void matchMaking(final List participations, final int matchingSize) { + final ArrayList memberIds = new ArrayList<>(participations.stream() + .map(Participation::getMemberId) + .toList() + ); + + final Map> results = matching.matchGroup(memberIds, matchingSize); + results.entrySet() + .stream() + .flatMap(entry -> entry.getValue() + .stream() .map(memberId -> new MatchedGroup(entry.getKey(), memberId))) .forEach(matchedGroupRepository::save); } diff --git a/backend/src/main/java/corea/repository/MemberRepository.java b/backend/src/main/java/corea/repository/MemberRepository.java index 6dd41b982..21347d678 100644 --- a/backend/src/main/java/corea/repository/MemberRepository.java +++ b/backend/src/main/java/corea/repository/MemberRepository.java @@ -3,5 +3,8 @@ import corea.domain.Member; import org.springframework.data.jpa.repository.JpaRepository; +import java.util.Optional; + public interface MemberRepository extends JpaRepository { + Optional findByEmail(String email); } diff --git a/backend/src/main/java/corea/repository/ParticipationRepository.java b/backend/src/main/java/corea/repository/ParticipationRepository.java index d48a7f409..ddd72ccda 100644 --- a/backend/src/main/java/corea/repository/ParticipationRepository.java +++ b/backend/src/main/java/corea/repository/ParticipationRepository.java @@ -3,5 +3,8 @@ import corea.domain.Participation; import org.springframework.data.jpa.repository.JpaRepository; +import java.util.List; + public interface ParticipationRepository extends JpaRepository { + List findAllByRoomId(long roomId); }