Unit and Integration Tests With Maven and JUnit Categories
Join the DZone community and get the full member experience.
Join For FreeIntroduction
This example shows how to split unit and integration tests using Maven and JUnit categories.
It is especially useful for existing test suites and can be implemented in minutes in 3 simple steps.
Why use this?
My previous post showed how we to use a maven profile to split unit and integration tests.
http://johndobie.blogspot.co.uk/2011/06/seperating-maven-unit-integration-tests.html
This has been a very well read post and I like how it uses seperate directories.
However this example show a much simpler technique that can easily be applied to legacy test suites.
It offers most of the benefits of the original, and sits more comfortably in the Maven world.
Code
The code for the example is here.svn co https://designbycontract.googlecode.com/svn/trunk/examples/maven/categories mvn clean install
JUnit Categories
As of JUnit 4.8 you can define your own categories for tests. This enables you to label and group tests.This example shows how easy it is to separate unit and integration test using the @Catgegory annotation. http://kentbeck.github.com/junit/javadoc/latest/org/junit/experimental/categories/Categories.html
Define the Marker Interface
The first step in grouping a test using categories is to create a marker interface.This interface will be used to mark all of the tests that you want to be run as integration tests.
public interface IntegrationTest {}
Mark your test classes
Next we add the category annotation to the top of your test class. It takes the name of your new interface as a parameter.
import org.junit.experimental.categories.Category; @Category(IntegrationTest.class) public class ExampleIntegrationTest{ @Test public void longRunningServiceTest() throws Exception { } }Categories can be used to mark classes or methods. Really in my opinion you should only mark a class.
If you have both unit and integration tests in a single class then split it.
Configure Maven Unit Tests
The beauty of this solution is that nothing really changes for the unit test side of things.We simply add some configuration to the maven surefire plugin to make it to ignore any integration tests.
<plugin> <groupid>org.apache.maven.plugins</groupid> <artifactid>maven-surefire-plugin</artifactid> <version>2.11</version> <dependencies> <dependency> <groupid>org.apache.maven.surefire</groupid> <artifactid>surefire-junit47</artifactid> <version>2.12</version> </dependency> </dependencies> <configuration> <includes> <include>**/*.class</include> </includes> <excludedgroups>com.test.annotation.type.IntegrationTest</excludedgroups> </configuration> </plugin>
Surefire will run all of your tests, except those marked as an integration test.
<excludedgroups>com.test.annotation.type.IntegrationTest</excludedgroups>The other important part is to make sure the surefire plugin uses the correct JUnit provider.
The JUnit47 provider is needed to correctly detect the categories.
<dependencies>
<dependency>
<groupid>org.apache.maven.surefire</groupid>
<artifactid>surefire-junit47</artifactid>
<version>2.12</version>
</dependency>
</dependencies>
Running the unit tests
To make sure this works correctly we can run the unit testsmvn clean testYou can see from the output below that the unit test is run, but not the integration test.
------------------------------------------------------- T E S T S ------------------------------------------------------- Running com.test.EmptyUnitTest Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0 sec Results : Tests run: 1, Failures: 0, Errors: 0, Skipped: 0 [INFO] ------------------------------------------------------------------------ [INFO] BUILD SUCCESS [INFO] ------------------------------------------------------------------------
Configure Maven Integration Tests
Again the configuration for this is very simple. We use the standard failsafe plugin and configure it to only run the integration tests.
<plugin>The configuration uses a standard execution goal to run the failsafe plugin during the integration-test phase of the build.
<artifactid>maven-failsafe-plugin</artifactid>
<version>2.12</version>
<dependencies>
<dependency>
<groupid>org.apache.maven.surefire</groupid>
<artifactid>surefire-junit47</artifactid>
<version>2.12</version>
</dependency>
</dependencies>
<configuration>
<groups>com.test.annotation.type.IntegrationTest</groups>
</configuration>
<executions>
<execution>
<goals>
<goal>integration-test</goal>
</goals>
<configuration>
<includes>
<include>**/*.class</include>
</includes>
</configuration>
</execution>
</executions>
</plugin>
The following configuration ensures only the integration tests are run.
<groups>com.test.annotation.type.IntegrationTest</groups>And again the JUnit provider must be correctly configured.
<dependencies>That’s it!
<dependency>
<groupid>org.apache.maven.surefire</groupid>
<artifactid>surefire-junit47</artifactid>
<version>2.12</version>
</dependency>
</dependencies>
Running the integration tests
We can now run the whole build.mvn clean installThis time as well as the unit test running, the integration tests are run during the integration-test phase.
------------------------------------------------------- T E S T S ------------------------------------------------------- Running com.test.AnotherEmptyIntegrationTest Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.016 sec Running com.test.EmptyIntegrationTest Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0 sec Results : Tests run: 2, Failures: 0, Errors: 0, Skipped: 0
Further Details
For more information and other articles follow this link.
http://johndobie.blogspot.co.uk/2012/04/unit-and-integration-tests-with-maven.html
Opinions expressed by DZone contributors are their own.
Comments