StrictMode: relax strictmode. Don't be fatal if running on FirebaseTestDevice
The only flavor running on firebase test device is the google one. The google
one log exception on Firebase so we will get the information anyway
/*
* Geekttrss is a RSS feed reader application on the Android Platform.
*
* Copyright (C) 2017-2019 by Frederic-Charles Barthelery.
*
* This file is part of Geekttrss.
*
* Geekttrss 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.
*
* Geekttrss 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 Geekttrss. If not, see <http://www.gnu.org/licenses/>.
*/
package com.geekorum.ttrss.debugtools
import android.app.Application
import android.content.ContentResolver
import android.os.Build
import android.os.StrictMode
import android.os.StrictMode.allowThreadDiskReads
import android.os.strictmode.Violation
import androidx.annotation.RequiresApi
import com.geekorum.geekdroid.dagger.AppInitializer
import com.geekorum.geekdroid.dagger.AppInitializersModule
import com.geekorum.ttrss.BuildConfig
import dagger.Binds
import dagger.Module
import dagger.multibindings.IntoSet
import kotlinx.coroutines.Dispatchers
import timber.log.Timber
import java.util.concurrent.Executors
import javax.inject.Inject
import kotlin.jvm.internal.Reflection
private const val TAG = "StrictMode"
/**
* Configure StrictMode policies
*/
class StrictModeInitializer @Inject constructor(
contentResolver: ContentResolver
) : AppInitializer {
private val listenerExecutor by lazy { Executors.newSingleThreadExecutor() }
private val shouldBeFatal: Boolean = (BuildConfig.DEBUG)
@delegate:RequiresApi(Build.VERSION_CODES.P)
private val violationListener: ViolationListener by lazy { ViolationListener() }
override fun initialize(app: Application) {
StrictMode.setVmPolicy(StrictMode.VmPolicy.Builder()
.detectActivityLeaks()
.detectCleartextNetwork()
.detectFileUriExposure()
.detectLeakedClosableObjects()
.detectLeakedRegistrationObjects()
.detectLeakedSqlLiteObjects()
.penaltyLog()
.apply {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
detectContentUriWithoutPermission()
// because crashlytics don't tag its socket
if (BuildConfig.FLAVOR != "google" || !shouldBeFatal) {
detectUntaggedSockets()
}
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
// appcompat use nonsdk
// detectNonSdkApiUsage()
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
penaltyListener(listenerExecutor, violationListener)
}
if (shouldBeFatal) {
penaltyDeath()
}
}
.build())
StrictMode.setThreadPolicy(StrictMode.ThreadPolicy.Builder()
.detectAll()
.penaltyLog()
.apply {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
penaltyListener(listenerExecutor, violationListener)
}
if (shouldBeFatal) {
penaltyDeath()
}
}
.build())
}
@RequiresApi(Build.VERSION_CODES.P)
private inner class ViolationListener : StrictMode.OnThreadViolationListener, StrictMode.OnVmViolationListener {
override fun onThreadViolation(v: Violation) = onViolation(v)
override fun onVmViolation(v: Violation) = onViolation(v)
private fun onViolation(v: Violation) {
Timber.tag(TAG).e(v, "StrictMode violation")
}
}
}
@Module(includes = [AppInitializersModule::class, KotlinInitializerModule::class])
abstract class StrictModeModule {
@Binds
@IntoSet
abstract fun bindStrictModeInitializer(strictModeInitializer: StrictModeInitializer): AppInitializer
}
/**
* Initialize some kotlin functionality eagerly to avoid a DiskReadViolation out of our control
*/
class KotlinInitializer @Inject constructor() : AppInitializer {
override fun initialize(app: Application) {
// load Dispatchers.Main at application start
withStrictMode(allowThreadDiskReads()) {
Dispatchers.Main
val k = Reflection.getOrCreateKotlinClass(Object::class.java)
Timber.d("initialize kotlin Klass with class $k")
}
}
}
@Module
abstract class KotlinInitializerModule {
@Binds
@IntoSet
abstract fun bindCoroutinesAsyncInitializer(oroutinesAsyncInitializer: KotlinInitializer): AppInitializer
}
/**
* Run [block] then restore [originalThreadPolicy]
* Allows to write code like
* ```
* withStrictMode(allowThreadDiskReads()) {
* //read disk data
* }
*
* ```
*/
inline fun withStrictMode(originalThreadPolicy: StrictMode.ThreadPolicy, block: () -> Unit) {
block()
StrictMode.setThreadPolicy(originalThreadPolicy)
}