为什么在这种情况下我必须使用GroovyMock?

时间:2020-03-20 10:00:55

标签: javafx groovy mocking spock testfx

这是MCVE:

main.groovy:

package core

import javafx.application.Application
import javafx.stage.Stage
import javafx.fxml.FXMLLoader

class App extends Application {
    FXMLLoader fxmlLoader = new FXMLLoader()
    static App instance
    FXMLController fxmlController
    void start(Stage primaryStage) {
        instance = this
        fxmlLoader.load( getClass().getResource("mainWindow.fxml").openStream() )
        primaryStage.show()
    }
}
class FXMLController {}

testfx.groovy:

package core

import javafx.fxml.FXMLLoader
import javafx.scene.Node
import javafx.stage.Stage
import org.testfx.framework.spock.ApplicationSpec

class FirstFXSpec extends ApplicationSpec {
    void start(Stage stage)  { }
    def 'at start, fxmlLoader should load mainWindow.fxml'() {
        given:
        App.instance = new App()
        App.instance.fxmlLoader = Mock(FXMLLoader) {
            getController() >> Mock(FXMLController)
        }
        Node mockNode = Mock(Node)
        // Stage mockStage = GroovyMock( Stage ){
        Stage mockStage = Mock( Stage ){

            show() >> null
        }

        when:
        App.instance.start( mockStage )

        then:
        true // details of test omitted
    }
}

build.gradle:

plugins {
    id 'groovy'
    id 'java'
    id 'application'
    id 'org.openjfx.javafxplugin' version '0.0.8'
}
repositories { mavenCentral() }
javafx {
    version = '11.0.2'
    modules = [ 'javafx.controls', 'javafx.fxml' ]
}
dependencies {
    implementation 'org.codehaus.groovy:groovy:2.5.+'
    testImplementation 'junit:junit:4.12'
    testImplementation 'org.spockframework:spock-core:1.3-groovy-2.5'
    testImplementation 'net.bytebuddy:byte-buddy:1.9.3'
    testImplementation 'org.objenesis:objenesis:2.6'
    testImplementation 'org.testfx:testfx-spock:4.0.15-alpha'
}
application {
    mainClassName = 'core.App'
}

文件src / main / resources / core / mainWindow.fxml的详细信息并不重要,但这是一个示例:

<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.layout.VBox?>
<VBox prefHeight="600.0" prefWidth="900.0" xmlns="http://javafx.com/javafx/11.0.1"
      xmlns:fx="http://javafx.com/fxml/1" fx:controller="core.FXMLController" >
    <children>
        <Label text="Some label"></Label>
    </children>
</VBox>

以上测试失败。调用了真实的方法Stage.show(),而忽略了它是模拟的事实。
如果我将其从Mock更改为GroovyMock,则使用模拟方法show()并通过测试。
为什么普通的Spock Mock会被忽略?

1 个答案:

答案 0 :(得分:2)

普通Java Mock无法模拟最终方法,并且根据JavaDocs Stage.show是最终方法。 GroovyMock通过使用groovy MOP起作用,因此仅在从groovy代码中调用时才起作用,但是在这方面它们更强大。