/*
* Geekttrss is a RSS feed reader application on the Android Platform.
*
* Copyright (C) 2017-2020 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.hilt.InstallIn
import dagger.hilt.components.SingletonComponent
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() : AppInitializer {
private val listenerExecutor by lazy { Executors.newSingleThreadExecutor() }
@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()
.apply {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
detectContentUriWithoutPermission()
// because crashlytics don't tag its socket
if (BuildConfig.FLAVOR != "google") {
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)
} else {
penaltyLog()
}
}
.build())
StrictMode.setThreadPolicy(StrictMode.ThreadPolicy.Builder()
.detectAll()
.apply {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
penaltyListener(listenerExecutor, violationListener)
} else {
penaltyLog()
}
}
.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])
@InstallIn(SingletonComponent::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
@InstallIn(SingletonComponent::class)
abstract class KotlinInitializerModule {
@Binds
@IntoSet
abstract fun bindKotlinInitializer(kotlinInitializer: KotlinInitializer): AppInitializer
}
/**
* Run [block] then restore [originalThreadPolicy]
* Allows to write code like
* ```
* withStrictMode(allowThreadDiskReads()) {
* //read disk data
* }
*
* ```
*/
inline fun <R> withStrictMode(originalThreadPolicy: StrictMode.ThreadPolicy, block: () -> R): R {
try {
return block()
} finally {
StrictMode.setThreadPolicy(originalThreadPolicy)
}
}