前面講了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依賴的版本也有關系,直接破防,放棄了這個方案。