From 887c671de6b3e54a1b01af20ba31fa2e5f2103bb Mon Sep 17 00:00:00 2001 From: DongGeon0908 Date: Wed, 8 Jan 2025 22:01:25 +0900 Subject: [PATCH] =?UTF-8?q?fix:=20ws=20=EB=B0=9C=EC=86=A1=20=EB=A1=9C?= =?UTF-8?q?=EC=A7=81=20=EB=A6=AC=ED=8E=99=ED=86=A0=EB=A7=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../domain/cheer/application/CheerUpFacade.kt | 4 +- .../domain/group/application/GroupFacade.kt | 2 +- .../ReactiveGroupUserWebSocketHandler.kt | 130 +++++++++++------- .../ws/model/GroupUserEventMessage.kt | 9 +- .../alignlab/ws/service/GroupUserWsFacade.kt | 12 +- 5 files changed, 90 insertions(+), 67 deletions(-) diff --git a/src/main/kotlin/com/hero/alignlab/domain/cheer/application/CheerUpFacade.kt b/src/main/kotlin/com/hero/alignlab/domain/cheer/application/CheerUpFacade.kt index b5efaac..e933c3d 100644 --- a/src/main/kotlin/com/hero/alignlab/domain/cheer/application/CheerUpFacade.kt +++ b/src/main/kotlin/com/hero/alignlab/domain/cheer/application/CheerUpFacade.kt @@ -51,9 +51,9 @@ class CheerUpFacade( logger.error(e) { "fail to create cheerUp" } }.onSuccess { createdCheerUp -> reactiveGroupUserWebSocketHandler.launchSendEventByCheerUp( - uid = createdCheerUp.targetUid, + actorUid = user.uid, + targetUid = createdCheerUp.targetUid, groupId = otherGroupUser.groupId, - senderUid = user.uid, ) }.getOrNull()?.targetUid } diff --git a/src/main/kotlin/com/hero/alignlab/domain/group/application/GroupFacade.kt b/src/main/kotlin/com/hero/alignlab/domain/group/application/GroupFacade.kt index 7ce31cc..b9cd95a 100644 --- a/src/main/kotlin/com/hero/alignlab/domain/group/application/GroupFacade.kt +++ b/src/main/kotlin/com/hero/alignlab/domain/group/application/GroupFacade.kt @@ -348,7 +348,7 @@ class GroupFacade( private fun sendEventWithDelay(groupUserScore: GroupUserScore) { CoroutineScope(Dispatchers.IO + Job()).launch { delay(3000) - wsHandler.launchSendEvent(groupUserScore.uid, groupUserScore.groupId) + wsHandler.launchSendStatusUpdateEvent(groupUserScore.uid) } } } diff --git a/src/main/kotlin/com/hero/alignlab/ws/handler/ReactiveGroupUserWebSocketHandler.kt b/src/main/kotlin/com/hero/alignlab/ws/handler/ReactiveGroupUserWebSocketHandler.kt index 44342c3..1a08db2 100644 --- a/src/main/kotlin/com/hero/alignlab/ws/handler/ReactiveGroupUserWebSocketHandler.kt +++ b/src/main/kotlin/com/hero/alignlab/ws/handler/ReactiveGroupUserWebSocketHandler.kt @@ -17,7 +17,6 @@ import org.springframework.web.reactive.socket.WebSocketHandler import org.springframework.web.reactive.socket.WebSocketSession import reactor.core.publisher.Mono import java.util.concurrent.ConcurrentHashMap -import java.util.concurrent.ConcurrentMap @Component class ReactiveGroupUserWebSocketHandler( @@ -29,35 +28,48 @@ class ReactiveGroupUserWebSocketHandler( /** * redis는 현 상태에서 사용하지 않는다. 현재 스펙상 오버엔지니어링 + * - session 정보들을 local에 캐싱해두어 사용. * - key : groupdId * - value * - key : uid * - value : WebSocketSession */ - private val groupUserByMap: ConcurrentMap> = ConcurrentHashMap() + private val groupUserByGroupId: MutableMap> = mutableMapOf() override fun handle(session: WebSocketSession): Mono { return mono { + /** uri 정보를 파싱 */ val uriContext = GroupUserUriContext.from(session) + /** uri에 들어있는 Token 정보로, ws로 요청이 들어온 user를 조회 */ val user = authFacade.resolveAuthUser(uriContext.token) + /** user가 속해 있는 그룹 정보를 전체 조회 */ val groupUsers = groupUserService.findAllByUid(user.uid) groupUsers.forEach { groupUser -> - val targetGroupUser = groupUserByMap[groupUser.groupId] ?: ConcurrentHashMap() + /** 현재 접속중인 그룹 유저들의 정보 */ + val groupUsersByUid = groupUserByGroupId[groupUser.groupId] ?: ConcurrentHashMap() - targetGroupUser[groupUser.uid] = session + /** ws를 요청한 사용자의 Session 정볼흘 업데이트 */ + groupUsersByUid[groupUser.uid] = session - groupUserByMap[groupUser.groupId] = targetGroupUser + /** local 정보 최신화 */ + groupUserByGroupId[groupUser.groupId] = groupUsersByUid } - groupUserByMap.forEach { (groupId, sessionByUid) -> + groupUserByGroupId.forEach { (groupId, sessionByUid) -> + /** ws 요청 사용자의 세션 */ sessionByUid[user.uid] ?: return@forEach - launchSendEvent(user.uid, groupId, sessionByUid) + /** 홰당 사용자의 session에 소켓 메세지 발송 및 같은 그룹원에게 소켓 메세지 발송 */ + launchSendConnectEvent( + groupId = groupId, + sessionByUid = sessionByUid + ) } + /** ws에 대한 ping-pong 및 종료 처리 로직 */ session.receive() .map { message -> message.payloadAsText } .flatMap { payload -> checkPingPong(payload, session) } @@ -85,7 +97,7 @@ class ReactiveGroupUserWebSocketHandler( } private fun handleSessionTermination(uid: Long) { - groupUserByMap.forEach { (groupId, uidBySession) -> + groupUserByGroupId.forEach { (groupId, uidBySession) -> val removedUser = uidBySession.remove(uid) if (removedUser != null) { @@ -93,89 +105,101 @@ class ReactiveGroupUserWebSocketHandler( when (uidBySession.isEmpty()) { true -> { - groupUserByMap.remove(groupId) + groupUserByGroupId.remove(groupId) logger.info { "Removed group $groupId as it has no more users." } } false -> { - launchSendEvent(uid, groupId, uidBySession) + launchSendConnectEvent(groupId, uidBySession) } } } } } - fun launchSendEvent(uid: Long, groupId: Long) { - groupUserByMap[groupId]?.let { groupUsers -> - launchSendEvent(uid, groupId, groupUsers) + fun forceCloseAllWebSocketSessions() { + groupUserByGroupId.forEach { (_, session) -> + session.forEach { (_, session) -> + session.close().subscribe() + } } - } - fun launchSendEventByCheerUp(uid: Long, groupId: Long, senderUid: Long) { - groupUserByMap[groupId]?.let { groupUsers -> - CoroutineScope(Dispatchers.IO + Job()).launch { - val eventMessage = groupUsers.keys.toList().let { uids -> - groupUserWsFacade.generateEventMessage( - uid = uid, - groupId = groupId, - uids = uids, - cheerUpSenderUid = senderUid - ) - } + /** Websocket Session Release */ + groupUserByGroupId.clear() + } - groupUsers[uid]?.let { session -> - session - .send(Mono.just(session.textMessage(eventMessage.message()))) - .subscribe() - } - } + fun getWsGroupUsers(): List { + return groupUserByGroupId.map { (groupId, groupUsers) -> + WsGroupUserModel(groupId, groupUsers.keys) } } /** 발송되는 순서가 중요하지 않다. */ - private fun launchSendEvent( - uid: Long, + private fun launchSendConnectEvent( groupId: Long, - sessionByUid: ConcurrentMap + sessionByUid: MutableMap, ) { CoroutineScope(Dispatchers.IO + Job()).launch { - sendUpdatedGroupStatus(uid, groupId, sessionByUid) + sendConnectEvent( + groupId = groupId, + sessionByUid = sessionByUid + ) } } - private suspend fun sendUpdatedGroupStatus( - uid: Long, + private suspend fun sendConnectEvent( groupId: Long, sessionByUid: MutableMap ) { - val eventMessage = sessionByUid.keys - .toList() - .let { uids -> groupUserWsFacade.generateEventMessage(uid, groupId, uids) } + sessionByUid.forEach { (uid, session) -> + val eventMessage = groupUserWsFacade.generateEventMessage( + uid = uid, + groupId = groupId, + spreadUids = sessionByUid.keys.toList() + ) - sessionByUid.forEach { (_, session) -> session .send(Mono.just(session.textMessage(eventMessage.message()))) .subscribe() } } - fun forceCloseAllWebSocketSessions() { - groupUserByMap.forEach { (_, session) -> - session.forEach { (_, session) -> + fun launchSendStatusUpdateEvent(groupId: Long) { + val sessions = groupUserByGroupId[groupId] ?: return + + launchSendConnectEvent(groupId, sessions) + } + + /** 응원을 보낸 사람과 받은 사람에게 WS 알림 진행 */ + fun launchSendEventByCheerUp(actorUid: Long, targetUid: Long, groupId: Long) { + CoroutineScope(Dispatchers.IO + Job()).launch { + /** 응원하기를 누른 action 대상자 */ + groupUserByGroupId[groupId]?.get(actorUid)?.let { session -> + val eventMessage = groupUserWsFacade.generateEventMessage( + uid = actorUid, + groupId = groupId, + spreadUids = listOf(actorUid, targetUid), + cheerUpSenderUid = actorUid, + ) + session - .close() + .send(Mono.just(session.textMessage(eventMessage.message()))) .subscribe() } - } - /** Websocket Session Release */ - groupUserByMap.clear() - } + /** 응원하기를 받은 대상자 */ + groupUserByGroupId[groupId]?.get(targetUid)?.let { session -> + val eventMessage = groupUserWsFacade.generateEventMessage( + uid = targetUid, + groupId = groupId, + spreadUids = listOf(actorUid, targetUid), + cheerUpSenderUid = actorUid, + ) - fun getWsGroupUsers(): List { - return groupUserByMap.map { - WsGroupUserModel(it.key, it.value.keys) + session + .send(Mono.just(session.textMessage(eventMessage.message()))) + .subscribe() + } } } } - diff --git a/src/main/kotlin/com/hero/alignlab/ws/model/GroupUserEventMessage.kt b/src/main/kotlin/com/hero/alignlab/ws/model/GroupUserEventMessage.kt index 9d4ecdc..f7ab1c8 100644 --- a/src/main/kotlin/com/hero/alignlab/ws/model/GroupUserEventMessage.kt +++ b/src/main/kotlin/com/hero/alignlab/ws/model/GroupUserEventMessage.kt @@ -1,7 +1,6 @@ package com.hero.alignlab.ws.model import com.hero.alignlab.common.extension.mapper -import com.hero.alignlab.domain.cheer.domain.CheerUp import com.hero.alignlab.domain.group.domain.GroupUser import com.hero.alignlab.domain.group.domain.GroupUserScore import com.hero.alignlab.domain.user.domain.UserInfo @@ -29,9 +28,9 @@ data class GroupUserEventMessage( /** 나에게 응원하기를 보낸 사용자의 uid */ val senderUid: Long?, /** 금일 받은 응원하기 수 */ - val countCheeredUp: Long?, + val countCheeredUp: Long, /** 내가 응원을 보낸 사용자 목록 */ - val sentUids: List?, + val sentUids: List, ) companion object { @@ -43,7 +42,7 @@ data class GroupUserEventMessage( scoreByUid: Map, cheerUpSenderUid: Long?, countCheeredUp: Long, - cheerUpsByTargetUid: Map>, + cheerUpsByTargetUid: List, ): GroupUserEventMessage { val rank = AtomicInteger(1) @@ -70,7 +69,7 @@ data class GroupUserEventMessage( cheerUp = CheerUpModel( senderUid = cheerUpSenderUid, countCheeredUp = countCheeredUp, - sentUids = cheerUpsByTargetUid[uid]?.map { cheerUp -> cheerUp.targetUid } + sentUids = cheerUpsByTargetUid, ) ) } diff --git a/src/main/kotlin/com/hero/alignlab/ws/service/GroupUserWsFacade.kt b/src/main/kotlin/com/hero/alignlab/ws/service/GroupUserWsFacade.kt index f3b5ea1..d5b72d2 100644 --- a/src/main/kotlin/com/hero/alignlab/ws/service/GroupUserWsFacade.kt +++ b/src/main/kotlin/com/hero/alignlab/ws/service/GroupUserWsFacade.kt @@ -19,17 +19,17 @@ class GroupUserWsFacade( suspend fun generateEventMessage( uid: Long, groupId: Long, - uids: List, + spreadUids: List, cheerUpSenderUid: Long? = null, ): GroupUserEventMessage { val now = LocalDate.now() return parZip( - { userInfoService.findAllByIds(uids) }, - { groupUserService.findAllByGroupIdAndUids(groupId, uids) }, - { groupUserScoreService.findAllByGroupIdAndUids(groupId, uids) }, + { userInfoService.findAllByIds(spreadUids) }, + { groupUserService.findAllByGroupIdAndUids(groupId, spreadUids) }, + { groupUserScoreService.findAllByGroupIdAndUids(groupId, spreadUids) }, { cheerUpService.countAllByCheeredAtAndUid(now, uid) }, - { cheerUpService.findAllByTargetUidInAndCheeredAt(uids.toSet(), now) } + { cheerUpService.findAllByUidAndCheeredAt(uid, now) }, ) { userInfoByUid, groupUsers, groupUserScores, countCheeredUp, cheerUps -> GroupUserEventMessage.of( uid = uid, @@ -39,7 +39,7 @@ class GroupUserWsFacade( scoreByUid = groupUserScores.associateBy { score -> score.uid }, cheerUpSenderUid = cheerUpSenderUid, countCheeredUp = countCheeredUp, - cheerUpsByTargetUid = cheerUps.groupBy { cheerUp -> cheerUp.targetUid }, + cheerUpsByTargetUid = cheerUps.map { cheerUp -> cheerUp.targetUid } ) } }