diff --git a/src/main/kotlin/app/cash/licensee/pluginExtension.kt b/src/main/kotlin/app/cash/licensee/pluginExtension.kt index 3d5f2330..622f87d1 100644 --- a/src/main/kotlin/app/cash/licensee/pluginExtension.kt +++ b/src/main/kotlin/app/cash/licensee/pluginExtension.kt @@ -18,14 +18,16 @@ package app.cash.licensee import app.cash.licensee.LicenseeExtension.AllowDependencyOptions import app.cash.licensee.LicenseeExtension.IgnoreDependencyOptions import java.util.Optional +import javax.inject.Inject import org.gradle.api.Action import org.gradle.api.Named +import org.gradle.api.NamedDomainObjectCollection import org.gradle.api.NamedDomainObjectContainer import org.gradle.api.artifacts.Dependency import org.gradle.api.provider.MapProperty import org.gradle.api.provider.Property import org.gradle.api.provider.Provider -import org.gradle.api.provider.SetProperty +import org.gradle.api.provider.ProviderFactory @Suppress("unused") // Public API for Gradle build scripts. interface LicenseeExtension { @@ -245,8 +247,19 @@ internal abstract class IgnoredCoordinate : Named { abstract val ignoredDatas: MapProperty } -internal abstract class MutableLicenseeExtension : LicenseeExtension { - internal abstract val allowedIdentifiers: SetProperty +internal abstract class SpdxId(private val spdxId: String) : Named { + override fun getName(): String = spdxId + init { + requireNotNull(SpdxLicenses.embedded.findByIdentifier(spdxId)) { + "$name is not a valid SPDX id." + } + } +} + +internal abstract class MutableLicenseeExtension @Inject constructor( + private val providers: ProviderFactory, +) : LicenseeExtension { + internal abstract val allowedIdentifiers: NamedDomainObjectContainer internal abstract val allowedUrls: MapProperty> internal abstract val allowedDependencies: MapProperty> internal abstract val ignoredGroupIds: MapProperty @@ -273,9 +286,11 @@ internal abstract class MutableLicenseeExtension : LicenseeExtension { } fun toLicenseValidationConfig(): Provider { - return allowedIdentifiers.zip(allowedUrls, allowedDependencies) { allowedIdentifiers, allowedUrls, allowedDependencies -> + return allowedIdentifiers.elements(providers) { + it.name + }.zip(allowedUrls, allowedDependencies) { allowedIdentifiers, allowedUrls, allowedDependencies -> ValidationConfig( - allowedIdentifiers.toSet(), + allowedIdentifiers, allowedUrls.mapValues { it.value.orElse(null) }, @@ -287,7 +302,7 @@ internal abstract class MutableLicenseeExtension : LicenseeExtension { } override fun allow(spdxId: String) { - allowedIdentifiers.add(spdxId) + allowedIdentifiers.register(spdxId) } override fun allowUrl(url: String, options: Action) { @@ -397,3 +412,13 @@ private fun NamedDomainObjectContainer.configure(name: String, config: Ac register(name, config) } } + +// https://github.com/gradle/gradle/issues/28043 +inline fun NamedDomainObjectCollection.elements( + providers: ProviderFactory, + crossinline transform: (T) -> R, +): Provider> { + return providers.provider { + mapTo(mutableSetOf(), transform) + } +} diff --git a/src/test/fixtures/allow-with-invalid-spdx/build.gradle b/src/test/fixtures/allow-with-invalid-spdx/build.gradle new file mode 100644 index 00000000..bc50465f --- /dev/null +++ b/src/test/fixtures/allow-with-invalid-spdx/build.gradle @@ -0,0 +1,8 @@ +plugins { + id("java-library") + alias(libs.plugins.licensee) +} + +licensee { + allow("ASDF") +} diff --git a/src/test/fixtures/allow-with-invalid-spdx/settings.gradle b/src/test/fixtures/allow-with-invalid-spdx/settings.gradle new file mode 100644 index 00000000..379dbf56 --- /dev/null +++ b/src/test/fixtures/allow-with-invalid-spdx/settings.gradle @@ -0,0 +1,7 @@ +pluginManagement { + includeBuild("../../test-build-logic") +} + +plugins { + id("licenseeTests") +} diff --git a/src/test/kotlin/app/cash/licensee/LicenseePluginFixtureTest.kt b/src/test/kotlin/app/cash/licensee/LicenseePluginFixtureTest.kt index df399c92..abdca42e 100644 --- a/src/test/kotlin/app/cash/licensee/LicenseePluginFixtureTest.kt +++ b/src/test/kotlin/app/cash/licensee/LicenseePluginFixtureTest.kt @@ -205,6 +205,16 @@ class LicenseePluginFixtureTest { ) } + @Test fun invalidSpdxFails( + @TestParameter("allow-with-invalid-spdx") fixtureName: String, + ) { + val fixtureDir = File(fixturesDir, fixtureName) + val result = createRunner(fixtureDir).buildAndFail() + assertThat(result.output).contains( + "ASDF is not a valid SPDX id.", + ) + } + @Test fun pluginMissingOnRootFails( @TestParameter("plugin-missing-on-root-fails") fixtureName: String, ) {