将Firebase Task <void>包装到RxJava CompletableEmitter中?

时间:2019-05-22 01:04:33

标签: android firebase google-cloud-firestore rx-java rx-java2

我正在尝试将 Firestore方法包装到CompletableEmitter中,以便能够在成功/完成时返回Completable.complete(),而在失败时返回Completable.error()。 (任务是Firestore操作,用于更新集合中的文档(该文档中的1个字段))

This是我要包装成Completablefirestore method,它返回一个Task<Void>。这是没有包装在Completable中的代码:

public static Completable updateDocument(DocumentReference ref, Map<String,Object> fieldsAndValues){
    ref.update(fieldsAndValues)
                .addOnSuccessListener(aVoid -> {
                    Timber.d("SUCCESS");
                }).addOnCompleteListener(task -> {
                    Timber.d("COMPLETE");
                }).addOnFailureListener(e -> {
                    Timber.d("ERROR");
                });
        return Completable.complete();    //just for the sake of returning complete to finish the method...
}

上面的代码按原样执行。

这是我包装该代码的尝试:

public static Completable updateDocument(DocumentReference ref, Map<String,Object> fieldsAndValues){
    return Completable.create(emitter -> ref.update(fieldsAndValues)
       .addOnSuccessListener(aVoid -> emitter.onComplete())
       .addOnCompleteListener(task -> emitter.onComplete())
       .addOnFailureListener(emitter::onError));
}

尽管第一个成功执行(调用onSuccess,Firestore中的文档已更新),但包装的代码永远不会从Task到达onSuccess(onSuccess尚未实现,Firestore中的文档从未更新)。但这不会引发任何错误,任务已完成但从未成功。

我不了解这种行为。有什么建议么?我在这里做什么错了?

2 个答案:

答案 0 :(得分:2)

您不必一定要对任务使用回调-我认为api在这种意义上是限制性的。

您可以使任务阻塞并在执行后对其进行评估,即

public static Completable updateDocument(DocumentReference ref, Map<String,Object> fieldsAndValues){
        return Completable.fromCallable(() -> {
            final Task<Void> task = ref.update(fieldsAndValues);
            Tasks.await(task);
            if(task.isSuccessful()) {
                return Completable.complete();
            } else {
                return Completable.error(new IllegalStateException("Task not successful", task.getException()));
            }
        });
    }

作为您的原始问题,请尝试在闭包中保留对任务的引用,即

public static Completable updateDocument(DocumentReference ref, Map<String, Object> fieldsAndValues) {
        return Completable.create(emitter -> {
            final Task<Void> task = ref.update(fieldsAndValues)
            task.addOnSuccessListener(aVoid -> emitter.onComplete())
                    .addOnCompleteListener(task -> emitter.onComplete())
                    .addOnFailureListener(emitter::onError);
        });
    }

答案 1 :(得分:1)

subscribe d是否已完成? 除非您订阅,否则ref.update()将永远不会执行

updateDocument(...)
    .subscribe(...) // <- this

注意:您的第一个代码也有问题。 Completable天时应执行subscribe个。但是,在您的第一个代码中,当您调用updateDocument()时将执行该代码。 例如,如果您执行Completable c = updateDocument(...),则不应更新文档,因为您尚未subscribe(),但是您的第一个代码将更新文档。