org.gradle.integtests.resolve.constraints.DependencyConstraintsIntegrationTest.groovy Maven / Gradle / Ivy
/*
* Copyright 2018 the original author or authors.
*
* Licensed 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.
*/
package org.gradle.integtests.resolve.constraints
import org.gradle.integtests.fixtures.AbstractPolyglotIntegrationSpec
import org.gradle.integtests.fixtures.ToBeFixedForConfigurationCache
import org.gradle.integtests.fixtures.resolve.ResolveTestFixture
import spock.lang.Issue
/**
* This is a variation of {@link PublishedDependencyConstraintsIntegrationTest} that tests dependency constraints
* declared in the build script (instead of published)
*/
class DependencyConstraintsIntegrationTest extends AbstractPolyglotIntegrationSpec {
private final String resolveName = 'resolve-fixture'
private final ResolveTestFixture resolve = new ResolveTestFixture(testDirectory.file("${resolveName}.gradle"), "conf").expectDefaultConfiguration("runtime")
def setup() {
buildSpec {
settings {
rootProjectName = 'test'
}
rootProject {
repositories {
maven(mavenRepo.uri)
}
configurations {
conf
}
applyFrom(resolveName)
}
}
resolve.prepare()
resolve.addDefaultVariantDerivationStrategy()
}
void "dependency constraint is not included in resolution without a hard dependency"() {
given:
mavenRepo.module("org", "foo", '1.0').publish()
writeSpec {
rootProject {
dependencies {
constraints {
conf('org:foo:1.0')
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {}
}
}
void "dependency constraint is included into the result of resolution when a hard dependency is also added"() {
given:
mavenRepo.module("org", "foo", '1.1').publish()
writeSpec {
rootProject {
dependencies {
conf 'org:foo'
constraints {
conf('org:foo:1.1')
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
edge("org:foo", "org:foo:1.1")
constraint("org:foo:1.1", "org:foo:1.1")
}
}
}
@ToBeFixedForConfigurationCache
void "dependency constraint can be used to declare incompatibility"() {
given:
mavenRepo.module("org", "foo", '1.1').publish()
mavenRepo.module("org", "bar", '1.0')
.dependsOn('org', 'foo', '1.1')
.publish()
writeSpec {
rootProject {
dependencies {
conf 'org:bar:1.0'
constraints {
conf('org:foo') {
version { rejectAll() }
}
}
}
}
}
when:
fails 'checkDeps'
then:
failure.assertHasCause("""Module 'org:foo' has been rejected:
Dependency path ':test:unspecified' --> 'org:bar:1.0' (runtime) --> 'org:foo:1.1'
Constraint path ':test:unspecified' --> 'org:foo:{reject all versions}'""")
}
void "dependency constraint is included into the result of resolution when a hard dependency is also added transitively"() {
given:
mavenRepo.module("org", "foo", '1.0').publish()
mavenRepo.module("org", "foo", '1.1').publish()
mavenRepo.module("org", "bar", "1.0").dependsOn("org", "foo", "1.0").publish()
writeSpec {
rootProject {
dependencies {
conf 'org:bar:1.0'
constraints {
conf 'org:foo:1.1'
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
module("org:bar:1.0") {
edge("org:foo:1.0", "org:foo:1.1").byConflictResolution("between versions 1.1 and 1.0")
}
constraint("org:foo:1.1", "org:foo:1.1")
}
}
}
/**
* Test demonstrates a bug in resolution of constraints, when real dependency is evicted via conflict resolution.
*/
@Issue("gradle/gradle#4610")
void "dependency constraint should not preserve hard dependency for evicted dependency"() {
given:
// "org:foo:1.0" -> "org:baz:1.0" -> "org:baz-transitive:1.0"
mavenRepo.module("org", "foo", '1.0')
.dependsOn("org", "baz", '1.0').publish()
mavenRepo.module("org", "baz", '1.0')
.dependsOn("org", "baz-transitive", "1.0").publish()
mavenRepo.module("org", "baz-transitive", "1.0").publish()
// "org:bar:1.0" -> "org:foo:1.1" (no further transitive deps)
mavenRepo.module("org", "bar", "1.0")
.dependsOn("org", "foo", '1.1').publish()
mavenRepo.module("org", "foo", '1.1').publish()
writeSpec {
rootProject {
dependencies {
conf 'org:foo:1.0' // Would bring in 'baz' and 'baz-transitive' (but will be evicted)
conf 'org:bar:1.0' // Brings in 'foo:1.1'
constraints {
conf 'org:baz:1.0' // Should not bring in 'baz' when 'foo:1.0' is evicted
}
}
section """
task resolve(type: Sync) {
from configurations.conf
into 'lib'
}
""", """
tasks.register("resolve") {
from(configurations["conf"])
into("lib")
}
"""
}
}
when:
run ':resolve'
then:
file('lib').assertHasDescendants("bar-1.0.jar", "foo-1.1.jar")
/*
* Cannot use ResolveTestFixture because the end graph cannot be handled
* - The edge ":test:" -> "org:baz:1.0" is included in graph
* - But "org:baz:1.0" is NOT in the "first level dependencies"
when:
run ':checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
edge("org:foo:1.0","org:foo:1.1").byConflictResolution()
module("org:bar:1.0") {
module("org:foo:1.1")
}
// BUG: This module should not be included, but it is
module("org:baz:1.0") {
module("org:baz-transitive:1.0")
}
}
}
*/
}
void "range resolution kicks in with dependency constraints"() {
given:
mavenRepo.module("org", "foo", '1.0').publish()
mavenRepo.module("org", "foo", '1.1').publish()
mavenRepo.module("org", "foo", '1.2').publish()
mavenRepo.module("org", "bar", "1.0").dependsOn("org", "foo", "[1.0,1.2]").publish()
writeSpec {
rootProject {
dependencies {
conf 'org:bar:1.0'
constraints {
conf('org:foo:[1.0,1.1]') {
because 'tested versions'
}
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
module("org:bar:1.0") {
edge("org:foo:[1.0,1.2]", "org:foo:1.1").byConstraint('didn\'t match version 1.2 because tested versions')
}
constraint("org:foo:[1.0,1.1]", "org:foo:1.1").byConstraint('didn\'t match version 1.2 because tested versions')
}
}
}
void "transitive dependencies of an dependency constraint do not participate in conflict resolution if it is not included elsewhere"() {
given:
mavenRepo.module("org", "foo", '1.0').dependsOn('org', 'bar', '1.1').publish()
mavenRepo.module("org", "bar", '1.0').publish()
mavenRepo.module("org", "bar", '1.1').publish()
writeSpec {
rootProject {
dependencies {
conf 'org:bar:1.0'
constraints {
conf 'org:foo:1.0'
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
module("org:bar:1.0")
}
}
}
void "dependency constraints on substituted module is recognized properly"() {
given:
mavenRepo.module("org", "foo", '1.0').publish()
mavenRepo.module("org", "foo", '1.1').publish()
mavenRepo.module("org", "bar", '1.1').publish()
writeSpec {
rootProject {
configurations {
conf {
section("""
resolutionStrategy.dependencySubstitution {
all { DependencySubstitution dependency ->
if (dependency.requested.module == 'bar') {
dependency.useTarget dependency.requested.group + ':foo:' + dependency.requested.version
}
}
}""", """resolutionStrategy.dependencySubstitution {
all {
val r = requested as ModuleComponentSelector
if (r.module == "bar") {
useTarget("\${r.group}:foo:\${r.version}")
}
}
}
""")
}
}
dependencies {
conf 'org:foo:1.0'
constraints {
conf 'org:bar:1.1'
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
constraint("org:bar:1.1", "org:foo:1.1").selectedByRule()
edge("org:foo:1.0", "org:foo:1.1").byConflictResolution("between versions 1.1 and 1.0")
}
}
}
void "dependency constraints are inherited"() {
given:
mavenRepo.module("org", "foo", '1.0').publish()
mavenRepo.module("org", "foo", '1.1').publish()
writeSpec {
rootProject {
configurations {
confSuper
conf {
extendsFrom("confSuper")
}
}
dependencies {
conf 'org:foo:1.0'
constraints {
confSuper 'org:foo:1.1'
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
edge("org:foo:1.0", "org:foo:1.1").byConflictResolution("between versions 1.1 and 1.0")
constraint("org:foo:1.1", "org:foo:1.1")
}
}
}
void "dependency constraints defined for a configuration are applied when resolving that configuration as part of a project dependency"() {
given:
mavenRepo.module("org", "foo", '1.0').publish()
mavenRepo.module("org", "foo", '1.1').publish()
writeSpec {
rootProject {
dependencies {
conf project(path: ':b', configuration: 'conf')
conf 'org:foo:1.0'
}
}
project("b") {
repositories {
maven(mavenRepo.uri)
}
configurations {
conf
}
dependencies {
constraints {
conf('org:foo:1.1') {
because 'transitive dependency constraint'
}
}
}
}
}
when:
run ':checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
edge("org:foo:1.0", "org:foo:1.1").byConflictResolution("between versions 1.1 and 1.0").byConstraint('transitive dependency constraint')
project(":b", "test:b:") {
configuration = "conf"
noArtifacts()
constraint("org:foo:1.1", "org:foo:1.1").byConstraint('transitive dependency constraint')
}
}
}
}
void "dependency constraints defined for a build are applied when resolving a configuration that uses that build as an included build"() {
given:
resolve.expectDefaultConfiguration('default')
mavenRepo.module("org", "foo", '1.0').publish()
mavenRepo.module("org", "foo", '1.1').publish()
writeSpec {
rootProject {
dependencies {
conf 'org:included:1.0'
conf 'org:foo:1.0'
}
}
includedBuild("includeBuild") {
settings {
rootProjectName = 'included'
}
rootProject {
group = 'org'
version = '1.0'
configurations {
conf
'default' { extendsFrom 'conf' }
}
dependencies {
constraints {
conf 'org:foo:1.1'
}
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
edge("org:foo:1.0", "org:foo:1.1:runtime").byConflictResolution("between versions 1.1 and 1.0")
edge("org:included:1.0", "project :includeBuild", "org:included:1.0") {
noArtifacts()
constraint("org:foo:1.1", "org:foo:1.1")
}.compositeSubstitute()
}
}
}
@Issue("gradle/gradle#4609")
def "dependency constraint does not invalidate excludes defined on hard dependency"() {
given:
mavenRepo.module("org", "baz", "1.0").publish()
mavenRepo.module("org", "foo", '1.0').dependsOn("org", "baz", '1.0').publish()
writeSpec {
rootProject {
dependencies {
conf('org:foo') {
section(
"exclude group: 'org', module: 'baz'",
'exclude(mapOf("group" to "org", "module" to "baz"))'
)
}
constraints {
conf 'org:foo:1.0'
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
edge("org:foo", "org:foo:1.0")
constraint("org:foo:1.0")
}
}
}
void "dependency constraints should not pull in additional artifacts"() {
given:
mavenRepo.module("org", "foo", '1.0').artifact(classifier: 'shaded').publish()
mavenRepo.module("org", "foo", '1.1').artifact(classifier: 'shaded').publish()
writeSpec {
rootProject {
dependencies {
conf 'org:foo:1.0:shaded'
constraints {
conf 'org:foo:1.1'
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
edge("org:foo:1.0", "org:foo:1.1")
constraint("org:foo:1.1", "org:foo:1.1") {
artifact(classifier: 'shaded')
}
}
}
}
void "dependency constraints should not pull in additional artifacts for transitive dependencies"() {
given:
def foo11 = mavenRepo.module("org", "foo", '1.0').artifact(classifier: 'shaded').publish()
mavenRepo.module("org", "foo", '1.1').artifact(classifier: 'shaded').publish()
mavenRepo.module("org", "bar", '1.0').dependsOn(classifier: 'shaded', foo11).publish()
writeSpec {
rootProject {
dependencies {
conf 'org:bar:1.0'
constraints {
conf 'org:foo:1.1'
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
constraint("org:foo:1.1", "org:foo:1.1") {
artifact(classifier: 'shaded')
}
module("org:bar:1.0") {
edge("org:foo:1.0", "org:foo:1.1")
}
}
}
}
void 'dependency updated through constraints has its transitive dependencies'() {
given:
def foo10 = mavenRepo.module('org', 'foo', '1.0').publish()
mavenRepo.module('org', 'bar', '1.0').publish()
mavenRepo.module('org', 'bar', '1.1').dependsOn(foo10).publish()
writeSpec {
rootProject {
dependencies {
conf 'org:bar:1.0'
constraints {
conf 'org:bar:1.1'
}
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
constraint("org:bar:1.1", "org:bar:1.1") {
edge('org:foo:1.0', 'org:foo:1.0')
}
edge('org:bar:1.0', 'org:bar:1.1')
}
}
}
void 'dependency without version updated through constraints has its transitive dependencies'() {
given:
def foo10 = mavenRepo.module('org', 'foo', '1.0').publish()
mavenRepo.module('org', 'bar', '1.0').publish()
mavenRepo.module('org', 'bar', '1.1').dependsOn(foo10).publish()
writeSpec {
rootProject {
dependencies {
constraints {
conf 'org:bar:1.1'
}
conf 'org:bar'
}
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
constraint("org:bar:1.1", "org:bar:1.1") {
edge('org:foo:1.0', 'org:foo:1.0')
}
edge('org:bar', 'org:bar:1.1')
}
}
}
void 'dependency constraint can be not pending, then pending, then not pending and still participate in resolution'() {
def constrainedBase = mavenRepo.module('org', 'constrained', '1.0').publish()
def constrained = mavenRepo.module('org', 'constrained', '1.1').publish()
def bom = mavenRepo.module('org', 'bom', '1.0').hasType('pom').dependencyConstraint(constrained).publish()
def user = mavenRepo.module('org', 'user', '1.0').dependsOn(constrainedBase).publish()
def higherUser = mavenRepo.module('org', 'user', '1.1').dependsOn(constrainedBase).publish()
def otherUser = mavenRepo.module('org', 'otherUser', '1.0').dependsOn(higherUser).publish()
mavenRepo.module('org', 'indirect', '1.0').dependsOn(user).dependsOn(otherUser).dependsOn(bom).publish()
writeSpec {
rootProject {
dependencies {
conf(platform('org:bom:1.0'))
conf 'org:indirect:1.0'
section("""
components {
withModule('org:indirect', PickPlatformRule)
}""",
"""
components {
withModule("org:indirect", PickPlatformRule::class.java)
}""")
}
section """
class PickPlatformRule implements ComponentMetadataRule {
ObjectFactory objects
@javax.inject.Inject
PickPlatformRule(ObjectFactory objects) {
this.objects = objects
}
@Override
void execute(ComponentMetadataContext context) {
def details = context.details
if (details.id.name == 'indirect') {
details.allVariants {
withDependencies {
it.each {
if (it.name == 'bom') {
it.attributes {
attribute(Category.CATEGORY_ATTRIBUTE, objects.named(Category, 'platform'))
}
}
}
}
}
}
}
}
""", """
open class PickPlatformRule @javax.inject.Inject constructor(val objects: ObjectFactory): ComponentMetadataRule {
override fun execute(context: ComponentMetadataContext) {
val details = context.details
if (details.id.name == "indirect") {
details.allVariants {
withDependencies {
forEach {
if (it.name == "bom") {
it.attributes {
attribute(Category.CATEGORY_ATTRIBUTE, objects.named(Category::class.java, "platform"))
}
}
}
}
}
}
}
}
"""
}
}
when:
run 'checkDeps'
then:
resolve.expectGraph {
root(':', ':test:') {
module("org:bom:1.0:platform-runtime") {
constraint("org:constrained:1.1", "org:constrained:1.1")
noArtifacts()
}
module("org:indirect:1.0") {
edge("org:user:1.0", "org:user:1.1") {
edge("org:constrained:1.0", "org:constrained:1.1")
}
module("org:otherUser:1.0") {
module("org:user:1.1")
}
module("org:bom:1.0:platform-runtime")
}
}
}
}
void 'dependency constraint on failed variant resolution needs to be in the right state'() {
mavenRepo.module('org', 'bar', '1.0').publish()
writeSpec {
rootProject {
dependencies {
constraints {
conf 'org:bar:1.0'
}
conf('org:bar') {
section("""attributes {
attribute(Usage.USAGE_ATTRIBUTE, project.objects.named(Usage, 'wrong'))
}""", """attributes {
attribute(Usage.USAGE_ATTRIBUTE, project.objects.named("wrong"))
}""")
}
}
}
}
when:
succeeds 'dependencyInsight', '--configuration', 'conf', '--dependency', 'org:bar'
then:
outputContains("org:bar: FAILED")
}
void 'multiple dependency constraints on single module are all taken into account (#one then #two)'() {
def bar10 = mavenRepo.module('org', 'bar', '1.0').publish()
def bar20 = mavenRepo.module('org', 'bar', '2.0').publish()
def foo10 = mavenRepo.module('org', 'foo', '1.0').dependsOn(bar10).publish()
def foo13 = mavenRepo.module('org', 'foo', '1.3').dependsOn(bar10).publish()
def bar15 = mavenRepo.module('org', 'bar', '1.5').publish()
def foo15 = mavenRepo.module('org', 'foo', '1.5').dependsOn(bar15).publish()
def third = mavenRepo.module('org', 'third', '1.0').dependsOn(foo15).publish()
def second = mavenRepo.module('org', 'second', '1.0').dependsOn(third).dependsOn(foo13).publish()
mavenRepo.module('org', 'first', '1.0').dependsOn(second).dependsOn(foo10).publish()
writeSpec {
rootProject {
dependencies {
constraints {
conf "org:bar:$one"
conf "org:bar:$two"
}
conf('org:first:1.0')
}
}
}
when:
succeeds 'dependencyInsight', '--configuration', 'conf', '--dependency', 'org:bar'
then:
outputContains("org:bar:2.0")
where:
one | two
'1.5' | '2.0'
'2.0' | '1.5'
}
@Issue("https://github.com/gradle/gradle/issues/19882")
void 'can use constraints on project to upgrade third party to higher version project'() {
def bar10 = mavenRepo.module('org', 'bar', '1.0').publish()
def foo10 = mavenRepo.module('org', 'foo', '1.0').dependencyConstraint(bar10).publish()
writeSpec {
rootProject {
dependencies {
conf "org:foo:1.0"
conf project(":bar")
}
}
project("foo") {
group = 'org'
version = '1.1'
configurations {
conf
'default' { extendsFrom 'conf' }
}
dependencies {
conf project(":bar")
}
}
project("bar") {
group = 'org'
version = '1.1'
configurations {
conf
'default' { extendsFrom 'conf' }
}
dependencies {
constraints {
conf project(":foo")
}
}
}
}
when:
run ':checkDeps'
then:
resolve.expectGraph {
root(':', ':test:') {
edge("org:foo:1.0", "project :foo", "org:foo:1.1") {
configuration = 'default'
noArtifacts()
project(":bar", "org:bar:1.1") {
configuration = 'default'
noArtifacts()
constraint("project :foo", "org:foo:1.1")
}
}
project(":bar", "org:bar:1.1") {
configuration = 'default'
noArtifacts()
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy