plugins { id("com.android.application") kotlin("android") id("kotlin-parcelize") id("zcash.android-build-conventions") id("com.github.triplet.play") id("com.osacky.fladle") } val packageName = "co.electriccoin.zcash" // Force orchestrator to be used for this module, because we need cleared state to generate screenshots val isOrchestratorEnabled = true android { defaultConfig { applicationId = packageName // If Google Play deployment is triggered, then these are placeholders which are overwritten // when the deployment runs versionCode = project.property("ZCASH_VERSION_CODE").toString().toInt() versionName = project.property("ZCASH_VERSION_NAME").toString() if (isOrchestratorEnabled) { testInstrumentationRunnerArguments["clearPackageData"] = "true" } } if (isOrchestratorEnabled) { testOptions { execution = "ANDROIDX_TEST_ORCHESTRATOR" } } compileOptions { isCoreLibraryDesugaringEnabled = true } flavorDimensions.add("network") val testNetFlavorName = "zcashtestnet" productFlavors { // would rather name them "testnet" and "mainnet" but product flavor names cannot start with the word "test" create(testNetFlavorName) { dimension = "network" applicationId = "$packageName.testnet" // allow to be installed alongside mainnet matchingFallbacks.addAll(listOf("zcashtestnet", "debug")) } create("zcashmainnet") { dimension = "network" applicationId = packageName matchingFallbacks.addAll(listOf("zcashmainnet", "release")) } } val releaseKeystorePath = project.property("ZCASH_RELEASE_KEYSTORE_PATH").toString() val releaseKeystorePassword = project.property("ZCASH_RELEASE_KEYSTORE_PASSWORD").toString() val releaseKeyAlias = project.property("ZCASH_RELEASE_KEY_ALIAS").toString() val releaseKeyAliasPassword = project.property("ZCASH_RELEASE_KEY_ALIAS_PASSWORD").toString() val isReleaseSigningConfigured = listOf( releaseKeystorePath, releaseKeystorePassword, releaseKeyAlias, releaseKeyAliasPassword ).all { !it.isNullOrBlank() } signingConfigs { if (isReleaseSigningConfigured) { // If this block doesn't execute, the output will be unsigned create("release").apply { storeFile = File(releaseKeystorePath) storePassword = releaseKeystorePassword keyAlias = releaseKeyAlias keyPassword = releaseKeyAliasPassword } } } buildTypes { getByName("release").apply { isMinifyEnabled = project.property("IS_MINIFY_ENABLED").toString().toBoolean() isShrinkResources = project.property("IS_MINIFY_ENABLED").toString().toBoolean() proguardFiles( getDefaultProguardFile("proguard-android-optimize.txt"), "proguard-project.txt" ) if (isReleaseSigningConfigured) { signingConfig = signingConfigs.getByName("release") } } } // TODO [#6]: Figure out how to move this into the build-conventions kotlinOptions { jvmTarget = libs.versions.java.get() allWarningsAsErrors = project.property("ZCASH_IS_TREAT_WARNINGS_AS_ERRORS").toString().toBoolean() } packagingOptions { resources.excludes.addAll( listOf( "**/*.kotlin_metadata", ".readme", "build-data.properties", "META-INF/*.kotlin_module", "META-INF/android.arch**", "META-INF/androidx**", "META-INF/com.android**", "META-INF/com.google.android.material_material.version", "META-INF/com.google.dagger_dagger.version", "META-INF/services/org.jetbrains.kotlin.compiler.plugin.CommandLineProcessor", "META-INF/services/org.jetbrains.kotlin.compiler.plugin.ComponentRegistrar", "META-INF/services/org.jetbrains.kotlin.diagnostics.rendering.DefaultErrorMessages\$Extension", "firebase-**.properties", "kotlin/**", "play-services-**.properties", "protolite-well-known-types.properties", "transport-api.properties", "transport-backend-cct.properties", "transport-runtime.properties" ) ) } playConfigs { register(testNetFlavorName) { enabled.set(false) } } } dependencies { coreLibraryDesugaring(libs.desugaring) implementation(libs.androidx.activity) implementation(libs.androidx.annotation) implementation(libs.androidx.core) implementation(libs.kotlin.stdlib) implementation(libs.kotlinx.coroutines.android) implementation(libs.kotlinx.coroutines.core) implementation(projects.uiLib) androidTestImplementation(libs.androidx.compose.test.junit) androidTestImplementation(libs.androidx.navigation.compose) androidTestImplementation(libs.androidx.uiAutomator) androidTestImplementation(libs.bundles.androidx.test) androidTestImplementation(projects.sdkExtLib) androidTestImplementation(projects.spackleLib) if (isOrchestratorEnabled) { androidTestUtil(libs.androidx.test.orchestrator) { artifact { type = "apk" } } } } val googlePlayServiceKeyFilePath = project.property("ZCASH_GOOGLE_PLAY_SERVICE_KEY_FILE_PATH").toString() if (googlePlayServiceKeyFilePath.isNotEmpty()) { // Update the versionName to reflect bumps in versionCode androidComponents { val versionCodeOffset = 0 // Change this to zero the final digit of the versionName onVariants { variant -> for (output in variant.outputs) { val processedVersionCode = output.versionCode.map { playVersionCode -> val defaultVersionName = project.property("ZCASH_VERSION_NAME").toString() // Version names will look like `myCustomVersionName.123` playVersionCode?.let { val delta = it - versionCodeOffset if (delta < 0) { defaultVersionName } else { "$defaultVersionName.$delta" } } ?: defaultVersionName } output.versionName.set(processedVersionCode) } } } configure { serviceAccountCredentials.set(File(googlePlayServiceKeyFilePath)) // For safety, only allow deployment to internal testing track track.set("internal") // Automatically manage version incrementing resolutionStrategy.set(com.github.triplet.gradle.androidpublisher.ResolutionStrategy.AUTO) val deployMode = project.property("ZCASH_GOOGLE_PLAY_DEPLOY_MODE").toString() if ("build" == deployMode) { releaseStatus.set(com.github.triplet.gradle.androidpublisher.ReleaseStatus.DRAFT) // Prevent upload; only generates a build with the correct version number commit.set(false) } else if ("deploy" == deployMode) { releaseStatus.set(com.github.triplet.gradle.androidpublisher.ReleaseStatus.COMPLETED) } } } val reportsDirectory = "${buildDir}/reports/androidTests/connected" // This is coordinated with `EccScreenCaptureProcessor` val onDeviceScreenshotsDirectory = "/sdcard/Pictures/zcash_screenshots" val clearScreenshotsTask = tasks.create("clearScreenshots") { executable = project.android.adbExecutable.absolutePath args = listOf("shell", "rm", "-r", onDeviceScreenshotsDirectory) } val fetchScreenshotsTask = tasks.create("fetchScreenshots") { executable = project.android.adbExecutable.absolutePath args = listOf("pull", onDeviceScreenshotsDirectory, reportsDirectory) finalizedBy(clearScreenshotsTask) } tasks.whenTaskAdded { if (name == "connectedZcashmainnetDebugAndroidTest") { finalizedBy(fetchScreenshotsTask) } } // Firebase Test Lab has min and max values that might differ from our project's // These are determined by `gcloud firebase test android models list` @Suppress("MagicNumber", "PropertyName", "VariableNaming") val FIREBASE_TEST_LAB_MIN_API = 23 @Suppress("MagicNumber", "PropertyName", "VariableNaming") val FIREBASE_TEST_LAB_MAX_API = 30 val firebaseTestLabKeyPath = project.properties["ZCASH_FIREBASE_TEST_LAB_API_KEY_PATH"].toString() if (firebaseTestLabKeyPath.isNotBlank()) { val minSdkVersion = run { val buildMinSdk = project.properties["ANDROID_MIN_SDK_VERSION"].toString().toInt() buildMinSdk.coerceAtLeast(FIREBASE_TEST_LAB_MIN_API).toString() } val targetSdkVersion = run { val buildTargetSdk = project.properties["ANDROID_TARGET_SDK_VERSION"].toString().toInt() buildTargetSdk.coerceAtMost(FIREBASE_TEST_LAB_MAX_API).toString() } fladle { serviceAccountCredentials.set(File(firebaseTestLabKeyPath)) configs { create("sanityConfig") { clearPropertiesForSanityRobo() debugApk.set( project.provider { "${buildDir}/outputs/apk/zcashmainnet/release/app-zcashmainnet-release.apk" } ) testTimeout.set("5m") devices.addAll( mapOf("model" to "Nexus6", "version" to minSdkVersion), mapOf("model" to "Pixel2", "version" to targetSdkVersion) ) } } } }