前面讲了Spock框架Mock对象、方法经验总结,今天分享一下Spock框架中Mock静态资源的实践经验汇总。分成静态资源和混合场景。
静态资源
静态变量
这个使用场景很少,如果需要Mock,直接把Mock的对象赋值给静态资源即可。所以这个场景pass。
静态方法
Mock静态方法我们使用PowerMock结合Mockito的方案,首先在测试类增加如下注解:
@RunWith(PowerMockRunner.class)
@PowerMockRunnerDelegate(Sputnik.class)
@PrepareForTest([NewUtil.class, HttpBase.class])
@PowerMockIgnore(["javax.management.*"])
@SuppressStaticInitializationFor(["com.funtester.util.NewUtil", "com.funtester.util.HttpBase"])
@RunWith
和@PowerMockRunnerDelegate
注解内容不用改动,直接复制即可,@PrepareForTest
注解后面的类就是需要被Mock的类。@PowerMockIgnore
这个注解用于忽略一些检查和异常。@SuppressStaticInitializationFor
这个注解处理类的初始化,这个注解后面跟的是不需要进行初始化的类的包路径,在现在的实践中通常和@PrepareForTest
后面的类是一致的。
其次我们需要在类初始化代码中对这个类进行Mock,语法如下:
PowerMockito.mockStatic(HttpBase.class)
PowerMockito.mockStatic(NewUtil.class)
下面演示一下如何自定义静态方法的行为:
PowerMockito.when(HttpBase.fetchServiceNames()).thenReturn(["service-prod", "api-pro", "prod", "service-prd", "write-pro"])
定义静态方法行为和非静态方法行为,在语法上是一致的,
混合场景
当一个测试用例中,既要Mock静态方法,也要Mock对象方法,就必须使用PowerMock提供的能力。原因之前提过,主要是因为增加了类注解之后,Spock和Mockito一的Mock对象和定义方法的功能会无法运行,这个没找到具体的文档做出区分,所以如果遇到混合场景,建议使用PowerMock进行对象的Mock。
使用语法上,就是混合了PowerMock处理静态和非静态资源,以及行为模拟的语法。Demo如下:
@RunWith(PowerMockRunner.class)
@PowerMockRunnerDelegate(Sputnik.class)
@PrepareForTest([NewUtil.class, HttpBase.class])
@PowerMockIgnore(["javax.management.*"])
@SuppressStaticInitializationFor(["com.funtester.util.newinterface.NewUtil", "com.funtester.util.slowapi.HttpBase"])
class TaskScheduledTest extends Specification {
@Shared
def service = PowerMockito.mock(IService)
def drive = new TaskScheduled(IService: service, cid: "")
def setupSpec() {
PowerMockito.mockStatic(HttpBase.class)
PowerMockito.mockStatic(NewUtil.class)
PowerMockito.when(HttpBase.fetch()).thenReturn(["ood", "ero"])
Mockito.when(newutil.filter(Mockito.any())).thenReturn(true)
Mockito.when(newser.selectAll()).thenReturn([new NewInterface() {
{
setUrl("/abc")
setNname("test")
setMethod("GET")
}
}, new NewInterface() {
{
setUrl("/abcd")
setNname("test")
setMethod("POST")
}
}, new NewInterface() {
{
setUrl("/abce")
setNname("test")
setMethod("GET")
}
}])
//这里因为send方法中用到了这个静态方法
PowerMockito.when(NewUtil.getsAll(anyList(), anyBoolean())).thenReturn([new NewInterface() {
{
setUrl("/abc")
setNname("test")
setMethod("GET")
}
}, new NewInterface() {
{
setUrl("/abc")
setNname("test")
setMethod("GET")
}
}])
}
def "Send"() {
given:
drive.send()
}
def "day"() {
}
}
PS:在Mockito高版本的依赖mockito-inline
中,也是支持对静态类和静态方法的Mock的,但在Spock中极难使用,资料说是因为项目pom中的Spock版本与Mockito版本不一致导致的,尝试了几个组合依然无法解决,又有人言,跟Groovy依赖的版本也有关系,直接破防,放弃了这个方案。