android - 在 android 中禁用 google ML Kit 库的 firebase 日志记录

标签 android firebase firebase-analytics google-mlkit

谁能帮我告诉我如何禁用 Google ML Kit 中的 firebase 登录安卓库。每 15 分钟,它会将一些信息发布到 https://firebaselogging.googleapis.com/v0cc/log/batch?format=json_proto3
我尝试使用谷歌的推荐 https://firebase.google.com/docs/perf-mon/disable-sdk?platform=android#kotlin+ktx而且我也不确定这是否是正确的方法。
欢迎提出建议。

最佳答案

Google's guide从这个问题对我不起作用,所以我一直在寻找替代方案。
该库是混淆的,因此很难确定,但似乎日志记录是硬编码的。但是,有一种非常hacky的方法可以通过一些脆弱的反射来禁用它:

import android.util.Log
import com.google.mlkit.common.sdkinternal.LazyInstanceMap
import java.lang.reflect.Field

/**
 * This class tries to disable MLKit's phoning home/logging.
 * This is extremely hacky and will probably break in the next update (obfuscated class names will probably need renaming).
 *
 * This class exploits the fact, that there are multiple options classes which control this
 * (look for "MLKitLoggingOptions" in toString implementation) and for some reason MLKit uses them as keys
 * in LazyInstanceMaps which exist as static (usually) variables (which are themselves lazy).
 *
 * This makes sure that the LazyInstanceMaps exist, then it hijacks their internal HashMap implementation
 * and replaces it with a custom map, that creates instances of whatever with logging disabled.
 *
 * The way to detect which holder classes need renaming, look at the stack trace, for example:
 * ```
    java.lang.NoClassDefFoundError: Failed resolution of: Lcom/google/android/datatransport/cct/CCTDestination;
    at com.google.android.gms.internal.mlkit_vision_barcode.zznu.<init>(com.google.android.gms:play-services-mlkit-barcode-scanning@@18.0.0:1)
    at com.google.android.gms.internal.mlkit_vision_barcode.zznf.<init>(com.google.android.gms:play-services-mlkit-barcode-scanning@@18.0.0:3)
    at com.google.android.gms.internal.mlkit_vision_barcode.zznw.create(com.google.android.gms:play-services-mlkit-barcode-scanning@@18.0.0:4)
    at com.google.mlkit.common.sdkinternal.LazyInstanceMap.get(com.google.mlkit:common@@18.0.0:3)
    at com.google.android.gms.internal.mlkit_vision_barcode.zznx.zza(com.google.android.gms:play-services-mlkit-barcode-scanning@@18.0.0:2)
    at com.google.android.gms.internal.mlkit_vision_barcode.zznx.zzb(com.google.android.gms:play-services-mlkit-barcode-scanning@@18.0.0:3)
    at com.google.mlkit.vision.barcode.internal.zzf.create(com.google.android.gms:play-services-mlkit-barcode-scanning@@18.0.0:3)
    at com.google.mlkit.common.sdkinternal.LazyInstanceMap.get(com.google.mlkit:common@@18.0.0:3)
    at com.google.mlkit.vision.barcode.internal.zze.zzb(com.google.android.gms:play-services-mlkit-barcode-scanning@@18.0.0:2)
    at com.google.mlkit.vision.barcode.BarcodeScanning.getClient(com.google.android.gms:play-services-mlkit-barcode-scanning@@18.0.0:3)
 * ```
 * here are two LazyInstanceMap lookups, of which only the second one (through trial and error or with debugger)
 * uses MLKitLoggingOptions keys. From here we can find that the holder class is com.google.android.gms.internal.mlkit_vision_barcode.zznx .
 */
object MLKitTrickery {

    private class mlkit_vision_barcodeLoggingOptions(base: com.google.android.gms.internal.mlkit_vision_barcode.zzne) : com.google.android.gms.internal.mlkit_vision_barcode.zzne() {
        private val libraryName: String = base.zzb()
        private val firelogEventType: Int = base.zza()
        override fun zza(): Int = firelogEventType
        override fun zzb(): String = libraryName
        override fun zzc(): Boolean = false //enableFirelog

        override fun equals(other: Any?): Boolean {
            if (this === other) return true
            if (javaClass != other?.javaClass) return false

            other as mlkit_vision_barcodeLoggingOptions
            if (libraryName != other.libraryName) return false
            if (firelogEventType != other.firelogEventType) return false
            return true
        }

        override fun hashCode(): Int {
            var result = libraryName.hashCode()
            result = 31 * result + firelogEventType
            return result
        }
    }

    private class mlkit_vision_commonLoggingOptions(base: com.google.android.gms.internal.mlkit_vision_common.zzjn) : com.google.android.gms.internal.mlkit_vision_common.zzjn() {
        private val libraryName: String = base.zzb()
        private val firelogEventType: Int = base.zza()
        override fun zza(): Int = firelogEventType
        override fun zzb(): String = libraryName
        override fun zzc(): Boolean = false //enableFirelog

        override fun equals(other: Any?): Boolean {
            if (this === other) return true
            if (javaClass != other?.javaClass) return false

            other as mlkit_vision_commonLoggingOptions
            if (libraryName != other.libraryName) return false
            if (firelogEventType != other.firelogEventType) return false
            return true
        }

        override fun hashCode(): Int {
            var result = libraryName.hashCode()
            result = 31 * result + firelogEventType
            return result
        }
    }

    private fun isMLKitLoggingOptions(obj: Any): Boolean {
        return obj is com.google.android.gms.internal.mlkit_vision_barcode.zzne
                || obj is com.google.android.gms.internal.mlkit_vision_common.zzjn
    }

    private fun convertMLKitLoggingOptions(obj: Any): Any? {
        if (obj is com.google.android.gms.internal.mlkit_vision_barcode.zzne) {
            return mlkit_vision_barcodeLoggingOptions(obj)
        }
        if (obj is com.google.android.gms.internal.mlkit_vision_common.zzjn) {
            return mlkit_vision_commonLoggingOptions(obj)
        }
        return null
    }

    @Suppress("UNCHECKED_CAST")
    private fun patchLazyMap(lazyMapHolder:Any?, lazyMapHolderClass: Class<*>) {
        val holderField = lazyMapHolderClass.declaredFields.find { LazyInstanceMap::class.java.isAssignableFrom(it.type) }!!
        var currentLazyInstanceMap = holderField.get(lazyMapHolder)
        if (currentLazyInstanceMap == null) {
            var lastError: Throwable? = null
            for (constructor in holderField.type.declaredConstructors) {
                try {
                    constructor.isAccessible = true
                    val params = arrayOfNulls<Any?>(constructor.parameterCount)
                    currentLazyInstanceMap = constructor.newInstance(*params)
                    holderField.set(lazyMapHolder, currentLazyInstanceMap)
                } catch (e:Throwable) {
                    lastError = e
                }
            }
            if (currentLazyInstanceMap == null) {
                throw java.lang.Exception("Failed to initialize LazyInstanceMap "+holderField.type, lastError)
            }
        }

        var mapHolderClass: Class<*> = currentLazyInstanceMap.javaClass
        val createMethod = mapHolderClass.getDeclaredMethod("create", Object::class.java)

        val mapField: Field
        while (true) {
            val mapFieldCandidate = mapHolderClass.declaredFields.firstOrNull { Map::class.java.isAssignableFrom(it.type) }
            if (mapFieldCandidate != null) {
                mapField = mapFieldCandidate
                break
            }
            mapHolderClass = mapHolderClass.superclass ?: error("It appears that ${currentLazyInstanceMap.javaClass} does not have a backing map field")
        }

        val oldMap = mapField.get(currentLazyInstanceMap) as MutableMap<Any, Any?>
        val customMap = object : MutableMap<Any, Any?> by oldMap {

            override fun containsKey(key: Any): Boolean {
                if (oldMap.containsKey(key)) {
                    return true
                }
                if (isMLKitLoggingOptions(key)) {
                    return true
                }
                return false
            }

            override fun get(key: Any): Any? {
                val existing = oldMap.get(key)
                if (existing != null) {
                    return existing
                }

                val convertedKey = convertMLKitLoggingOptions(key)
                if (convertedKey != null) {
                    val created = createMethod.invoke(currentLazyInstanceMap, convertedKey)
                    oldMap.put(key, created)
                    return created
                }

                return null
            }
        }
        mapField.isAccessible = true
        mapField.set(currentLazyInstanceMap, customMap)
    }

    private var initialized = false

    /**
     * Call this to attempt to disable MLKit logging.
     */
    fun init() {
        try {
            patchLazyMap(null, com.google.android.gms.internal.mlkit_vision_barcode.zznx::class.java)
            patchLazyMap(null, com.google.android.gms.internal.mlkit_vision_common.zzkc::class.java)
            initialized = true
        } catch (e: Throwable) {
            Log.e("MLKitTrickery", "Failed to disable MLKit phoning home")
        }
    }
}
当您还使用以下命令填充 GMS TelemetryLogging 时:
@file:Suppress("unused", "UNUSED_PARAMETER")

package com.google.android.gms.common.internal

import android.app.Activity
import android.content.Context
import android.os.Parcel
import com.google.android.gms.tasks.OnFailureListener
import com.google.android.gms.tasks.OnSuccessListener
import com.google.android.gms.tasks.Task
import java.util.concurrent.Executor

class TelemetryLoggingOptions {
    class Builder {
        fun setApi(api: String?): Builder = this
        fun build(): TelemetryLoggingOptions = TelemetryLoggingOptions()
    }

    companion object {
        @JvmStatic
        fun builder(): Builder = Builder()
    }
}

private object DummyLogTask : Task<Void?>() {
    override fun addOnFailureListener(p0: OnFailureListener): Task<Void?> {
        // Implemented, because failing tells MLKit to back-off for 30 minutes, which is a win for performance
        p0.onFailure(exception)
        return this
    }
    override fun addOnFailureListener(p0: Activity, p1: OnFailureListener): Task<Void?> = addOnFailureListener(p1)
    override fun addOnFailureListener(p0: Executor, p1: OnFailureListener): Task<Void?> = addOnFailureListener(p1)

    override fun addOnSuccessListener(p0: OnSuccessListener<in Void?>): Task<Void?> = this
    override fun addOnSuccessListener(p0: Activity, p1: OnSuccessListener<in Void?>): Task<Void?> = addOnSuccessListener(p1)
    override fun addOnSuccessListener(p0: Executor, p1: OnSuccessListener<in Void?>): Task<Void?> = addOnSuccessListener(p1)

    override fun getException(): Exception? = exception
    override fun getResult(): Void? = null
    override fun <X : Throwable?> getResult(p0: Class<X>): Void? = null

    override fun isCanceled(): Boolean = false
    override fun isComplete(): Boolean = true
    override fun isSuccessful(): Boolean = false

    private val exception = Exception("Success was never an option")
}

object TelemetryLogging {
    @JvmStatic
    fun getClient(context: Context): TelemetryLoggingClient {
        return object : TelemetryLoggingClient {
            override fun log(data: TelemetryData): Task<Void?> {
                return DummyLogTask
            }
        }
    }

    @JvmStatic
    fun getClient(context: Context, options: TelemetryLoggingOptions): TelemetryLoggingClient {
        return getClient(context)
    }
}

interface TelemetryLoggingClient {
    fun log(data: TelemetryData): Task<Void?>
}

class TelemetryData(var1: Int, var2:List<MethodInvocation>?) {
    fun writeToParcel(var1: Parcel, var2: Int) {}
}

class MethodInvocation {

    constructor(methodKey:Int, resultStatusCode:Int, connectionResultStatusCode:Int,
                startTimeMillis:Long, endTimeMillis:Long,
                callingModuleId: String?, callingEntryPoint: String?, serviceId:Int)

    constructor(methodKey:Int, resultStatusCode:Int, connectionResultStatusCode:Int,
                startTimeMillis:Long, endTimeMillis:Long,
                callingModuleId: String?, callingEntryPoint: String?,
                serviceId:Int, var11:Int)

    fun writeToParcel(var1: Parcel, var2: Int) {}
}
可以修剪许多传递依赖项并节省 apk 大小:
implementation("com.google.mlkit:barcode-scanning:17.0.2") {
   exclude("com.google.android.gms", "play-services-base")
   exclude("com.google.android.datatransport", "transport-api")
   exclude("com.google.android.datatransport", "transport-backend-cct")
   exclude("com.google.android.datatransport", "transport-runtime")
   exclude("com.google.firebase", "firebase-encoders-json")
   exclude("com.google.firebase", "firebase-encoders")
}
但是,如上所述,这是非常脆弱的,并且可能会在 MLKit 更新后以某种方式中断。如果不需要,那就太好了。

关于android - 在 android 中禁用 google ML Kit 库的 firebase 日志记录,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/71465067/

相关文章:

android - SQLiteOpenHelper getInstance() 方法抛出空指针异常

android - 创建数据库 firebase

google-analytics - Android 实时分析

swift - Firebase Analytics - 记录 screen_view 在 SwiftUI 中不起作用

ios - Firebase REST API - 如何观察和监听变化

android - 监控用户在应用程序不同部分花费的时间

android - 启动 ACTION_SEND Intent 时如何附加位图

java - 未收到来自服务器 : Java.net.SocketException 的响应:recvfrom 失败:ECONNRESET

android - 使用按钮滚动图库

android - 添加/修改子项时的 Firebase 通知