Maven Artifacts

An artifact is a file (more precisely, a sequence of bytes) that can be addressed using its coordinates and which Maven downloads, installs, or deploys for you. Most artifacts are POMs and JARs, but an artifact can be really any file. A very important thing about artifacts is that they have coordinates, so they are not “just files”, but they are files that are in some way addressable by Maven.

Artifact coordinates are most often represented as groupId:artifactId:version, or GAV in short. (Please note that artifact coordinates have more fields, but for brevity we still call the coordinates “GAV”, not “GAVCE”). The artifact coordinates uniquely identify an artifact, but do not specify anything about its source. It is up to Maven to figure out (or you to tell Maven how to figure out) how and from where to retrieve the artifact.

A word about uniqueness: as stated above, GAV coordinates uniquely identify an artifact, but only within one repository. Different repositories can contain artifacts with the same GAV. (This is normal with mirror repositories.) If those files are not identical, it can cause severe issues without you noticing it. In short, these cases should be avoided.

Artifact Properties

The artifacts that Maven uses have the following coordinate properties:

Name Description
groupId The project group
artifactId The artifact ID
version The artifact version (linked with baseVersion)
baseVersion The artifact base version (linked with version)
classifier The artifact classifier (optional)
extension The artifact extension (default: “jar”)

One property worth explaining is a bit of special one: baseVersion is derived from/linked to version (or the other way around, depending on the context). For release artifacts, it has the same value as version, whereas for snapshot artifacts, it has the “non-timestamped snapshot version”. For example, snapshot version “1.0-20220119.164608-1” has the baseVersion “1.0-SNAPSHOT”. So, version and baseVersion are linked, derived from each other, but they have different values only in the case of snapshots.

Maven pom.xml files identify artifacts via coordinates in four different ways, depending on what the artifact is and how it will be used.

  • A dependency of the project, often though not always a jar archive, is referenced by a dependency element in either the dependencies or dependenciesManagement section.
  • The pom.xml file itself has coordinates given by the top-level groupId, artifactId, and version elements.
  • A build plugin is referenced by a plugin element in the plugins section.
  • A build extension is referenced by an extension element in the extensions section.

Coordinates of Dependencies

Artifact coordinates for dependencies are calculated from a dependency element:

Artifact Coordinate POM element Coordinate Value
groupId project/dependencies/dependency/groupId -> group ID
artifactId project/dependencies/dependency/artifactId -> artifact ID
version project/dependencies/dependency/version -> version string
classifier project/dependencies/dependency/classifier -> classifier, or type handler provided
extension project/dependencies/dependency/type -> type handler provided, or same as type

This also applies when the dependency element is a child of a dependencyManagement element.

Notice that the dependency element does not have an extension element. Instead it has a type element which is used to derive the extension and sometimes the classifier. Out of the box, Maven Core defines 11 “types” (handled by the same named ArtifactHandler components):

Type Name Extension Classifier
pom pom
jar jar
maven-plugin jar
ear ear
ejb jar
ejb-client jar ejb-client
javadoc jar javadoc
java-source jar sources
rar rar
test-jar jar tests
war war
any any

From the table above, you can see that if the dependency type is “war”, the extension is also war and the classifier is the value of the classifier element (if present) or the empty string if the classifier element is not present. If the type is “test-jar”, the extension is “jar” and the classifier is “tests”. If the type is not one of these 11 names, then the value of the “type” is used as the extension. For example, if the type element is <type>tar.gz</type>, the extension will be tar.gz, and the classifier will be set by the classifier element. This table may be extended by plugins and extensions used in the build.

This has “interesting” consequences. Consider the artifact org.project:reusable-test-support:1.0:tests:jar. Maybe surprisingly, a dependency on this artifact can be described in two ways:

<dependency>
  <groupId>org.project</groupId>
  <artifactId>reusable-test-support</artifactId>
  <version>1.0</version>
  <classifier>tests</classifier>
</dependency>

and the equivalent dependency would be:

<dependency>
  <groupId>org.project</groupId>
  <artifactId>reusable-test-support</artifactId>
  <version>1.0</version>
  <type>test-jar</type>
</dependency>

The obvious difference is that the first example has the <classifier>tests</classifier>, while the second has the <type>test-jar</type>. The type “test-jar” implies a classifier of “tests”. In both cases, the extension is “jar”. The first uses the default value for the extension, while the second derives it from the type.

Note: In this case, the first way is somewhat “explicit”, and is recommended. When the type handler carries important extra information such as custom packaging, using type is more appropriate. Simply put, in this example the type “test-jar” is like an alias for ordinary JARs with the “tests” classifier.

Coordinates of POM Files

The coordinates of a POM artifact that contain a project's pom.xml file are derived from the top-level groupId, artifactId, and version elements in that pom.xml file:

Artifact Coordinate POM element Coordinate Value
groupId project/groupId -> group ID
artifactId project/artifactId -> artifact ID
version project/version -> version string
classifier - "" (always)
extension - “pom” (always)

Coordinate values are computed after the POM is made into an effective POM; that is, after parent values have been inherited.

Coordinates of Build Plugins and Build Extensions

Build plugin and build extension artifacts are JARs. For build plugins, this is how the corresponding coordinates are computed from a plugin element:

Artifact Coordinate POM element Coordinate Value
groupId project/build/plugins/plugin/groupId -> group ID
artifactId project/build/plugins/plugin/artifactId -> artifact ID
version project/build/plugins/plugin/version -> version string
classifier - -> "" (always)
extension - -> “jar” (always)

This also applies when the plugin element is a child of a pluginManagement element.

Build extensions are similarly computed from an extension element (which is not the same as and should not be confused with the extension artifact coordinate):

Artifact Coordinate POM element Coordinate Value
groupId project/build/extensions/extension/groupId -> group ID
artifactId project/build/extensions/extension/artifactId -> artifact ID
version project/build/extensions/extension/version -> version string
classifier - -> "" (always)
extension - -> “jar” (always)

Note: The extension artifact coordinate and a Maven build extension are two completely different things that unfortunately share the name “extension”. A Maven build extension is a JAR archive that is added to the project class loader's classpath. It is referenced by an extension element in pom.xml. An extension coordinate is usually the filename extension of an artifact's JAR file such as jar, zip, or txt. This extension is most often set to an implicit default value, but can be changed by the type child of a dependency element.

Summary

This is how various Maven bits like “project”, “parent project”, “plugin”, “extension”, and “dependency” derive artifact coordinates from POM elements.