pachli-android/app/build.gradle.kts
Nik Clayton a3d45ca9ec
refactor: Convert from Gson to Moshi (#428)
Moshi is faster to decode JSON at runtime, is actively maintained, has a
smaller memory and method footprint, and a slightly smaller APK size.
Moshi also correctly creates default constructor arguments instead of
leaving them null, which was a source of `NullPointerExceptions` when
using Gson.

The conversion broadly consisted of:

- Adding `@JsonClass(generateAdapter = true)` to data classes that
marshall to/from JSON.

- Replacing `@SerializedName(value = ...)` with `@Json(name = ...)`.

- Replacing Gson instances with Moshi in Retrofit, Hilt, and tests.

- Using Moshi adapters to marshall to/from JSON instead of Gson `toJson`
/ `fromJson`.

- Deleting `Rfc3339DateJsonAdapter` and related code, and using the
equivalent adapter bundled with Moshi.

- Rewriting `GuardedBooleanAdapter` as a more generic `GuardedAdapter`.

- Deleting unused ProGuard rules; Moshi generates adapters using code
generation, not runtime reflection.

The conversion surfaced some bugs which have been fixed.

- Not all audio attachments have attachment size metadata. Don't show
the attachment preview if the metadata is missing.

- Some `throwable` were not being logged correctly.

- The wrong type was being used when parsing the response when sending a
scheduled status.

- Exceptions other than `HttpException` or `IoException` would also
cause a status to be resent. If there's a JSON error parsing a response
the status would be repeatedly sent.

- In tests strings containing error responses were not valid JSON.

- Workaround Mastodon a bug and ensure `filter.keywords` is populated,
https://github.com/mastodon/mastodon/issues/29142
2024-02-09 12:41:13 +01:00

206 lines
6.2 KiB
Plaintext

/*
* Copyright 2023 Pachli Association
*
* This file is a part of Pachli.
*
* This program is free software; you can redistribute it and/or modify it under the terms of the
* GNU General Public License as published by the Free Software Foundation; either version 3 of the
* License, or (at your option) any later version.
*
* Pachli is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
* the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
* Public License for more details.
*
* You should have received a copy of the GNU General Public License along with Pachli; if not,
* see <http://www.gnu.org/licenses>.
*/
import com.android.build.gradle.internal.api.ApkVariantOutputImpl
plugins {
alias(libs.plugins.pachli.android.application)
alias(libs.plugins.pachli.android.hilt)
alias(libs.plugins.kotlin.parcelize)
}
apply(from = "gitTools.gradle")
val getGitSha: groovy.lang.Closure<String> by extra
val getGitRevCount: groovy.lang.Closure<Int> by extra
android {
namespace = "app.pachli"
defaultConfig {
applicationId = "app.pachli"
versionCode = 11
versionName = "2.2.0"
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
testInstrumentationRunnerArguments["disableAnalytics"] = "true"
vectorDrawables.useSupportLibrary = true
}
buildTypes {
debug {
isDefault = true
}
release {
isMinifyEnabled = true
isShrinkResources = true
proguardFiles(getDefaultProguardFile("proguard-android-optimize.txt"), "proguard-rules.pro")
}
}
compileOptions {
isCoreLibraryDesugaringEnabled = true
}
packaging {
resources.excludes.apply {
add("LICENSE_OFL")
add("LICENSE_UNICODE")
}
}
bundle {
language {
// bundle all languages in every apk so the dynamic language switching works
enableSplit = false
}
}
dependenciesInfo {
includeInApk = false
includeInBundle = false
}
testOptions {
unitTests.all {
it.systemProperty("robolectric.logging.enabled", "true")
it.systemProperty("robolectric.lazyload", "ON")
}
}
applicationVariants.configureEach {
tasks.register("printVersionInfo${name.replaceFirstChar { it.uppercaseChar() }}") {
notCompatibleWithConfigurationCache("Should always print the version info")
doLast {
println("$versionCode $versionName")
}
}
outputs.configureEach {
this as ApkVariantOutputImpl
// Set the "orange" release versionCode to the number of commits on the
// branch, to ensure the versionCode updates on every release. Include the
// SHA of the current commit to help with troubleshooting bug reports
if (flavorName.startsWith("orange")) {
versionNameOverride = "$versionName+${getGitSha()}"
}
if (buildType.name == "release" && flavorName.startsWith("orange")) {
versionCodeOverride = getGitRevCount()
}
outputFileName = "Pachli_${versionName}_${versionCode}_${getGitSha()}_${flavorName}_${buildType.name}.apk"
}
}
}
configurations {
// JNI-only libraries don't play nicely with Robolectric
// see https://github.com/tuskyapp/Tusky/pull/3367 and
// https://github.com/google/conscrypt/issues/649
testImplementation {
exclude(group = "org.conscrypt", module = "conscrypt-android")
}
}
dependencies {
coreLibraryDesugaring(libs.desugar.jdk.libs)
// CachedTimelineRemoteMediator needs the @Transaction annotation from Room
compileOnly(libs.bundles.room)
testCompileOnly(libs.bundles.room)
implementation(projects.core.accounts)
implementation(projects.core.activity)
implementation(projects.core.common)
implementation(projects.core.data)
implementation(projects.core.database)
implementation(projects.core.designsystem)
implementation(projects.core.navigation)
implementation(projects.core.network)
implementation(projects.core.preferences)
implementation(projects.core.ui)
implementation(projects.feature.about)
implementation(projects.feature.login)
implementation(libs.kotlinx.coroutines.android)
implementation(libs.kotlinx.coroutines.rx3)
implementation(libs.bundles.androidx)
implementation(libs.android.material)
implementation(libs.moshi)
implementation(libs.moshi.adapters)
ksp(libs.moshi.codegen)
implementation(libs.bundles.retrofit)
implementation(libs.bundles.okhttp)
implementation(libs.conscrypt.android)
implementation(libs.bundles.glide)
ksp(libs.glide.compiler)
implementation(libs.bundles.rxjava3)
implementation(libs.bundles.autodispose)
implementation(libs.sparkbutton)
implementation(libs.touchimageview)
implementation(libs.bundles.material.drawer)
implementation(libs.material.typeface)
implementation(libs.image.cropper)
implementation(libs.bundles.filemojicompat)
implementation(libs.bouncycastle)
implementation(libs.unified.push)
implementation(libs.bundles.xmldiff)
implementation(libs.timber)
googleImplementation(libs.app.update)
googleImplementation(libs.app.update.ktx)
implementation(libs.semver)
debugImplementation(libs.leakcanary)
testImplementation(projects.core.testing)
testImplementation(libs.androidx.test.junit)
testImplementation(libs.robolectric)
testImplementation(libs.bundles.mockito)
testImplementation(libs.androidx.core.testing)
testImplementation(libs.kotlinx.coroutines.test)
testImplementation(libs.androidx.work.testing)
testImplementation(libs.truth)
testImplementation(libs.turbine)
testImplementation(libs.androidx.test.core.ktx)
androidTestImplementation(libs.espresso.core)
androidTestImplementation(libs.androidx.room.testing)
androidTestImplementation(libs.androidx.test.junit)
androidTestImplementation(libs.androidx.test.core.ktx)
lintChecks(projects.checks)
}