我怎样才能运行所有JUnit单元测试,除了那些以“ IntegrationTest”在使用集成测试运行器的IntelliJ IDEA项目中?
-
05-07-2019 - |
题
我基本上想要在我的IntelliJ IDEA项目中运行所有JUnit 单元 测试(不包括JUnit集成测试),使用JUnit的static suite()方法。为什么要使用static suite()方法?因为我可以使用IntelliJ IDEA的JUnit测试运行器在我的应用程序中运行所有单元测试(并通过命名约定轻松排除所有集成测试)。到目前为止,代码如下所示:
package com.acme;
import junit.framework.Test;
import junit.framework.TestCase;
import junit.framework.TestSuite;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
public class AllUnitTests extends TestCase {
public static Test suite() {
List classes = getUnitTestClasses();
return createTestSuite(classes);
}
private static List getUnitTestClasses() {
List classes = new ArrayList();
classes.add(CalculatorTest.class);
return classes;
}
private static TestSuite createTestSuite(List allClasses) {
TestSuite suite = new TestSuite("All Unit Tests");
for (Iterator i = allClasses.iterator(); i.hasNext();) {
suite.addTestSuite((Class<? extends TestCase>) i.next());
}
return suite;
}
}
应重写方法getUnitTestClasses()以添加扩展TestCase的所有项目类,除非类名以“IntegrationTest”结尾。
我知道我可以在Maven中轻松地做到这一点,但是我需要在IntelliJ IDEA中这样做,所以我可以使用集成测试运行器 - 我喜欢绿色条:)
解决方案
我已经编写了一些代码来完成大部分工作。仅当您的文件位于本地磁盘而不是JAR中时,它才有效。你需要的只是包中的一个类。为此,您可以创建一个Locator.java类,以便能够找到包。
public class ClassEnumerator {
public static void main(String[] args) throws ClassNotFoundException {
List<Class<?>> list = listClassesInSamePackage(Locator.class, true);
System.out.println(list);
}
private static List<Class<?>> listClassesInSamePackage(Class<?> locator, boolean includeLocator)
throws ClassNotFoundException {
File packageFile = getPackageFile(locator);
String ignore = includeLocator ? null : locator.getSimpleName() + ".class";
return toClassList(locator.getPackage().getName(), listClassNames(packageFile, ignore));
}
private static File getPackageFile(Class<?> locator) {
URL url = locator.getClassLoader().getResource(locator.getName().replace(".", "/") + ".class");
if (url == null) {
throw new RuntimeException("Cannot locate " + Locator.class.getName());
}
try {
return new File(url.toURI()).getParentFile();
}
catch (URISyntaxException e) {
throw new RuntimeException(e);
}
}
private static String[] listClassNames(File packageFile, final String ignore) {
return packageFile.list(new FilenameFilter(){
@Override
public boolean accept(File dir, String name) {
if (name.equals(ignore)) {
return false;
}
return name.endsWith(".class");
}
});
}
private static List<Class<?>> toClassList(String packageName, String[] classNames)
throws ClassNotFoundException {
List<Class<?>> result = new ArrayList<Class<?>>(classNames.length);
for (String className : classNames) {
// Strip the .class
String simpleName = className.substring(0, className.length() - 6);
result.add(Class.forName(packageName + "." + simpleName));
}
return result;
}
}
其他提示
如何将每个主要的junit测试组放入他们自己的root包中。我在我的项目中使用这个包结构:
test.
quick.
com.acme
slow.
com.acme
如果没有任何编码,您可以设置IntelliJ来运行所有测试,只需要快速测试或只是慢速测试。
如何使用JUnit4和Suite-Runner?
示例:
@RunWith(Suite.class)
@Suite.SuiteClasses({
UserUnitTest.class,
AnotherUnitTest.class
})
public class UnitTestSuite {}
我制作了一个小型Shell脚本来查找所有单元测试,另一个用于查找我的集成测试。看看我的博客文章: http://blog.timomeinen.de/ 2010/02 /发现,所有的JUnit测试-IN-A-项目/
如果使用Spring TestContext,可以使用@IfProfile Annotation声明不同的测试。
亲切的问候, 蒂莫梅宁
Spring在PathMatchingResourcePatternResolver中实现了一个优秀的类路径搜索功能。如果使用classpath *:前缀,则可以找到所有资源,包括给定层次结构中的类,甚至可以根据需要过滤它们。然后,您可以使用AbstractTypeHierarchyTraversingFilter,AnnotationTypeFilter和AssignableTypeFilter的子项在类级别注释或它们实现的接口上过滤这些资源。
解决方案: https://github.com/MichaelTamm/junit-toolbox
使用以下功能
@RunWith(WildcardPatternSuite.class)
@SuiteClasses({"**/*.class", "!**/*IntegrationTest.class"})
public class AllTestsExceptionIntegrationSuit {
}
假设您遵循命名模式,其中集成测试结束于... IntegrationTest并将文件放在最顶层的包中(因此** / * .class搜索将有机会获取所有测试)