Как включить тестовые классы в банку Maven и выполнить их?

В проекте Maven у меня есть тестовые классы и исходные классы в одном пакете, но в разных физических местах.

.../src/main/java/package/** <-- application code
.../src/test/java/package/** <-- test code

Нет проблем с доступом к исходным классам в тестовых классах, но я хотел бы запустить тестовый бегун в основном методе и получить доступ к AllTest.class, чтобы я мог создать jar и выполнить мои тесты.

 public static void main(String[] args) {
    // AllTest not found
    Result result = JUnitCore.runClasses(AllTest.class);
    for (Failure failure : result.getFailures()) {
        System.out.println(failure.toString());
    }
    System.out.println(result.wasSuccessful());
}

Но это не работает, поскольку у меня нет доступа к тестовому коду. Я не понимаю, так как они находятся в одном пакете.

Вопрос: как получить доступ к классам тестов из классов приложений? В качестве альтернативы, как Maven может упаковать толстую банку, включая тестовые классы и выполнить тесты?

Ответы

Ответ 1

Вам не следует обращаться к тестовым классам из кода приложения, а скорее создавать основную (ту же основную) в области проверки и создавать дополнительный артефакт для вашего проекта.

Однако в этом дополнительном артефакте (jar) вам нужно будет:

  • Тест-классы
  • Классы кода приложения
  • Внешние зависимости, требуемые кодом приложения (в области compile)
  • Внешние зависимости, требуемые тестовым кодом (в области test)

Что в основном означает живую банку с добавлением тестовых классов (и их зависимостей). Maven Jar Plugin и его test-jar не будут удовлетворить эту потребность. Maven Shade Plugin и его параметр shadeTestJar не будет не помогайте никому.

Итак, как создать в Maven жирную банку с тестовыми классами и внешними зависимостями?

Плагин сборки Maven является идеальным кандидатом в этом случае.

Вот минимальный образец POM:

<project>
    <modelVersion>4.0.0</modelVersion>
    <groupId>com.sample</groupId>
    <artifactId>sample-project</artifactId>
    <version>1.0-SNAPSHOT</version>

    <build>
        <plugins>
            <plugin>
                <artifactId>maven-assembly-plugin</artifactId>
                <version>2.3</version>
                <configuration>
                    <descriptor>src/main/assembly/assembly.xml</descriptor>
                </configuration>
                <executions>
                    <execution>
                        <id>make-assembly</id>
                        <phase>package</phase>
                        <goals>
                            <goal>single</goal>
                        </goals>
                        <configuration>
                            <archive>
                                <manifest>
                                    <mainClass>com.sample.TestMain</mainClass>
                                </manifest>
                            </archive>
                        </configuration>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>

    <dependencies>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.11</version>
            <scope>test</scope>
        </dependency>
    </dependencies>
</project>

В приведенной выше конфигурации задается основной класс, определенный вами в ваших тестовых классах. Но этого недостаточно.

Вам также необходимо создать файл дескриптора в папке src\main\assembly и assembly.xml со следующим содержимым:

<assembly
    xmlns="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.3"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.3 http://maven.apache.org/xsd/assembly-1.1.3.xsd">
    <id>fat-tests</id>
    <formats>
        <format>jar</format>
    </formats>
    <includeBaseDirectory>false</includeBaseDirectory>
    <dependencySets>
        <dependencySet>
            <outputDirectory>/</outputDirectory>
            <useProjectArtifact>true</useProjectArtifact>
            <unpack>true</unpack>
            <scope>test</scope>
        </dependencySet>
    </dependencySets>
    <fileSets>
        <fileSet>
            <directory>${project.build.directory}/test-classes</directory>
            <outputDirectory>/</outputDirectory>
            <includes>
                <include>**/*.class</include>
            </includes>
            <useDefaultExcludes>true</useDefaultExcludes>
        </fileSet>
    </fileSets>
</assembly>

Конфигурация выше:

  • установка внешних зависимостей из области test (которая также примет область compile)
  • установка fileset для включения скомпилированных тестовых классов как часть упакованной толщины жира
  • установка последней банки с fat-tests классификатором (следовательно, ваш окончательный файл будет чем-то вроде sampleproject-1.0-SNAPSHOT-fat-tests.jar).

Затем вы можете вызвать main следующим образом (из папки target):

java -jar sampleproject-1.0-SNAPSHOT-fat-tests.jar

Из такого основного вы также можете вызвать все ваши тестовые примеры следующим образом:

  • Создайте тестовый пакет JUni
  • Добавьте к тестовому набору соответствующие тесты
  • Вызовите тестовый пакет с простой Java-основной

Пример набора тестов:

import org.junit.runner.RunWith;
import org.junit.runners.Suite;
import org.junit.runners.Suite.SuiteClasses;

@RunWith(Suite.class)
@SuiteClasses({ AppTest.class })
public class AllTests {

}

Примечание: в этом случае набор тестов относится только к тесту AppTest.

Тогда у вас может быть основной класс следующим образом:

import org.junit.internal.TextListener;
import org.junit.runner.JUnitCore;

public class MainAppTest {

    public static void main(String[] args) {
        System.out.println("Running tests!");

        JUnitCore engine = new JUnitCore();
        engine.addListener(new TextListener(System.out)); // required to print reports
        engine.run(AllTests.class);
    }
}

В приведенном выше разделе будет выполнен набор тестов, который в цепочке выполнит все сконфигурированные тесты.