plugins { id("com.android.application") kotlin("android") id("kotlin-parcelize") id("zcash.android-build-conventions") id("com.github.triplet.play") id("com.osacky.fladle") id("wtf.emulator.gradle") id("zcash.emulator-wtf-conventions") } 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" } testInstrumentationRunner = "co.electriccoin.zcash.test.ZcashUiTestRunner" } 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("debug").apply { // Note that the build-conventions defines the res configs isPseudoLocalesEnabled = true } 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") } } } 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(libs.zcash.sdk) // just to configure logging implementation(projects.crashAndroidLib) implementation(projects.spackleAndroidLib) 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.sdkExtUiLib) androidTestImplementation(projects.spackleLib) androidTestImplementation(projects.testLib) 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) } } } fladle { // 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_SDK = 23 @Suppress("MagicNumber", "PropertyName", "VariableNaming") val FIREBASE_TEST_LAB_MAX_SDK = 30 val minSdkVersion = run { val buildMinSdk = project.properties["ANDROID_APP_MIN_SDK_VERSION"].toString().toInt() buildMinSdk.coerceAtLeast(FIREBASE_TEST_LAB_MIN_SDK).toString() } val targetSdkVersion = run { val buildTargetSdk = project.properties["ANDROID_TARGET_SDK_VERSION"].toString().toInt() buildTargetSdk.coerceAtMost(FIREBASE_TEST_LAB_MAX_SDK).toString() } val firebaseTestLabKeyPath = project.properties["ZCASH_FIREBASE_TEST_LAB_API_KEY_PATH"].toString() val firebaseProject = project.properties["ZCASH_FIREBASE_TEST_LAB_PROJECT"].toString() if (firebaseTestLabKeyPath.isNotEmpty()) { serviceAccountCredentials.set(File(firebaseTestLabKeyPath)) } else if (firebaseProject.isNotEmpty()) { projectId.set(firebaseProject) } configs { create("sanityConfig") { clearPropertiesForSanityRobo() debugApk.set( project.provider { "${buildDir}/outputs/universal_apk/zcashmainnetRelease/app-zcashmainnet-release-universal.apk" } ) testTimeout.set("3m") devices.addAll( mapOf("model" to "Pixel2", "version" to minSdkVersion), mapOf("model" to "Pixel2", "version" to targetSdkVersion) ) flankVersion.set(libs.versions.flank.get()) } } } emulatorwtf { directoriesToPull.set(listOf("/sdcard/googletest/test_outputfiles")) val appMinSdkVersion = run { @Suppress("MagicNumber", "PropertyName", "VariableNaming") val EMULATOR_WTF_MIN_SDK = 23 val buildMinSdk = project.properties["ANDROID_APP_MIN_SDK_VERSION"].toString().toInt() buildMinSdk.coerceAtLeast(EMULATOR_WTF_MIN_SDK).toString() } val targetSdkVersion = run { @Suppress("MagicNumber", "PropertyName", "VariableNaming") val EMULATOR_WTF_MAX_SDK = 31 val buildTargetSdk = project.properties["ANDROID_TARGET_SDK_VERSION"].toString().toInt() buildTargetSdk.coerceAtMost(EMULATOR_WTF_MAX_SDK).toString() } devices.set( listOf( mapOf("model" to "Pixel2", "version" to appMinSdkVersion), mapOf("model" to "Pixel2", "version" to targetSdkVersion) ) ) }