我有一个模拟类,其中包含我从模块提供的服务的简单实现。我正在使用 OpenJDK 11.03、gradle 5.2.1 和 IntelliJ 2019.2。
在 /main/code/myPackage/myService.java
我有:
package myPackage;
class myService {
public abstract void someFunction();
}
在我的 test/code/somePackage/myMockService
我有:
package myPackage;
// no import, they're in the same package.
class myMockService extends myService {
@Override
public void someFunction() { System.out.prinln("Hello World"); }
}
在我的 main/code/module-info.java
我有:
module myModule {
exports somePackage;
}
我已经在 test/code/module-info.java
上尝试了几种变体,但没有成功。例如:
// "open module" lets anyone use reflection within (mostly JUnit 5 in my case)
import myPackage.myService;
import myPackage.myMockService;
open module myTestModule {
exports myPackage;
provides myService with myMockService
}
上面的 module-info.java
吐出关于“模块名称 myTestModule 与预期名称 myModule 不匹配”、“包‘myPackage’不可见”(来自 myMockModule.java)的错误,解释“包 myPackage 在模块 myModule 中声明,但模块 myTestModule 不读取它”
另一方面,使用以下 module-info.java
,我得到了不同批处理的错误(在代码下方)
import myPackage.myService;
import myPackage.myMockService;
open module myModule {
provides myService with myMockService;
}
如果没有 requires myModule;
,我的测试代码中对主代码分支的每次引用都会给出“错误:找不到符号”。 使用 requires myModule;
,我收到“错误:涉及 myModule 的循环依赖”。
所以...我的测试不能在不同的模块中。而且它们不能是同一个模块! [删除了一长串咒骂]
如何在测试代码中引入服务的模拟版本,而不是创建完全不同的模块/gradle 子项目?
或者这是不可能的情况,虽然您可以有一个单独的测试模块信息,但您不能用它做很多事情?
或者是否有某种方法可以在运行时动态加载内容,这样我就不必将每个小模拟服务都放在任何模块信息、测试或其他方面?这样
ServiceLoader.load()
就会找到它们。嗯...也许扩展ServiceLoader
并将其用法包装在主代码中,这样它将在生产代码或测试代码中使用正确的代码...
最佳答案
a) 欢迎来到“模块化世界中的测试”!
长话短说 https://sormuras.github.io/blog/2018-09-11-testing-in-the-modular-world.html
拥有一个或多个专用测试模块是好。有了所有花里胡哨的东西,请阅读 module-info.java
声明。这些测试模块是您的主要模块的第一批客户。只要确保您的构建工具在编译和运行测试模块之前打包了所有主要模块。否则,您不会尽可能接近现实地测试您的主要模块——其他人会将您的主要模块作为 JAR 文件使用。你也应该如此。这也解决了服务和多版本 JAR 的所有问题。
现在是有趣的部分:模块内 测试,也称为白盒测试。或者如何测试驻留在非导出包中的类型或导出包中的包私有(private)类型?使用知道如何在测试编译和/或测试运行时将测试模块修补到主模块(反之亦然)的构建。喜欢pro或 Bach.java (我维护),或者在您使用 Gradle 的情况下,请参阅此答案的 b) 下面部分。
b) Gradle 和 Java main
, test
, …模块不是开箱即用的 friend
基于插件的最佳解决方案:https://github.com/java9-modularity/gradle-modules-plugin -- 它尊重 在测试运行时传递这些 java 命令行选项 module-info.test
配置文件(我发明的)。在这里,您基本上通过详细的命令行选项描述了您的测试模块需求,尽管已经存在一个完美的 DSL:module-info-java
... 循环回到 a) 和模块感知构建工具。
c) IntelliJ IDEA 和 Java test 模块正在......改进!
- https://youtrack.jetbrains.com/issue/IDEA-171419
module-info.java
2019.3 支持 - https://youtrack.jetbrains.com/issue/IDEA-222831
module-info.test
支持,很快吗?
关于java - 生产代码 + 测试模块信息 = 不可能?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58979522/