This is probably the beginning of a series of articles about testing Maven plugins.
I’ll start with the Maven Embedder, which allows you to run an embedded Maven from a Java program. Note that we’re not simply running a locally installed Maven binary from a Java program; we run Maven taken from a Java library. So, we’re not forking any process.
Whether this is useful or not for your integration tests is your decision 😉
The source code of the example used in this tutorial can be found here: https://github.com/LorenzoBettini/maven-embedder-example/
I like to use the Maven Embedder when using the Maven Verifier Component (described in another blog post). Since it’s not trivial to get the dependencies to run the Maven Embedder properly, I decided to write this tutorial, where I’ll show a basic Java class running the Maven Embedder and a few JUnit tests that use this Java class to build (with the embedded Maven) a test Maven project.
This is the website of the Maven Embedder and its description:
https://maven.apache.org/ref/3.9.6/maven-embedder/
Maven embeddable component, with CLI and logging support.
Remember: this post will NOT describe integration testing for Maven plugins; however, getting to know the Maven Embedder in a simpler context was helpful for me.
Let’s create a simple Java Maven project with the quickstart archetype
1 2 3 4 5 6 7 |
mvn archetype:generate \ -DarchetypeGroupId=org.apache.maven.archetypes \ -DarchetypeArtifactId=maven-archetype-quickstart \ -DarchetypeVersion=1.4 \ -DgroupId=com.examples \ -DartifactId=maven-embedder-example \ -DinteractiveMode=false |
Let’s change the Java version in the POM to Java 17, use a more recent version of JUnit, and add another test dependency we’ll use later:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
<properties> ... <maven.compiler.source>17</maven.compiler.source> <maven.compiler.target>17</maven.compiler.target> </properties> <dependencies> <!-- Testing dependencies --> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.13.2</version> <scope>test</scope> </dependency> <!-- For the FileUtils.deleteDirectory that we use in tests. --> <dependency> <groupId>commons-io</groupId> <artifactId>commons-io</artifactId> <version>2.6</version> <scope>test</scope> </dependency> ... |
Let’s import the Maven Java project into Eclipse (assuming you have m2e installed in Eclipse).
Let’s add the dependencies for the Maven Embedder:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 |
<properties> ... <maven-embedder-version>3.9.6</maven-embedder-version> </properties> ... <dependency> <groupId>org.apache.maven</groupId> <artifactId>maven-embedder</artifactId> <version>${maven-embedder-version}</version> </dependency> <!-- For Maven logging --> <dependency> <groupId>org.slf4j</groupId> <artifactId>slf4j-simple</artifactId> <version>1.7.36</version> </dependency> <!-- Required due to https://issues.apache.org/jira/browse/MNG-6561 Otherwise, you get the error: [main] WARN Sisu - Error injecting: org.apache.maven.project.DefaultProjectBuildingHelper com.google.inject.ProvisionException: Unable to provision, see the following errors: 1) No implementation for RepositorySystem was bound. while locating DefaultProjectBuildingHelper --> <dependency> <groupId>org.apache.maven</groupId> <artifactId>maven-compat</artifactId> <version>${maven-embedder-version}</version> </dependency> <!-- Required to let Maven download artifacts. Otherwise, you get errors of the shape: Plugin ... or one of its dependencies could not be resolved: Failed to read artifact descriptor for ...: Could not transfer artifact ... from/to central (https://repo.maven.apache.org/maven2): No connector factories available. --> <dependency> <groupId>org.apache.maven.resolver</groupId> <artifactId>maven-resolver-connector-basic</artifactId> <version>1.9.18</version> </dependency> <dependency> <groupId>org.apache.maven.resolver</groupId> <artifactId>maven-resolver-transport-http</artifactId> <version>1.9.18</version> </dependency> |
Getting all the needed dependencies right for the Maven Embedder is not trivial due to the dynamic nature of Maven components and dependency injection. The requirements are properly documented above.
Let’s replace the “App.java” inside “src/main/java/” with this Java class:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
package com.examples; import org.apache.maven.cli.MavenCli; public class MavenEmbedderRunner { public int run(String baseDir, String...args) { MavenCli cli = new MavenCli(); // Required to avoid the error: // "-Dmaven.multiModuleProjectDirectory system property is not set." System.setProperty("maven.multiModuleProjectDirectory", baseDir); return cli.doMain(args, baseDir, System.out, System.err); } } |
That’s just a simple example of using the Maven Embedder. We rely on its “doMain” method that takes the arguments to pass to the embedded Maven, the base directory from where we want to launch the embedded Maven, and the standard output/error where Maven will log all its information. In a more advanced scenario, we could store the logging in a file instead of the console by passing the proper “PrintStream” streams acting on files.
Let’s create the folder “src/test/resources” (it will be used by default as a source folder in Eclipse); this is where we’ll store the test Maven project to build with the Maven Embedder.
Inside that folder, let’s create another Maven project (remember, this will be used only for testing purposes: we’ll use the Maven Embedder to build that project from a JUnit test):
1 2 3 4 5 6 7 |
mvn archetype:generate \ -DarchetypeGroupId=org.apache.maven.archetypes \ -DarchetypeArtifactId=maven-archetype-quickstart \ -DarchetypeVersion=1.4 \ -DgroupId=com.examples \ -DartifactId=maven-quickstart-example \ -DinteractiveMode=false |
We rely on the fact that the contents of “src/test/resources” are automatically copied recursively into the “target/test-classes” folder. Eclipse and m2e will take care of such a copy; during the Maven build, there’s a dedicated phase (coming before the phase “test”) that performs the copy: “process-test-resources”.
Let’s replace the “AppTest.java” inside “src/test/java/” with this JUnit class:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 |
package com.examples; import static org.junit.Assert.assertEquals; import java.io.File; import java.io.IOException; import org.apache.commons.io.FileUtils; import org.junit.Before; import org.junit.Test; public class MavenEmbedderRunnerTest { private MavenEmbedderRunner runner; @Before public void setup() { runner = new MavenEmbedderRunner(); } @Test public void testRunnerOnSimpleProject() { String baseDir = new File("target/test-classes/maven-quickstart-example").getAbsolutePath(); assertEquals(0, runner.run(baseDir, "clean", "verify")); } @Test public void testRunnerWithLocalRepo() throws IOException { File localRepo = new File("target/test-classes/local-repo"); FileUtils.deleteDirectory(localRepo); String baseDir = new File("target/test-classes/maven-quickstart-example").getAbsolutePath(); assertEquals(0, runner.run(baseDir, "-Dmaven.repo.local=" + localRepo.getAbsolutePath(), "clean", "verify")); } } |
The first test is simpler: it runs the embedded Maven with the goals “clean” and “verify” on the test project we created above. The second one is more oriented to a proper integration test since it also passes the standard system property to tell Maven to use another local repository (not the default one “~/.m2/repository”). In such a test, we use a temporary local repository inside the target folder and always wipe its contents before the test. This way, Maven will always start with an empty local repository and download everything from scratch for building the test project in this test. On the contrary, the first test, when running the embedded Maven, will use the same local repository of your user.
The first test will be faster but will add Maven artifacts to your local Maven repository. This might be bad if you run the “install” phase on the test project because the test project artifacts will be uselessly stored in your local Maven repository.
The second test will be slower since it will always download dependencies and plugins from scratch. However, it will be completely isolated, which is good for tests and makes it more reproducible.
Note that we are not running Maven on the test project stored in “src/test/reources” to avoid cluttering the test project with generated Maven artifacts: we build the test project copied in the “target/test-classes”.
In both cases, we expect success (as usual, a 0 return value means success).
In a more realistic integration test, we should also verify the presence of some generated artifacts, like the JAR and the executed tests. However, this is easier with the Maven Verifier Component, which I’ll describe in another post.
IMPORTANT: if you run these tests from Eclipse and they fail because the Embedded Maven cannot find the test project to build, run “Project -> Clean” so that Eclipse will force the copying of the test project from “src/test/resources” to “target/test-classes” directory, where the tests expect the test project. Such a copy should happen automatically, but sometimes Eclipse goes out of sync and removes the copied test resources.
If you run such tests, you’ll see the logging of the embedded Maven on the console while it builds the test project. For example, something like that (the log is actually full of additional information like the Java class of the current goal; I replaced such noise with “…” in the shown log below):
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 |
[main] INFO ... - Scanning for projects... [main] INFO ... - [main] INFO ... - ---------------< com.examples:maven-quickstart-example >---------------- [main] INFO ... - Building maven-quickstart-example 1.0-SNAPSHOT [main] INFO ... - from pom.xml [main] INFO ... - --------------------------------[ jar ]--------------------------------- [main] INFO ... - [main] INFO ... - --- clean:3.1.0:clean (default-clean) @ maven-quickstart-example --- [main] INFO ... - [main] INFO ... - --- resources:3.0.2:resources (default-resources) @ maven-quickstart-example --- [main] INFO ... - Using 'UTF-8' encoding to copy filtered resources. [main] INFO ... - skip non existing resourceDirectory /Users/bettini/work/maven/maven-embedder-example/target/test-classes/maven-quickstart-example/src/main/resources [main] INFO ... - [main] INFO ... - --- compiler:3.8.0:compile (default-compile) @ maven-quickstart-example --- [main] INFO ... - Changes detected - recompiling the module! [main] INFO ... - Compiling 1 source file to /Users/bettini/work/maven/maven-embedder-example/target/test-classes/maven-quickstart-example/target/classes [main] INFO ... - [main] INFO ... - --- resources:3.0.2:testResources (default-testResources) @ maven-quickstart-example --- [main] INFO ... - Using 'UTF-8' encoding to copy filtered resources. [main] INFO ... - skip non existing resourceDirectory /Users/bettini/work/maven/maven-embedder-example/target/test-classes/maven-quickstart-example/src/test/resources [main] INFO ... - [main] INFO ... - --- compiler:3.8.0:testCompile (default-testCompile) @ maven-quickstart-example --- [main] INFO org.apache.maven.plugin.compiler.TestCompilerMojo - Changes detected - recompiling the module! [main] INFO ... - Compiling 1 source file to /Users/bettini/work/maven/maven-embedder-example/target/test-classes/maven-quickstart-example/target/test-classes [main] INFO ... - [main] INFO ... - --- surefire:2.22.1:test (default-test) @ maven-quickstart-example --- [main] INFO ... - [main] INFO ... - ------------------------------------------------------- [main] INFO ... - T E S T S [main] INFO ... - ------------------------------------------------------- [ThreadedStreamConsumer] INFO ... - Running com.examples.AppTest [ThreadedStreamConsumer] INFO ... - Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.013 s - in com.examples.AppTest [main] INFO ... - [main] INFO ... - Results: [main] INFO ... - [main] INFO ... - Tests run: 1, Failures: 0, Errors: 0, Skipped: 0 [main] INFO ... - [main] INFO ... - [main] INFO ... - --- jar:3.0.2:jar (default-jar) @ maven-quickstart-example --- [main] INFO ... - Building jar: /Users/bettini/work/maven/maven-embedder-example/target/test-classes/maven-quickstart-example/target/maven-quickstart-example-1.0-SNAPSHOT.jar [main] INFO ... - ------------------------------------------------------------------------ [main] INFO ... - BUILD SUCCESS [main] INFO ... - ------------------------------------------------------------------------ [main] INFO ... - Total time: 1.340 s [main] INFO ... - Finished at: 2024-02-14T20:32:32+01:00 [main] INFO ... - ------------------------------------------------------------------------ |
REMEMBER: this is not the output of the main project’s build; it is the embedded Maven running the build from our JUnit test on the test project.
Note that the two tests will build the same test project. In a more realistic integration test scenario, each test should build a different test project.
If you only run the second test after it finishes, you can inspect the “target/test-classes” to see the results of the build (note the “local-repo” containing all the downloaded dependencies and plugins for the test project and the generated artifacts, including test results, for the test project):
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 |
ls target/test-classes/local-repo/ backport-util-concurrent classworlds com commons-io junit org tree target/test-classes/maven-quickstart-example/ target/test-classes/maven-quickstart-example/ ├── pom.xml ├── src │ ├── main │ │ └── java │ │ └── com │ │ └── examples │ │ └── App.java │ └── test │ └── java │ └── com │ └── examples │ └── AppTest.java └── target ├── classes │ └── com │ └── examples │ └── App.class ├── generated-sources │ └── annotations ├── generated-test-sources │ └── test-annotations ├── maven-archiver │ └── pom.properties ├── maven-quickstart-example-1.0-SNAPSHOT.jar ├── maven-status │ └── maven-compiler-plugin │ ├── compile │ │ └── default-compile │ │ ├── createdFiles.lst │ │ └── inputFiles.lst │ └── testCompile │ └── default-testCompile │ ├── createdFiles.lst │ └── inputFiles.lst ├── surefire-reports │ ├── 2024-02-17T10-26-50_728.dumpstream │ ├── com.examples.AppTest.txt │ └── TEST-com.examples.AppTest.xml └── test-classes └── com └── examples └── AppTest.class 28 directories, 14 files |
Now, you can continue experimenting with the Maven Embedder.
In the next articles, we’ll see how to use the Maven Embedder when running Maven integration tests (typically, for integration tests of Maven plugins), e.g., together with the Maven Verifier Component.
Stay tuned 🙂