------ Introduction to the Dependency Mechanism ------ Brett Porter Trygve Laugstol ------ 2005-10-12 ------ ~~ Licensed to the Apache Software Foundation (ASF) under one ~~ or more contributor license agreements. See the NOTICE file ~~ distributed with this work for additional information ~~ regarding copyright ownership. The ASF licenses this file ~~ to you under the Apache License, Version 2.0 (the ~~ "License"); you may not use this file except in compliance ~~ with the License. You may obtain a copy of the License at ~~ ~~ http://www.apache.org/licenses/LICENSE-2.0 ~~ ~~ Unless required by applicable law or agreed to in writing, ~~ software distributed under the License is distributed on an ~~ "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY ~~ KIND, either express or implied. See the License for the ~~ specific language governing permissions and limitations ~~ under the License. ~~ NOTE: For help with the syntax of this file, see: ~~ http://maven.apache.org/doxia/references/apt-format.html Introduction to the Dependency Mechanism Dependency management is one of the features of Maven that is best known to users and is one of the areas where Maven excels. There is not much difficulty in managing dependencies for a single a project, but when you start getting into dealing with multi-module projects and applications that consist of tens or hundreds of modules this is where Maven can help you a great deal in maintaining a high degree of control and stability. Learn more about: * {{{Transitive_Dependencies}Transitive Dependencies}} * Excluded/Optional Dependencies * {{{Dependency_Scope}Dependency Scope}} * {{{Dependency_Management}Dependency Management}} * {{{Importing_Dependencies}Importing Dependencies}} * {{{System_Dependencies}System Dependencies}} [] * {Transitive Dependencies} Transitive dependencies are a new feature in Maven 2.0. This allows you to avoid needing to discover and specify the libraries that your own dependencies require, and including them automatically. This feature is facilitated by reading the project files of your dependencies from the remote repositories specified. In general, all dependencies of those projects are used in your project, as are any that the project inherits from its parents, or from its dependencies, and so on. There is no limit to the number of levels that dependencies can be gathered from, and will only cause a problem if a cyclic dependency is discovered. With transitive dependencies, the graph of included libraries can quickly grow quite large. For this reason, there are some additional features that will limit which dependencies are included: * - this determines what version of a dependency will be used when multiple versions of an artifact are encountered. Currently, Maven 2.0 only supports using the "nearest definition" which means that it will use the version of the closest dependency to your project in the tree of dependencies. You can always guarantee a version by declaring it explicitly in your project's POM. Note that if two dependency versions are at the same depth in the dependency tree, until Maven 2.0.8 it was not defined which one would win, but since Maven 2.0.9 it's the order in the declaration that counts: the first declaration wins. * "nearest definition" means that the version used will be the closest one to your project in the tree of dependencies, eg. if dependencies for A, B, and C are defined as A -> B -> C -> D 2.0 and A -> E -> D 1.0, then D 1.0 will be used when building A because the path from A to D through E is shorter. You could explicitly add a dependency to D 2.0 in A to force the use of D 2.0 * - this allows project authors to directly specify the versions of artifacts to be used when they are encountered in transitive dependencies or in dependencies where no version has been specified. In the example in the preceding section a dependency was directly added to A even though it is not directly used by A. Instead, A can include D as a dependency in its dependencyManagement section and directly control which version of D is used when, or if, it is ever referenced. * - this allows you to only include dependencies appropriate for the current stage of the build. This is described in more detail below. * - If project X depends on project Y, and project Y depends on project Z, the owner of project X can explicitly exclude project Z as a dependency, using the "exclusion" element. * - If project Y depends on project Z, the owner of project Y can mark project Z as an optional dependency, using the "optional" element. When project X depends on project Y, X will depend only on Y and not on Y's optional dependency Z. The owner of project X may then explicitly add a dependency on Z, at her option. (It may be helpful to think of optional dependencies as "excluded by default.") [] * {Dependency Scope} Dependency scope is used to limit the transitivity of a dependency, and also to affect the classpath used for various build tasks. There are 6 scopes available: * <>\ This is the default scope, used if none is specified. Compile dependencies are available in all classpaths of a project. Furthermore, those dependencies are propagated to dependent projects. * <>\ This is much like <<>>, but indicates you expect the JDK or a container to provide the dependency at runtime. For example, when building a web application for the Java Enterprise Edition, you would set the dependency on the Servlet API and related Java EE APIs to scope <<>> because the web container provides those classes. This scope is only available on the compilation and test classpath, and is not transitive. * <>\ This scope indicates that the dependency is not required for compilation, but is for execution. It is in the runtime and test classpaths, but not the compile classpath. * <>\ This scope indicates that the dependency is not required for normal use of the application, and is only available for the test compilation and execution phases. * <>\ This scope is similar to <<>> except that you have to provide the JAR which contains it explicitly. The artifact is always available and is not looked up in a repository. * <> <(only available in Maven 2.0.9 or later)>\ This scope is only used on a dependency of type <<>> in the <<<\>>> section. It indicates that the specified POM should be replaced with the dependencies in that POM's <<<\>>> section. Since they are replaced, dependencies with a scope of <<>> do not actually participate in limiting the transitivity of a dependency. [] Each of the scopes (except for <<>>) affects transitive dependencies in different ways, as is demonstrated in the table below. If a dependency is set to the scope in the left column, transitive dependencies of that dependency with the scope across the top row will result in a dependency in the main project with the scope listed at the intersection. If no scope is listed, it means the dependency will be omitted. *----------+------------+----------+----------+------+ | | compile | provided | runtime | test *----------+------------+----------+----------+------+ | compile | compile(*) | - | runtime | - *----------+------------+----------+----------+------+ | provided | provided | - | provided | - *----------+------------+----------+----------+------+ | runtime | runtime | - | runtime | - *----------+------------+----------+----------+------+ | test | test | - | test | - *----------+------------+----------+----------+------+ <<(*) Note:>> it is intended that this should be runtime scope instead, so that all compile dependencies must be explicitly listed - however, there is the case where the library you depend on extends a class from another library, forcing you to have available at compile time. For this reason, compile time dependencies remain as compile scope even when they are transitive. * {Dependency Management} The dependency management section is a mechanism for centralizing dependency information. When you have a set of projects that inherits a common parent it's possible to put all information about the dependency in the common POM and have simpler references to the artifacts in the child POMs. The mechanism is best illustrated through some examples. Given these two POMs which extend the same parent: Project A: +----+ ... group-a artifact-a 1.0 group-c excluded-artifact group-a artifact-b 1.0 bar runtime +----+ Project B: +----+ ... group-c artifact-b 1.0 war runtime group-a artifact-b 1.0 bar runtime +----+ These two example POMs share a common dependency and each has one non-trivial dependency. This information can be put in the parent POM like this: +----+ ... group-a artifact-a 1.0 group-c excluded-artifact group-c artifact-b 1.0 war runtime group-a artifact-b 1.0 bar runtime +----+ And then the two child poms would become much simpler: +----+ ... group-a artifact-a group-a artifact-b bar +----+ +----+ ... group-c artifact-b war group-a artifact-b bar +----+ <> In two of these dependency references, we had to specify the \ element. This is because the minimal set of information for matching a dependency reference against a dependencyManagement section is actually <<\{groupId, artifactId, type, classifier\}>>. In many cases, these dependencies will refer to jar artifacts with no classifier. This allows us to shorthand the identity set to <<\{groupId, artifactId\}>>, since the default for the type field is <<>>, and the default classifier is null. A second, and very important use of the dependency management section is to control the versions of artifacts used in transitive dependencies. As an example consider these projects: Project A: +----+ 4.0.0 maven A pom A 1.0 test a 1.2 test b 1.0 compile test c 1.0 compile test d 1.2 +----+ Project B: +----+ A maven 1.0 4.0.0 maven B pom B 1.0 test d 1.0 test a 1.0 runtime test c runtime +----+ When maven is run on project B version 1.0 of artifacts a, b, c, and d will be used regardless of the version specified in their pom. * a and c both are declared as dependencies of the project so version 1.0 is used due to dependency mediation. Both will also have runtime scope since it is directly specified. * b is defined in B's parent's dependency management section and since dependency management takes precedence over dependency mediation for transitive dependencies, version 1.0 will be selected should it be referenced in a or c's pom. b will also have compile scope. * Finally, since d is specified in B's dependency management section, should d be a dependency (or transitive dependency) of a or c, version 1.0 will be chosen - again because dependency management takes precedence over dependency mediation and also because the current pom's declaration takes precedence over its parent's declaration. [] The reference information about the dependency management tags is available from the {{{../../ref/current/maven-model/maven.html#class_DependencyManagement}project descriptor reference}}. ** {Importing Dependencies} The examples in the previous section describe how to specify managed dependencies through inheritence. However, in larger projects it may be impossible to accomplish this since a project can only inherit from a single parent. To accommodate this, projects can import managed dependencies from other projects. This is accomplished by declaring a pom artifact as a dependency with a scope of "import". Project B: +----+ 4.0.0 maven B pom B 1.0 maven A 1.0 pom import test d 1.0 test a 1.0 runtime test c runtime +----+ Assuming A is the pom defined in the preceding example, the end result would be the same. All of A's managed dependencies would be incorporated into B except for d since it is defined in this pom. Project X: +----+ 4.0.0 maven X pom X 1.0 test a 1.1 test b 1.0 compile +----+ Project Y: +----+ 4.0.0 maven Y pom Y 1.0 test a 1.2 test c 1.0 compile +----+ Project Z: +----+ 4.0.0 maven Z pom Z 1.0 maven X 1.0 pom import maven Y 1.0 pom import +----+ In the example above Z imports the managed dependencies from both X and Y. However, both X and Y contain dependency a. Here, version 1.1 of a would be used since X is declared first and a is not declared in Z's dependencyManagement. This process is recursive. For example, if X imports another pom, Q, when Z is processed it will simply appear that all of Q's managed dependencies are defined in X. Imports are most effective when used for defining a "library" of related artifacts that are generally part of a multiproject build. It is fairly common for one project to use one or more artifacts from these libraries. However, it has sometimes been difficult to keep the versions in the project using the artifacts in synch with the versions distributed in the library. The pattern below illustrates how a "bill of materials" (BOM) can be created for use by other projects. The root of the project is the BOM pom. It defines the versions of all the artifacts that will be created in the library. Other projects that wish to use the library should import this pom into the dependencyManagement section of their pom. +----+ 4.0.0 com.test bom 1.0.0 pom 1.0.0 1.0.0 com.test project1 ${project1Version} com.test project2 ${project1Version} parent +----+ The parent subproject has the BOM pom as its parent. It is a normal multiproject pom. +----+ 4.0.0 com.test 1.0.0 bom com.test parent 1.0.0 pom log4j log4j 1.2.12 commons-logging commons-logging 1.1.1 project1 project2 +----+ Next are the actual project poms. +----+ 4.0.0 com.test 1.0.0 parent com.test project1 ${project1Version} jar log4j log4j 4.0.0 com.test 1.0.0 parent com.test project2 ${project2Version} jar commons-logging commons-logging +----+ The project that follows shows how the library can now be used in another project without having to specify the dependent project's versions. +----+ 4.0.0 com.test use 1.0.0 jar com.test bom 1.0.0 pom import com.test project1 com.test project2 +----+ Finally, when creating projects that import dependencies beware of the following: * Do not attempt to import a pom that is defined in a submodule of the current pom. Attempting to do that will result in the build failing since it won't be able to locate the pom. * Never declare the pom importing a pom as the parent (or grandparent, etc) of the target pom. There is no way to resolve the circularity and an exception will be thrown. * When referring to artifacts whose poms have transitive dependencies the project will need to specify versions of those artifacts as managed dependencies. Not doing so will result in a build failure since the artifact may not have a version specified. (This should be considered a best practice in any case as it keeps the versions of artifacts from changing from one build to the next). * {System Dependencies} Dependencies with the scope are always available and are not looked up in repository. They are usually used to tell Maven about dependencies which are provided by the JDK or the VM. Thus, system dependencies are especially useful for resolving dependencies on artifacts which are now provided by the JDK, but where available as separate downloads earlier. Typical example are the JDBC standard extensions or the Java Authentication and Authorization Service (JAAS). A simple example would be: +----+ ... javax.sql jdbc-stdext 2.0 system ${java.home}/lib/rt.jar ... +----+ If your artifact is provided by the JDK's <<>> the system path would be defined as follows: +----+ ... sun.jdk tools 1.5.0 system ${java.home}/../lib/tools.jar ... +----+