From 68bd80bde419c3b59c6f2a7a32f6043151eb1ce6 Mon Sep 17 00:00:00 2001
From: seonghaejo <jsh990324@snu.ac.kr>
Date: Sun, 9 Feb 2025 13:52:59 +0900
Subject: [PATCH 1/6] Add PushOptOut data model and repository

---
 .../kotlin/notification/data/PushCategory.kt  | 33 +++++++++++++++++++
 .../kotlin/notification/data/PushOptOut.kt    | 21 ++++++++++++
 .../repository/PushOptOutRepository.kt        | 17 ++++++++++
 3 files changed, 71 insertions(+)
 create mode 100644 core/src/main/kotlin/notification/data/PushCategory.kt
 create mode 100644 core/src/main/kotlin/notification/data/PushOptOut.kt
 create mode 100644 core/src/main/kotlin/notification/repository/PushOptOutRepository.kt

diff --git a/core/src/main/kotlin/notification/data/PushCategory.kt b/core/src/main/kotlin/notification/data/PushCategory.kt
new file mode 100644
index 00000000..6f1ff1e2
--- /dev/null
+++ b/core/src/main/kotlin/notification/data/PushCategory.kt
@@ -0,0 +1,33 @@
+package com.wafflestudio.snutt.notification.data
+
+import com.fasterxml.jackson.annotation.JsonValue
+import org.springframework.core.convert.converter.Converter
+import org.springframework.data.convert.ReadingConverter
+import org.springframework.data.convert.WritingConverter
+import org.springframework.stereotype.Component
+
+enum class PushCategory(
+    @JsonValue val value: Int,
+) {
+    LECTURE_UPDATE(1),
+    VACANCY_NOTIFICATION(2),
+    ;
+
+    companion object {
+        private val valueMap = PushCategory.entries.associateBy { e -> e.value }
+
+        fun getOfValue(value: Int): PushCategory? = valueMap[value]
+    }
+}
+
+@ReadingConverter
+@Component
+class PushCategoryReadConverter : Converter<Int, PushCategory> {
+    override fun convert(source: Int): PushCategory = PushCategory.getOfValue(source)!!
+}
+
+@WritingConverter
+@Component
+class PushCategoryWriteConverter : Converter<PushCategory, Int> {
+    override fun convert(source: PushCategory): Int = source.value
+}
diff --git a/core/src/main/kotlin/notification/data/PushOptOut.kt b/core/src/main/kotlin/notification/data/PushOptOut.kt
new file mode 100644
index 00000000..30ef5888
--- /dev/null
+++ b/core/src/main/kotlin/notification/data/PushOptOut.kt
@@ -0,0 +1,21 @@
+package com.wafflestudio.snutt.notification.data
+
+import org.springframework.data.annotation.Id
+import org.springframework.data.mongodb.core.index.CompoundIndex
+import org.springframework.data.mongodb.core.index.Indexed
+import org.springframework.data.mongodb.core.mapping.Document
+import org.springframework.data.mongodb.core.mapping.Field
+import org.springframework.data.mongodb.core.mapping.FieldType
+
+@Document(collection = "push_opt_out")
+@CompoundIndex(def = "{ 'user_id': 1, 'push_category': 1 }")
+data class PushOptOut(
+    @Id
+    val id: String? = null,
+    @Indexed
+    @Field("user_id", targetType = FieldType.OBJECT_ID)
+    val userId: String,
+    @Field("push_category")
+    @Indexed
+    val pushCategory: PushCategory,
+)
diff --git a/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt b/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
new file mode 100644
index 00000000..f8751019
--- /dev/null
+++ b/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
@@ -0,0 +1,17 @@
+package com.wafflestudio.snutt.notification.repository
+
+import com.wafflestudio.snutt.notification.data.PushCategory
+import com.wafflestudio.snutt.notification.data.PushOptOut
+import org.springframework.data.repository.kotlin.CoroutineCrudRepository
+
+interface PushOptOutRepository : CoroutineCrudRepository<PushOptOut, String> {
+    suspend fun existsByUserIdAndPushCategory(
+        userId: String,
+        pushCategory: PushCategory,
+    ): Boolean
+
+    suspend fun deleteByUserIdAndPushCategory(
+        userId: String,
+        pushCategory: PushCategory,
+    ): Long
+}

From d13cd599dcbd91f4a2aaf3c4134bc1fe9da7e2e9 Mon Sep 17 00:00:00 2001
From: seonghaejo <jsh990324@snu.ac.kr>
Date: Sun, 9 Feb 2025 14:42:44 +0900
Subject: [PATCH 2/6] Add send-categorical-push method to PushService

---
 .../repository/PushOptOutRepository.kt        |  5 ++
 .../notification/service/PushService.kt       | 65 +++++++++++++++++++
 2 files changed, 70 insertions(+)

diff --git a/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt b/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
index f8751019..88972f54 100644
--- a/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
+++ b/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
@@ -14,4 +14,9 @@ interface PushOptOutRepository : CoroutineCrudRepository<PushOptOut, String> {
         userId: String,
         pushCategory: PushCategory,
     ): Long
+
+    suspend fun findByUserIdInAndPushCategory(
+        userIds: List<String>,
+        pushCategory: PushCategory,
+    ): List<PushOptOut>
 }
diff --git a/core/src/main/kotlin/notification/service/PushService.kt b/core/src/main/kotlin/notification/service/PushService.kt
index 0627a91e..8dccf9d0 100644
--- a/core/src/main/kotlin/notification/service/PushService.kt
+++ b/core/src/main/kotlin/notification/service/PushService.kt
@@ -3,6 +3,8 @@ package com.wafflestudio.snutt.notification.service
 import com.wafflestudio.snutt.common.push.PushClient
 import com.wafflestudio.snutt.common.push.dto.PushMessage
 import com.wafflestudio.snutt.common.push.dto.TargetedPushMessageWithToken
+import com.wafflestudio.snutt.notification.data.PushCategory
+import com.wafflestudio.snutt.notification.repository.PushOptOutRepository
 import org.springframework.stereotype.Service
 
 /**
@@ -22,12 +24,30 @@ interface PushService {
     suspend fun sendGlobalPush(pushMessage: PushMessage)
 
     suspend fun sendTargetPushes(userToPushMessage: Map<String, PushMessage>)
+
+    suspend fun sendCategoricalPush(
+        pushMessage: PushMessage,
+        userId: String,
+        pushCategory: PushCategory,
+    )
+
+    suspend fun sendCategoricalPushes(
+        pushMessage: PushMessage,
+        userIds: List<String>,
+        pushCategory: PushCategory,
+    )
+
+    suspend fun sendCategoricalTargetPushes(
+        userToPushMessage: Map<String, PushMessage>,
+        pushCategory: PushCategory,
+    )
 }
 
 @Service
 class PushServiceImpl internal constructor(
     private val deviceService: DeviceService,
     private val pushClient: PushClient,
+    private val pushOptOutRepository: PushOptOutRepository,
 ) : PushService {
     override suspend fun sendPush(
         pushMessage: PushMessage,
@@ -60,4 +80,49 @@ class PushServiceImpl internal constructor(
             deviceService.getUserDevices(userId).map { it.fcmRegistrationId to pushMessage }
         }.map { (fcmRegistrationId, message) -> TargetedPushMessageWithToken(fcmRegistrationId, message) }
             .let { pushClient.sendMessages(it) }
+
+    override suspend fun sendCategoricalPush(
+        pushMessage: PushMessage,
+        userId: String,
+        pushCategory: PushCategory,
+    ) {
+        if (!pushOptOutRepository.existsByUserIdAndPushCategory(userId, pushCategory)) {
+            sendPush(pushMessage, userId)
+        }
+    }
+
+    override suspend fun sendCategoricalPushes(
+        pushMessage: PushMessage,
+        userIds: List<String>,
+        pushCategory: PushCategory,
+    ) {
+        val filteredUserIds =
+            pushOptOutRepository
+                .findByUserIdInAndPushCategory(userIds, pushCategory)
+                .map { it.userId }
+                .toSet()
+                .let { optOutUserIds -> userIds.filterNot { it in optOutUserIds } }
+
+        if (filteredUserIds.isNotEmpty()) {
+            sendPushes(pushMessage, filteredUserIds)
+        }
+    }
+
+    override suspend fun sendCategoricalTargetPushes(
+        userToPushMessage: Map<String, PushMessage>,
+        pushCategory: PushCategory,
+    ) {
+        val userIds = userToPushMessage.keys.toList()
+
+        val filteredUserToPushMessage =
+            pushOptOutRepository
+                .findByUserIdInAndPushCategory(userIds, pushCategory)
+                .map { it.userId }
+                .toSet()
+                .let { optOutUserIds -> userToPushMessage.filterKeys { it !in optOutUserIds } }
+
+        if (filteredUserToPushMessage.isNotEmpty()) {
+            sendTargetPushes(filteredUserToPushMessage)
+        }
+    }
 }

From 8c67b4bda4798b8a9532d24df0fca534e830070b Mon Sep 17 00:00:00 2001
From: seonghaejo <jsh990324@snu.ac.kr>
Date: Sun, 9 Feb 2025 15:08:51 +0900
Subject: [PATCH 3/6] =?UTF-8?q?SugangSnuNotificationService=20/=20PushWith?=
 =?UTF-8?q?NotificationService=EC=97=90=20=ED=91=B8=EC=8B=9C=20=EC=B9=B4?=
 =?UTF-8?q?=ED=85=8C=EA=B3=A0=EB=A6=AC=20=EC=A0=81=EC=9A=A9?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

---
 .../common/service/SugangSnuNotificationService.kt     |  3 ++-
 core/src/main/kotlin/notification/data/PushCategory.kt |  8 ++++++++
 .../main/kotlin/notification/service/PushService.kt    | 10 +++++++++-
 .../service/PushWithNotificationService.kt             |  5 +++--
 4 files changed, 22 insertions(+), 4 deletions(-)

diff --git a/batch/src/main/kotlin/sugangsnu/common/service/SugangSnuNotificationService.kt b/batch/src/main/kotlin/sugangsnu/common/service/SugangSnuNotificationService.kt
index 2b78d01b..69d735b2 100644
--- a/batch/src/main/kotlin/sugangsnu/common/service/SugangSnuNotificationService.kt
+++ b/batch/src/main/kotlin/sugangsnu/common/service/SugangSnuNotificationService.kt
@@ -5,6 +5,7 @@ import com.wafflestudio.snutt.common.push.dto.PushMessage
 import com.wafflestudio.snutt.coursebook.data.Coursebook
 import com.wafflestudio.snutt.notification.data.Notification
 import com.wafflestudio.snutt.notification.data.NotificationType
+import com.wafflestudio.snutt.notification.data.PushCategory
 import com.wafflestudio.snutt.notification.service.NotificationService
 import com.wafflestudio.snutt.notification.service.PushService
 import com.wafflestudio.snutt.notification.service.PushWithNotificationService
@@ -74,7 +75,7 @@ class SugangSnuNotificationServiceImpl(
                         urlScheme = DeeplinkType.NOTIFICATIONS,
                     )
                 }
-            pushService.sendTargetPushes(userIdToMessage)
+            pushService.sendCategoricalTargetPushes(userIdToMessage, PushCategory.LECTURE_UPDATE)
         }
 
     override suspend fun notifyCoursebookUpdate(coursebook: Coursebook) {
diff --git a/core/src/main/kotlin/notification/data/PushCategory.kt b/core/src/main/kotlin/notification/data/PushCategory.kt
index 6f1ff1e2..abb2c537 100644
--- a/core/src/main/kotlin/notification/data/PushCategory.kt
+++ b/core/src/main/kotlin/notification/data/PushCategory.kt
@@ -9,6 +9,7 @@ import org.springframework.stereotype.Component
 enum class PushCategory(
     @JsonValue val value: Int,
 ) {
+    NORMAL(0),
     LECTURE_UPDATE(1),
     VACANCY_NOTIFICATION(2),
     ;
@@ -31,3 +32,10 @@ class PushCategoryReadConverter : Converter<Int, PushCategory> {
 class PushCategoryWriteConverter : Converter<PushCategory, Int> {
     override fun convert(source: PushCategory): Int = source.value
 }
+
+fun PushCategory(notificationType: NotificationType) =
+    when (notificationType) {
+        NotificationType.LECTURE_UPDATE -> PushCategory.LECTURE_UPDATE
+        NotificationType.LECTURE_VACANCY -> PushCategory.VACANCY_NOTIFICATION
+        else -> PushCategory.NORMAL
+    }
diff --git a/core/src/main/kotlin/notification/service/PushService.kt b/core/src/main/kotlin/notification/service/PushService.kt
index 8dccf9d0..f14d8632 100644
--- a/core/src/main/kotlin/notification/service/PushService.kt
+++ b/core/src/main/kotlin/notification/service/PushService.kt
@@ -86,7 +86,7 @@ class PushServiceImpl internal constructor(
         userId: String,
         pushCategory: PushCategory,
     ) {
-        if (!pushOptOutRepository.existsByUserIdAndPushCategory(userId, pushCategory)) {
+        if (pushCategory == PushCategory.NORMAL || !pushOptOutRepository.existsByUserIdAndPushCategory(userId, pushCategory)) {
             sendPush(pushMessage, userId)
         }
     }
@@ -96,6 +96,10 @@ class PushServiceImpl internal constructor(
         userIds: List<String>,
         pushCategory: PushCategory,
     ) {
+        if (pushCategory == PushCategory.NORMAL) {
+            sendPushes(pushMessage, userIds)
+        }
+
         val filteredUserIds =
             pushOptOutRepository
                 .findByUserIdInAndPushCategory(userIds, pushCategory)
@@ -112,6 +116,10 @@ class PushServiceImpl internal constructor(
         userToPushMessage: Map<String, PushMessage>,
         pushCategory: PushCategory,
     ) {
+        if (pushCategory == PushCategory.NORMAL) {
+            sendTargetPushes(userToPushMessage)
+        }
+
         val userIds = userToPushMessage.keys.toList()
 
         val filteredUserToPushMessage =
diff --git a/core/src/main/kotlin/notification/service/PushWithNotificationService.kt b/core/src/main/kotlin/notification/service/PushWithNotificationService.kt
index 0da50ba8..1d25c8a4 100644
--- a/core/src/main/kotlin/notification/service/PushWithNotificationService.kt
+++ b/core/src/main/kotlin/notification/service/PushWithNotificationService.kt
@@ -2,6 +2,7 @@ package com.wafflestudio.snutt.notification.service
 
 import com.wafflestudio.snutt.common.push.dto.PushMessage
 import com.wafflestudio.snutt.notification.data.NotificationType
+import com.wafflestudio.snutt.notification.data.PushCategory
 import kotlinx.coroutines.coroutineScope
 import kotlinx.coroutines.launch
 import org.springframework.stereotype.Service
@@ -46,7 +47,7 @@ class PushWithNotificationServiceImpl internal constructor(
     ): Unit =
         coroutineScope {
             launch { notificationService.sendNotification(pushMessage.toNotification(notificationType, userId)) }
-            launch { pushService.sendPush(pushMessage, userId) }
+            launch { pushService.sendCategoricalPush(pushMessage, userId, PushCategory(notificationType)) }
         }
 
     override suspend fun sendPushesAndNotifications(
@@ -65,7 +66,7 @@ class PushWithNotificationServiceImpl internal constructor(
                     },
                 )
             }
-            launch { pushService.sendPushes(pushMessage, userIds) }
+            launch { pushService.sendCategoricalPushes(pushMessage, userIds, PushCategory(notificationType)) }
         }
 
     override suspend fun sendGlobalPushAndNotification(

From 2cf26d5381918c91ce91713091b4658ec0cc00d8 Mon Sep 17 00:00:00 2001
From: seonghaejo <jsh990324@snu.ac.kr>
Date: Sun, 9 Feb 2025 23:27:33 +0900
Subject: [PATCH 4/6] Add PushPreference Service/Handler

---
 .../kotlin/handler/PushPreferenceHandler.kt   | 47 +++++++++++++
 api/src/main/kotlin/router/MainRouter.kt      | 14 ++++
 .../kotlin/notification/data/PushOptOut.kt    |  2 +-
 .../kotlin/notification/dto/PushPreference.kt |  8 +++
 .../dto/PushPreferenceResponse.kt             | 12 ++++
 .../repository/PushOptOutRepository.kt        |  2 +
 .../service/PushPreferenceService.kt          | 67 +++++++++++++++++++
 7 files changed, 151 insertions(+), 1 deletion(-)
 create mode 100644 api/src/main/kotlin/handler/PushPreferenceHandler.kt
 create mode 100644 core/src/main/kotlin/notification/dto/PushPreference.kt
 create mode 100644 core/src/main/kotlin/notification/dto/PushPreferenceResponse.kt
 create mode 100644 core/src/main/kotlin/notification/service/PushPreferenceService.kt

diff --git a/api/src/main/kotlin/handler/PushPreferenceHandler.kt b/api/src/main/kotlin/handler/PushPreferenceHandler.kt
new file mode 100644
index 00000000..c73426e7
--- /dev/null
+++ b/api/src/main/kotlin/handler/PushPreferenceHandler.kt
@@ -0,0 +1,47 @@
+package com.wafflestudio.snutt.handler
+
+import com.wafflestudio.snutt.middleware.SnuttRestApiDefaultMiddleware
+import com.wafflestudio.snutt.notification.data.PushCategory
+import com.wafflestudio.snutt.notification.dto.PushPreferenceResponse
+import com.wafflestudio.snutt.notification.service.PushPreferenceService
+import org.springframework.stereotype.Component
+import org.springframework.web.reactive.function.server.ServerRequest
+
+@Component
+class PushPreferenceHandler(
+    private val pushPreferenceService: PushPreferenceService,
+    snuttRestApiDefaultMiddleware: SnuttRestApiDefaultMiddleware,
+) : ServiceHandler(
+        handlerMiddleware = snuttRestApiDefaultMiddleware,
+    ) {
+    suspend fun getPushPreferences(req: ServerRequest) =
+        handle(req) {
+            val user = req.getContext().user!!
+            val pushPreferences = pushPreferenceService.getPushPreferences(user)
+            pushPreferences.map { PushPreferenceResponse(it) }
+        }
+
+    suspend fun enableLectureUpdate(req: ServerRequest) =
+        handle(req) {
+            val user = req.getContext().user!!
+            pushPreferenceService.enablePush(user, PushCategory.LECTURE_UPDATE)
+        }
+
+    suspend fun disableLectureUpdate(req: ServerRequest) =
+        handle(req) {
+            val user = req.getContext().user!!
+            pushPreferenceService.disablePush(user, PushCategory.LECTURE_UPDATE)
+        }
+
+    suspend fun enableVacancyNotification(req: ServerRequest) =
+        handle(req) {
+            val user = req.getContext().user!!
+            pushPreferenceService.enablePush(user, PushCategory.VACANCY_NOTIFICATION)
+        }
+
+    suspend fun disableVacancyNotification(req: ServerRequest) =
+        handle(req) {
+            val user = req.getContext().user!!
+            pushPreferenceService.disablePush(user, PushCategory.VACANCY_NOTIFICATION)
+        }
+}
diff --git a/api/src/main/kotlin/router/MainRouter.kt b/api/src/main/kotlin/router/MainRouter.kt
index 463de490..00b51023 100644
--- a/api/src/main/kotlin/router/MainRouter.kt
+++ b/api/src/main/kotlin/router/MainRouter.kt
@@ -15,6 +15,7 @@ import com.wafflestudio.snutt.handler.FriendTableHandler
 import com.wafflestudio.snutt.handler.LectureSearchHandler
 import com.wafflestudio.snutt.handler.NotificationHandler
 import com.wafflestudio.snutt.handler.PopupHandler
+import com.wafflestudio.snutt.handler.PushPreferenceHandler
 import com.wafflestudio.snutt.handler.StaticPageHandler
 import com.wafflestudio.snutt.handler.TagHandler
 import com.wafflestudio.snutt.handler.TimetableHandler
@@ -70,6 +71,7 @@ class MainRouter(
     private val feedbackHandler: FeedbackHandler,
     private val staticPageHandler: StaticPageHandler,
     private val evServiceHandler: EvServiceHandler,
+    private val pushPreferenceHandler: PushPreferenceHandler,
 ) {
     @Bean
     fun healthCheck() =
@@ -343,4 +345,16 @@ class MainRouter(
             GET("/privacy_policy").invoke { staticPageHandler.privacyPolicy() }
             GET("/terms_of_service").invoke { staticPageHandler.termsOfService() }
         }
+
+    @Bean
+    fun pushPreferenceRouter() =
+        v1CoRouter {
+            "/push/preferences".nest {
+                GET("", pushPreferenceHandler::getPushPreferences)
+                POST("lecture-update", pushPreferenceHandler::enableLectureUpdate)
+                DELETE("lecture-update", pushPreferenceHandler::disableLectureUpdate)
+                POST("vacancy-notification", pushPreferenceHandler::enableVacancyNotification)
+                DELETE("vacancy-notification", pushPreferenceHandler::disableVacancyNotification)
+            }
+        }
 }
diff --git a/core/src/main/kotlin/notification/data/PushOptOut.kt b/core/src/main/kotlin/notification/data/PushOptOut.kt
index 30ef5888..c6c7c1b1 100644
--- a/core/src/main/kotlin/notification/data/PushOptOut.kt
+++ b/core/src/main/kotlin/notification/data/PushOptOut.kt
@@ -8,7 +8,7 @@ import org.springframework.data.mongodb.core.mapping.Field
 import org.springframework.data.mongodb.core.mapping.FieldType
 
 @Document(collection = "push_opt_out")
-@CompoundIndex(def = "{ 'user_id': 1, 'push_category': 1 }")
+@CompoundIndex(def = "{ 'user_id': 1, 'push_category': 1 }", unique = true)
 data class PushOptOut(
     @Id
     val id: String? = null,
diff --git a/core/src/main/kotlin/notification/dto/PushPreference.kt b/core/src/main/kotlin/notification/dto/PushPreference.kt
new file mode 100644
index 00000000..b3d9ff95
--- /dev/null
+++ b/core/src/main/kotlin/notification/dto/PushPreference.kt
@@ -0,0 +1,8 @@
+package com.wafflestudio.snutt.notification.dto
+
+import com.wafflestudio.snutt.notification.data.PushCategory
+
+data class PushPreference(
+    val pushCategory: PushCategory,
+    val enabled: Boolean,
+)
diff --git a/core/src/main/kotlin/notification/dto/PushPreferenceResponse.kt b/core/src/main/kotlin/notification/dto/PushPreferenceResponse.kt
new file mode 100644
index 00000000..66b8df99
--- /dev/null
+++ b/core/src/main/kotlin/notification/dto/PushPreferenceResponse.kt
@@ -0,0 +1,12 @@
+package com.wafflestudio.snutt.notification.dto
+
+data class PushPreferenceResponse(
+    val pushCategoryName: String,
+    val enabled: Boolean,
+)
+
+fun PushPreferenceResponse(pushPreference: PushPreference) =
+    PushPreferenceResponse(
+        pushCategoryName = pushPreference.pushCategory.name,
+        enabled = pushPreference.enabled,
+    )
diff --git a/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt b/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
index 88972f54..c63be8a4 100644
--- a/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
+++ b/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
@@ -19,4 +19,6 @@ interface PushOptOutRepository : CoroutineCrudRepository<PushOptOut, String> {
         userIds: List<String>,
         pushCategory: PushCategory,
     ): List<PushOptOut>
+
+    suspend fun findByUserId(userId: String): List<PushOptOut>
 }
diff --git a/core/src/main/kotlin/notification/service/PushPreferenceService.kt b/core/src/main/kotlin/notification/service/PushPreferenceService.kt
new file mode 100644
index 00000000..aedefc4a
--- /dev/null
+++ b/core/src/main/kotlin/notification/service/PushPreferenceService.kt
@@ -0,0 +1,67 @@
+package com.wafflestudio.snutt.notification.service
+
+import com.wafflestudio.snutt.notification.data.PushCategory
+import com.wafflestudio.snutt.notification.data.PushOptOut
+import com.wafflestudio.snutt.notification.dto.PushPreference
+import com.wafflestudio.snutt.notification.repository.PushOptOutRepository
+import com.wafflestudio.snutt.users.data.User
+import org.springframework.stereotype.Service
+
+interface PushPreferenceService {
+    suspend fun enablePush(
+        user: User,
+        pushCategory: PushCategory,
+    )
+
+    suspend fun disablePush(
+        user: User,
+        pushCategory: PushCategory,
+    )
+
+    suspend fun getPushPreferences(user: User): List<PushPreference>
+}
+
+@Service
+class PushPreferenceServiceImpl(
+    private val pushOptOutRepository: PushOptOutRepository,
+) : PushPreferenceService {
+    override suspend fun enablePush(
+        user: User,
+        pushCategory: PushCategory,
+    ) {
+        pushOptOutRepository.save(
+            PushOptOut(
+                userId = user.id!!,
+                pushCategory = pushCategory,
+            ),
+        )
+    }
+
+    override suspend fun disablePush(
+        user: User,
+        pushCategory: PushCategory,
+    ) {
+        pushOptOutRepository.deleteByUserIdAndPushCategory(
+            userId = user.id!!,
+            pushCategory = pushCategory,
+        )
+    }
+
+    override suspend fun getPushPreferences(user: User): List<PushPreference> {
+        val allPushCategories = PushCategory.entries.filterNot { it == PushCategory.NORMAL }
+        val disabledPushCategories = pushOptOutRepository.findByUserId(user.id!!).map { it.pushCategory }.toSet()
+        return allPushCategories.map {
+            if (it in disabledPushCategories) {
+                return@map PushPreference(
+                    pushCategory = it,
+                    enabled = false,
+                )
+            } else {
+                return@map PushPreference(
+                    pushCategory = it,
+                    enabled = true,
+                )
+            }
+        }
+    }
+}

From 2e3bb3abb6bf0d440d1b58fe1b5a2ab21d5fae1c Mon Sep 17 00:00:00 2001
From: seonghaejo <jsh990324@snu.ac.kr>
Date: Wed, 19 Mar 2025 21:46:56 +0900
Subject: [PATCH 5/6] =?UTF-8?q?=EC=97=94=ED=8B=B0=ED=8B=B0=20=EB=AA=A8?=
 =?UTF-8?q?=EB=8D=B8=20=EB=B3=80=EA=B2=BD=20&=20PushPreferenceService=20?=
 =?UTF-8?q?=EA=B0=9C=EC=84=A0?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

---
 .../kotlin/handler/PushPreferenceHandler.kt   |  30 ++----
 api/src/main/kotlin/router/MainRouter.kt      |   5 +-
 .../service/SugangSnuNotificationService.kt   |   4 +-
 .../kotlin/notification/data/PushCategory.kt  |  41 -------
 .../data/{PushOptOut.kt => PushPreference.kt} |  14 +--
 .../notification/data/PushPreferenceItem.kt   |   6 ++
 .../notification/data/PushPreferenceType.kt   |  14 +++
 .../kotlin/notification/dto/PushPreference.kt |   8 --
 .../notification/dto/PushPreferenceDto.kt     |  10 ++
 .../dto/PushPreferenceResponse.kt             |  12 ---
 .../repository/PushOptOutRepository.kt        |  24 -----
 .../repository/PushPreferenceRepository.kt    |  10 ++
 .../service/PushPreferenceService.kt          | 102 ++++++++++--------
 .../notification/service/PushService.kt       |  57 ++++------
 .../service/PushWithNotificationService.kt    |   6 +-
 15 files changed, 137 insertions(+), 206 deletions(-)
 delete mode 100644 core/src/main/kotlin/notification/data/PushCategory.kt
 rename core/src/main/kotlin/notification/data/{PushOptOut.kt => PushPreference.kt} (53%)
 create mode 100644 core/src/main/kotlin/notification/data/PushPreferenceItem.kt
 create mode 100644 core/src/main/kotlin/notification/data/PushPreferenceType.kt
 delete mode 100644 core/src/main/kotlin/notification/dto/PushPreference.kt
 create mode 100644 core/src/main/kotlin/notification/dto/PushPreferenceDto.kt
 delete mode 100644 core/src/main/kotlin/notification/dto/PushPreferenceResponse.kt
 delete mode 100644 core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
 create mode 100644 core/src/main/kotlin/notification/repository/PushPreferenceRepository.kt

diff --git a/api/src/main/kotlin/handler/PushPreferenceHandler.kt b/api/src/main/kotlin/handler/PushPreferenceHandler.kt
index c73426e7..8fc7bedd 100644
--- a/api/src/main/kotlin/handler/PushPreferenceHandler.kt
+++ b/api/src/main/kotlin/handler/PushPreferenceHandler.kt
@@ -1,11 +1,11 @@
 package com.wafflestudio.snutt.handler
 
 import com.wafflestudio.snutt.middleware.SnuttRestApiDefaultMiddleware
-import com.wafflestudio.snutt.notification.data.PushCategory
-import com.wafflestudio.snutt.notification.dto.PushPreferenceResponse
+import com.wafflestudio.snutt.notification.dto.PushPreferenceDto
 import com.wafflestudio.snutt.notification.service.PushPreferenceService
 import org.springframework.stereotype.Component
 import org.springframework.web.reactive.function.server.ServerRequest
+import org.springframework.web.reactive.function.server.awaitBody
 
 @Component
 class PushPreferenceHandler(
@@ -17,31 +17,13 @@ class PushPreferenceHandler(
     suspend fun getPushPreferences(req: ServerRequest) =
         handle(req) {
             val user = req.getContext().user!!
-            val pushPreferences = pushPreferenceService.getPushPreferences(user)
-            pushPreferences.map { PushPreferenceResponse(it) }
+            pushPreferenceService.getPushPreferenceDto(user)
         }
 
-    suspend fun enableLectureUpdate(req: ServerRequest) =
+    suspend fun savePushPreferences(req: ServerRequest) =
         handle(req) {
             val user = req.getContext().user!!
-            pushPreferenceService.enablePush(user, PushCategory.LECTURE_UPDATE)
-        }
-
-    suspend fun disableLectureUpdate(req: ServerRequest) =
-        handle(req) {
-            val user = req.getContext().user!!
-            pushPreferenceService.disablePush(user, PushCategory.LECTURE_UPDATE)
-        }
-
-    suspend fun enableVacancyNotification(req: ServerRequest) =
-        handle(req) {
-            val user = req.getContext().user!!
-            pushPreferenceService.enablePush(user, PushCategory.VACANCY_NOTIFICATION)
-        }
-
-    suspend fun disableVacancyNotification(req: ServerRequest) =
-        handle(req) {
-            val user = req.getContext().user!!
-            pushPreferenceService.disablePush(user, PushCategory.VACANCY_NOTIFICATION)
+            val pushPreferenceDto = req.awaitBody<PushPreferenceDto>()
+            pushPreferenceService.savePushPreference(user, pushPreferenceDto)
         }
 }
diff --git a/api/src/main/kotlin/router/MainRouter.kt b/api/src/main/kotlin/router/MainRouter.kt
index 00b51023..64215da8 100644
--- a/api/src/main/kotlin/router/MainRouter.kt
+++ b/api/src/main/kotlin/router/MainRouter.kt
@@ -351,10 +351,7 @@ class MainRouter(
         v1CoRouter {
             "/push/preferences".nest {
                 GET("", pushPreferenceHandler::getPushPreferences)
-                POST("lecture-update", pushPreferenceHandler::enableLectureUpdate)
-                DELETE("lecture-update", pushPreferenceHandler::disableLectureUpdate)
-                POST("vacancy-notification", pushPreferenceHandler::enableVacancyNotification)
-                DELETE("vacancy-notification", pushPreferenceHandler::disableVacancyNotification)
+                POST("", pushPreferenceHandler::savePushPreferences)
             }
         }
 }
diff --git a/batch/src/main/kotlin/sugangsnu/common/service/SugangSnuNotificationService.kt b/batch/src/main/kotlin/sugangsnu/common/service/SugangSnuNotificationService.kt
index 69d735b2..46f408e1 100644
--- a/batch/src/main/kotlin/sugangsnu/common/service/SugangSnuNotificationService.kt
+++ b/batch/src/main/kotlin/sugangsnu/common/service/SugangSnuNotificationService.kt
@@ -5,7 +5,7 @@ import com.wafflestudio.snutt.common.push.dto.PushMessage
 import com.wafflestudio.snutt.coursebook.data.Coursebook
 import com.wafflestudio.snutt.notification.data.Notification
 import com.wafflestudio.snutt.notification.data.NotificationType
-import com.wafflestudio.snutt.notification.data.PushCategory
+import com.wafflestudio.snutt.notification.data.PushPreferenceType
 import com.wafflestudio.snutt.notification.service.NotificationService
 import com.wafflestudio.snutt.notification.service.PushService
 import com.wafflestudio.snutt.notification.service.PushWithNotificationService
@@ -75,7 +75,7 @@ class SugangSnuNotificationServiceImpl(
                         urlScheme = DeeplinkType.NOTIFICATIONS,
                     )
                 }
-            pushService.sendCategoricalTargetPushes(userIdToMessage, PushCategory.LECTURE_UPDATE)
+            pushService.sendTargetPushes(userIdToMessage, PushPreferenceType.LECTURE_UPDATE)
         }
 
     override suspend fun notifyCoursebookUpdate(coursebook: Coursebook) {
diff --git a/core/src/main/kotlin/notification/data/PushCategory.kt b/core/src/main/kotlin/notification/data/PushCategory.kt
deleted file mode 100644
index abb2c537..00000000
--- a/core/src/main/kotlin/notification/data/PushCategory.kt
+++ /dev/null
@@ -1,41 +0,0 @@
-package com.wafflestudio.snutt.notification.data
-
-import com.fasterxml.jackson.annotation.JsonValue
-import org.springframework.core.convert.converter.Converter
-import org.springframework.data.convert.ReadingConverter
-import org.springframework.data.convert.WritingConverter
-import org.springframework.stereotype.Component
-
-enum class PushCategory(
-    @JsonValue val value: Int,
-) {
-    NORMAL(0),
-    LECTURE_UPDATE(1),
-    VACANCY_NOTIFICATION(2),
-    ;
-
-    companion object {
-        private val valueMap = PushCategory.entries.associateBy { e -> e.value }
-
-        fun getOfValue(value: Int): PushCategory? = valueMap[value]
-    }
-}
-
-@ReadingConverter
-@Component
-class PushCategoryReadConverter : Converter<Int, PushCategory> {
-    override fun convert(source: Int): PushCategory = PushCategory.getOfValue(source)!!
-}
-
-@WritingConverter
-@Component
-class PushCategoryWriteConverter : Converter<PushCategory, Int> {
-    override fun convert(source: PushCategory): Int = source.value
-}
-
-fun PushCategory(notificationType: NotificationType) =
-    when (notificationType) {
-        NotificationType.LECTURE_UPDATE -> PushCategory.LECTURE_UPDATE
-        NotificationType.LECTURE_VACANCY -> PushCategory.VACANCY_NOTIFICATION
-        else -> PushCategory.NORMAL
-    }
diff --git a/core/src/main/kotlin/notification/data/PushOptOut.kt b/core/src/main/kotlin/notification/data/PushPreference.kt
similarity index 53%
rename from core/src/main/kotlin/notification/data/PushOptOut.kt
rename to core/src/main/kotlin/notification/data/PushPreference.kt
index c6c7c1b1..5699fc9a 100644
--- a/core/src/main/kotlin/notification/data/PushOptOut.kt
+++ b/core/src/main/kotlin/notification/data/PushPreference.kt
@@ -1,21 +1,17 @@
 package com.wafflestudio.snutt.notification.data
 
 import org.springframework.data.annotation.Id
-import org.springframework.data.mongodb.core.index.CompoundIndex
 import org.springframework.data.mongodb.core.index.Indexed
 import org.springframework.data.mongodb.core.mapping.Document
 import org.springframework.data.mongodb.core.mapping.Field
 import org.springframework.data.mongodb.core.mapping.FieldType
 
-@Document(collection = "push_opt_out")
-@CompoundIndex(def = "{ 'user_id': 1, 'push_category': 1 }", unique = true)
-data class PushOptOut(
+@Document(collection = "pushPreference")
+data class PushPreference(
     @Id
     val id: String? = null,
-    @Indexed
-    @Field("user_id", targetType = FieldType.OBJECT_ID)
+    @Indexed(unique = true)
+    @Field(targetType = FieldType.OBJECT_ID)
     val userId: String,
-    @Field("push_category")
-    @Indexed
-    val pushCategory: PushCategory,
+    val pushPreferences: List<PushPreferenceItem>,
 )
diff --git a/core/src/main/kotlin/notification/data/PushPreferenceItem.kt b/core/src/main/kotlin/notification/data/PushPreferenceItem.kt
new file mode 100644
index 00000000..df835e57
--- /dev/null
+++ b/core/src/main/kotlin/notification/data/PushPreferenceItem.kt
@@ -0,0 +1,6 @@
+package com.wafflestudio.snutt.notification.data
+
+data class PushPreferenceItem(
+    val type: PushPreferenceType,
+    val isEnabled: Boolean,
+)
diff --git a/core/src/main/kotlin/notification/data/PushPreferenceType.kt b/core/src/main/kotlin/notification/data/PushPreferenceType.kt
new file mode 100644
index 00000000..9f8814ea
--- /dev/null
+++ b/core/src/main/kotlin/notification/data/PushPreferenceType.kt
@@ -0,0 +1,14 @@
+package com.wafflestudio.snutt.notification.data
+
+enum class PushPreferenceType {
+    NORMAL,
+    LECTURE_UPDATE,
+    VACANCY_NOTIFICATION,
+}
+
+fun PushPreferenceType(notificationType: NotificationType) =
+    when (notificationType) {
+        NotificationType.LECTURE_UPDATE -> PushPreferenceType.LECTURE_UPDATE
+        NotificationType.LECTURE_VACANCY -> PushPreferenceType.VACANCY_NOTIFICATION
+        else -> PushPreferenceType.NORMAL
+    }
diff --git a/core/src/main/kotlin/notification/dto/PushPreference.kt b/core/src/main/kotlin/notification/dto/PushPreference.kt
deleted file mode 100644
index b3d9ff95..00000000
--- a/core/src/main/kotlin/notification/dto/PushPreference.kt
+++ /dev/null
@@ -1,8 +0,0 @@
-package com.wafflestudio.snutt.notification.dto
-
-import com.wafflestudio.snutt.notification.data.PushCategory
-
-data class PushPreference(
-    val pushCategory: PushCategory,
-    val enabled: Boolean,
-)
diff --git a/core/src/main/kotlin/notification/dto/PushPreferenceDto.kt b/core/src/main/kotlin/notification/dto/PushPreferenceDto.kt
new file mode 100644
index 00000000..60cd9375
--- /dev/null
+++ b/core/src/main/kotlin/notification/dto/PushPreferenceDto.kt
@@ -0,0 +1,10 @@
+package com.wafflestudio.snutt.notification.dto
+
+import com.wafflestudio.snutt.notification.data.PushPreference
+import com.wafflestudio.snutt.notification.data.PushPreferenceItem
+
+data class PushPreferenceDto(
+    val pushPreferences: List<PushPreferenceItem>,
+)
+
+fun PushPreferenceDto(pushPreference: PushPreference) = PushPreferenceDto(pushPreference.pushPreferences.toList())
diff --git a/core/src/main/kotlin/notification/dto/PushPreferenceResponse.kt b/core/src/main/kotlin/notification/dto/PushPreferenceResponse.kt
deleted file mode 100644
index 66b8df99..00000000
--- a/core/src/main/kotlin/notification/dto/PushPreferenceResponse.kt
+++ /dev/null
@@ -1,12 +0,0 @@
-package com.wafflestudio.snutt.notification.dto
-
-data class PushPreferenceResponse(
-    val pushCategoryName: String,
-    val enabled: Boolean,
-)
-
-fun PushPreferenceResponse(pushPreference: PushPreference) =
-    PushPreferenceResponse(
-        pushCategoryName = pushPreference.pushCategory.name,
-        enabled = pushPreference.enabled,
-    )
diff --git a/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt b/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
deleted file mode 100644
index c63be8a4..00000000
--- a/core/src/main/kotlin/notification/repository/PushOptOutRepository.kt
+++ /dev/null
@@ -1,24 +0,0 @@
-package com.wafflestudio.snutt.notification.repository
-
-import com.wafflestudio.snutt.notification.data.PushCategory
-import com.wafflestudio.snutt.notification.data.PushOptOut
-import org.springframework.data.repository.kotlin.CoroutineCrudRepository
-
-interface PushOptOutRepository : CoroutineCrudRepository<PushOptOut, String> {
-    suspend fun existsByUserIdAndPushCategory(
-        userId: String,
-        pushCategory: PushCategory,
-    ): Boolean
-
-    suspend fun deleteByUserIdAndPushCategory(
-        userId: String,
-        pushCategory: PushCategory,
-    ): Long
-
-    suspend fun findByUserIdInAndPushCategory(
-        userIds: List<String>,
-        pushCategory: PushCategory,
-    ): List<PushOptOut>
-
-    suspend fun findByUserId(userId: String): List<PushOptOut>
-}
diff --git a/core/src/main/kotlin/notification/repository/PushPreferenceRepository.kt b/core/src/main/kotlin/notification/repository/PushPreferenceRepository.kt
new file mode 100644
index 00000000..213e39b0
--- /dev/null
+++ b/core/src/main/kotlin/notification/repository/PushPreferenceRepository.kt
@@ -0,0 +1,10 @@
+package com.wafflestudio.snutt.notification.repository
+
+import com.wafflestudio.snutt.notification.data.PushPreference
+import org.springframework.data.repository.kotlin.CoroutineCrudRepository
+
+interface PushPreferenceRepository : CoroutineCrudRepository<PushPreference, String> {
+    suspend fun findByUserId(userId: String): PushPreference?
+
+    suspend fun findByUserIdIn(userIds: List<String>): List<PushPreference>
+}
diff --git a/core/src/main/kotlin/notification/service/PushPreferenceService.kt b/core/src/main/kotlin/notification/service/PushPreferenceService.kt
index aedefc4a..5a8f93e1 100644
--- a/core/src/main/kotlin/notification/service/PushPreferenceService.kt
+++ b/core/src/main/kotlin/notification/service/PushPreferenceService.kt
@@ -1,67 +1,85 @@
 package com.wafflestudio.snutt.notification.service
 
-import com.wafflestudio.snutt.notification.data.PushCategory
-import com.wafflestudio.snutt.notification.data.PushOptOut
-import com.wafflestudio.snutt.notification.dto.PushPreference
-import com.wafflestudio.snutt.notification.repository.PushOptOutRepository
+import com.wafflestudio.snutt.notification.data.PushPreference
+import com.wafflestudio.snutt.notification.data.PushPreferenceType
+import com.wafflestudio.snutt.notification.dto.PushPreferenceDto
+import com.wafflestudio.snutt.notification.repository.PushPreferenceRepository
 import com.wafflestudio.snutt.users.data.User
 import org.springframework.stereotype.Service
 
 interface PushPreferenceService {
-    suspend fun enablePush(
+    suspend fun savePushPreference(
         user: User,
-        pushCategory: PushCategory,
+        pushPreferenceDto: PushPreferenceDto,
     )
 
-    suspend fun disablePush(
-        user: User,
-        pushCategory: PushCategory,
-    )
+    suspend fun getPushPreferenceDto(user: User): PushPreferenceDto
+
+    suspend fun isPushPreferenceEnabled(
+        userId: String,
+        pushPreferenceType: PushPreferenceType,
+    ): Boolean
 
-    suspend fun getPushPreferences(user: User): List<PushPreference>
+    suspend fun filterUsersByPushPreference(
+        userIds: List<String>,
+        pushPreferenceType: PushPreferenceType,
+    ): List<String>
 }
 
 @Service
 class PushPreferenceServiceImpl(
-    private val pushOptOutRepository: PushOptOutRepository,
+    private val pushPreferenceRepository: PushPreferenceRepository,
 ) : PushPreferenceService {
-    override suspend fun enablePush(
+    override suspend fun savePushPreference(
         user: User,
-        pushCategory: PushCategory,
+        pushPreferenceDto: PushPreferenceDto,
     ) {
-        pushOptOutRepository.save(
-            PushOptOut(
-                userId = user.id!!,
-                pushCategory = pushCategory,
-            ),
+        pushPreferenceRepository.save(
+            pushPreferenceRepository.findByUserId(user.id!!)
+                ?.copy(pushPreferences = pushPreferenceDto.pushPreferences)
+                ?: PushPreference(
+                    userId = user.id,
+                    pushPreferences = pushPreferenceDto.pushPreferences,
+                ),
         )
     }
 
-    override suspend fun disablePush(
-        user: User,
-        pushCategory: PushCategory,
-    ) {
-        pushOptOutRepository.deleteByUserIdAndPushCategory(
-            userId = user.id!!,
-            pushCategory = pushCategory,
-        )
+    override suspend fun getPushPreferenceDto(user: User): PushPreferenceDto =
+        pushPreferenceRepository.findByUserId(user.id!!)
+            ?.let { PushPreferenceDto(it) }
+            ?: PushPreferenceDto(
+                pushPreferences = emptyList(),
+            )
+
+    override suspend fun isPushPreferenceEnabled(
+        userId: String,
+        pushPreferenceType: PushPreferenceType,
+    ): Boolean {
+        if (pushPreferenceType == PushPreferenceType.NORMAL) {
+            return true
+        }
+
+        return pushPreferenceRepository
+            .findByUserId(userId)
+            ?.pushPreferences
+            ?.any { it.type == pushPreferenceType && it.isEnabled }
+            ?: false
     }
 
-    override suspend fun getPushPreferences(user: User): List<PushPreference> {
-        val allPushCategories = PushCategory.entries.filterNot { it == PushCategory.NORMAL }
-        val disabledPushCategories = pushOptOutRepository.findByUserId(user.id!!).map { it.pushCategory }.toSet()
-        return allPushCategories.map {
-            if (it in disabledPushCategories) {
-                return@map PushPreference(
-                    pushCategory = it,
-                    enabled = false,
-                )
-            } else {
-                return@map PushPreference(
-                    pushCategory = it,
-                    enabled = true,
-                )
-            }
+    override suspend fun filterUsersByPushPreference(
+        userIds: List<String>,
+        pushPreferenceType: PushPreferenceType,
+    ): List<String> {
+        if (pushPreferenceType == PushPreferenceType.NORMAL) {
+            return userIds
         }
+
+        return pushPreferenceRepository
+            .findByUserIdIn(userIds)
+            .filter { pushPreference ->
+                pushPreference.pushPreferences
+                    .any { it.type == pushPreferenceType && it.isEnabled }
+            }
+            .map { it.userId }
     }
 }
diff --git a/core/src/main/kotlin/notification/service/PushService.kt b/core/src/main/kotlin/notification/service/PushService.kt
index f14d8632..655bb184 100644
--- a/core/src/main/kotlin/notification/service/PushService.kt
+++ b/core/src/main/kotlin/notification/service/PushService.kt
@@ -3,8 +3,7 @@ package com.wafflestudio.snutt.notification.service
 import com.wafflestudio.snutt.common.push.PushClient
 import com.wafflestudio.snutt.common.push.dto.PushMessage
 import com.wafflestudio.snutt.common.push.dto.TargetedPushMessageWithToken
-import com.wafflestudio.snutt.notification.data.PushCategory
-import com.wafflestudio.snutt.notification.repository.PushOptOutRepository
+import com.wafflestudio.snutt.notification.data.PushPreferenceType
 import org.springframework.stereotype.Service
 
 /**
@@ -25,21 +24,21 @@ interface PushService {
 
     suspend fun sendTargetPushes(userToPushMessage: Map<String, PushMessage>)
 
-    suspend fun sendCategoricalPush(
+    suspend fun sendPush(
         pushMessage: PushMessage,
         userId: String,
-        pushCategory: PushCategory,
+        pushPreferenceType: PushPreferenceType,
     )
 
-    suspend fun sendCategoricalPushes(
+    suspend fun sendPushes(
         pushMessage: PushMessage,
         userIds: List<String>,
-        pushCategory: PushCategory,
+        pushPreferenceType: PushPreferenceType,
     )
 
-    suspend fun sendCategoricalTargetPushes(
+    suspend fun sendTargetPushes(
         userToPushMessage: Map<String, PushMessage>,
-        pushCategory: PushCategory,
+        pushPreferenceType: PushPreferenceType,
     )
 }
 
@@ -47,7 +46,7 @@ interface PushService {
 class PushServiceImpl internal constructor(
     private val deviceService: DeviceService,
     private val pushClient: PushClient,
-    private val pushOptOutRepository: PushOptOutRepository,
+    private val pushPreferenceService: PushPreferenceService,
 ) : PushService {
     override suspend fun sendPush(
         pushMessage: PushMessage,
@@ -81,53 +80,37 @@ class PushServiceImpl internal constructor(
         }.map { (fcmRegistrationId, message) -> TargetedPushMessageWithToken(fcmRegistrationId, message) }
             .let { pushClient.sendMessages(it) }
 
-    override suspend fun sendCategoricalPush(
+    override suspend fun sendPush(
         pushMessage: PushMessage,
         userId: String,
-        pushCategory: PushCategory,
+        pushPreferenceType: PushPreferenceType,
     ) {
-        if (pushCategory == PushCategory.NORMAL || !pushOptOutRepository.existsByUserIdAndPushCategory(userId, pushCategory)) {
+        if (pushPreferenceService.isPushPreferenceEnabled(userId, pushPreferenceType)) {
             sendPush(pushMessage, userId)
         }
     }
 
-    override suspend fun sendCategoricalPushes(
+    override suspend fun sendPushes(
         pushMessage: PushMessage,
         userIds: List<String>,
-        pushCategory: PushCategory,
+        pushPreferenceType: PushPreferenceType,
     ) {
-        if (pushCategory == PushCategory.NORMAL) {
-            sendPushes(pushMessage, userIds)
-        }
-
-        val filteredUserIds =
-            pushOptOutRepository
-                .findByUserIdInAndPushCategory(userIds, pushCategory)
-                .map { it.userId }
-                .toSet()
-                .let { optOutUserIds -> userIds.filterNot { it in optOutUserIds } }
+        val filteredUserIds = pushPreferenceService.filterUsersByPushPreference(userIds, pushPreferenceType)
 
         if (filteredUserIds.isNotEmpty()) {
             sendPushes(pushMessage, filteredUserIds)
         }
     }
 
-    override suspend fun sendCategoricalTargetPushes(
+    override suspend fun sendTargetPushes(
         userToPushMessage: Map<String, PushMessage>,
-        pushCategory: PushCategory,
+        pushPreferenceType: PushPreferenceType,
     ) {
-        if (pushCategory == PushCategory.NORMAL) {
-            sendTargetPushes(userToPushMessage)
-        }
-
-        val userIds = userToPushMessage.keys.toList()
-
         val filteredUserToPushMessage =
-            pushOptOutRepository
-                .findByUserIdInAndPushCategory(userIds, pushCategory)
-                .map { it.userId }
-                .toSet()
-                .let { optOutUserIds -> userToPushMessage.filterKeys { it !in optOutUserIds } }
+            userToPushMessage.filterKeys {
+                    userId ->
+                pushPreferenceService.isPushPreferenceEnabled(userId, pushPreferenceType)
+            }
 
         if (filteredUserToPushMessage.isNotEmpty()) {
             sendTargetPushes(filteredUserToPushMessage)
diff --git a/core/src/main/kotlin/notification/service/PushWithNotificationService.kt b/core/src/main/kotlin/notification/service/PushWithNotificationService.kt
index 1d25c8a4..1ccbcfea 100644
--- a/core/src/main/kotlin/notification/service/PushWithNotificationService.kt
+++ b/core/src/main/kotlin/notification/service/PushWithNotificationService.kt
@@ -2,7 +2,7 @@ package com.wafflestudio.snutt.notification.service
 
 import com.wafflestudio.snutt.common.push.dto.PushMessage
 import com.wafflestudio.snutt.notification.data.NotificationType
-import com.wafflestudio.snutt.notification.data.PushCategory
+import com.wafflestudio.snutt.notification.data.PushPreferenceType
 import kotlinx.coroutines.coroutineScope
 import kotlinx.coroutines.launch
 import org.springframework.stereotype.Service
@@ -47,7 +47,7 @@ class PushWithNotificationServiceImpl internal constructor(
     ): Unit =
         coroutineScope {
             launch { notificationService.sendNotification(pushMessage.toNotification(notificationType, userId)) }
-            launch { pushService.sendCategoricalPush(pushMessage, userId, PushCategory(notificationType)) }
+            launch { pushService.sendPush(pushMessage, userId, PushPreferenceType(notificationType)) }
         }
 
     override suspend fun sendPushesAndNotifications(
@@ -66,7 +66,7 @@ class PushWithNotificationServiceImpl internal constructor(
                     },
                 )
             }
-            launch { pushService.sendCategoricalPushes(pushMessage, userIds, PushCategory(notificationType)) }
+            launch { pushService.sendPushes(pushMessage, userIds, PushPreferenceType(notificationType)) }
         }
 
     override suspend fun sendGlobalPushAndNotification(

From 2e896ce5182b684ee8353a2a5af2bed37b7c2377 Mon Sep 17 00:00:00 2001
From: seonghaejo <jsh990324@snu.ac.kr>
Date: Wed, 19 Mar 2025 22:34:50 +0900
Subject: [PATCH 6/6] Add docs

---
 api/src/main/kotlin/router/MainRouter.kt    |  2 +
 api/src/main/kotlin/router/docs/PushDocs.kt | 95 +++++++++++++++++++++
 2 files changed, 97 insertions(+)
 create mode 100644 api/src/main/kotlin/router/docs/PushDocs.kt

diff --git a/api/src/main/kotlin/router/MainRouter.kt b/api/src/main/kotlin/router/MainRouter.kt
index 64215da8..fe60d3a3 100644
--- a/api/src/main/kotlin/router/MainRouter.kt
+++ b/api/src/main/kotlin/router/MainRouter.kt
@@ -35,6 +35,7 @@ import com.wafflestudio.snutt.router.docs.FriendDocs
 import com.wafflestudio.snutt.router.docs.LectureSearchDocs
 import com.wafflestudio.snutt.router.docs.NotificationDocs
 import com.wafflestudio.snutt.router.docs.PopupDocs
+import com.wafflestudio.snutt.router.docs.PushDocs
 import com.wafflestudio.snutt.router.docs.TagDocs
 import com.wafflestudio.snutt.router.docs.ThemeDocs
 import com.wafflestudio.snutt.router.docs.TimetableDocs
@@ -347,6 +348,7 @@ class MainRouter(
         }
 
     @Bean
+    @PushDocs
     fun pushPreferenceRouter() =
         v1CoRouter {
             "/push/preferences".nest {
diff --git a/api/src/main/kotlin/router/docs/PushDocs.kt b/api/src/main/kotlin/router/docs/PushDocs.kt
new file mode 100644
index 00000000..37bf28a0
--- /dev/null
+++ b/api/src/main/kotlin/router/docs/PushDocs.kt
@@ -0,0 +1,95 @@
+package com.wafflestudio.snutt.router.docs
+
+import com.wafflestudio.snutt.notification.dto.PushPreferenceDto
+import io.swagger.v3.oas.annotations.Operation
+import io.swagger.v3.oas.annotations.media.Content
+import io.swagger.v3.oas.annotations.media.ExampleObject
+import io.swagger.v3.oas.annotations.media.Schema
+import io.swagger.v3.oas.annotations.parameters.RequestBody
+import io.swagger.v3.oas.annotations.responses.ApiResponse
+import org.springdoc.core.annotations.RouterOperation
+import org.springdoc.core.annotations.RouterOperations
+import org.springframework.http.MediaType
+import org.springframework.web.bind.annotation.RequestMethod
+
+@RouterOperations(
+    RouterOperation(
+        path = "/v1/push/preferences",
+        method = [RequestMethod.GET],
+        produces = [MediaType.APPLICATION_JSON_VALUE],
+        operation =
+            Operation(
+                operationId = "getPushPreferences",
+                responses = [
+                    ApiResponse(
+                        responseCode = "200",
+                        content = [
+                            Content(
+                                schema = Schema(implementation = PushPreferenceDto::class),
+                                examples = [
+                                    ExampleObject(
+                                        value = """
+                                            {
+                                              "pushPreferences": [
+                                                {
+                                                    "type": "LECTURE_UPDATE",
+                                                    "isEnabled": "true"
+                                                },
+                                                {
+                                                    "type": "VACANCY_NOTIFICATION",
+                                                    "isEnabled": "true"
+                                                }
+                                              ]
+                                            }
+                                            """,
+                                    ),
+                                ],
+                            ),
+                        ],
+                    ),
+                ],
+            ),
+    ),
+    RouterOperation(
+        path = "/v1/push/preferences",
+        method = [RequestMethod.POST],
+        produces = [MediaType.APPLICATION_JSON_VALUE],
+        operation =
+            Operation(
+                operationId = "savePushPreferences",
+                requestBody =
+                    RequestBody(
+                        content = [
+                            Content(
+                                schema = Schema(implementation = PushPreferenceDto::class),
+                                mediaType = MediaType.APPLICATION_JSON_VALUE,
+                                examples = [
+                                    ExampleObject(
+                                        value = """
+                                            {
+                                                "pushPreferences": [
+                                                    {
+                                                        "type": "LECTURE_UPDATE",
+                                                        "isEnabled": false
+                                                    },
+                                                    {
+                                                        "type": "VACANCY_NOTIFICATION",
+                                                        "isEnabled": false
+                                                    }
+                                                ]
+                                            }
+                                        """,
+                                    ),
+                                ],
+                            ),
+                        ],
+                    ),
+                responses = [
+                    ApiResponse(
+                        responseCode = "200",
+                    ),
+                ],
+            ),
+    ),
+)
+annotation class PushDocs()