3

我在 Kotlin 有一个项目。

我创建了一个@ConfigurationProperties类,我想知道单元测试的最佳实践。

以下是我的属性类:

@ConstructorBinding
@ConfigurationProperties(prefix = "myapp")
data class MyAppProperties(
    /**
     * Base path to be used by myapp. Default is '/search'.
     */
    val basePath: String = "/myapp"
)

我在我的控制器中注入 MyAppProperties:

@RestController
final class MyAppController(
    myAppProperties: MyAppProperties
) {

    ...

}

我想测试我的控制器:

@ExtendWith(MockitoExtension::class)
internal class MyAppControllerTest {

    @Mock
    lateinit var myAppProperties: MyAppProperties

    @InjectMocks
    lateinit var myAppController: MyAppController

    ...

}

但我有以下 Mockito 错误:

org.mockito.exceptions.base.MockitoException: 
Cannot mock/spy class com.myapp.MyAppProperties
Mockito cannot mock/spy because :
 - final class

解决此问题的最佳解决方案是什么:

    /src/test/resources/mockito-extensions/org.mockito.plugins.MockMaker: mock-maker-inline

这个解决方案看起来不错,因为我们不修改现有代码,但它为整个项目和所有最终类添加了 Mockito 行为。

  • 使我的属性类open
    open class MyAppProperties...

这种解决方案需要修改代码并使类可扩展,这可能不是一件好事?

  • 通过 Maven 配置打开所有 @ConfigurationProperties 类:
    <plugin>
        <groupId>org.jetbrains.kotlin</groupId>
        <artifactId>kotlin-maven-plugin</artifactId>
        <configuration>
            <args>
                <arg>-Xjsr305=strict</arg>
                <arg>-Xjvm-default=enable</arg>
            </args>
            <compilerPlugins>
                <plugin>all-open</plugin>
            </compilerPlugins>
            <pluginOptions>
                <option>all-open:annotation=org.springframework.boot.context.properties.ConfigurationProperties</option>
            </pluginOptions>
        </configuration>
        ...
    </plugin>

这个解决方案不需要修改代码,而是让所有@ConfigurationProperties 类open,这可能不是一件好事?

  • 不要模拟属性类并在 MyAppControllerTest 中对其进行初始化:
    @ExtendWith(MockitoExtension::class)
    internal class MyAppControllerTest {
    
        val myAppProperties: MyAppProperties = MyAppProperties("/mypath")
    
        ...
    }

这不允许我们根据测试给出特定的行为。

4

0 回答 0