org.gradle.integtests.resolve.capabilities.CapabilitiesRulesIntegrationTest.groovy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* 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.capabilities
import org.gradle.integtests.fixtures.GradleMetadataResolveRunner
import org.gradle.integtests.fixtures.RequiredFeature
import org.gradle.integtests.fixtures.ToBeFixedForConfigurationCache
import org.gradle.integtests.resolve.AbstractModuleDependencyResolveTest
import spock.lang.Issue
import spock.lang.Unroll
class CapabilitiesRulesIntegrationTest extends AbstractModuleDependencyResolveTest {
@ToBeFixedForConfigurationCache
def "can declare capabilities using a component metadata rule"() {
given:
repository {
'cglib:cglib:3.2.5'()
'cglib:cglib-nodep:3.2.5'()
}
buildFile << """
class CapabilityRule implements ComponentMetadataRule {
@Override
void execute(ComponentMetadataContext context) {
def details = context.details
details.allVariants {
withCapabilities {
addCapability('cglib', 'cglib', details.id.version)
}
}
}
}
dependencies {
conf "cglib:cglib-nodep:3.2.5"
conf "cglib:cglib:3.2.5"
components {
withModule('cglib:cglib-nodep', CapabilityRule)
}
}
"""
when:
repositoryInteractions {
'cglib:cglib-nodep:3.2.5' {
expectGetMetadata()
}
'cglib:cglib:3.2.5' {
expectGetMetadata()
}
}
fails ':checkDeps'
then:
def variant = 'runtime'
if (!isGradleMetadataPublished() && useIvy()) {
variant = 'default'
}
failure.assertHasCause("""Module 'cglib:cglib-nodep' has been rejected:
Cannot select module with conflict on capability 'cglib:cglib:3.2.5' also provided by [cglib:cglib:3.2.5($variant)]""")
failure.assertHasCause("""Module 'cglib:cglib' has been rejected:
Cannot select module with conflict on capability 'cglib:cglib:3.2.5' also provided by [cglib:cglib-nodep:3.2.5($variant)]""")
}
@ToBeFixedForConfigurationCache
def "implicit capability conflict is detected if implicit capability is discovered late"() {
given:
repository {
'cglib:cglib:3.2.5'()
'cglib:cglib-nodep:3.2.5'()
'org:lib:1.0' {
dependsOn 'cglib:cglib:3.2.5'
}
}
buildFile << """
class CapabilityRule implements ComponentMetadataRule {
@Override
void execute(ComponentMetadataContext context) {
def details = context.details
details.allVariants {
withCapabilities {
addCapability('cglib', 'cglib', details.id.version)
}
}
}
}
dependencies {
conf "cglib:cglib-nodep:3.2.5"
conf "org:lib:1.0"
components {
withModule('cglib:cglib-nodep', CapabilityRule)
}
}
"""
when:
repositoryInteractions {
'cglib:cglib-nodep:3.2.5' {
expectGetMetadata()
}
'cglib:cglib:3.2.5' {
expectGetMetadata()
}
'org:lib:1.0' {
expectGetMetadata()
}
}
fails ':checkDeps'
then:
def variant = 'runtime'
if (!isGradleMetadataPublished() && useIvy()) {
variant = 'default'
}
failure.assertHasCause("""Module 'cglib:cglib-nodep' has been rejected:
Cannot select module with conflict on capability 'cglib:cglib:3.2.5' also provided by [cglib:cglib:3.2.5($variant)]""")
failure.assertHasCause("""Module 'cglib:cglib' has been rejected:
Cannot select module with conflict on capability 'cglib:cglib:3.2.5' also provided by [cglib:cglib-nodep:3.2.5($variant)]""")
}
@Unroll
def "can detect conflict with capability in different versions (#rule)"() {
given:
repository {
'cglib:cglib:3.2.5'()
'cglib:cglib-nodep:3.2.4'()
}
buildFile << """
class CapabilityRule implements ComponentMetadataRule {
@Override
void execute(ComponentMetadataContext context) {
def details = context.details
details.allVariants {
withCapabilities {
addCapability('cglib', 'cglib', details.id.version)
}
}
}
}
dependencies {
conf "cglib:cglib-nodep:3.2.4"
conf "cglib:cglib:3.2.5"
components {
withModule('cglib:cglib-nodep', CapabilityRule)
}
}
configurations.conf.resolutionStrategy.capabilitiesResolution {
$rule
}
"""
when:
repositoryInteractions {
'cglib:cglib-nodep:3.2.4' {
expectGetMetadata()
}
'cglib:cglib:3.2.5' {
expectResolve()
}
}
run ':checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
edge('cglib:cglib-nodep:3.2.4', 'cglib:cglib:3.2.5')
.byConflictResolution(reason)
module('cglib:cglib:3.2.5')
}
}
where:
rule | reason
'all { selectHighestVersion() }' | 'latest version of capability cglib:cglib'
'withCapability("cglib:cglib") { selectHighestVersion() }' | 'latest version of capability cglib:cglib'
'withCapability("cglib", "cglib") { selectHighestVersion() }' | 'latest version of capability cglib:cglib'
'all { select(candidates.find { it.id.module == "cglib" }) because "custom reason" }' | 'On capability cglib:cglib custom reason'
}
@ToBeFixedForConfigurationCache
def "can detect conflict between local project and capability from external dependency"() {
given:
repository {
'org:test:1.0'()
}
buildFile << """
apply plugin: 'java-library'
class CapabilityRule implements ComponentMetadataRule {
@Override
void execute(ComponentMetadataContext context) {
context.details.allVariants {
withCapabilities {
addCapability('org', 'capability', '1.0')
}
}
}
}
configurations.api.outgoing {
capability 'org:capability:1.0'
}
dependencies {
conf 'org:test:1.0'
components {
withModule('org:test', CapabilityRule)
}
}
configurations {
conf.extendsFrom(api)
}
"""
when:
repositoryInteractions {
'org:test:1.0' {
expectGetMetadata()
}
}
fails ':checkDeps'
then:
failure.assertHasCause("""Module 'org:test' has been rejected:
Cannot select module with conflict on capability 'org:capability:1.0' also provided by [:test:unspecified(conf)]""")
}
@RequiredFeature(feature = GradleMetadataResolveRunner.GRADLE_METADATA, value="true")
def "can remove a published capability"() {
given:
repository {
'org:a:1.0' {
variant('runtime') {
capability('test_capability')
}
}
'org:b:1.0' {
variant('runtime') {
capability('test_capability')
}
}
}
buildFile << """
class CapabilityRule implements ComponentMetadataRule {
@Override
void execute(ComponentMetadataContext context) {
context.details.allVariants {
withCapabilities {
removeCapability('org.test', 'test_capability')
}
}
}
}
dependencies {
conf 'org:a:1.0'
conf 'org:b:1.0'
components.all(CapabilityRule)
}
"""
when:
repositoryInteractions {
'org:a:1.0' {
expectResolve()
}
'org:b:1.0' {
expectResolve()
}
}
run ':checkDeps'
then:
resolve.expectGraph {
root(":", ":test:") {
module('org:a:1.0')
module('org:b:1.0')
}
}
}
@Issue("gradle/gradle#12011")
@ToBeFixedForConfigurationCache
@Unroll
def "can detect capability conflict even when participants belong to a virtual platform (#first, #second)"() {
given:
repository {
'aligned:foo:1.0'()
'indirect:bar:1.0' {
dependsOn 'other:baz:1.0'
}
'other:baz:1.0'()
}
buildFile << """
class TheOneRule implements ComponentMetadataRule {
@Override
void execute(ComponentMetadataContext context) {
def id = context.details.id
if (id.group == "aligned") {
context.details.belongsTo('aligned:virtual:1.0')
context.details.allVariants {
withCapabilities {
addCapability('org.test', 'test_capability', '1.0')
}
}
} else if (id.group == "other") {
context.details.allVariants {
withCapabilities {
addCapability('org.test', 'test_capability', '1.0')
}
}
}
}
}
dependencies {
conf '$first'
conf '$second'
components.all(TheOneRule)
}
"""
when:
repositoryInteractions {
'aligned:foo:1.0' {
expectGetMetadata()
}
'indirect:bar:1.0' {
expectGetMetadata()
}
'other:baz:1.0' {
expectGetMetadata()
}
}
fails ":checkDeps"
then:
failure.assertHasCause("Module 'aligned:foo' has been rejected")
where:
first | second
'aligned:foo:1.0' | 'indirect:bar:1.0'
'indirect:bar:1.0' | 'aligned:foo:1.0'
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy