feat(ranking): 크리에이터 랭킹 홈 API를 추가한다
This commit is contained in:
@@ -102,6 +102,7 @@ class SecurityConfig(
|
|||||||
.antMatchers(HttpMethod.GET, "/api/chat/original/list").permitAll()
|
.antMatchers(HttpMethod.GET, "/api/chat/original/list").permitAll()
|
||||||
.antMatchers(HttpMethod.POST, "/charge/payverse/webhook").permitAll()
|
.antMatchers(HttpMethod.POST, "/charge/payverse/webhook").permitAll()
|
||||||
.antMatchers(HttpMethod.GET, "/api/v2/home/recommendations").permitAll()
|
.antMatchers(HttpMethod.GET, "/api/v2/home/recommendations").permitAll()
|
||||||
|
.antMatchers(HttpMethod.GET, "/api/v2/home/rankings/creators").permitAll()
|
||||||
// 페이지네이션 하위 경로(/lives, /debut-creators 등)는 인증 필수
|
// 페이지네이션 하위 경로(/lives, /debut-creators 등)는 인증 필수
|
||||||
.antMatchers(HttpMethod.GET, "/api/v2/home/recommendations/**").authenticated()
|
.antMatchers(HttpMethod.GET, "/api/v2/home/recommendations/**").authenticated()
|
||||||
.anyRequest().authenticated()
|
.anyRequest().authenticated()
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
package kr.co.vividnext.sodalive.v2.api.home.adapter.`in`.web
|
||||||
|
|
||||||
|
import kr.co.vividnext.sodalive.common.ApiResponse
|
||||||
|
import kr.co.vividnext.sodalive.member.Member
|
||||||
|
import kr.co.vividnext.sodalive.v2.api.home.application.HomeCreatorRankingFacade
|
||||||
|
import org.springframework.security.core.annotation.AuthenticationPrincipal
|
||||||
|
import org.springframework.web.bind.annotation.GetMapping
|
||||||
|
import org.springframework.web.bind.annotation.RequestMapping
|
||||||
|
import org.springframework.web.bind.annotation.RestController
|
||||||
|
|
||||||
|
@RestController
|
||||||
|
@RequestMapping("/api/v2/home/rankings")
|
||||||
|
class CreatorRankingController(
|
||||||
|
private val homeCreatorRankingFacade: HomeCreatorRankingFacade
|
||||||
|
) {
|
||||||
|
@GetMapping("/creators")
|
||||||
|
fun getCreatorRankings(
|
||||||
|
@AuthenticationPrincipal(expression = "#this == 'anonymousUser' ? null : member") member: Member?
|
||||||
|
) = run {
|
||||||
|
ApiResponse.ok(homeCreatorRankingFacade.getCreatorRankings(member))
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
package kr.co.vividnext.sodalive.v2.api.home.application
|
||||||
|
|
||||||
|
import kr.co.vividnext.sodalive.member.Member
|
||||||
|
import kr.co.vividnext.sodalive.v2.api.home.dto.ranking.CreatorRankingResponse
|
||||||
|
import kr.co.vividnext.sodalive.v2.ranking.application.CreatorRankingQueryService
|
||||||
|
import org.springframework.stereotype.Component
|
||||||
|
|
||||||
|
@Component
|
||||||
|
class HomeCreatorRankingFacade(
|
||||||
|
private val creatorRankingQueryService: CreatorRankingQueryService
|
||||||
|
) {
|
||||||
|
fun getCreatorRankings(member: Member?): CreatorRankingResponse {
|
||||||
|
return CreatorRankingResponse.from(
|
||||||
|
creatorRankingQueryService.getCreatorRankings(viewerMemberId = member?.id)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
package kr.co.vividnext.sodalive.v2.api.home.dto.ranking
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty
|
||||||
|
import kr.co.vividnext.sodalive.v2.ranking.application.CreatorRankingResult
|
||||||
|
import kr.co.vividnext.sodalive.v2.ranking.domain.CreatorRankingItem
|
||||||
|
|
||||||
|
data class CreatorRankingResponse(
|
||||||
|
val showRankChange: Boolean,
|
||||||
|
val items: List<CreatorRankingResponseItem>
|
||||||
|
) {
|
||||||
|
companion object {
|
||||||
|
fun from(result: CreatorRankingResult): CreatorRankingResponse {
|
||||||
|
return CreatorRankingResponse(
|
||||||
|
showRankChange = result.showRankChange,
|
||||||
|
items = result.items.map { CreatorRankingResponseItem.from(it) }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
data class CreatorRankingResponseItem(
|
||||||
|
val rank: Int,
|
||||||
|
val rankChange: Int?,
|
||||||
|
@JsonProperty("isNew")
|
||||||
|
val isNew: Boolean,
|
||||||
|
val creatorId: Long,
|
||||||
|
val nickname: String,
|
||||||
|
val profileImageUrl: String?
|
||||||
|
) {
|
||||||
|
companion object {
|
||||||
|
fun from(item: CreatorRankingItem): CreatorRankingResponseItem {
|
||||||
|
return CreatorRankingResponseItem(
|
||||||
|
rank = item.rank,
|
||||||
|
rankChange = item.rankChange,
|
||||||
|
isNew = item.isNew,
|
||||||
|
creatorId = item.creatorId,
|
||||||
|
nickname = item.nickname,
|
||||||
|
profileImageUrl = item.profileImageUrl
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,151 @@
|
|||||||
|
package kr.co.vividnext.sodalive.v2.api.home
|
||||||
|
|
||||||
|
import kr.co.vividnext.sodalive.member.Member
|
||||||
|
import kr.co.vividnext.sodalive.member.MemberAdapter
|
||||||
|
import kr.co.vividnext.sodalive.member.MemberRepository
|
||||||
|
import kr.co.vividnext.sodalive.member.MemberRole
|
||||||
|
import kr.co.vividnext.sodalive.member.block.BlockMember
|
||||||
|
import kr.co.vividnext.sodalive.support.EmbeddedRedisInitializer
|
||||||
|
import kr.co.vividnext.sodalive.v2.ranking.adapter.out.persistence.CreatorRankingSnapshot
|
||||||
|
import org.junit.jupiter.api.DisplayName
|
||||||
|
import org.junit.jupiter.api.Test
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired
|
||||||
|
import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc
|
||||||
|
import org.springframework.boot.test.context.SpringBootTest
|
||||||
|
import org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.user
|
||||||
|
import org.springframework.test.context.ContextConfiguration
|
||||||
|
import org.springframework.test.web.servlet.MockMvc
|
||||||
|
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get
|
||||||
|
import org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath
|
||||||
|
import org.springframework.test.web.servlet.result.MockMvcResultMatchers.status
|
||||||
|
import org.springframework.transaction.annotation.Transactional
|
||||||
|
import java.time.LocalDateTime
|
||||||
|
import javax.persistence.EntityManager
|
||||||
|
|
||||||
|
@SpringBootTest
|
||||||
|
@AutoConfigureMockMvc
|
||||||
|
@Transactional
|
||||||
|
@ContextConfiguration(initializers = [EmbeddedRedisInitializer::class])
|
||||||
|
class CreatorRankingControllerTest @Autowired constructor(
|
||||||
|
private val mockMvc: MockMvc,
|
||||||
|
private val memberRepository: MemberRepository,
|
||||||
|
private val entityManager: EntityManager
|
||||||
|
) {
|
||||||
|
@Test
|
||||||
|
@DisplayName("크리에이터 랭킹 조회는 허용된 응답 필드만 반환하고 점수와 기간은 노출하지 않는다")
|
||||||
|
fun shouldReturnCreatorRankingSchemaWithoutScoreAndPeriodFields() {
|
||||||
|
saveSnapshot(
|
||||||
|
creatorId = 1L,
|
||||||
|
nickname = "creator-one",
|
||||||
|
profileImageUrl = "profile-one.png",
|
||||||
|
finalScore = 100.0
|
||||||
|
)
|
||||||
|
entityManager.flush()
|
||||||
|
entityManager.clear()
|
||||||
|
|
||||||
|
mockMvc.perform(get("/api/v2/home/rankings/creators"))
|
||||||
|
.andExpect(status().isOk)
|
||||||
|
.andExpect(jsonPath("$.success").value(true))
|
||||||
|
.andExpect(jsonPath("$.data.showRankChange").value(false))
|
||||||
|
.andExpect(jsonPath("$.data.items[0].rank").value(1))
|
||||||
|
.andExpect(jsonPath("$.data.items[0].rankChange").doesNotExist())
|
||||||
|
.andExpect(jsonPath("$.data.items[0].isNew").value(false))
|
||||||
|
.andExpect(jsonPath("$.data.items[0].creatorId").value(1L))
|
||||||
|
.andExpect(jsonPath("$.data.items[0].nickname").value("creator-one"))
|
||||||
|
.andExpect(jsonPath("$.data.items[0].profileImageUrl").value("profile-one.png"))
|
||||||
|
.andExpect(jsonPath("$.data.items[0].finalScore").doesNotExist())
|
||||||
|
.andExpect(jsonPath("$.data.items[0].aggregationStartAtUtc").doesNotExist())
|
||||||
|
.andExpect(jsonPath("$.data.items[0].aggregationEndAtUtc").doesNotExist())
|
||||||
|
.andExpect(jsonPath("$.data.aggregationStartAtUtc").doesNotExist())
|
||||||
|
.andExpect(jsonPath("$.data.aggregationEndAtUtc").doesNotExist())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@DisplayName("크리에이터 랭킹 조회는 비회원도 호출 가능하고 빈 랭킹을 성공 응답으로 반환한다")
|
||||||
|
fun shouldReturnEmptyCreatorRankingsForAnonymous() {
|
||||||
|
mockMvc.perform(get("/api/v2/home/rankings/creators"))
|
||||||
|
.andExpect(status().isOk)
|
||||||
|
.andExpect(jsonPath("$.success").value(true))
|
||||||
|
.andExpect(jsonPath("$.data.showRankChange").value(false))
|
||||||
|
.andExpect(jsonPath("$.data.items").isArray)
|
||||||
|
.andExpect(jsonPath("$.data.items.length()").value(0))
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@DisplayName("크리에이터 랭킹 조회는 인증 회원 id를 전달해 차단 크리에이터 정보를 마스킹한다")
|
||||||
|
fun shouldMaskBlockedCreatorForAuthenticatedMember() {
|
||||||
|
val viewer = saveMember("ranking-viewer", MemberRole.USER)
|
||||||
|
val blockedCreator = saveMember("blocked-creator", MemberRole.CREATOR)
|
||||||
|
saveSnapshot(
|
||||||
|
creatorId = blockedCreator.id!!,
|
||||||
|
nickname = blockedCreator.nickname,
|
||||||
|
profileImageUrl = "blocked-profile.png",
|
||||||
|
finalScore = 100.0
|
||||||
|
)
|
||||||
|
saveBlock(member = viewer, blockedMember = blockedCreator)
|
||||||
|
entityManager.flush()
|
||||||
|
entityManager.clear()
|
||||||
|
|
||||||
|
mockMvc.perform(
|
||||||
|
get("/api/v2/home/rankings/creators")
|
||||||
|
.with(user(MemberAdapter(viewer)))
|
||||||
|
)
|
||||||
|
.andExpect(status().isOk)
|
||||||
|
.andExpect(jsonPath("$.success").value(true))
|
||||||
|
.andExpect(jsonPath("$.data.items[0].rank").value(1))
|
||||||
|
.andExpect(jsonPath("$.data.items[0].creatorId").value(0L))
|
||||||
|
.andExpect(jsonPath("$.data.items[0].nickname").value(""))
|
||||||
|
.andExpect(jsonPath("$.data.items[0].profileImageUrl").value("/profile/default-profile.png"))
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun saveMember(seed: String, role: MemberRole): Member {
|
||||||
|
return memberRepository.saveAndFlush(
|
||||||
|
Member(
|
||||||
|
email = "$seed@test.com",
|
||||||
|
password = "password",
|
||||||
|
nickname = seed,
|
||||||
|
role = role
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun saveBlock(member: Member, blockedMember: Member) {
|
||||||
|
entityManager.persist(
|
||||||
|
BlockMember().apply {
|
||||||
|
this.member = member
|
||||||
|
this.blockedMember = blockedMember
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun saveSnapshot(
|
||||||
|
creatorId: Long,
|
||||||
|
nickname: String,
|
||||||
|
profileImageUrl: String?,
|
||||||
|
finalScore: Double
|
||||||
|
) {
|
||||||
|
entityManager.persist(
|
||||||
|
CreatorRankingSnapshot(
|
||||||
|
aggregationStartAtUtc = LocalDateTime.of(2026, 5, 31, 15, 0, 0),
|
||||||
|
aggregationEndAtUtc = LocalDateTime.of(2026, 6, 7, 15, 0, 0),
|
||||||
|
creatorId = creatorId,
|
||||||
|
nickname = nickname,
|
||||||
|
profileImageUrl = profileImageUrl,
|
||||||
|
finalScore = finalScore,
|
||||||
|
contentLiveScore = 0.0,
|
||||||
|
engagementScore = 0.0,
|
||||||
|
supportScore = 0.0,
|
||||||
|
fanLoyaltyScore = 0.0,
|
||||||
|
liveCanAmount = 0,
|
||||||
|
contentPurchaseCanAmount = 0,
|
||||||
|
contentLikeCount = 0,
|
||||||
|
contentCommentCount = 0,
|
||||||
|
channelDonationCanAmount = 0,
|
||||||
|
channelDonationCount = 0,
|
||||||
|
fanTalkCount = 0,
|
||||||
|
finalFollowerCount = 0,
|
||||||
|
followIncrease = 0
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user