使用 Groovy 应用程序和测试代码结合 jlink 解决方案来捆绑 JavaFX

Use Groovy app and test code in combination with jlink solution for bundling JavaFX

这是从 到如何让 Gradle 将 JavaFX 与您的发行版捆绑在一起的问题。

NB 规格:LinuxMint 18.3,Java11,JavaFX 13。

那些涉及 jlink 和模块的东西 info.java,超出了我的工资等级(尽管我正在努力阅读这些东西)。

我想在我的应用程序和测试代码(即 Spock)中使用 Groovy 而不是 Java。问题是,当我在我的 build.gradle 中包含 "normal" 依赖项时,即

implementation 'org.codehaus.groovy:groovy-all:2.5.9'

并尝试构建,我收到多个错误:

mike@M17A ~/IdeaProjects/TestProj $  ./gradlew build

> Configure project :
Found module name 'javafx.jlink.example.main'

> Task :compileTestJava FAILED
error: the unnamed module reads package org.codehaus.groovy.tools.shell.util from both org.codehaus.groovy.groovysh and org.codehaus.groovy
[...]
error: the unnamed module reads package groovy.xml from both org.codehaus.groovy and org.codehaus.groovy.xml
[...]
error: module org.codehaus.groovy.ant reads package groovy.lang from both org.codehaus.groovy and org.codehaus.groovy.test
error: module org.codehaus.groovy.ant reads package groovy.util from both org.codehaus.groovy.xml and org.codehaus.groovy.ant
100 errors

FAILURE: Build failed with an exception.

* What went wrong:
Execution failed for task ':compileTestJava'.

是的,100 个错误...可能更多!通过评论各种事情,我认为我得出的结论是 jlink 东西注入了一些 Groovy 依赖项。好吧,我可以接受(尽管很高兴知道 Groovy 是什么版本)。

麻烦的是,即使我省略了 Groovy 依赖行,当我尝试引入 Spock 依赖时也会出现同样的错误:

testImplementation 'org.spockframework:spock-core:1.2-groovy-2.5'

有没有人知道这里发生了什么以及该怎么办?

我搜索了一个答案。我没有找到好的解决方案。

根据,Groovy目前似乎与Java模块不兼容。这是由于某些包包含在库的多个 jar 中(与模块不兼容)。您必须等待 Groovy 4 才能获得兼容版本。

我发现 JavaFX 插件在内部使用 this plugin。这个插件似乎认为所有依赖项都是模块(这不是默认的 Gradle 行为)。

要使您的应用程序正常运行,您似乎必须:

  • 强制Gradle将Groovy放在类路径中而不是模块路径中(它不会被视为模块,但如果使用javafx插件似乎不可能)
  • 使用 "patch-module" 系统:它允许 Gradle 将库 jar 融合到一个模块中,以防止包在不同 jar 中的问题

我用 IDEA(项目 structure/Libraries)搜索了 Groovy 罐子,我尝试使用插件提供的语法来使用 "patch-module":

patchModules.config = [
    "org.codehaus.groovy=groovy-ant-3.0.1.jar",
    "org.codehaus.groovy=groovy-cli-picocli-3.0.1.jar",
    "org.codehaus.groovy=groovy-console-3.0.1.jar",
    "org.codehaus.groovy=groovy-datetime-3.0.1.jar",
    "org.codehaus.groovy=groovy-docgenerator-3.0.1.jar",
    "org.codehaus.groovy=groovy-groovydoc-3.0.1.jar",
    "org.codehaus.groovy=groovy-groovysh-3.0.1.jar",
    "org.codehaus.groovy=groovy-jmx-3.0.1.jar",
    "org.codehaus.groovy=groovy-json-3.0.1.jar",
    "org.codehaus.groovy=groovy-jsr-3.0.1.jar",
    "org.codehaus.groovy=groovy-macro-3.0.1.jar",
    "org.codehaus.groovy=groovy-nio-3.0.1.jar",
    "org.codehaus.groovy=groovy-servlet-3.0.1.jar",
    "org.codehaus.groovy=groovy-sql-3.0.1.jar",
    "org.codehaus.groovy=groovy-swing-3.0.1.jar",
    "org.codehaus.groovy=groovy-templates-3.0.1.jar",
    "org.codehaus.groovy=groovy-test-junit-3.0.1.jar",
    "org.codehaus.groovy=groovy-test-3.0.1.jar",
    "org.codehaus.groovy=groovy-testng-3.0.1.jar",
    "org.codehaus.groovy=groovy-xml-3.0.1.jar"
]

它只适用于单行 "org.codehaus.groovy=X.jar",但一个错误阻止它适用于所有库 jar(查看 Github 上的 this issue)。

所以你有多个选择:

  • 使用Java代替Groovy
  • 等待新的 Groovy 版本,或新版本的插件(模块插件,以及内部使用此插件的 javafx 插件版本)
  • 使用旧的 javafx 配置:默认情况下依赖项不是模块,您必须在 build.gradle 中手动指定 JavaFX 依赖项应被视为模块(检查 )