-
Notifications
You must be signed in to change notification settings - Fork 1
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
refactor: notification을 타입계층으로 리팩터링 #929
Changes from all commits
b8b9287
f77cdf0
ec435d7
211ab56
4d8775d
2b71319
deb422d
2d4decd
f10288e
9b999e1
69a6179
42bb6f1
3f3f69d
6c22260
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -14,8 +14,15 @@ | |
import com.ody.meeting.domain.Meeting; | ||
import com.ody.meeting.dto.response.MateEtaResponsesV2; | ||
import com.ody.member.domain.Member; | ||
import com.ody.notification.domain.FcmTopic; | ||
import com.ody.notification.domain.Notification; | ||
import com.ody.notification.domain.types.DepartureReminder; | ||
import com.ody.notification.domain.types.Entry; | ||
import com.ody.notification.domain.types.MateLeave; | ||
import com.ody.notification.domain.types.MemberDeletion; | ||
import com.ody.notification.domain.types.Nudge; | ||
import com.ody.notification.service.NotificationService; | ||
import com.ody.route.domain.DepartureTime; | ||
import com.ody.route.domain.RouteTime; | ||
import com.ody.route.service.RouteService; | ||
import com.ody.util.TimeUtil; | ||
|
@@ -43,14 +50,32 @@ public MateSaveResponseV2 saveAndSendNotifications( | |
Member member, | ||
Meeting meeting | ||
) { | ||
validateMeetingOverdue(meeting); | ||
validateAlreadyAttended(member, meeting); | ||
Mate mate = saveMateAndEta(mateSaveRequest, member, meeting); | ||
|
||
FcmTopic fcmTopic = new FcmTopic(meeting); | ||
sendEntry(mate, fcmTopic); | ||
notificationService.subscribeTopic(member.getDeviceToken(), fcmTopic); | ||
sendDepartureReminder(meeting, mate, fcmTopic); | ||
return MateSaveResponseV2.from(meeting); | ||
} | ||
|
||
private void sendEntry(Mate mate, FcmTopic fcmTopic) { | ||
Entry entry = new Entry(mate, fcmTopic); | ||
notificationService.saveAndSchedule(entry.toNotification()); | ||
} | ||
|
||
private void sendDepartureReminder(Meeting meeting, Mate mate, FcmTopic fcmTopic) { | ||
DepartureTime departureTime = new DepartureTime(meeting, mate.getEstimatedMinutes()); | ||
DepartureReminder departureReminder = new DepartureReminder(mate, departureTime, fcmTopic); | ||
notificationService.saveAndSchedule(departureReminder.toNotification()); | ||
} | ||
|
||
private void validateMeetingOverdue(Meeting meeting) { | ||
if (meeting.isOverdue()) { | ||
throw new OdyBadRequestException("참여 가능한 시간이 지난 약속에 참여할 수 없습니다."); | ||
} | ||
|
||
Mate mate = saveMateAndEta(mateSaveRequest, member, meeting); | ||
notificationService.saveAndSendNotifications(meeting, mate, member.getDeviceToken()); | ||
return MateSaveResponseV2.from(meeting); | ||
} | ||
|
||
public void validateAlreadyAttended(Member member, Meeting meeting) { | ||
|
@@ -78,7 +103,8 @@ public void nudge(NudgeRequest nudgeRequest) { | |
Mate requestMate = findFetchedMate(nudgeRequest.requestMateId()); | ||
Mate nudgedMate = findFetchedMate(nudgeRequest.nudgedMateId()); | ||
validateNudgeCondition(requestMate, nudgedMate); | ||
notificationService.sendNudgeMessage(requestMate, nudgedMate); | ||
Nudge nudge = new Nudge(nudgedMate); | ||
notificationService.sendNudgeMessage(requestMate, nudge); | ||
Comment on lines
+106
to
+107
There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 commentThe reason will be displayed to describe this comment to others. Learn more. 2안 스케줄링 로직을 외부에서 알 필요가 있을까요? 외부에서 NotificationService를 사용할 땐, 알림을 보낸다/안 보낸다만 알면 되지 않을까요?
단, NotificationService 내에서 private 메서드로 스케줄링이 필요한지 아닌지 분기처리가 필요해지는 단점이 있을 수 있어요. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 이슈 파서 작업해주겠습니다~ |
||
} | ||
|
||
private void validateNudgeCondition(Mate requestMate, Mate nudgedMate) { | ||
|
@@ -133,15 +159,15 @@ public void deleteAllByMember(Member member) { | |
|
||
@Transactional | ||
public void withdraw(Mate mate) { | ||
Notification memberDeletionNotification = Notification.createMemberDeletion(mate); | ||
Notification memberDeletionNotification = new MemberDeletion(mate).toNotification(); | ||
notificationService.save(memberDeletionNotification); | ||
delete(mate); | ||
} | ||
|
||
@Transactional | ||
public void leaveByMeetingIdAndMemberId(Long meetingId, Long memberId) { | ||
Mate mate = findByMeetingIdAndMemberId(meetingId, memberId); | ||
Notification leaveNotification = Notification.createMateLeave(mate); | ||
Notification leaveNotification = new MateLeave(mate).toNotification(); | ||
notificationService.save(leaveNotification); | ||
delete(mate); | ||
} | ||
|
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. ~ 👍 |
Original file line number | Diff line number | Diff line change | ||||
---|---|---|---|---|---|---|
@@ -0,0 +1,43 @@ | ||||||
package com.ody.notification.domain.types; | ||||||
|
||||||
import com.ody.mate.domain.Mate; | ||||||
import com.ody.notification.domain.FcmTopic; | ||||||
import com.ody.notification.domain.Notification; | ||||||
import com.ody.notification.domain.NotificationStatus; | ||||||
import com.ody.notification.domain.NotificationType; | ||||||
import java.time.LocalDateTime; | ||||||
|
||||||
public abstract class AbstractNotification { | ||||||
|
||||||
private final Mate mate; | ||||||
private final NotificationType type; | ||||||
private final LocalDateTime sendAt; | ||||||
private final NotificationStatus status; | ||||||
private final FcmTopic fcmTopic; | ||||||
|
||||||
protected AbstractNotification( | ||||||
Mate mate, | ||||||
LocalDateTime sendAt, | ||||||
NotificationStatus status, | ||||||
FcmTopic fcmTopic | ||||||
) { | ||||||
this.mate = mate; | ||||||
this.type = getType(); | ||||||
this.sendAt = calculateSendAt(sendAt); | ||||||
this.status = status; | ||||||
this.fcmTopic = fcmTopic; | ||||||
} | ||||||
|
||||||
private static LocalDateTime calculateSendAt(LocalDateTime sendAt) { | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. [제안]
Suggested change
|
||||||
if (sendAt.isBefore(LocalDateTime.now())) { | ||||||
return LocalDateTime.now(); | ||||||
} | ||||||
return sendAt; | ||||||
} | ||||||
Comment on lines
+31
to
+36
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. [질문] 해당 부분은 Departure Reminder를 위한 로직인데 AbstractNotification에 위치시킨 이유가 있나요? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 처음에는 Departure Reminder에 위치시켰었는데요. 이전 Notifiaction 로직에서 NotificationService의 private method로 보내기 전에 calualteSendAt으로 모든 notification의 sendAt을 체크해서 공통로직인 줄 알았습니다.
제리는 어떻게 생각하시나요? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 음.. DepartureReminder의 특수 규칙이었을 때 더 확장성이 좋을 것 같아요. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 지금까지 공통 로직으로 서비스되긴 했지만.... 😭 |
||||||
|
||||||
abstract NotificationType getType(); | ||||||
|
||||||
public Notification toNotification() { | ||||||
return new Notification(null, mate, type, sendAt, status, fcmTopic); | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. [제안]
Suggested change
|
||||||
} | ||||||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
package com.ody.notification.domain.types; | ||
|
||
import com.ody.mate.domain.Mate; | ||
import com.ody.notification.domain.FcmTopic; | ||
import com.ody.notification.domain.NotificationStatus; | ||
import com.ody.notification.domain.NotificationType; | ||
import com.ody.route.domain.DepartureTime; | ||
|
||
public class DepartureReminder extends AbstractNotification { | ||
|
||
public DepartureReminder(Mate mate, DepartureTime departureTime, FcmTopic fcmTopic) { | ||
super(mate, departureTime.getValue(), NotificationStatus.PENDING, fcmTopic); | ||
} | ||
|
||
@Override | ||
public NotificationType getType() { | ||
return NotificationType.DEPARTURE_REMINDER; | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
package com.ody.notification.domain.types; | ||
|
||
import com.ody.mate.domain.Mate; | ||
import com.ody.notification.domain.FcmTopic; | ||
import com.ody.notification.domain.NotificationStatus; | ||
import com.ody.notification.domain.NotificationType; | ||
import java.time.LocalDateTime; | ||
|
||
public class Entry extends AbstractNotification { | ||
|
||
public Entry(Mate mate, FcmTopic fcmTopic) { | ||
super(mate, LocalDateTime.now(), NotificationStatus.DONE, fcmTopic); | ||
} | ||
|
||
@Override | ||
public NotificationType getType() { | ||
return NotificationType.ENTRY; | ||
} | ||
} | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
package com.ody.notification.domain.types; | ||
|
||
import com.ody.mate.domain.Mate; | ||
import com.ody.notification.domain.NotificationStatus; | ||
import com.ody.notification.domain.NotificationType; | ||
import java.time.LocalDateTime; | ||
|
||
public class MateLeave extends AbstractNotification { | ||
|
||
public MateLeave(Mate mate) { | ||
super(mate, LocalDateTime.now(), NotificationStatus.DONE, null); | ||
} | ||
|
||
@Override | ||
public NotificationType getType() { | ||
return NotificationType.LEAVE; | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
package com.ody.notification.domain.types; | ||
|
||
import com.ody.mate.domain.Mate; | ||
import com.ody.notification.domain.NotificationStatus; | ||
import com.ody.notification.domain.NotificationType; | ||
import java.time.LocalDateTime; | ||
|
||
public class MemberDeletion extends AbstractNotification { | ||
|
||
public MemberDeletion(Mate mate) { | ||
super(mate, LocalDateTime.now(), NotificationStatus.DONE, null); | ||
} | ||
|
||
@Override | ||
public NotificationType getType() { | ||
return NotificationType.MEMBER_DELETION; | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
package com.ody.notification.domain.types; | ||
|
||
import com.ody.mate.domain.Mate; | ||
import com.ody.notification.domain.NotificationStatus; | ||
import com.ody.notification.domain.NotificationType; | ||
import java.time.LocalDateTime; | ||
|
||
public class Nudge extends AbstractNotification { | ||
|
||
public Nudge(Mate nudgeMate) { | ||
super(nudgeMate, LocalDateTime.now(), NotificationStatus.DONE, null); | ||
} | ||
|
||
@Override | ||
public NotificationType getType() { | ||
return NotificationType.NUDGE; | ||
} | ||
} |
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.
[질문] 노션에 작성해주신 부분 중에 해당 로직을 MateService 로직에 종속적인 로직이라고 본 이유가 있나요?
저는 콜리와 반대로 '토픽을 구독한다'를 MateService가 알고 있을 필요가 없다고 생각하는데, 그 이유는 토픽을 구독하는 것은 모임원 모두에게 알림을 보내기 위한 방법이라서 입니다. MateService는 모두에게 알림을 보낸다 정도만 알아도 충분하다고 생각해서 오히려 내부 구현 + fcm 관련 배경 지식이 노출된 느낌이 들어요.
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.
첫째로, 해당 메서드가 약속 참여 플로우 이외에 그 어떤 비즈니스 로직에도 확장성있게 사용될수가 없었어요. 참여 알림 보내고 + 구독하고 + 약속리마인더 알림을 보낸다 -> 라는 흐름은 약속 참여시 약속을 보내는 방법이지 알림 서비스가 알아야할 약속 참여 시 필요한 알림 로직이 아니라 생각했습니다.
두번째로, saveAndSendNotification으로 추상화된 메서드 명을 보면 그저 알림을 저장하고 보내는 것으로 생각이 되는데, 저장을 하는 알림이 약속 리마인더와 출발알림으로 정해져 있습니다. 다른 알림은 저장될수가 없고 보낼수도 없습니다. mateService 참여 시 필요한 알림을 해당 메서드에서 보내고 있었습니다.
세번째로, fcm 구현이 밖으로 드러나있다는 점에 대해서는 동의합니다. 해당 구현을 캡슐화하기 위해 sendToDevice, sendToTopic 등의 메서드를 만들어주어 subscribe를 드러내지 않는 것은 동의합니다. 그러나, notificationService가 특정 알림 객체를 위한 메서드를 가지는 것은 비즈니스 로직상 어떤 알림을 보냈는지를 상위 서비스가 아니라 하위 서비스인 notificationService가 알게된다는 점에서 너무 많은 것을 알고 있다고 생각합니다. 또한, 기존에 notificationServic가 unsubscribe라는 메서드를 가지고 있다는 점, fcmTopic 필드를 meeting이 가지고 있다는 점에서 fcm 로직을 도메인들도 알고 있다고 판단한 부분도 있습니다.
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.
오 납득했습니다!! 자세하게 설명해주어 고마워요 🥦🙇♀️