foundry.gradle.ApkVersioningPlugin.kt Maven / Gradle / Ivy
/*
* Copyright (C) 2022 Slack Technologies, LLC
*
* 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
*
* https://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.
*/
@file:Suppress("UnstableApiUsage")
package foundry.gradle
import com.android.build.api.variant.ApplicationAndroidComponentsExtension
import com.android.build.gradle.AppPlugin
import foundry.gradle.properties.localGradleProperty
import foundry.gradle.properties.setDisallowChanges
import java.util.Properties
import org.gradle.api.DefaultTask
import org.gradle.api.Plugin
import org.gradle.api.Project
import org.gradle.api.file.RegularFileProperty
import org.gradle.api.provider.Property
import org.gradle.api.provider.Provider
import org.gradle.api.tasks.CacheableTask
import org.gradle.api.tasks.Input
import org.gradle.api.tasks.OutputFile
import org.gradle.api.tasks.TaskAction
/**
* Uses AGP 4.0's new APIs for manipulating variant properties in a more cache-friendly way.
*
* Normally we should use cacheable tasks and reuse their values, but manifest manipulation appears
* to break their task dependencies in a way we don't understand. Since our computations are cheap,
* we just compute them on the fly for now.
*
* Based on examples here: https://github.com/android/gradle-recipes
*/
@Suppress("unused")
internal class ApkVersioningPlugin : Plugin {
@Suppress("LongMethod")
override fun apply(project: Project) {
project.plugins.withType(AppPlugin::class.java) {
val versionMajor = project.localGradleProperty("versionMajor")
val versionMinor = project.localGradleProperty("versionMinor")
val versionPatch = project.localGradleProperty("versionPatch")
val foundryProperties = FoundryProperties.getOrCreate(project)
val debugVersionNameProvider =
versionNameProvider(
versionMajor,
versionMinor,
versionPatch,
user = project.provider { foundryProperties.debugUserString },
)
val defaultVersionNameProvider =
versionNameProvider(
versionMajor,
versionMinor,
versionPatch,
user =
project.ciBuildNumber
.map { "" }
// Only provider the user if this is _not_ running on CI. Composition of properties
// is still a little weird in Gradle.
.orElse(project.providers.environmentVariable("USER")),
)
val debugVersionCodeProvider: Provider =
project.provider { foundryProperties.debugVersionCode }
val ciVersionFileProvider =
project.rootProject.layout.projectDirectory.file("ci/release.version")
val ciContent = project.providers.fileContents(ciVersionFileProvider)
val defaultVersionCodeProvider: Provider =
project.ciBuildNumber
.flatMap { ciContent.asText.map { it.toInt() } }
.orElse(debugVersionCodeProvider)
configureVariants(
project,
debugVersionNameProvider,
debugVersionCodeProvider,
defaultVersionNameProvider,
defaultVersionCodeProvider,
)
// Register a version properties task. This is run on ci via android_preflight.sh
val shortGitShaProvider = project.gitShaProvider()
val longGitShaProvider = project.fullGitShaProvider()
project.tasks.register("generateVersionProperties") {
outputFile.setDisallowChanges(
project.layout.buildDirectory.file("intermediates/versioning/version.properties")
)
versionCode.setDisallowChanges(defaultVersionCodeProvider)
versionName.setDisallowChanges(defaultVersionNameProvider)
shortGitSha.setDisallowChanges(shortGitShaProvider)
longGitSha.setDisallowChanges(longGitShaProvider)
this.versionMajor.setDisallowChanges(versionMajor)
this.versionMinor.setDisallowChanges(versionMinor)
}
}
}
private fun configureVariants(
project: Project,
debugVersionNameProvider: Provider,
debugVersionCodeProvider: Provider,
defaultVersionNameProvider: Provider,
defaultVersionCodeProvider: Provider,
) {
// Register version calculations for variants
project.configure {
onVariants { variant ->
val flavorName = variant.flavorName
val versionCodeProvider =
if (variant.buildType == "debug") {
debugVersionCodeProvider
} else {
defaultVersionCodeProvider
}
val versionNameProvider =
if (variant.buildType == "debug") {
debugVersionNameProvider
} else {
defaultVersionNameProvider
}
val mappedVersionNameProvider =
if (flavorName != "external") {
// Replacement for versionNameSuffix, which this overrides
versionNameProvider.map { "$it-$flavorName" }
} else {
versionNameProvider
}
// Have to iterate outputs because of APK splits.
variant.outputs.forEach { variantOutput ->
variantOutput.versionName.setDisallowChanges(mappedVersionNameProvider)
variantOutput.versionCode.setDisallowChanges(versionCodeProvider)
}
}
}
}
private fun versionNameProvider(
versionMajor: Provider,
versionMinor: Provider,
versionPatch: Provider,
user: Provider,
): Provider {
return versionMajor
.zip(versionMinor) { major, minor -> "$major.$minor" }
.zip(versionPatch) { prev, patch -> "$prev.$patch" }
.zip(user) { prev, possibleUser ->
val addOn =
possibleUser.takeIf { it.isNotEmpty() }?.let { presentUser -> "-$presentUser" } ?: ""
"$prev$addOn"
}
}
}
/**
* This generates a `version.properties` file that CI uses (see
* `ci/master/master.multibranch.jenkinsfile`).
*/
@CacheableTask
internal abstract class VersionPropertiesTask : DefaultTask() {
@get:Input abstract val versionMajor: Property
@get:Input abstract val versionMinor: Property
@get:Input abstract val shortGitSha: Property
@get:Input abstract val longGitSha: Property
@get:Input abstract val versionName: Property
@get:Input abstract val versionCode: Property
@get:OutputFile abstract val outputFile: RegularFileProperty
@TaskAction
fun generate() {
val releaseVersion = "${versionMajor.get()}.${versionMinor.get()}"
val resolvedVersionCode = versionCode.get()
val resolvedVersionName = versionName.get()
val versionProps =
Properties().apply {
setProperty("tag", "$resolvedVersionName-$resolvedVersionCode")
setProperty("git_short", shortGitSha.get())
setProperty("version_code", "" + resolvedVersionCode)
setProperty("full_git_sha", longGitSha.get())
setProperty("release_version", releaseVersion)
}
outputFile.asFile.get().writer().use {
versionProps.store(it, "Auto-generated by build. Do not check in")
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy