-
Notifications
You must be signed in to change notification settings - Fork 101
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
feat(mail): implement mail reservation #731
base: develop
Are you sure you want to change the base?
Conversation
- update functionality is replaced with deleting and creating
dd76881
to
8bb541f
Compare
최신 |
제이슨과의 오프라인 코드 리뷰 후 변경한 사항 정리합니다. 도메인예약 관련컨벤션 |
- update functionality is replaced with deleting and creating
8bb541f
to
6f23fa7
Compare
fun sendReservedMail(standardTime: LocalDateTime = LocalDateTime.now()) { | ||
val reservations = mailReservationRepository.findByReservationTimeBetweenAndStatus( | ||
standardTime.minusMinutes(1), | ||
standardTime.plusMinutes(1), | ||
MailReservationStatus.WAITING | ||
) | ||
val messagesById = findMessageMapById(reservations.map { it.mailMessageId }) | ||
|
||
reservations.forEach { mailReservation -> | ||
mailReservation.send() | ||
mailReservationRepository.save(mailReservation) | ||
mailService.sendMailsByBccSynchronous(MailData(messagesById.getValue(mailReservation.id)), emptyMap()) | ||
mailReservation.finish() | ||
} | ||
} |
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.
회의 시간에 제이슨이 언급하셨던 예약 메일 전송 시 트랜잭션 처리와 관련된 코드입니다.
기존의 메일 전송은 Async로 동작하지만, 이번에는 문제를 단순화하여 동기 방식으로 처리하기로 했어요. (현재 설정 상 초당 300건의 메일 전송이 가능)
설계 시에는 멀티 환경에서의 전송을 생각하고 WATING(대기), SENDING(발송 중), FINISHED(처리 완료) 세 가지의 메일 예약 상태를 가지도록 했는데요.
쓰면서 생각해보니 한 번의 요청으로 하나의 서버에서 동기로 처리를 한다면 중간 상태 변경이 없어도 괜찮지 않나 싶기도 하네요.
혹시 제가 잘못 생각하고 있거나 아니면 더 좋은 동작 아이디어가 있다면 의견 부탁드립니다!
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.
브리 몇가지 의견 남겼습니다~!
그리고 이번에 AwsMailSender에서 예외 처리와 로그도 남기면 좋을 것 같아요!
mailMessageRepository.deleteById(mailReservation.mailMessageId) | ||
} | ||
|
||
fun sendReservedMail(standardTime: LocalDateTime = LocalDateTime.now()) { |
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.
r: @ Transactional
을 추가하고 타임아웃을 정해두시죠!
|
||
fun sendReservedMail(standardTime: LocalDateTime = LocalDateTime.now()) { | ||
val reservations = mailReservationRepository.findByReservationTimeBetweenAndStatus( | ||
standardTime.minusMinutes(1), |
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.
1분 범위에 예약된 메일을 모두 발송하기 위해 1분 추가, 빼기를 한게 맞을까요?
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.
네 맞습니다! 트리거가 정확하게 시작하는걸 보장하기 어렵기도 하고, 메일 예약이 15분 단위로 설정되니 다른 시간의 예약을 조회할일은 없어 앞뒤로 1분을 넣었어요.
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.
r: 1시 10분이 예약되어 있다고 했을 때 1시 9분, 1시 10분, 1시 11분에 요청이 안오면 해당 메일은 발송을 못하는 케이스가 존재할 것 같은데요.
standardTime보다 이전에 발송되어야 할 메시지 중 상태가 WAITING인 애들을 조회해서 발송하면 문제가 없어질 것 같은데 어떻게 생각하시나용
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.
오 그렇네요. 아마 이전에 코드를 짤 때 예약시간 수정을 생각하다 보니 사이값을 검색하도록 만들었던 것 같아요.
현재 로직은 예약 시간/내용 수정 대신 기존 예약을 삭제 후 재생성하도록 하고있어서, 말씀주신 것처럼 예약 시간이 현재보다 이전인 대기 상태의 모든 예약을 조회하는게 더 좋을 것 같습니다!
|
||
reservations.forEach { mailReservation -> | ||
mailReservation.send() | ||
mailReservationRepository.save(mailReservation) |
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.
c: 메일 예약을 개별 저장하는 것보다 한 번에 처리하는건 어떨까요
배치 사이즈를 지정했다면 더 효율적으로 처리할 수 있을 것 같아요
mailReservationRepository.saveAll(reservations)
saveMailHistories(mailMessage.id, succeeded, failed) | ||
} | ||
|
||
fun sendMailsByBccSynchronous(request: MailData, files: Map<String, ByteArrayResource>) { |
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.
- 단건으로 보내는 거라 s는 빼는게 맞겠네요.
fun sendMailsByBccSynchronous(request: MailData, files: Map<String, ByteArrayResource>) { | |
fun sendMailByBccSynchronous(request: MailData, files: Map<String, ByteArrayResource>) { |
-
files도 MailData에 합치는게 좋을 것 같은데 스프링 리소스를 쓰고 있어서 따로 분리했나보네요. 저는 인터페이스 추가해서 하나로 합치는게 나을 것 같은데 다른 분들 어떠신가요. 따로 다루는게 나을까요?
-
첨부 파일이 없는 경우가 많을거라 기본값으로 emptyMap()을 설정하는게 좋을 것 같습니다.
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.
이번에는 메일 예약에서 파일 첨부 기능을 지원하지 않기로 해서 크게 신경쓰지는 않았는데, 장기적인 관점으로는 구구 말씀처럼 합치는 방향으로 가는게 좋아보여요 ㅎㅎ
@@ -24,11 +26,13 @@ private const val MAIL_SENDING_UNIT: Int = 50 | |||
class MailService( |
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.
c: MailMessageService가 있고 MailService가 있으니까 헷갈리네요;;;
MailService는 실제 메일 발송과 관련된 작업만 존재하니 SedingMailService 같은 이름으로 바꾸는건 어떨까요?
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.
저도 만들면서 메일 관련 서비스들이 헷갈렸습니다... (MailService
, MailHistoryService
, MailMessageService
)
현재 MailService의 역할에 공감해서 좋은 네이밍 인 것 같아 반영하겠습니다!
MailHistoryService
는 현재 어드민에서 사용중이라 이후 어드민 작업 하면서 필요성을 보고 함께 정리할게요.
mailMessageRepository.deleteById(mailReservation.mailMessageId) | ||
} | ||
|
||
fun sendReservedMail(standardTime: LocalDateTime = LocalDateTime.now()) { |
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.
c: 예약된 메일들을 보내는거라 s 붙이는게 좋겠네요
fun sendReservedMail(standardTime: LocalDateTime = LocalDateTime.now()) { | |
fun sendReservedMails(standardTime: LocalDateTime = LocalDateTime.now()) { |
standardTime.plusMinutes(1), | ||
MailReservationStatus.WAITING | ||
) | ||
val messagesById = findMessageMapById(reservations.map { it.mailMessageId }) |
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.
MailReservation이랑 MailMessage를 한 번에 불러오면 쿼리 호출 횟수도 줄이고 코드 복잡도도 줄어들거 같아요
아래 예시 쿼리로 작성했어요. JPQL 문법이 맞는지 확인이 필요합니다 ㅎㅎ;
@Query("select m" +
" from MailMessage m" +
" join MailReservation ms on ms.mailMessageId = m.id" +
" where ms.reservationTime between :from and :to" +
" and ms.status = :status")
fun findByReservationTimeBetweenAndStatus(
from: LocalDateTime,
to: LocalDateTime,
status: MailReservationStatus
): List<MailMessage>
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.
오 이렇게하면 바로 예약 MailMessage 를 가져올 수 있군요! 넘 좋습니다 ㅎㅎㅎ
한 가지 궁금한게 이 때 조회한 MailReservation도 함께 받을 수 있나요? 메일 전송만 생각하면 MailMessage만 있어도 되는데, 상태 변경이 필요해서요!
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.
쿼리에서 select에 ms를 추가하면 될거에요. 그런데 MailMessage가 MailReservation를 간접참조 하고 있으면 좀 더 수정해야 해요. dto를 새로 만들거나 객체를 참조하도록 바꾸는 방법이 있겠네요
reservations.forEach { mailReservation -> | ||
mailReservation.send() | ||
mailReservationRepository.save(mailReservation) | ||
mailService.sendMailsByBccSynchronous(MailData(messagesById.getValue(mailReservation.id)), emptyMap()) |
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.
c: MailService를 인프라용 객체가 아닌 도메인용으로 본다면 MailMessage 객체를 전달하는게 나을 것 같아요.
val mailMessage = messagesById.getValue(mailReservation.id)
mailService.sendMailsByBccSynchronous(mailMessage)
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.
어드민에서 직접 사용하는 비동기 방식의 메일 전송 메서드와 맞추긴 했는데, 저도 고민되긴 하더라구요 ㅋㅋ
MailMessage를 만들테고 실제로 전송할 때 MailService(SendingMailService)
를 사용한다면, MailMessage를 전달하는게 좋아보이긴 해요.
이건 변경 여파가 어드민에 미쳐서 작업을 안했었는데, 이부분도 이후에 어드민 작업과 함께 정리해볼게요.
|
||
@Column(nullable = false) | ||
val reservationTime: LocalDateTime, | ||
id: Long = 0L |
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.
a: id를 하단에 위치시키니 mailMessageId랑 헷갈리네요. 아이디 생략하고 기본값으로 0을 설정하려고 마지막에 배치한걸까요?
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.
이건 다른 엔티티들을 보니 컨벤션으로 보여 맞추긴 했습니다. ㅎㅎ 의도는 @woowahan-pjs 에게 토스,,,
fun sendReservedMail(standardTime: LocalDateTime = LocalDateTime.now()) { | ||
val reservations = mailReservationRepository.findByReservationTimeBetweenAndStatus( | ||
standardTime.minusMinutes(1), | ||
standardTime.plusMinutes(1), | ||
MailReservationStatus.WAITING | ||
) | ||
val messagesById = findMessageMapById(reservations.map { it.mailMessageId }) | ||
|
||
reservations.forEach { mailReservation -> | ||
mailReservation.send() | ||
mailReservationRepository.save(mailReservation) | ||
mailService.sendMailsByBccSynchronous(MailData(messagesById.getValue(mailReservation.id)), emptyMap()) | ||
mailReservation.finish() | ||
} | ||
} |
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.
고생 많으셨습니다 :)
) | ||
val messagesById = findMessageMapById(reservations.map { it.mailMessageId }) | ||
|
||
reservations.forEach { mailReservation -> |
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.
c: 해당 메일 발송 작업이 하나의 트랜잭션으로 묶여야 할 이유가 있을까요? 개별적으로 성공 실패가 나뉘는게 더 자연스럽지 않을까 합니다.
|
||
@PostMapping("/reserved") | ||
fun sendMail( | ||
@Accessor("lambda") ignored: Unit |
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.
r: 채점을 위한 token이 변경되면 같이 변경이 되야 하는 문제가 있을 것 같은데, 동일한 token으로 가는 것 보다 다른 토큰으로 분리가 되면 어떨까요?
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.
몰래 같이 쓰려다 걸렸네요... 🙄
mail-scheduler 로 분리했어요 ㅋㅋㅋ
val messagesById = findMessageMapById(reservations.map { it.mailMessageId }) | ||
|
||
reservations.forEach { mailReservation -> | ||
mailReservation.send() |
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.
질문) 지금 트랜잭션으로 묶여 있는데 MailReservation.status가 SENDING로 DB에 존재하는 케이스가 생길 수 있나요?
|
||
fun sendReservedMail(standardTime: LocalDateTime = LocalDateTime.now()) { | ||
val reservations = mailReservationRepository.findByReservationTimeBetweenAndStatus( | ||
standardTime.minusMinutes(1), |
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.
r: 1시 10분이 예약되어 있다고 했을 때 1시 9분, 1시 10분, 1시 11분에 요청이 안오면 해당 메일은 발송을 못하는 케이스가 존재할 것 같은데요.
standardTime보다 이전에 발송되어야 할 메시지 중 상태가 WAITING인 애들을 조회해서 발송하면 문제가 없어질 것 같은데 어떻게 생각하시나용
e77a87d
to
9f0e2a2
Compare
9f0e2a2
to
0fe7829
Compare
@kang-hyungu @woowahan-neo @woowahan-pjs 그리고 (말도 많고 탈도 많던...) 예약 메시지 발송 관련 로직은 트랜잭션 걸지 않게 변경했어요. (0fe7829)
어제 프리코스 안내 메일 처리 속도를 보니 메일 전송 요청을 비동기로 변경해도 괜찮을 것� 같은데, 이건 AwsMailSender에 RateLimit이 걸려있어서 실제 요청 수치들을 더 확인하고 변경하는게 좋아보여 이번에 반영하지 않았습니다. |
Resolves #715
해결하려는 문제가 무엇인가요?
어떻게 해결했나요?
MailMessage
(메일 내용),MailReservation
(예약),MailHistory
(전송 이력) 의 객체로 확장하였습니다.어떤 부분에 집중하여 리뷰해야 할까요?
RCA 룰
r: 꼭 반영해 주세요. 적극적으로 고려해 주세요. (Request changes)
c: 웬만하면 반영해 주세요. (Comment)
a: 반영해도 좋고 넘어가도 좋습니다. 그냥 사소한 의견입니다. (Approve)