如何模拟测试 Kotlin Spring Boot 2 应用程序

Posted

技术标签:

【中文标题】如何模拟测试 Kotlin Spring Boot 2 应用程序【英文标题】:how to mock test Kotlin Spring boot 2 application 【发布时间】:2019-09-28 02:08:49 【问题描述】:

我正在尝试测试一个完全用 kotlin 编写的 Spring Boot 应用程序。

我正在尝试测试模拟存储库对象的服务层。代码如下

@RunWith(MockitoJUnitRunner::class)
class TaskServiceTest 

    @Mock
    lateinit var taskRepository: TaskRepository
    @InjectMocks
    lateinit var taskService: TaskService

    @Test
    fun createNewTaskFailTest() 
        val taskRequest = TaskRequest("fullName", "94123456789", "", "")
        val newTask = Task(
                id = null,
                status = PENDING)
        val savedTask = newTask.copy(id = 1L)
        given(taskRepository.save(newTask)).willReturn(savedTask)
        val createdTask = taskService.createTask(taskRequest)
        Assert.assertEquals(createdTask.status, PENDING)
    

当我运行此代码时,我收到以下错误

java.lang.IllegalStateException: savedTask must not be null

    at com.hsenid.servicestatusinquiry.service.TaskService.createTask(TaskService.kt:46)
    at com.hsenid.servicestatusinquiry.service.TaskServiceTest.createNewTaskFailTest(TaskServiceTest.kt:41)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.mockito.internal.runners.DefaultInternalRunner$1.run(DefaultInternalRunner.java:79)
    at org.mockito.internal.runners.DefaultInternalRunner.run(DefaultInternalRunner.java:85)
    at org.mockito.internal.runners.StrictRunner.run(StrictRunner.java:39)
    at org.mockito.junit.MockitoJUnitRunner.run(MockitoJUnitRunner.java:163)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
    at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
    at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
    at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
    at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)

[MockitoHint] TaskServiceTest.createNewTaskFailTest (see javadoc for MockitoHint):
[MockitoHint] 1. Unused... -> at com.hsenid.servicestatusinquiry.service.TaskServiceTest.createNewTaskFailTest(TaskServiceTest.kt:40)
[MockitoHint]  ...args ok? -> at com.hsenid.servicestatusinquiry.service.TaskService.createTask(TaskService.kt:45)

当我使用 http 调用从控制器调用它时,创建任务工作正常

下面是创建任务代码示例

    fun createTask(taskRequest: TaskRequest): Task 
        val newTask = Task(
                id = null,
                status = PENDING)
        val savedTask = repository.save(newTask)
        return savedTask
    

如何解决这个问题?

【问题讨论】:

我不是 Kotlin 专家,但在 Java 中,您在 createTask 函数中创建的 newTask 与您在设置时所指的 newTask 不同模拟,因为它们是两个独立的实例。我还没有测试过,但我认为如果你在定义模拟的地方使用 Mockito 的 eq matcher,它应该可以工作,所以:given(taskRepository.save(eq(newTask))).willReturn(savedTask)。尝试refEqany(Task.class) 失败可能会有所帮助。 any(Task.class) 成功了,谢谢 【参考方案1】:

以下代码对我有用

given(taskRepository.save(any(Task.class))).willReturn(savedTask)

【讨论】:

【参考方案2】:

我这样做了:

首先我用以下内容注释我的类:

@ExtendWith(MockitoExtension::class)
class TaskServiceTest 

然后:

private fun <T> anyObject(): T 
    Mockito.anyObject<T>()
    return uninitialized()


Mockito.`when`<Any>(taskRepository.save(anyObject()))
            .thenReturn(validTask())

为我工作。

【讨论】:

以上是关于如何模拟测试 Kotlin Spring Boot 2 应用程序的主要内容,如果未能解决你的问题,请参考以下文章

如何在 Spring Boot 测试中模拟 Spring amqp/rabbit

在 Kotlin 和 JUnit5 中测试 Spring Boot 缓存

如何在 Spring Boot 测试中模拟会话关闭/过期?

kotlin + Spring boot 中的单元测试休息控制器

使用 spring boot、kotlin 和 junit 进行休息控制器单元测试

Kotlin Spring Boot 单元测试 - 添加 @TestExecutionListeners 不会注入依赖项