generated from aura-ascend/template-service
add TransactionController
This commit is contained in:
8
http/transaction.http
Normal file
8
http/transaction.http
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
### Deposit
|
||||||
|
POST {{url}}/deposit
|
||||||
|
Content-Type: application/json
|
||||||
|
|
||||||
|
{
|
||||||
|
"account": "00000000-0000-0000-0000-000000000000",
|
||||||
|
"amount": 1.23
|
||||||
|
}
|
||||||
40
src/main/kotlin/ltd/lulz/controller/TransactionController.kt
Normal file
40
src/main/kotlin/ltd/lulz/controller/TransactionController.kt
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
package ltd.lulz.controller
|
||||||
|
|
||||||
|
import io.github.oshai.kotlinlogging.KotlinLogging
|
||||||
|
import jakarta.validation.Valid
|
||||||
|
import ltd.lulz.exception.AccountNotFoundException
|
||||||
|
import ltd.lulz.model.Transaction
|
||||||
|
import ltd.lulz.service.TransactionService
|
||||||
|
import org.springframework.http.HttpStatus.CREATED
|
||||||
|
import org.springframework.http.HttpStatus.INTERNAL_SERVER_ERROR
|
||||||
|
import org.springframework.http.HttpStatus.NOT_FOUND
|
||||||
|
import org.springframework.validation.annotation.Validated
|
||||||
|
import org.springframework.web.bind.annotation.PostMapping
|
||||||
|
import org.springframework.web.bind.annotation.RequestBody
|
||||||
|
import org.springframework.web.bind.annotation.ResponseStatus
|
||||||
|
import org.springframework.web.bind.annotation.RestController
|
||||||
|
import org.springframework.web.server.ResponseStatusException
|
||||||
|
import reactor.core.publisher.Mono
|
||||||
|
|
||||||
|
private val log = KotlinLogging.logger {}
|
||||||
|
|
||||||
|
@Validated
|
||||||
|
@RestController
|
||||||
|
class TransactionController(
|
||||||
|
private val transactionService: TransactionService,
|
||||||
|
) {
|
||||||
|
|
||||||
|
@PostMapping("/deposit")
|
||||||
|
@ResponseStatus(CREATED)
|
||||||
|
fun deposit(
|
||||||
|
@Valid @RequestBody request: Transaction.Request,
|
||||||
|
): Mono<Void> = transactionService.deposit(request.account, request.amount)
|
||||||
|
.onErrorResume {
|
||||||
|
when (it) {
|
||||||
|
is AccountNotFoundException -> Mono.error(ResponseStatusException(NOT_FOUND))
|
||||||
|
else -> Mono.error(ResponseStatusException(INTERNAL_SERVER_ERROR))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.doOnError { log.warn { "deposit account ${request.account}: " + it.localizedMessage } }
|
||||||
|
.then()
|
||||||
|
}
|
||||||
105
src/test/kotlin/ltd/lulz/controller/TransactionControllerTest.kt
Normal file
105
src/test/kotlin/ltd/lulz/controller/TransactionControllerTest.kt
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
package ltd.lulz.controller
|
||||||
|
|
||||||
|
import io.mockk.every
|
||||||
|
import io.mockk.mockk
|
||||||
|
import java.math.BigDecimal
|
||||||
|
import java.util.UUID
|
||||||
|
import ltd.lulz.controller.AccountControllerTest.Companion.name
|
||||||
|
import ltd.lulz.exception.AccountNotFoundException
|
||||||
|
import ltd.lulz.model.AccountEntity
|
||||||
|
import ltd.lulz.model.Transaction
|
||||||
|
import ltd.lulz.service.TransactionService
|
||||||
|
import org.junit.jupiter.api.BeforeEach
|
||||||
|
import org.junit.jupiter.api.Test
|
||||||
|
import org.springframework.http.MediaType.APPLICATION_JSON
|
||||||
|
import org.springframework.test.web.reactive.server.WebTestClient
|
||||||
|
import reactor.core.publisher.Mono
|
||||||
|
|
||||||
|
@Suppress("ReactiveStreamsUnusedPublisher")
|
||||||
|
class TransactionControllerTest {
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
val amount: BigDecimal = BigDecimal.valueOf(0.01)
|
||||||
|
val uuid: UUID = UUID.fromString("00000000-0000-0000-0000-000000000000")
|
||||||
|
}
|
||||||
|
|
||||||
|
private val transactionService: TransactionService = mockk()
|
||||||
|
private lateinit var webTestClient: WebTestClient
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
fun setUp() {
|
||||||
|
webTestClient = WebTestClient.bindToController(TransactionController(transactionService)).build()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `deposit success`() {
|
||||||
|
// given
|
||||||
|
val request = Transaction.Request(uuid, amount)
|
||||||
|
|
||||||
|
every { transactionService.deposit(any(), any()) } returns Mono.just(
|
||||||
|
AccountEntity(id = uuid, name = name, amount = amount),
|
||||||
|
)
|
||||||
|
|
||||||
|
// when
|
||||||
|
val result = webTestClient.post()
|
||||||
|
.uri("/deposit")
|
||||||
|
.contentType(APPLICATION_JSON)
|
||||||
|
.bodyValue(request)
|
||||||
|
.exchange()
|
||||||
|
|
||||||
|
// then
|
||||||
|
result.expectStatus().isCreated
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `deposit fail amount to small`() {
|
||||||
|
// given
|
||||||
|
val request = Transaction.Request(uuid, BigDecimal.valueOf(0.009))
|
||||||
|
|
||||||
|
// when
|
||||||
|
val result = webTestClient.post()
|
||||||
|
.uri("/deposit")
|
||||||
|
.contentType(APPLICATION_JSON)
|
||||||
|
.bodyValue(request)
|
||||||
|
.exchange()
|
||||||
|
|
||||||
|
// then
|
||||||
|
result.expectStatus().isBadRequest
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `deposit default error`() {
|
||||||
|
// given
|
||||||
|
val request = Transaction.Request(uuid, amount)
|
||||||
|
|
||||||
|
every { transactionService.deposit(any(), any()) } returns Mono.error(RuntimeException())
|
||||||
|
|
||||||
|
// when
|
||||||
|
val result = webTestClient.post()
|
||||||
|
.uri("/deposit")
|
||||||
|
.contentType(APPLICATION_JSON)
|
||||||
|
.bodyValue(request)
|
||||||
|
.exchange()
|
||||||
|
|
||||||
|
// then
|
||||||
|
result.expectStatus().is5xxServerError
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `deposit account not found`() {
|
||||||
|
// given
|
||||||
|
val request = Transaction.Request(uuid, amount)
|
||||||
|
|
||||||
|
every { transactionService.deposit(any(), any()) } returns Mono.error(AccountNotFoundException())
|
||||||
|
|
||||||
|
// when
|
||||||
|
val result = webTestClient.post()
|
||||||
|
.uri("/deposit")
|
||||||
|
.contentType(APPLICATION_JSON)
|
||||||
|
.bodyValue(request)
|
||||||
|
.exchange()
|
||||||
|
|
||||||
|
// then
|
||||||
|
result.expectStatus().isNotFound
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user