Firebase ランループ (3.0.0) でキャッチされない例外 質問する

Firebase ランループ (3.0.0) でキャッチされない例外 質問する

最新のFirebase(9.0.2)を使用しています:build.gradle:

dependencies {
     ...
     compile "com.google.firebase:firebase-database:9.0.2"
     compile 'com.google.firebase:firebase-auth:9.0.2'
}
apply plugin: 'com.google.gms.google-services'

プロジェクトbuild.gradle

classpath 'com.google.gms:google-services:3.0.0'

しばらくすると、アプリケーションは次の例外でクラッシュし始めます。

  Fatal Exception: java.lang.RuntimeException: Uncaught exception in Firebase runloop (3.0.0). Please report to [email protected]
       at com.google.android.gms.internal.zzadp$1$1.run(Unknown Source)
       at android.os.Handler.handleCallback(Handler.java:739)
       at android.os.Handler.dispatchMessage(Handler.java:95)
       at android.os.Looper.loop(Looper.java:135)
       at android.app.ActivityThread.main(ActivityThread.java:5274)
       at java.lang.reflect.Method.invoke(Method.java)
       at java.lang.reflect.Method.invoke(Method.java:372)
       at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:909)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:704)
Caused by java.lang.AssertionError: hardAssert failed: 
       at com.google.android.gms.internal.zzaiv.zzb(Unknown Source)
       at com.google.android.gms.internal.zzaiv.zzaN(Unknown Source)
       at com.google.android.gms.internal.zzagh.zzb(Unknown Source)
       at com.google.android.gms.internal.zzagh.<init>(Unknown Source)
       at com.google.android.gms.internal.zzaga.<init>(Unknown Source)
       at com.google.android.gms.internal.zzaga.<init>(Unknown Source)
       at com.google.android.gms.internal.zzadp.zza(Unknown Source)
       at com.google.android.gms.internal.zzaeu.zzic(Unknown Source)
       at com.google.android.gms.internal.zzafc.zzRy(Unknown Source)
       at com.google.android.gms.internal.zzafc.zza(Unknown Source)
       at com.google.android.gms.internal.zzafc$1.run(Unknown Source)
       at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:422)
       at java.util.concurrent.FutureTask.run(FutureTask.java:237)
       at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.access$201(ScheduledThreadPoolExecutor.java:152)
       at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.run(ScheduledThreadPoolExecutor.java:265)
       at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
       at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
       at java.lang.Thread.run(Thread.java:818)

アプリケーションの onCreate には次のものがあります:

@Override
    public void onCreate() {
    ...
    FirebaseDatabase.getInstance().setPersistenceEnabled(true);
}

また、アクティビティ (同じプロセス内のすべてのアクティビティ)/フラグメントから呼び出される Firebase 用のシングルトン ヘルパー クラスも作成しました。

 private FirebaseHelper() {
        mFirebaseRef = FirebaseDatabase.getInstance().getReference();
        mFirebaseAuth = FirebaseAuth.getInstance();
        mFirebaseAuth.addAuthStateListener(this);
        authentication();
    }

    public static synchronized FirebaseHelper getInstance() {
        if (mInstance == null || mInstance.getFirebaseRef() == null) {
            mInstance = new FirebaseHelper();
        }
        return mInstance;
    }

ライブラリ:

dependencies {
    testCompile 'junit:junit:4.12'
    compile('com.crashlytics.sdk.android:crashlytics:2.5.6@aar') {
        transitive = true;
    }
    compile 'com.google.code.gson:gson:2.6.2'
    compile 'com.android.support:support-v4:23.4.0'
    compile 'com.android.support:support-v13:23.4.0'
    compile 'com.android.support:appcompat-v7:23.4.0'
    compile 'com.android.support:design:23.4.0'
    compile 'com.android.support:support-annotations:23.4.0'
    compile 'com.android.support:gridlayout-v7:23.4.0'
    compile 'com.google.android.gms:play-services-base:9.0.2'
    compile 'com.google.android.gms:play-services-maps:9.0.2'
    compile 'com.google.android.gms:play-services-location:9.0.2'
    compile 'com.google.android.gms:play-services-appindexing:9.0.2'
    compile 'com.google.android.gms:play-services-analytics:9.0.2'
    compile 'com.google.firebase:firebase-messaging:9.0.2'
    compile 'com.facebook.android:facebook-android-sdk:4.11.0'
    compile 'de.greenrobot:eventbus:2.4.0'
    compile 'com.amazonaws:aws-android-sdk-core:2.2.12'
    compile 'com.amazonaws:aws-android-sdk-cognito:2.2.12'
    compile 'com.amazonaws:aws-android-sdk-s3:2.2.12'
    compile 'com.android.support:multidex:1.0.1'
    compile 'com.squareup.retrofit2:retrofit:2.0.2'
    compile 'com.squareup.retrofit2:converter-gson:2.0.2'
    compile 'com.squareup.retrofit2:adapter-rxjava:2.0.2'
    compile 'io.reactivex:rxandroid:1.2.0'
    compile 'io.reactivex:rxjava:1.1.5'
    compile 'com.squareup.okhttp3:logging-interceptor:3.3.1'
    compile 'com.github.curioustechizen.android-ago:library:1.3.0'
    compile 'com.cedarsoftware:json-io:4.4.0'
    compile 'com.timehop.stickyheadersrecyclerview:library:0.4.3@aar'
    compile 'joda-time:joda-time:2.9.3'
    compile 'com.facebook.fresco:fresco:0.10.0'
    compile 'com.facebook.fresco:imagepipeline-okhttp3:0.10.0'
    compile 'com.google.firebase:firebase-core:9.0.2'
    compile 'com.google.firebase:firebase-invites:9.0.2'
    compile 'com.google.firebase:firebase-database:9.0.1'
    compile 'com.google.firebase:firebase-auth:9.0.1'
    compile 'com.github.jd-alexander:LikeButton:0.2.0'

    debugCompile 'com.squareup.leakcanary:leakcanary-android:1.4-beta2'
    releaseCompile 'com.squareup.leakcanary:leakcanary-android-no-op:1.4-beta2'
    androidTestCompile 'junit:junit:4.12'
    androidTestCompile 'com.android.support:support-annotations:23.4.0'
    androidTestCompile 'com.android.support.test:runner:0.5'
    androidTestCompile 'com.android.support.test:rules:0.5'
    compile files('libs/core-3.2.1.jar')
}

ベストアンサー1

アップデート:Kristijan がコメントで述べたように、この症状には別の原因があることが判明しました。startAt() または endAt() 呼び出しに整数エンドポイント (例: startAt(10)) を指定すると、この種のキャッシュ破損が引き起こされる可能性があるというバグを特定しました。このバグは SDK の次のリリースで修正される予定です。それまでの間、回避策として非整数エンドポイント (例: startAt(10.001)) を使用できる場合があります。

これらの症状は、Android アプリの複数のプロセスで永続性が有効になっている場合に Firebase Realtime Database が機能しなくなるという既知の制限と一致しています。

マルチプロセス Android アプリでは、すべてのApplication.onCreate()コードがすべてのプロセスで実行されることに注意してください。そのため、アプリがマルチプロセスの場合は、複数のプロセスで永続性を有効にして Firebase データベースを初期化することになり、オフライン キャッシュが破損し、報告されている hardAssert エラーが発生する可能性があります。

気付かないうちにアプリがマルチプロセスになっている場合があることに注意してください。たとえば、firebase-crash を使用している場合、クラッシュをより確実に報告するためにバックグラウンド プロセスが作成されます。そのため、firebase-crash を使用すると、マルチプロセス アプリになります。他のサードパーティ ライブラリでも同様の動作が考えられます。

テストするには、Application.onCreate() に次のようなコードを追加します。

System.out.println('INITIALIZING APP FROM PID: ' + android.os.Process.myPid());

logcat に 2 回ログが記録されている場合 (2 つの異なる PID を使用)、アプリが複数のプロセスで実行されており、前述した制限に達していることを意味します。

回避策として、次のいずれかを実行できます。

  1. 単一のプロセスのみを使用するようにアプリを変更します。
  2. setPersistenceEnabled()アプリケーション クラスからコードを削除し、メイン プロセスでのみ実行される場所に配置します。

注意: hardAssert エラーが発生した場合、このエラーを解消するにはアプリ データを消去する必要がある可能性があります。このエラーはオフライン キャッシュが無効な状態になったことを示しているため、これを修正するには完全に消去する必要があります。

今後のリリースでは、このシナリオの検出機能が向上し、より適切なエラー メッセージが表示されるようになります。さらに、firebase-crash は将来的に 2 番目のプロセスの生成を回避するため、この問題が発生する可能性は低くなる可能性があります。

おすすめ記事