This feature is really a bucket for key things we want to fix in the short-term for Dependency Management, many of which have require (or have the potential for) a strategic solution.
As this 'feature' is a list of bug fixes, this feature spec will not follow the usual template.
See GRADLE-2861
Currently, the POM parser (inherited from Ivy) attaches special extra attributes to the ModuleDescriptor
for a POM. These are later used by the POM parser
when it parses a child POM. Sometimes these attributes cause badly formed XML to be generated, hence the failure listed in the jira issue.
The solution is to have the parser request the parent POM artifact directly, rather than indirectly via the module meta-data:
- Add a
LocallyAvailableExternalResource getArtifact(Artifact)
method toDescriptorParseContext
.- Implementation can reuse the
ModuleVersionResolveResult
from the existinggetModuleDescriptor()
method. This result includes anArtifactResolver
which can be used to resolve anArtifact
to aFile
. There's an example of how to adapt aFile
to aLocallyAvailableExternalResource
instance inAbstractModuleDescriptorParser.parseMetaData()
.
- Implementation can reuse the
- Change the
GradlePomModuleDescriptorParser.parseOtherPom()
to use this new method to fetch and parse the parent POM artifact, rather than using the parsedModuleDescriptor
for the parent. For this step, can continue to represent the parent pom using aModuleDescriptor
inside the parser. - Change
GradlePomModuleDescriptorParser
to represent the parent POM using aPomReader
rather than aModuleDescriptor
. - Clean out
GradlePomModuleDescriptorBuilder
so that it no longer defines any extra properties on the parsedModuleDescriptor
. - Change
IvyXmlModuleDescriptorParser.parseOtherIvyFile()
to use the new method to fetch and parse the Ivy descriptor artifact. - Remove
DescriptorParseContext.getModuleDescriptor()
. It should no longer be required.
- Unignore the existing test case in
BadPomFileResolveIntegrationTest
. - Add a test case to
MavenParentPomResolveIntegrationTest
to cover two Maven modules that share a common parent. - Add a test case to
MavenParentPomResolveIntegrationTest
to cover a Maven module that has a parent and grandparent module.
See this post See GRADLE-2752
- Allow self-resolving dependencies to imply zero or more module versions.
- Change the resolution algorithm to consider this during conflict resolution. Self-resolving dependencies cannot be evicted and must always be selected. Assert that the self-resolving dependency is compatible-with the requested version. Fail resolution if not.
- Change the Java plugin to attach current project's identifier to the main and test classes.
- To infer the identifier of the current project.
- Identifier of any publication.
- Project (group, name, version).
- Maven deployer.
- Project (group, archivesBaseName, version).
- To infer the identifier of the current project.
- Change the resolution algorithm so that the root node is not considered by conflict resolution.
TBD - A later story should travel back from each publication to determine if the main or test binary is included in the publication. Use this publication's identifier if so.
See GRADLE-2516
- Change the local dependencies (eg
gradleApi()
,localGroovy()
) to imply the various modules that they contribute to the result.
See GRADLE-2502
- Using
latest.integration
- Empty repository fails with not found.
- Publish
1.0
and1.1
with statusintegration
. Resolves to1.1
. - Publish
1.2
with statusrelease
. Resolves to1.2
- Publish
1.3
with no ivy.xml. Resolves to1.3
.
- Using
latest.milestone
- Empty repository fails with not found.
- Publish
2.0
with no ivy.xml. Fails with not found. - Publish
1.3
with statusintegration
. Fails with not found. - Publish
1.0
and1.1
with ivy.xml and statusmilestone
. Resolves to1.1
. - Publish
1.2
with statusrelease
. Resolves to1.2
- Using
latest.release
- Empty repository fails with not found.
- Publish
2.0
with no ivy.xml. Fails with not found. - Publish
1.3
with statusmilestone
. Fails with not found. - Publish
1.0
and1.1
with ivy.xml and statusrelease
. Resolves to1.1
.
- Multiple repositories.
- Checking for changes. Using
latest.release
- Publish
1.0
with statusrelease
and2.0
with statusmilestone
. - Resolve and assert directory listing and
1.0
artifacts downloaded. - Resolve and assert directory listing downloaded.
- Publish
1.1
with statusrelease
. - Resolve and assert directory listing and
1.1
artifacts downloaded.
- Publish
- Maven integration
- Publish
1.0
. Checklatest.integration
resolves to1.0
andlatest.release
fails with not found. - Publish
1.1-SNAPSHOT
. Checklatest.integration
resolves to1.1-SNAPSHOT
andlatest.release
fails with not found.
- Publish
- Version ranges
- Repository with multiple patterns.
- Repository with
[type]
in pattern before[revision]
. - Multiple dynamic versions match the same remote revision.
Change ExternalResourceResolver.getDependency() to use the following algorithm:
- Calculate an ordered list of candidate versions.
- For a static version selector the list contains a single candidate.
- For a dynamic version selector the list is the full set of versions for the module.
- For a Maven repository, this is determined using
maven-metadata.xml
if available, falling back to a directory listing. - For an Ivy repository, this is determined using a directory listing.
- Fail if directory listing is not available.
- For a Maven repository, this is determined using
- For each candidate version:
- If the version matcher does not accept the module version, continue.
- Fetch the module version meta-data, as described below. If not found, continue.
- If the version matcher requires the module meta-data and it does not accept the meta-data, continue.
- Use the module version.
- Return not found.
To fetch the meta-data for a module version:
- Download the meta data descriptor resource, via the resource cache. If found, parse.
- Validate module version in meta-data == the expected module version.
- Check for a jar artifact, via the resource cache. If found, use default meta-data. The meta-data must have
default
set totrue
andstatus
set tointegration
. - Return not found.
- GRADLE-2328 - invalidate cached HTTP/HTTPS resource when user credentials change.
- Invalidate cached HTTP/HTTPS resource when proxy settings change.
- Invalidate cached HTTPS resource when SSL settings change.
TBD
TBD
There is currently no consistent approach to exception handling or reporting in our dependency resolution code. Sometimes failures are ignored, sometimes they are reported as a 'not found', and sometimes they bubble up and break resolution, even when the module is available at another repository.
There are 3 goals:
- Add a couple of key firewalls in the code, where we handle failures and make a high-level decision whether to proceed (and warn), or fail the resolution (and add some context).
- Remove all known places in our (inherited) code where we catch and ignore, catch and mark as unknown, or catch and discard the cause.
- Wrap/change each parser to add context on parse failure.
There are 4 possible outcomes when resolving a module or downloading an artifact from a given repository:
- Success
- A remote resource is not found (a 404 status code for an HTTP request, or empty directory listing).
- An expected failure occurs accessing a remote resource (Connect exception, a non-200 status code, or a parse error).
- An unexpected failure occurs (all other failures).
In general, when an expected or unexpected failure occurs resolving from a repository:
- The resolve is aborted for that repository (this means, for example, if we can't parse a .sha1 resource, then we bail).
- If the failure is an unexpected failure, then propogate it and fail the resolve.
- Log the details of the failure.
- If there are no other repositories defined, then fail the resolve.
- If there is another repository defined, proceed to the next repository.
TODO - there are some exceptions to this rule, for example:
- if we get a 401 fetching the .sha1, we should continue to the artifact
- if we get a 401 fetching an artifact, and the repository has multiple root URLs, we should continue on to the next URL.
TODO - flesh this out
TODO - flesh this out
- GRADLE-2188: Artifact not found resolving dependencies with packaging/type "orbit"
Our engine for parsing Maven pom files is borrowed from ivy, and assumes the 'packaging' element equals the artifact type, with a few exceptions (ejb, bundle, eclipse-plugin, maven-plugin). This is different from the way Maven does the calculation, which is:
- Type defaults to 'jar' but can be explicitly declared.
- Maven maps the type to an [extension, classifier] combination using some hardcoded rules. Unknown types are mapped to [type, ""].
- To resolve the artefact, maven looks for an artefact with the given artifactId, version, classifier and extension.
At present, our model of an Artifact is heavily based on ivy; for this fix we can introduce the concept of mapping between our internal model and a repository-centric artifact model. This will be a small step toward an independent Gradle model of artifacts, which then maps to repository specific things link extension, classifier, etc.
- When the dependency declaration has no 'type' specified, or a 'type' that maps to the extension 'jar'
- Resolution of a POM module with packaging in ['', 'pom', 'jar', 'ejb', 'bundle', 'maven-plugin', 'eclipse-plugin'] will not change
- Resolution of a POM with packaging 'foo' that maps to 'module.foo', a deprecation warning will be emitted and the artifact 'module.foo' will be used
- Resolution of a POM with packaging 'foo' that maps to 'module.jar', the artifact 'module.jar' will be successfully found. (ie 'orbit'). An extra HTTP request will be required to first look for 'module.foo'.
- When the dependency declaration has a 'type' specified that maps to an extension 'ext' (other than 'jar')
- Resolution of a POM module with packaging in ['pom', 'jar', 'ejb', 'bundle', 'maven-plugin', 'eclipse-plugin'] will emit a deprecation warning before using 'module.jar' if it exists
- Resolution of a POM with packaging 'foo' and actual artifact 'module.foo', a deprecation warning will be emitted and the artifact 'module.foo' will be used
- Resolution of a POM with packaging 'foo' and actual artifact 'module.ext', the artifact 'module.ext' will be successfully found. An extra HTTP request will be required to first look for 'module.foo'.
- Coverage for resolving pom dependencies referenced in various ways:
- Need modules published in maven repositories with packaging = ['', 'pom', 'jar', 'war', 'eclipse-plugin', 'custom']
- Test resolution of artifacts in these modules via
- Direct dependency in a Gradle project
- Transitive dependency in a maven module (pom) which is itself a dependency of a Gradle project
- Transitive dependency in an ivy module (ivy.xml) which is itself a dependency of a Gradle project
- For 1. and 2., need dependency declaration with and without type attribute specified
- Must verify that deprecation warning is logged appropriately
- Sad-day coverage for the case where neither packaging nor type can successfully locate the maven artifact. Error message should report 'type'-based location.
- Determine 2 locations for the primary artifact:
- The 'packaging' location: apply the current logic to determine location from module packaging attribute
- Retain current packaging->extension mapping for specific packaging types
- The 'type' location: Use maven3 rules to map type->extension+classifier, and construct a location
- The 'packaging' location: apply the current logic to determine location from module packaging attribute
- If both locations are the same, use the artifact at that location.
- If not, look for the artifact in the packaging location
- If found, emit a deprecation warning and use that location
- If not found, use the artifact from the type location
- In 2.0, we will remove the packaging->extension mapping and the deprecation warning
- GRADLE-2210: Migrate to maven 3
- GRADLE-2238: Use maven 3 classes to locate maven local repository
- GRADLE-2366: Have mavenLocal() check M2_HOME/conf/settings.xml
- http://forums.gradle.org/gradle/topics/why\_does\_maven\_deployer\_ignore\_the\_specified_repository
- http://forums.gradle.org/gradle/topics/crash\_when\_use\_gradle\_1\_0\_rc\_1\_on\_mac_osx\_10\_7\_3
As part of the effort to include Java software in Fedora (http://fedoraproject.org) we are in the process of building and packaging Gradle. One of the issues we have is that Fedora has a very strict requirement: any software in Fedora should be buildable with software already existing in Fedora. In order to meet this requirement we are preparing using an ant build script to build a first version of Gradle that we can then to auto-build Gradle.
One of the issues we find with this approach is the following error: Caused by: org.gradle.api.internal.artifacts.mvnsettings.CannotLocateLocalMavenRepositoryException: java.lang.NoSuchFieldException: userSettingsFile at org.gradle.api.internal.artifacts.mvnsettings.DefaultLocalMavenRepositoryLocator.buildSettings(DefaultLocalMavenRepositoryLocator.java:75)
The Fedora project already has the Maven3 jars available for use in this build, but not the Maven2 jars that we use in the DefaultLocalMavenRepositoryLocator.
While a number of Maven2 classes leak out through the Gradle DSL and cannot be immediately replaced/removed, it is a long-term goal to remove these classes from our public API. This would allow us to upgrade to use Maven3 classes for various maven/gradle integration points (POM parsing, handling maven-metadata.xml, ...): Maven2 classes are very often unsuitable for this purpose.
In order to start using Maven3 classes without removing Maven2 from our API, we can try to use JarJar to provide the Maven3 classes under a different namespace. This would allow us to migrate internally to Maven3; with a goal of deprecating and removing Maven2 for Gradle 2.0.
No intentional user visible change.
We currently have inadequate coverage for mavenLocal repositories, which depend on the LocalMavenRepositoryLocator that is being converted.
The following 'resolve from mavenLocal' happy-day tests would be useful:
- no settings.xml is defined
- repo location defined in ~/.m2/settings.xml
- repo location defined in settings.xml in M2_HOME/conf
- repo location defined in ~/.m2/settings.xml and in M2_HOME (settings file precedence) And sad-day tests for 'resolve from mavenLocal':
- repo directory does not exist
- settings.xml file is invalid
And a test that regular resolve succeeds from http repository when settings.xml is invalid. The local artifact reuse stuff tries to find candidates in mavenLocal.
- Implement all of the integration tests
- Implement m2 repository location with Maven3
- Use jarjar to repackage the required maven3 classes and include them in the Gradle distro.
Allow resolution of java-source and javadoc types from maven repositories (and other types: tests, ejb-client)
- GRADLE-201: Enable support for retrieving source artifacts of a module
- GRADLE-1444: Sources are not downloaded when dependency is using a classifier
- GRADLE-2320: Support for multiple artifacts with source jars in Eclipse plugin
Maven plugins publish various artifact 'types' using well-known naming schemes.
- The maven-source-plugin packages sources for a project into ${name}-sources.jar, and test sources into ${name}-test-sources.jar.
- The maven-javadoc-plugin packages javadoc for a project into ${name}-javadoc.jar and test code into ${name}-test-javadoc.jar.
- The maven-ejb-plugin packages the client jar into ${name}-client.jar. This plugin allows dependencies with type='ejb' to reference the standard jar file, and dependencies with type='ejb-client' to reference the client jar.
- The maven-jar plugin creates ${name}-tests.jar when run with the jar:test-jar plugin. This guide: http://maven.apache.org/guides/mini/guide-attached-tests.html states that the recommended way to reference the test jar as a dependency is by using type='test-jar'. Using classifier='tests' may also work, but has issues with some other plugins.
An example of a module containing a bunch of permutations is http://repo1.maven.org/maven2/org/codehaus/httpcache4j/httpcache4j-core/2.2/
Currently, the only way to resolve 'source' or 'javadoc' artifacts for a module is by using well-known classifiers ('sources' & 'javadoc'). This means that you need to explicitly model these as dependency artifacts, or programmatically add these artifacts (with classifiers) to a detached configuration (see IdeDependenciesExtractor).
Currently, the only way to resolve 'tests' or 'ejb-client' artifacts for a module is by using well-known classifiers.
This would be a good opportunity to introduce some stronger typing to our dependency model, and to map to/from the maven model for these artifact/dependency types. The proposed model for dependencies is outlined in ./dependency-model.md. A good start would be to explicitly map 'source' and 'javadoc' artifact types to/from the maven repository model when resolving.
An attempt to resolve a dependency of type 'source' with no classifier would map to the ${name}-sources.${ext} in the maven repository.
If a classifier other than 'sources' was on the artifact, then we would try to locate the artifact at ${name}-${classifier}-sources.${ext}.
For backward compatibility, we will continue to honour the current model of locating an artifact of type 'source' with classifier!='sources' via the
${name}-${classifier}.${ext} pattern, but we should emit a deprecation warning for this behaviour.
When we reach 2.0, we could:
i) remove the support for 'source' artifacts with a pattern other than ${name}-${classifier}-sources.${ext}
ii) stop adding the 'sources' classifier to the Gradle model of artifacts with type='source'.
We can apply similar changes for artifacts of type 'javadoc', 'test-jar' and 'ejb-client' for maven repositories
- Coverage for resolving typed dependencies on maven modules referenced in various ways:
- Need module published in maven repository with various 'classifier' artifacts: ['source', 'javadoc', 'client', 'test-jar']
- Test resolution of artifacts in these modules via
- Direct dependency in a Gradle project with relevant type specified
- Transitive dependency in a maven module (pom) which is itself a dependency of a Gradle project
- Transitive dependency in an ivy module (ivy.xml) which is itself a dependency of a Gradle project
- Coverage for maven module published with "${name}-src.jar" pattern: this will require use of classifiers to resolve, and should emit deprecation warning.
- Sad-day coverage for the case where neither type nor classifier can successfully locate the maven artifact. Error message should report 'type'-based location as expected.
This will mean that we will be checking 2 locations for such an artifact: ${name}-${classifier}.${ext} and ${name}-${classifier}-sources.${ext}. An example where the former is required would be classifier='src', for the latter classifier='jdk15' (where jdk15 has a different source jar).
- For an artifact of type "source", construct 2 possible locations for the artifact:
- The 'classifier' location: ${name}-${classifier}.${ext}
- The 'type' location: ${name}-sources.${ext}
- If both locations are the same, use the artifact at that location.
- If not, look for the artifact in the 'classifier' location
- If found, emit a deprecation warning and use that location
- If not found, use the artifact from the 'type' location
- In 2.0, we will remove the use of the classifier location for 'source' artifacts, and the deprecation warning
Similarly:
- The above changes apply to type="javadoc", classifier="javadoc" and type pattern = "${name}-javadoc.${ext}"
- The above changes apply to type="test-jar", classifier="tests" and type pattern = "${name}-tests.${ext}"
- The above changes apply to type="ejb-client", classifier="client" and type pattern = "${name}-client.${ext}"
It would be good to try to use Maven3 classes to assist with the mapping of [type]->URL and [type,classifier]->URL if possible.
Until we map these types into the ivy repository model as well:
- The IDEDependenciesExtractor will need to continue using type+classifier
- We cannot deprecate the use of classifier='sources'
- GRADLE-2211: Resolved binary executables and libraries do not use the platform specific naming scheme
- GRADLE-2034: Existence of pom file requires that declared artifacts can be found in the same repository
- GRADLE-2369: Dependency resolution fails for mavenLocal(), mavenCentral() if artifact partially in mavenLocal()
- GRADLE-2335: Provide the ability to implement a custom HTTP authentication scheme for repository access