feat(home): 추천 크리에이터 동시 팔로우 API를 추가한다

This commit is contained in:
2026-06-01 10:19:49 +09:00
parent 8300b1875c
commit cdff31422c
3 changed files with 259 additions and 0 deletions

View File

@@ -0,0 +1,40 @@
package kr.co.vividnext.sodalive.v2.api.home.adapter.`in`.web
import kr.co.vividnext.sodalive.common.ApiResponse
import kr.co.vividnext.sodalive.common.SodaException
import kr.co.vividnext.sodalive.member.Member
import kr.co.vividnext.sodalive.v2.api.home.dto.FollowRecommendedCreatorsRequest
import kr.co.vividnext.sodalive.v2.recommend.application.RecommendedCreatorFollowService
import org.springframework.security.core.annotation.AuthenticationPrincipal
import org.springframework.web.bind.annotation.PostMapping
import org.springframework.web.bind.annotation.RequestBody
import org.springframework.web.bind.annotation.RequestMapping
import org.springframework.web.bind.annotation.RestController
@RestController
@RequestMapping("/api/v2/home/recommendations")
class HomeRecommendationController(
private val recommendedCreatorFollowService: RecommendedCreatorFollowService
) {
@PostMapping("/creators/follow")
fun followRecommendedCreators(
@RequestBody request: FollowRecommendedCreatorsRequest,
@AuthenticationPrincipal(expression = "#this == 'anonymousUser' ? null : member") member: Member?
) = run {
if (member == null) throw SodaException(messageKey = "common.error.bad_credentials")
val creatorIds = request.creatorIds
if (creatorIds.isNullOrEmpty() || creatorIds.size > MAX_CREATOR_IDS) {
throw SodaException(messageKey = "common.error.invalid_request")
}
recommendedCreatorFollowService.followCreators(
member = member,
creatorIds = creatorIds
)
ApiResponse.ok<Unit>()
}
companion object {
private const val MAX_CREATOR_IDS = 50
}
}

View File

@@ -0,0 +1,5 @@
package kr.co.vividnext.sodalive.v2.api.home.dto
data class FollowRecommendedCreatorsRequest(
val creatorIds: List<Long>?
)

View File

@@ -0,0 +1,214 @@
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.following.CreatorFollowing
import kr.co.vividnext.sodalive.member.following.CreatorFollowingRepository
import kr.co.vividnext.sodalive.support.EmbeddedRedisInitializer
import org.junit.jupiter.api.Assertions.assertEquals
import org.junit.jupiter.api.Assertions.assertNotNull
import org.junit.jupiter.api.Assertions.assertTrue
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.http.MediaType
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.post
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 javax.persistence.EntityManager
@SpringBootTest
@AutoConfigureMockMvc
@Transactional
@ContextConfiguration(initializers = [EmbeddedRedisInitializer::class])
class HomeRecommendationControllerTest @Autowired constructor(
private val mockMvc: MockMvc,
private val memberRepository: MemberRepository,
private val creatorFollowingRepository: CreatorFollowingRepository,
private val entityManager: EntityManager
) {
@Test
@DisplayName("추천 크리에이터 동시 팔로우 비로그인 요청은 Spring Security에서 거부한다")
fun shouldRejectAnonymousFollowRequest() {
mockMvc.perform(
post("/api/v2/home/recommendations/creators/follow")
.contentType(MediaType.APPLICATION_JSON)
.content("""{"creatorIds":[1,2]}""")
)
.andExpect(status().isUnauthorized)
}
@Test
@DisplayName("추천 크리에이터 동시 팔로우 성공 응답은 id 목록 없이 성공 여부만 반환하고 신규 팔로우만 저장한다")
fun shouldReturnSuccessOnlyAndPersistOnlyNewFollows() {
val member = saveMember("viewer", MemberRole.USER)
val newCreator = saveMember("new-creator", MemberRole.CREATOR)
val followedCreator = saveMember("followed-creator", MemberRole.CREATOR)
saveFollowing(member = member, creator = followedCreator)
entityManager.flush()
entityManager.clear()
mockMvc.perform(
post("/api/v2/home/recommendations/creators/follow")
.with(user(MemberAdapter(member)))
.contentType(MediaType.APPLICATION_JSON)
.content("""{"creatorIds":[${newCreator.id},${followedCreator.id},${member.id}]}""")
)
.andExpect(status().isOk)
.andExpect(jsonPath("$.success").value(true))
.andExpect(jsonPath("$.data").doesNotExist())
entityManager.flush()
entityManager.clear()
assertNotNull(creatorFollowingRepository.findByCreatorIdAndMemberId(newCreator.id!!, member.id!!))
assertNotNull(creatorFollowingRepository.findByCreatorIdAndMemberId(followedCreator.id!!, member.id!!))
assertEquals(2, creatorFollowingRepository.findAll().size)
}
@Test
@DisplayName("추천 크리에이터 동시 팔로우는 비활성 팔로우 이력을 신규 row 없이 다시 활성화한다")
fun shouldReactivateInactiveFollowingThroughApi() {
val member = saveMember("viewer", MemberRole.USER)
val creator = saveMember("reactivate-creator", MemberRole.CREATOR)
val inactiveFollowing = saveFollowing(member = member, creator = creator).apply {
isNotify = false
isActive = false
}
entityManager.flush()
entityManager.clear()
mockMvc.perform(
post("/api/v2/home/recommendations/creators/follow")
.with(user(MemberAdapter(member)))
.contentType(MediaType.APPLICATION_JSON)
.content("""{"creatorIds":[${creator.id}]}""")
)
.andExpect(status().isOk)
.andExpect(jsonPath("$.success").value(true))
.andExpect(jsonPath("$.data").doesNotExist())
entityManager.flush()
entityManager.clear()
val reactivatedFollowing = creatorFollowingRepository.findByCreatorIdAndMemberId(creator.id!!, member.id!!)
assertNotNull(reactivatedFollowing)
assertEquals(inactiveFollowing.id, reactivatedFollowing!!.id)
assertTrue(reactivatedFollowing.isNotify)
assertTrue(reactivatedFollowing.isActive)
assertEquals(1, creatorFollowingRepository.findAll().size)
}
@Test
@DisplayName("추천 크리에이터 동시 팔로우 요청에 유효하지 않은 id가 있으면 실패하고 신규 저장하지 않는다")
fun shouldFailAndSaveNothingWhenInvalidCreatorIdIsIncluded() {
val member = saveMember("viewer", MemberRole.USER)
val validCreator = saveMember("valid-creator", MemberRole.CREATOR)
entityManager.flush()
entityManager.clear()
mockMvc.perform(
post("/api/v2/home/recommendations/creators/follow")
.with(user(MemberAdapter(member)))
.contentType(MediaType.APPLICATION_JSON)
.content("""{"creatorIds":[${validCreator.id},999999]}""")
)
.andExpect(status().isOk)
.andExpect(jsonPath("$.success").value(false))
.andExpect(jsonPath("$.message").value("크리에이터 정보를 확인해주세요."))
entityManager.flush()
entityManager.clear()
assertEquals(0, creatorFollowingRepository.findAll().size)
}
@Test
@DisplayName("추천 크리에이터 동시 팔로우 요청 creatorIds가 비어 있으면 실패하고 신규 저장하지 않는다")
fun shouldRejectEmptyCreatorIdsAndSaveNothing() {
val member = saveMember("viewer", MemberRole.USER)
entityManager.flush()
entityManager.clear()
mockMvc.perform(
post("/api/v2/home/recommendations/creators/follow")
.with(user(MemberAdapter(member)))
.contentType(MediaType.APPLICATION_JSON)
.content("""{"creatorIds":[]}""")
)
.andExpect(status().isOk)
.andExpect(jsonPath("$.success").value(false))
entityManager.flush()
entityManager.clear()
assertEquals(0, creatorFollowingRepository.findAll().size)
}
@Test
@DisplayName("추천 크리에이터 동시 팔로우 요청 creatorIds가 null이면 실패하고 신규 저장하지 않는다")
fun shouldRejectNullCreatorIdsAndSaveNothing() {
val member = saveMember("viewer", MemberRole.USER)
entityManager.flush()
entityManager.clear()
mockMvc.perform(
post("/api/v2/home/recommendations/creators/follow")
.with(user(MemberAdapter(member)))
.contentType(MediaType.APPLICATION_JSON)
.content("""{"creatorIds":null}""")
)
.andExpect(status().isOk)
.andExpect(jsonPath("$.success").value(false))
entityManager.flush()
entityManager.clear()
assertEquals(0, creatorFollowingRepository.findAll().size)
}
@Test
@DisplayName("추천 크리에이터 동시 팔로우 요청 creatorIds가 50개를 초과하면 실패하고 신규 저장하지 않는다")
fun shouldRejectTooManyCreatorIdsAndSaveNothing() {
val member = saveMember("viewer", MemberRole.USER)
val creatorIds = (1..51).joinToString(",") { it.toString() }
entityManager.flush()
entityManager.clear()
mockMvc.perform(
post("/api/v2/home/recommendations/creators/follow")
.with(user(MemberAdapter(member)))
.contentType(MediaType.APPLICATION_JSON)
.content("""{"creatorIds":[$creatorIds]}""")
)
.andExpect(status().isOk)
.andExpect(jsonPath("$.success").value(false))
entityManager.flush()
entityManager.clear()
assertEquals(0, creatorFollowingRepository.findAll().size)
}
private fun saveMember(seed: String, role: MemberRole): Member {
return memberRepository.saveAndFlush(
Member(
email = "$seed@test.com",
password = "password",
nickname = seed,
role = role
)
)
}
private fun saveFollowing(member: Member, creator: Member): CreatorFollowing {
return creatorFollowingRepository.saveAndFlush(
CreatorFollowing().apply {
this.member = member
this.creator = creator
}
)
}
}