org.gradle.integtests.resolve.strict.StrictVersionsInPlatformCentricDevelopmentIntegrationTest.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 2019 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.strict
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.Unroll
import static org.gradle.integtests.resolve.strict.StrictVersionsInPlatformCentricDevelopmentIntegrationTest.PlatformType.ENFORCED_PLATFORM
import static org.gradle.integtests.resolve.strict.StrictVersionsInPlatformCentricDevelopmentIntegrationTest.PlatformType.LEGACY_PLATFORM
import static org.gradle.integtests.resolve.strict.StrictVersionsInPlatformCentricDevelopmentIntegrationTest.PlatformType.MODULE
import static org.gradle.integtests.resolve.strict.StrictVersionsInPlatformCentricDevelopmentIntegrationTest.PlatformType.PLATFORM
@RequiredFeature(feature = GradleMetadataResolveRunner.GRADLE_METADATA, value = "true")
class StrictVersionsInPlatformCentricDevelopmentIntegrationTest extends AbstractModuleDependencyResolveTest {
enum PlatformType {
// The recommended way of doing platforms
PLATFORM, // constraints in platform are published with strict constraints, consumer uses 'platform()' dependencies
// The recommended way of dealing with existing platforms
LEGACY_PLATFORM, // constraints in platform are published without strict constraints, consumer uses 'platform()' dependencies + component metadata rules to make all published constraints strict
// The discouraged way of dealing with existing platforms
ENFORCED_PLATFORM, // constraints in platform are published without strict constraints, consumer uses 'enforcedPlatform()' dependencies (to be deprecated)
// Using a normal module as "platform" (i.e. endorsing it's strict constraints) also works
MODULE // constraints in module are published with strict constraints, consumer uses normal dependencies with 'endorseStrictVersions()'
}
def setup() {
resolve.withStrictReasonsCheck()
}
String platformDependency(platformType, String dependency) {
switch (platformType) {
case PLATFORM:
case LEGACY_PLATFORM:
return "conf(platform('$dependency'))"
case ENFORCED_PLATFORM:
return "conf(enforcedPlatform('$dependency'))"
case MODULE:
return "conf('$dependency') { endorseStrictVersions() }"
}
""
}
private singleLibraryBuildFile(platformType) {
buildFile << """
dependencies {
${platformDependency(platformType, 'org:platform:1.+')}
conf('org:bar')
}
"""
}
private void initialRepository(platformType) {
repository {
'org:platform:1.0'() {
variant('apiElements') {
attributes = ['org.gradle.category': 'platform']
noArtifacts = true
}
if (platformType in [PLATFORM, MODULE]) {
constraint(group: 'org', artifact: 'bar', strictly: '2.0')
constraint(group: 'org', artifact: 'foo', strictly: '3.0', rejects: ['3.1', '3.2'])
} else {
constraint(group: 'org', artifact: 'bar', version: '2.0')
constraint(group: 'org', artifact: 'foo', version: '3.0', rejects: ['3.1', '3.2'])
}
}
'org:foo' {
'3.0'()
'3.1'() // bad version
'3.2'() // bad version
}
'org:bar:2.0'() {
dependsOn 'org:foo:3.1'
}
}
if (platformType == LEGACY_PLATFORM) {
// we use component metadata rules to get behavior similar to 'enforcedPlatform()' for 'platform()'
buildFile << """
dependencies {
components.withModule('org:platform') { ComponentMetadataDetails details ->
details.withVariant('apiElements') {
withDependencyConstraints {
it.each {
def rejected = it.versionConstraint.rejectedVersions
it.version {
strictly(it.requiredVersion);
if (rejected) { reject(*rejected) }
}
}
}
}
}
}
"""
}
}
private void updatedRepository(platformType) {
initialRepository(platformType)
repository {
'org:platform:1.1'() {
variant('apiElements') {
attributes = ['org.gradle.category': 'platform']
noArtifacts = true
}
if (platformType in [PLATFORM, MODULE]) {
constraint(group: 'org', artifact: 'bar', strictly: '2.0')
constraint(group: 'org', artifact: 'foo', strictly: '3.1.1', rejects: ['3.1', '3.2'])
} else {
constraint(group: 'org', artifact: 'bar', version: '2.0')
constraint(group: 'org', artifact: 'foo', version: '3.1.1', rejects: ['3.1', '3.2'])
}
}
'org:foo' {
'3.1.1'()
}
}
}
static String expectStrictVersion(platformType, String requiredVersion, String rejectedVersions = '') {
boolean strictVersion = platformType != ENFORCED_PLATFORM
if (strictVersion && rejectedVersions.isEmpty()) {
return "{strictly $requiredVersion}"
}
if (!strictVersion && !rejectedVersions.isEmpty()) {
return "{require $requiredVersion; reject $rejectedVersions}"
}
if (strictVersion && !rejectedVersions.isEmpty()) {
return "{strictly $requiredVersion; reject $rejectedVersions}"
}
requiredVersion
}
@Unroll
void "(1) all future releases of org:foo:3.0 are bad and the platform enforces 3.0 [#platformType]"() {
initialRepository(platformType)
singleLibraryBuildFile(platformType)
when:
repositoryInteractions {
'org:platform' {
expectVersionListing()
}
'org:platform:1.0' {
expectGetMetadata()
if (platformType == MODULE) {
expectGetArtifact()
}
}
'org:bar:2.0' {
expectGetMetadata()
expectGetArtifact()
}
'org:foo:3.0' {
expectGetMetadata()
expectGetArtifact()
}
}
run ':checkDeps'
then:
resolve.expectGraph {
root(':', ':test:') {
edge('org:platform:1.+', 'org:platform:1.0') {
byRequest()
if (platformType != MODULE) {
configuration(platformType == ENFORCED_PLATFORM ? 'enforcedApiElements' : 'apiElements')
noArtifacts()
}
constraint("org:bar:${StrictVersionsInPlatformCentricDevelopmentIntegrationTest.expectStrictVersion(platformType, '2.0')}", 'org:bar:2.0').byConstraint()
constraint("org:foo:${StrictVersionsInPlatformCentricDevelopmentIntegrationTest.expectStrictVersion(platformType, '3.0', '3.1 & 3.2')}", 'org:foo:3.0').byConstraint()
}
edge('org:bar', 'org:bar:2.0') {
byRequest()
edge('org:foo:3.1', 'org:foo:3.0') {
if (platformType != ENFORCED_PLATFORM) {
byAncestor()
} else {
byRequest()
}
}
}
}
if (platformType == ENFORCED_PLATFORM) {
nodesWithoutRoot.each { it.forced() }
}
}
where:
platformType << PlatformType.values()
}
@Unroll
void "(2) org:foo:3.1.1 and platform upgrade 1.1 are release [#platformType]"() {
updatedRepository(platformType)
singleLibraryBuildFile(platformType)
when:
repositoryInteractions {
'org:platform' {
expectVersionListing()
}
'org:platform:1.1' {
expectGetMetadata()
if (platformType == MODULE) {
expectGetArtifact()
}
}
'org:bar:2.0' {
expectGetMetadata()
expectGetArtifact()
}
'org:foo:3.1.1' {
expectGetMetadata()
expectGetArtifact()
}
}
run ':checkDeps'
then:
resolve.expectGraph {
root(':', ':test:') {
edge('org:platform:1.+', 'org:platform:1.1') {
byRequest()
if (platformType != MODULE) {
configuration(platformType == ENFORCED_PLATFORM ? 'enforcedApiElements' : 'apiElements')
noArtifacts()
}
constraint("org:bar:${StrictVersionsInPlatformCentricDevelopmentIntegrationTest.expectStrictVersion(platformType, '2.0')}", 'org:bar:2.0').byConstraint()
constraint("org:foo:${StrictVersionsInPlatformCentricDevelopmentIntegrationTest.expectStrictVersion(platformType, '3.1.1', '3.1 & 3.2')}", 'org:foo:3.1.1').byConstraint()
}
edge('org:bar', 'org:bar:2.0') {
byRequest()
edge('org:foo:3.1', 'org:foo:3.1.1') {
if (platformType != ENFORCED_PLATFORM) {
byAncestor()
} else {
byRequest()
}
}
}
}
if (platformType == ENFORCED_PLATFORM) {
nodesWithoutRoot.each { it.forced() }
}
}
where:
platformType << PlatformType.values()
}
@Unroll
@ToBeFixedForConfigurationCache(iterationMatchers = [
".*\\[PLATFORM\\].*",
".*\\[LEGACY_PLATFORM\\].*",
".*\\[MODULE\\].*"
])
void "(3) library developer has issues with org:foo:3.1.1 and overrides platform decision with 3.2 which fails due to reject [#platformType]"() {
updatedRepository(platformType)
singleLibraryBuildFile(platformType)
buildFile << """
dependencies {
constraints {
conf('org:foo:3.2')
}
}
"""
when:
repositoryInteractions {
'org:platform' {
expectVersionListing()
}
'org:platform:1.1' {
expectGetMetadata()
}
'org:bar:2.0' {
expectGetMetadata()
if (platformType == ENFORCED_PLATFORM) {
expectGetArtifact()
}
}
'org:foo:3.1.1' {
expectGetMetadata()
if (platformType == ENFORCED_PLATFORM) {
expectGetArtifact()
}
}
'org:foo:3.2' {
if (platformType != ENFORCED_PLATFORM) {
expectGetMetadata()
}
}
}
if (platformType == ENFORCED_PLATFORM) {
// issue with enforced platform: the forced version is always used and the conflict is 'hidden'
succeeds ':checkDeps'
} else {
fails ':checkDeps'
}
then:
def platformVariant = platformType == MODULE ? 'runtime' : 'apiElements'
(platformType == ENFORCED_PLATFORM && !failure) || failure.assertHasCause(
"""Cannot find a version of 'org:foo' that satisfies the version constraints:
Dependency path ':test:unspecified' --> 'org:bar:2.0' (runtime) --> 'org:foo:3.1'
Constraint path ':test:unspecified' --> 'org:platform:1.1' (${platformVariant}) --> 'org:foo:{strictly 3.1.1; reject 3.1 & 3.2}'
Constraint path ':test:unspecified' --> 'org:foo:3.2'""")
where:
platformType << PlatformType.values()
}
@Unroll
@ToBeFixedForConfigurationCache(iterationMatchers = [
".*\\[ENFORCED_PLATFORM\\].*"
])
void "(4) library developer has issues with org:foo:3.1.1 and forces an override of the platform decision with strictly [#platformType]"() {
// issue with enforced platform: consumer can not override platform decision via constraint
// (an override via an own forced dependency is possible)
def expectedFooVersion = platformType == ENFORCED_PLATFORM ? '3.1.1' : '3.2'
updatedRepository(platformType)
singleLibraryBuildFile(platformType)
buildFile << """
dependencies {
constraints {
conf('org:foo') {
version { strictly '3.2' }
}
}
}
"""
when:
repositoryInteractions {
'org:platform' {
expectVersionListing()
expectGetArtifact()
}
'org:platform:1.1' {
expectGetMetadata()
if (platformType == MODULE) {
expectGetArtifact()
}
}
'org:bar:2.0' {
expectGetMetadata()
if (platformType != ENFORCED_PLATFORM) {
expectGetArtifact()
}
}
"org:foo:$expectedFooVersion" {
expectGetMetadata()
if (platformType != ENFORCED_PLATFORM) {
expectGetArtifact()
}
}
}
if (platformType != ENFORCED_PLATFORM) {
run ':checkDeps'
} else {
fails ':checkDeps'
}
then:
if (platformType == ENFORCED_PLATFORM) {
failure.assertHasCause """Cannot find a version of 'org:foo' that satisfies the version constraints:
Dependency path ':test:unspecified' --> 'org:bar:2.0' (runtime) --> 'org:foo:3.1'
Constraint path ':test:unspecified' --> 'org:platform:1.1' (enforcedApiElements) --> 'org:foo:{require 3.1.1; reject 3.1 & 3.2}'
Constraint path ':test:unspecified' --> 'org:foo:{strictly 3.2}'"""
} else {
resolve.expectGraph {
root(':', ':test:') {
constraint('org:foo:{strictly 3.2}', "org:foo:$expectedFooVersion").byConstraint()
edge('org:platform:1.+', 'org:platform:1.1') {
byRequest()
if (platformType != MODULE) {
configuration(platformType == ENFORCED_PLATFORM ? 'enforcedApiElements' : 'apiElements')
noArtifacts()
}
constraint("org:bar:${StrictVersionsInPlatformCentricDevelopmentIntegrationTest.expectStrictVersion(platformType, '2.0')}", 'org:bar:2.0').byConstraint()
constraint("org:foo:${StrictVersionsInPlatformCentricDevelopmentIntegrationTest.expectStrictVersion(platformType, '3.1.1', '3.1 & 3.2')}", "org:foo:$expectedFooVersion").byConstraint()
}
edge('org:bar', 'org:bar:2.0') {
edge('org:foo:3.1', "org:foo:$expectedFooVersion").byAncestor()
}.byRequest()
}
if (platformType == ENFORCED_PLATFORM) {
nodesWithoutRoot.each { it.forced() }
}
}
}
where:
platformType << PlatformType.values()
}
@Unroll
@ToBeFixedForConfigurationCache
void "(5) if two libraries are combined without agreeing on an override, the original platform constraint is brought back [#platformType]"() {
updatedRepository(platformType)
settingsFile << "\ninclude 'recklessLibrary', 'secondLibrary'"
buildFile << """
project(':recklessLibrary') {
configurations { conf }
dependencies {
${platformDependency(platformType, 'org:platform:1.+')}
conf('org:bar')
constraints {
conf('org:foo') {
version { strictly '3.2' } // ignoring platform's reject
}
}
}
}
project(':secondLibrary') {
configurations { conf }
dependencies {
${platformDependency(platformType, 'org:platform:1.+')}
conf('org:bar')
}
}
dependencies {
conf(project(path: ':recklessLibrary', configuration: 'conf'))
conf(project(path: ':secondLibrary', configuration: 'conf'))
}
"""
when:
repositoryInteractions {
'org:platform' {
expectVersionListing()
}
'org:platform:1.1' {
expectGetMetadata()
}
'org:bar:2.0' {
expectGetMetadata()
}
'org:foo:3.1.1' {
expectGetMetadata()
}
'org:foo:3.2' {
if (platformType != ENFORCED_PLATFORM) {
expectGetMetadata()
}
}
}
fails ':checkDeps'
then:
if (platformType == ENFORCED_PLATFORM) {
failure.assertHasCause """Cannot find a version of 'org:foo' that satisfies the version constraints:
Dependency path ':test:unspecified' --> 'test:recklessLibrary:unspecified' (conf) --> 'org:bar:2.0' (runtime) --> 'org:foo:3.1'
Constraint path ':test:unspecified' --> 'test:recklessLibrary:unspecified' (conf) --> 'org:platform:1.1' (enforcedApiElements) --> 'org:foo:{require 3.1.1; reject 3.1 & 3.2}'
Constraint path ':test:unspecified' --> 'test:recklessLibrary:unspecified' (conf) --> 'org:foo:{strictly 3.2}'"""
} else {
def platformVariant = platformType == MODULE ? 'runtime' : 'apiElements'
failure.assertHasCause(
"""Cannot find a version of 'org:foo' that satisfies the version constraints:
Dependency path ':test:unspecified' --> 'test:recklessLibrary:unspecified' (conf) --> 'org:bar:2.0' (runtime) --> 'org:foo:3.1'
Constraint path ':test:unspecified' --> 'test:recklessLibrary:unspecified' (conf) --> 'org:platform:1.1' (${platformVariant}) --> 'org:foo:{strictly 3.1.1; reject 3.1 & 3.2}'
Constraint path ':test:unspecified' --> 'test:recklessLibrary:unspecified' (conf) --> 'org:foo:{strictly 3.2}'""")
}
where:
platformType << PlatformType.values()
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy