회사에서 단위 테스트 작성 시, 가독성이 너무 떨어져 알아보기 힘들 경우가 종종 있었다. 어떻게 하면 조금 더 예쁘게 코드를 작성할 수 있을 지 간단하게 정리해보겠다.
class RegisterEmoticonFeature(
_emoticonRepository: EmoticonRepository
) : BehaviorSpec() {
private val emoticonRepository = spyk(_emoticonRepository)
private val accountService = mockk<AccountService>()
private val emoticonService = EmoticonService(repository = emoticonRepository)
private val emoticonHandler = EmoticonHandler(
accountService = accountService,
emoticonService = emoticonService,
emailService = mockk()
)
override fun isolationMode(): IsolationMode = InstancePerLeaf
init {
}
}
object Mock {
fun account(identified: Boolean) = Account(
id = Arb.long(min = 1).single(),
name = Arb.string(5..20).single(),
email = Arb.stringPattern("([a-zA-Z0-9]{5,20})\\@test\\.kakao\\.com").single(),
identified = identified
)
fun emoticonEntity(createdDate: LocalDate, zone: ZoneId) = arb { rs ->
val accountId = Arb.long(100_000L..200_000L)
val title = Arb.string(10..100)
val description = Arb.string(100..300)
val choco = Arb.int(100..500)
val images = Arb.stringPattern("([a-zA-Z0-9]{1,10})/([a-zA-Z0-9]{1,10})\\.jpg")
.chunked(1..10)
val created = Arb.instant(
minValue = createdDate.atTime(LocalTime.MIN).atZone(zone).toInstant(),
maxValue = createdDate.atTime(LocalTime.MAX).atZone(zone).toInstant()
)
generateSequence {
created.next(rs).let { createdAt ->
EmoticonEntity(
accountId = accountId.next(rs),
title = title.next(rs),
description = description.next(rs),
choco = choco.next(rs),
images = images.next(rs),
createdAt = createdAt,
updatedAt = createdAt
)
}
}
}
}
Mock.kt
객체를 통해 아래처럼 Given
절에 정의할 수 있다.init {
Given("본인 인증된 사용자가 로그인된 상황에서") {
val token = Arb.stringPattern("([a-zA-Z0-9]{20})").single()
val account = Mock.account(identified = true) //
every { accountService.take(ofType(String::class)) } returns account
'''
}
}
@ContextConfiguration(classes = [SpringDataConfig::class])
class RegisterEmoticonFeature(
_emoticonRepository: EmoticonRepository
) : BehaviorSpec() {
private val emoticonRepository = spyk(_emoticonRepository)
private val accountService = mockk<AccountService>() // MSA
private val emoticonService = EmoticonService(repository = emoticonRepository)
private val emoticonHandler = EmoticonHandler(
accountService = accountService,
emoticonService = emoticonService,
emailService = mockk()
)
override fun isolationMode(): IsolationMode = InstancePerLeaf
init {
Given("본인 인증된 사용자가 로그인된 상황에서") {
val token = Arb.stringPattern("([a-zA-Z0-9]{20})").single()
val account = Mock.account(identified = true)
every { accountService.take(ofType(String::class)) } returns account
When("검수 정보를 입력란에") {
And("검수 정보를 입력하고 검수 등록 버튼을 누르면") {
val request = request()
val response = emoticonHandler.register(token, request)
Then("등록 결과가 포함된 검수 진행 목록 화면으로 이동한다") {
response.authorId shouldBe account.id
response.title shouldBe request.title
response.description shouldBe request.description
response.choco shouldBe request.choco
response.images shouldContainAll request.images
verify(exactly = 1) {
accountService.take(token = token)
emoticonRepository.save(match<EmoticonEntity> {
it.accountId == account.id
})
}
}
}
And("검수 정보를 입력하지 않고 검수 등록 버튼을 누르면") {
val ex = shouldThrowExactly<DeniedRegisterEmoticonException> {
emoticonHandler.register(token, null)
}
Then("검수 등록 실패 사유가 화면에 표시되어야 한다") {
ex.message.isNotBlank() shouldBe true
verify(exactly = 0) {
accountService.take(any())
emoticonRepository.save(ofType<EmoticonEntity>())
}
}
}
}
}
}
private fun request() = RegisterEmoticon(
title = Arb.string(10..100).single(),
description = Arb.string(250..300).single(),
choco = Arb.int(100..500).single(),
images = Arb.stringPattern("([a-zA-Z0-9]{1,10})/([a-zA-Z0-9]{1,10})\\.jpg")
.chunked(1..10)
.single()
)
}
Spec
beforeSpec, AfterSpec 메서드로 LifeCycle을 관리할 수 있다.Container
, beforeContainer, afterContainer 메서드로 LifeCycle을 관리할 수 있다.Each
, beforeEach, afterEach라는 메서드로 LifeCycle을 관리할 수 있다.https://github.com/harry-jk/ifkakao-2020-code/tree/master
kotest, mockk 공식 문서