Запускаете JUnit Test параллельно на уровне Suite?

У меня есть несколько тестов, которые организованы в тестовых наборах JUnit. Эти тесты в значительной степени используют селен для тестирования веб-приложений. Итак, естественно для селена, время выполнения этих тестов довольно продолжительное. Поскольку тестовые классы в наборах не могут работать параллельно из-за некоторого перекрытия в тестовой базе данных, я хотел бы запускать наборы параллельно.

JUnit ParallelComputer может выполнять тесты только на уровне класса или метода параллельно, есть ли какие-либо стандартные способы для JUnit делать это с наборами?

Если я просто передаю классы наборов средству выполнения junit и настраиваю компьютер для распараллеливания на уровне классов, он выбирает сами тестовые классы, а не наборы.

br Франк


person Frank    schedule 15.04.2011    source источник


Ответы (2)


Вот код, который у меня сработал. Я этого не писал. Если вы используете @RunWith(ConcurrentSuite.class) вместо @RunWith(Suite.class), это должно сработать. Также необходима аннотация, которая находится ниже.

package utilities.runners;

import org.junit.internal.builders.AllDefaultPossibilitiesBuilder;
import org.junit.runner.Runner;
import org.junit.runners.Suite;
import org.junit.runners.model.InitializationError;
import org.junit.runners.model.RunnerBuilder;
import org.junit.runners.model.RunnerScheduler;

import utilities.annotations.Concurrent;

import java.util.Arrays;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;
import java.util.concurrent.CompletionService;
import java.util.concurrent.ExecutorCompletionService;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * @author Mathieu Carbou ([email protected])
 */
public final class ConcurrentSuite extends Suite {
    public ConcurrentSuite(final Class<?> klass) throws InitializationError {
        super(klass, new AllDefaultPossibilitiesBuilder(true) {
            @Override
            public Runner runnerForClass(Class<?> testClass) throws Throwable {
                List<RunnerBuilder> builders = Arrays.asList(
                        new RunnerBuilder() {
                            @Override
                            public Runner runnerForClass(Class<?> testClass) throws Throwable {
                                Concurrent annotation = testClass.getAnnotation(Concurrent.class);
                                if (annotation != null)
                                    return new ConcurrentJunitRunner(testClass);
                                return null;
                            }
                        },
                        ignoredBuilder(),
                        annotatedBuilder(),
                        suiteMethodBuilder(),
                        junit3Builder(),
                        junit4Builder());
                for (RunnerBuilder each : builders) {
                    Runner runner = each.safeRunnerForClass(testClass);
                    if (runner != null)
                        return runner;
                }
                return null;
            }
        });
        setScheduler(new RunnerScheduler() {
            ExecutorService executorService = Executors.newFixedThreadPool(
                    klass.isAnnotationPresent(Concurrent.class) ?
                            klass.getAnnotation(Concurrent.class).threads() :
                            (int) (Runtime.getRuntime().availableProcessors() * 1.5),
                    new NamedThreadFactory(klass.getSimpleName()));
            CompletionService<Void> completionService = new ExecutorCompletionService<Void>(executorService);
            Queue<Future<Void>> tasks = new LinkedList<Future<Void>>();

            @Override
            public void schedule(Runnable childStatement) {
                tasks.offer(completionService.submit(childStatement, null));
            }

            @Override
            public void finished() {
                try {
                    while (!tasks.isEmpty())
                        tasks.remove(completionService.take());
                } catch (InterruptedException e) {
                    Thread.currentThread().interrupt();
                } finally {
                    while (!tasks.isEmpty())
                        tasks.poll().cancel(true);
                    executorService.shutdownNow();
                }
            }
        });
    }

    static final class NamedThreadFactory implements ThreadFactory {
        static final AtomicInteger poolNumber = new AtomicInteger(1);
        final AtomicInteger threadNumber = new AtomicInteger(1);
        final ThreadGroup group;

        NamedThreadFactory(String poolName) {
            group = new ThreadGroup(poolName + "-" + poolNumber.getAndIncrement());
        }

        @Override
        public Thread newThread(Runnable r) {
            return new Thread(group, r, group.getName() + "-thread-" + threadNumber.getAndIncrement(), 0);
        }
    }

}

А аннотация такая.

package utilities.annotations;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * @author Mathieu Carbou ([email protected])
 */
@Retention(RetentionPolicy.RUNTIME)
@Target({ ElementType.TYPE })
public @interface Concurrent {
    int threads() default 5;
}
person Reid Mac    schedule 16.04.2012
comment
Похоже, копия этого кода доступна по этому URL-адресу mycila.googlecode.com/svn/sandbox/src/main/java/com/mycila/ - person Lucas Holt; 21.11.2013
comment
Да, мне следовало включить ссылку. Спасибо. - person Reid Mac; 13.12.2013
comment
Это сработало после добавления дополнительного конструктора: public ConcurrentSuite (Class ‹?› Klass, RunnerBuilder builder) выбрасывает InitializationError {this (klass); } - person Todd Flanders; 31.03.2014
comment
Поскольку ссылка сейчас мертва - вот предварительно скомпилированный jar и исходники на Maven: mvnrepository. com / artifact / com.mycila / mycila-junit и случайная выборка - person Simon Sobisch; 23.10.2018

Поскольку Suite используется для аннотирования класса, запустите класс, аннотированный Suite, JUnitCore.runClasses(ParallelComputer.classes(), cls) способом. cls - классы с аннотациями Suite.

@RunWith(Suite.class)
@Suite.SuiteClasses({
Test1.class,
Test2.class})
public class Suite1 {
}

@RunWith(Suite.class)
@Suite.SuiteClasses({
Test3.class,
Test4.class})
public class Suite2 {
}
...
JUnitCore.runClasses(ParallelComputer.classes(), new Class[]{Suite1.class, Suite2.class})
person 卢声远 Shengyuan Lu    schedule 15.04.2011
comment
Привет, спасибо за быстрый ответ. К сожалению, это не так, как я планировал. Как я написал в конце своего вопроса, при этом параллельно запускаются классы тестов внутри набора, а не сами наборы. - person Frank; 15.04.2011
comment
Ваш пример действительно выполняется следующим образом: Test1 и Test2 работают параллельно, именно такое поведение я должен предотвратить. - person Frank; 15.04.2011
comment
Добро пожаловать! Если мой код не может вам помочь, вы можете написать многопоточность для запуска своих наборов (если нет лучшего решения). - person 卢声远 Shengyuan Lu; 15.04.2011
comment
Я думал об этом, но если я это сделаю, мне придется самому агрегировать результат junit, потому что он мне нужен для отчета о тестировании. Но спасибо за вашу быструю помощь! - person Frank; 15.04.2011