Как я могу запустить все модульные тесты JUnit, кроме тех, которые заканчиваются на "IntegrationTest" в моем проекте IntelliJ IDEA, используя встроенный тестовый прогон?
В основном я хочу запустить все модульные тесты JUnit в моем проекте IntelliJ IDEA (исключая интеграционные тесты JUnit), используя метод static suite() из JUnit. Зачем использовать метод 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, чтобы я мог использовать встроенный тестовый прогон - мне нравится зеленая полоса:)
5 ответов
Я написал некоторый код, чтобы сделать большую часть работы. Это работает, только если ваши файлы находятся на локальном диске, а не в 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 в их собственный корневой пакет. Я использую эту структуру пакета в моем проекте:
test.
quick.
com.acme
slow.
com.acme
Без какого-либо кодирования вы можете настроить IntelliJ для запуска всех тестов, только быстрых или медленных.
Как насчет использования JUnit4 и Suite-Runner?
Пример:
@RunWith(Suite.class)
@Suite.SuiteClasses({
UserUnitTest.class,
AnotherUnitTest.class
})
public class UnitTestSuite {}
Я сделал небольшой Shell-Script, чтобы найти все юнит-тесты, и еще один, чтобы найти мои интеграционные тесты. Посмотрите на мою запись в блоге: http://blog.timomeinen.de/2010/02/find-all-junit-tests-in-a-project/
Если вы используете Spring TestContext, вы можете использовать аннотацию @IfProfile для объявления различных тестов.
С уважением, Тимо Майнен
Решение: https://github.com/MichaelTamm/junit-toolbox
Используйте следующие функции
@RunWith(WildcardPatternSuite.class)
@SuiteClasses({"**/*.class", "!**/*IntegrationTest.class"})
public class AllTestsExceptionIntegrationSuit {
}
при условии, что вы следуете шаблону именования, где интеграционные тесты заканчиваются на...IntegrationTest и вы помещаете файл в самый верхний пакет (поэтому при поиске класса **/*. будет возможность собрать все ваши тесты)
Spring реализовал отличную функцию поиска по пути к классам в PathMatchingResourcePatternResolver. Если вы используете префикс classpath*:, вы можете найти все ресурсы, включая классы в данной иерархии, и даже отфильтровать их, если хотите. Затем вы можете использовать дочерние элементы AbstractTypeHierarchyTraversingFilter, AnnotationTypeFilter и AssignableTypeFilter для фильтрации этих ресурсов либо по аннотациям уровня класса, либо по интерфейсам, которые они реализуют.