我在 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
解决此问题的最佳解决方案是什么:
- 允许 Mockito 模拟最终课程:https ://blog.mindorks.com/mockito-cannot-mock-in-kotlin
/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")
...
}
这不允许我们根据测试给出特定的行为。