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

Why am I obliged to use a GroovyMock in this case?

这是一个 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 的事实被忽略了。
如果我将其从 Mock 更改为 GroovyMock,则使用模拟方法 show() 并且测试通过。
为什么正常的 Spock Mock 会被忽略?

Plain Java Mocks 不能模拟 final 方法,根据 JavaDocs Stage.show 是 final。 GroovyMocks 通过使用 groovy MOP 工作,因此仅在从 groovy 代码调用时工作,但它们在这方面更强大。