Maven CI Friendly Versions

Starting with Maven 3.5.0-beta-1 you can use the ${revision}, ${sha1} and/or ${changelist} as placeholders for the version in your pom file.

Single Project Setup

This can look like this:

  1. <project>
  2. <modelVersion>4.0.0</modelVersion>
  3. <parent>
  4. <groupId>org.apache</groupId>
  5. <artifactId>apache</artifactId>
  6. <version>18</version>
  7. </parent>
  8. <groupId>org.apache.maven.ci</groupId>
  9. <artifactId>ci-parent</artifactId>
  10. <name>First CI Friendly</name>
  11. <version>${revision}</version>
  12. ...
  13. </project>

This is of course a simple situation where we use only ${revision} for brevity to show the general course.

Based on the above pom you can build your project using:

mvn clean package

But wait there is a problem? Which version will the artifacts have? So you need to define the version for your artifacts. The first possibility is to use the command line like this:

mvn -Drevision=1.0.0-SNAPSHOT clean package

This will become cumbersome over the time. So the other solution for this is to simply use a property inside the pom file which looks like this:

  1. <project>
  2. <modelVersion>4.0.0</modelVersion>
  3. <parent>
  4. <groupId>org.apache</groupId>
  5. <artifactId>apache</artifactId>
  6. <version>18</version>
  7. </parent>
  8. <groupId>org.apache.maven.ci</groupId>
  9. <artifactId>ci-parent</artifactId>
  10. <name>First CI Friendly</name>
  11. <version>${revision}</version>
  12. ...
  13. <properties>
  14. <revision>1.0.0-SNAPSHOT</revision>
  15. </properties>
  16. </project>

So now you can simply call Maven as usual like mvn clean package.

You can of course change the version via the command line like this:

mvn -Drevision=2.0.0-SNAPSHOT clean package

Of course, you can use the .mvn/maven.config file for this.

A note about the used properties. You can only use those named ${revision}, ${sha1} and/or ${changelist} and not other named properties like this:

  1. <project>
  2. <modelVersion>4.0.0</modelVersion>
  3. <parent>
  4. <groupId>org.apache</groupId>
  5. <artifactId>apache</artifactId>
  6. <version>18</version>
  7. </parent>
  8. <groupId>org.apache.maven.ci</groupId>
  9. <artifactId>ci-parent</artifactId>
  10. <name>First CI Friendly</name>
  11. <version>${revision}</version>
  12. ...
  13. <properties>
  14. <revision>1.0.0-${buildNumber}-SNAPSHOT</revision>
  15. </properties>
  16. </project>

The above example will not work as expected. If you like to have more flexibility you can use a combination of the different properties like this:

  1. <project>
  2. <modelVersion>4.0.0</modelVersion>
  3. <parent>
  4. <groupId>org.apache</groupId>
  5. <artifactId>apache</artifactId>
  6. <version>18</version>
  7. </parent>
  8. <groupId>org.apache.maven.ci</groupId>
  9. <artifactId>ci-parent</artifactId>
  10. <name>First CI Friendly</name>
  11. <version>${revision}${sha1}${changelist}</version>
  12. ...
  13. <properties>
  14. <revision>1.3.1</revision>
  15. <changelist>-SNAPSHOT</changelist>
  16. <sha1/>
  17. </properties>
  18. </project>

If you like to make a version 2.0.0-SNAPSHOT this can simply being achieved by using this:

mvn -Drevision=2.0.0 clean package

Another usage example can be to make a release which can be done via (version 1.3.1):

mvn -Dchangelist= clean package

Or if you like to make a release with another version:

mvn -Drevision=2.7.8 -Dchangelist= clean package

Please read until the end of this documentation!

Multi Module Setup

So now let us take a look into a situation where we have a multi module build. We have a parent pom and one or more children. The parent pom will look like this:

  1. <project>
  2. <modelVersion>4.0.0</modelVersion>
  3. <parent>
  4. <groupId>org.apache</groupId>
  5. <artifactId>apache</artifactId>
  6. <version>18</version>
  7. </parent>
  8. <groupId>org.apache.maven.ci</groupId>
  9. <artifactId>ci-parent</artifactId>
  10. <name>First CI Friendly</name>
  11. <version>${revision}</version>
  12. ...
  13. <properties>
  14. <revision>1.0.0-SNAPSHOT</revision>
  15. </properties>
  16. <modules>
  17. <module>child1</module>
  18. ..
  19. </modules>
  20. </project>

The child will look like this:

  1. <project>
  2. <modelVersion>4.0.0</modelVersion>
  3. <parent>
  4. <groupId>org.apache.maven.ci</groupId>
  5. <artifactId>ci-parent</artifactId>
  6. <version>${revision}</version>
  7. </parent>
  8. <groupId>org.apache.maven.ci</groupId>
  9. <artifactId>ci-child</artifactId>
  10. ...
  11. </project>

A multi module build can of course build the same way as the single project setup. You should define the version either via property in the parent or use the .mvn/maven.config file.

Dependencies

In a multi module build you have often the case that you define dependencies between module(s). The usual way of defining dependencies and their appropriate versions has been to use ${project.version} and this has not been changed.

So the correct way to do such things can be seen in the following example:

  1. <project>
  2. <modelVersion>4.0.0</modelVersion>
  3. <parent>
  4. <groupId>org.apache</groupId>
  5. <artifactId>apache</artifactId>
  6. <version>18</version>
  7. </parent>
  8. <groupId>org.apache.maven.ci</groupId>
  9. <artifactId>ci-parent</artifactId>
  10. <name>First CI Friendly</name>
  11. <version>${revision}</version>
  12. ...
  13. <properties>
  14. <revision>1.0.0-SNAPSHOT</revision>
  15. </properties>
  16. <modules>
  17. <module>child1</module>
  18. ..
  19. </modules>
  20. </project>

The child will look like this:

  1. <project>
  2. <modelVersion>4.0.0</modelVersion>
  3. <parent>
  4. <groupId>org.apache.maven.ci</groupId>
  5. <artifactId>ci-parent</artifactId>
  6. <version>${revision}</version>
  7. </parent>
  8. <groupId>org.apache.maven.ci</groupId>
  9. <artifactId>ci-child</artifactId>
  10. ...
  11. <dependencies>
  12. <dependency>
  13. <groupId>org.apache.maven.ci</groupId>
  14. <artifactId>child2</artifactId>
  15. <version>${project.version}</version>
  16. </dependency>
  17. </dependencies>
  18. </project>

If you try to use ${revision} instead of ${project.version} your build will fail.

Install / Deploy

If you like to install or deploy artifacts by using the above setup you have to use the flatten-maven-plugin otherwise you will install/deploy artifacts in your repository which will not be consumable by Maven anymore. Such kind of setup will look like this:

  1. <project>
  2. <modelVersion>4.0.0</modelVersion>
  3. <parent>
  4. <groupId>org.apache</groupId>
  5. <artifactId>apache</artifactId>
  6. <version>18</version>
  7. </parent>
  8. <groupId>org.apache.maven.ci</groupId>
  9. <artifactId>ci-parent</artifactId>
  10. <name>First CI Friendly</name>
  11. <version>${revision}</version>
  12. ...
  13. <properties>
  14. <revision>1.0.0-SNAPSHOT</revision>
  15. </properties>
  16. <build>
  17. <plugins>
  18. <plugin>
  19. <groupId>org.codehaus.mojo</groupId>
  20. <artifactId>flatten-maven-plugin</artifactId>
  21. <version>1.1.0</version>
  22. <configuration>
  23. <updatePomFile>true</updatePomFile>
  24. <flattenMode>resolveCiFriendliesOnly</flattenMode>
  25. </configuration>
  26. <executions>
  27. <execution>
  28. <id>flatten</id>
  29. <phase>process-resources</phase>
  30. <goals>
  31. <goal>flatten</goal>
  32. </goals>
  33. </execution>
  34. <execution>
  35. <id>flatten.clean</id>
  36. <phase>clean</phase>
  37. <goals>
  38. <goal>clean</goal>
  39. </goals>
  40. </execution>
  41. </executions>
  42. </plugin>
  43. </plugins>
  44. </build>
  45. <modules>
  46. <module>child1</module>
  47. ..
  48. </modules>
  49. </project>