Context
- I am using Gradle 8.4.
- The project is a multi-module project.
- The project has several modules.
- The modules have a lot of overlap in the structure of the build.gradle, and I have gotten them to the point where they are practically the same.
- I have written custom plugins in Kotlin.
- The plugins react to the application of other plugins.
- One plugin in particular, creates and configures custom source sets. Right now, the source sets are created by using the
NamedDomainObjectContainer
method, and mybuild.gradle
files have the relevant DSL block like so:
scalaVariants {
create("2.12") // The plugin uses this to create two source sets, 'scala212' and 'testScala212', wire up the configurations, and tasks.
create("2.13") // The plugin uses this to create two source sets, 'scala213' and 'testScala213', wire up the configurations, and tasks.
}
This is worked, however I am faced with a lot of duplication in my dependencies
block. I have this for example:
scalaVariants {
create("2.12")
create("2.13")
}
ext {
junitVersion = "5.10.1"
// a bunch of other dependency version
sparkVersion = "3.4.2"
scalaBinaryVersion = project.findProperty("scala.binary.version")
}
dependencies {
implementation("org.apache.spark:spark-sql_${scalaBinaryVersion}:${sparkVersion}")
// other regular and scala flavoured dependencies affecting the 'main' source set
testImplementation("org.junit.jupiter:junit-jupiter:${junitVersion}")
// other regular and scala flavoured dependencies affecting the 'test' source set
scala212Implementation("org.apache.spark:spark-sql_2.12:${sparkVersion}")
// other regular and scala flavoured dependencies affecting the 'scala212' source set
testScala212Implementation("org.junit.jupiter:junit-jupiter:${junitVersion}")
// other regular and scala flavoured dependencies affecting the 'testScala212' source set
scala213Implementation("org.apache.spark:spark-sql_2.13:${sparkVersion}")
// other regular and scala flavoured dependencies affecting the 'scala213' source set
testScala213Implementation("org.junit.jupiter:junit-jupiter:${junitVersion}")
// other regular and scala flavoured dependencies affecting the 'testScala213' source set
}
Now this pattern occurs across all the modules where it has to be compiled using different Scala variants of Apache Spark.
So I did some research, and I came up with an approach I’d like to take. Namely, I want to have a “DSL” that looks like this:
scalaNg {
defaultScalaBinaryVersion = "2.12"
supportedScalaBinaryVersions = ["2.12", "2.13"]
scalaImpl("org.apache.spark", "spark-sql", project.findProperty("spark.version"))
testImpl("org.junit.jupiter", "junit-jupiter", project.findProperty("junit.version"))
}
And the plugin will then configure the dependencies for each source set. scalaImpl
means that a “Scala Binary Version Aware” dependency should be created. The inspiration of this comes from sbt
, which uses the %%
operator to automatically “inject” the current Scala binary version into the artefact’s name, and it will place the dependency in the implementation
, scala212Implementation
, and scala213Implementation
configurations. Similarly, the testImpl
will do the same for the ‘test’ source sets.
The Problem
So I wrote a plugin (as we do), and whilst I get no compilation errors, when I check what configurations and source sets are present, its as if the plugin was never applied.
Here is my plugin code:
class ScalaVariantPluginNg : Plugin<Project> {
private fun String.scalaSourceSetName() = "scala${this.replace(".", "")}"
private fun String.testScalaSourceSetName() = "testScala${this.replace(".", "")}"
override fun apply(target: Project) {
target.plugins.withType<JavaPlugin> {
val ext = target.extensions.create("scalaNg", VariantExtension::class.java)
val sourceSetContainer = target.extensions.getByType<SourceSetContainer>()
createSourceSets(sourceSetContainer, ext)
configureConfigurations(target, ext)
configureDependencies(target, ext)
}
}
private fun createSourceSets(
sourceSetContainer: SourceSetContainer,
variant: VariantExtension
) = variant.supportedScalaBinaryVersions.map { scalaBinaryVersions ->
scalaBinaryVersions.forEach { scalaBinaryVersion ->
val sourceSetName = scalaBinaryVersion.scalaSourceSetName()
val testSourceSetName = scalaBinaryVersion.testScalaSourceSetName()
sourceSetContainer.create(sourceSetName) {
java.setSrcDirs(listOf("src/main/java", "src/${sourceSetName}/java"))
resources.setSrcDirs(listOf("src/main/resources", "src/${sourceSetName}/resources"))
}
sourceSetContainer.create(testSourceSetName) {
compileClasspath += sourceSetContainer[sourceSetName].output
runtimeClasspath += sourceSetContainer[sourceSetName].output
}
}
}
private fun configureConfigurations(target: Project, ext: VariantExtension) =
ext.supportedScalaBinaryVersions.map { scalaBinaryVersions ->
configureConfigurations(target, scalaBinaryVersions)
}
private fun configureConfigurations(target: Project, scalaBinaryVersions: List<String>) =
scalaBinaryVersions.forEach {
configureConfigurations(target, it)
}
private fun configureConfigurations(target: Project, scalaBinaryVersion: String) {
val configurations = target.configurations
val sourceSetName = scalaBinaryVersion.scalaSourceSetName()
val testSourceSetName = scalaBinaryVersion.testScalaSourceSetName()
val api = configurations.create("${sourceSetName}Api")
configurations.create("${sourceSetName}ApiElements") {
extendsFrom(api)
isCanBeResolved = false
isCanBeConsumed = true
}
val implementation = configurations.named("${sourceSetName}Implementation") {
extendsFrom(api)
}
val runtimeOnly = configurations.named("${sourceSetName}RuntimeOnly")
val compileOnly = configurations.named("${sourceSetName}CompileOnly")
configurations.create("${sourceSetName}RuntimeElements") {
extendsFrom(implementation.get(), runtimeOnly.get())
isCanBeResolved = false
isCanBeConsumed = true
}
configurations.named("${sourceSetName}CompileClasspath") {
extendsFrom(compileOnly.get(), implementation.get())
isCanBeResolved = true
}
configurations.named("${sourceSetName}RuntimeClasspath") {
extendsFrom(implementation.get(), runtimeOnly.get())
isCanBeResolved = true
}
val testImplementation = configurations.named("${testSourceSetName}Implementation") {
extendsFrom(implementation.get())
}
val testRuntimeOnly = configurations.named("${testSourceSetName}RuntimeOnly") {
extendsFrom(runtimeOnly.get())
}
val testCompileOnly = configurations.named("${testSourceSetName}CompileOnly")
configurations.named("${testSourceSetName}CompileClasspath") {
extendsFrom(testCompileOnly.get(), testImplementation.get())
isCanBeResolved = true
}
configurations.named("${testSourceSetName}RuntimeClasspath") {
extendsFrom(testImplementation.get(), testRuntimeOnly.get())
isCanBeResolved = true
}
}
private fun configureDependencies(target: Project, variant: VariantExtension) {
val partitionedDependencies = variant.dependencies.map { deps ->
deps.partition { it.configuration.startsWith("test") }
}
partitionedDependencies.map { (testDependencies, mainDependencies) ->
val defaultScalaBinaryVersion = variant.defaultScalaBinaryVersion.get()
applyDependenciesToMainSourceSet(target, mainDependencies, defaultScalaBinaryVersion)
applyDependenciesToTestSourceSet(target, testDependencies, defaultScalaBinaryVersion)
variant.supportedScalaBinaryVersions.get().map { scalaBinaryVersion ->
applyDependenciesToMainScalaSourceSet(target, mainDependencies, scalaBinaryVersion)
applyDependenciesToTestScalaSourceSet(target, testDependencies, scalaBinaryVersion)
}
}
}
private fun applyDependenciesToMainSourceSet(
target: Project,
mainDependencies: List<DependencySpec>,
scalaBinaryVersion: String
) = mainDependencies.forEach { dep ->
val dependencyNotation = resolveDependencyNotation(dep, scalaBinaryVersion)
target.dependencies.add(dep.configuration, dependencyNotation)
}
private fun applyDependenciesToTestSourceSet(
target: Project,
testDependencies: List<DependencySpec>,
scalaBinaryVersion: String
) = testDependencies.forEach { dep ->
val dependencyNotation = resolveDependencyNotation(dep, scalaBinaryVersion)
target.dependencies.add(dep.configuration, dependencyNotation)
}
private fun applyDependenciesToMainScalaSourceSet(
target: Project,
mainDependencies: List<DependencySpec>,
scalaBinaryVersion: String
) {
val scalaSourceSetName = scalaBinaryVersion.scalaSourceSetName()
mainDependencies.forEach { dep ->
val dependencyNotation = resolveDependencyNotation(dep, scalaBinaryVersion)
target.dependencies.add(
"${scalaSourceSetName}${dep.configuration.capitalize()}",
dependencyNotation
)
}
}
private fun applyDependenciesToTestScalaSourceSet(
target: Project,
testDependencies: List<DependencySpec>,
scalaBinaryVersion: String
) {
val testSourceSetName = scalaBinaryVersion.testScalaSourceSetName()
testDependencies.forEach { dep ->
val dependencyNotation = resolveDependencyNotation(dep, scalaBinaryVersion)
target.dependencies.add(
"${testSourceSetName}${dep.configuration.capitalize()}",
dependencyNotation
)
}
}
private fun resolveDependencyNotation(dep: DependencySpec, scalaBinaryVersion: String): String {
return when (dep) {
is DependencySpec.ScalaAwareDependencySpec -> {
"${dep.group}:${
dep.name.replace(
"{scalaBinaryVersion}",
scalaBinaryVersion ?: ""
)
}:${dep.version}"
}
is DependencySpec.RegularDependencySpec -> {
"${dep.group}:${dep.name}:${dep.version}"
}
}
}
}
sealed interface DependencySpec {
val group: String
val name: String
val version: String
val configuration: String
class RegularDependencySpec(
override val group: String,
override val name: String,
override val version: String,
override val configuration: String
) : DependencySpec
class ScalaAwareDependencySpec(
override val group: String,
override val name: String,
override val version: String,
override val configuration: String
) : DependencySpec
}
interface VariantExtension {
val defaultScalaBinaryVersion: Property<String>
val supportedScalaBinaryVersions: ListProperty<String>
val dependencies: ListProperty<DependencySpec>
fun add(dependency: DependencySpec) {
dependencies.add(dependency)
}
fun impl(group: String, module: String, version: String) {
add(DependencySpec.RegularDependencySpec(group, module, version, "implementation"))
}
fun scalaImpl(group: String, module: String, version: String) {
add(DependencySpec.ScalaAwareDependencySpec(group, module, version, "implementation"))
}
fun compileOnly(group: String, module: String, version: String) {
add(DependencySpec.RegularDependencySpec(group, module, version, "compileOnly"))
}
fun scalaCompileOnly(group: String, module: String, version: String) {
add(DependencySpec.ScalaAwareDependencySpec(group, module, version, "compileOnly"))
}
fun runtimeOnly(group: String, module: String, version: String) {
add(DependencySpec.RegularDependencySpec(group, module, version, "runtimeOnly"))
}
fun scalaRuntimeOnly(group: String, module: String, version: String) {
add(DependencySpec.ScalaAwareDependencySpec(group, module, version, "runtimeOnly"))
}
fun api(group: String, module: String, version: String) {
add(DependencySpec.RegularDependencySpec(group, module, version, "api"))
}
fun scalaApi(group: String, module: String, version: String) {
add(DependencySpec.ScalaAwareDependencySpec(group, module, version, "api"))
}
fun testImplementation(group: String, module: String, version: String) {
add(DependencySpec.RegularDependencySpec(group, module, version, "testImplementation"))
}
fun scalaTestImplementation(group: String, module: String, version: String) {
add(DependencySpec.ScalaAwareDependencySpec(group, module, version, "testImplementation"))
}
fun testCompileOnly(group: String, module: String, version: String) {
add(DependencySpec.RegularDependencySpec(group, module, version, "testCompileOnly"))
}
fun scalaTestCompileOnly(group: String, module: String, version: String) {
add(DependencySpec.ScalaAwareDependencySpec(group, module, version, "testCompileOnly"))
}
fun testRuntimeOnly(group: String, module: String, version: String) {
add(DependencySpec.RegularDependencySpec(group, module, version, "testRuntimeOnly"))
}
fun scalaTestRuntimeOnly(group: String, module: String, version: String) {
add(DependencySpec.ScalaAwareDependencySpec(group, module, version, "testRuntimeOnly"))
}
}