24

我正在尝试在我的 Android 应用程序中编写删除用户方法,但每次执行它时都会遇到一些问题。当用户按下Activity 上的Delete account按钮时,将执行此方法。我的应用程序适用于 FirebaseUI 身份验证。

这是方法:

private void deleteAccount() {
    Log.d(TAG, "ingreso a deleteAccount");
    FirebaseAuth firebaseAuth = FirebaseAuth.getInstance();
    final FirebaseUser currentUser = firebaseAuth.getCurrentUser();

    currentUser.delete().addOnCompleteListener(new OnCompleteListener<Void>() {
        @Override
        public void onComplete(@NonNull Task<Void> task) {
            if (task.isSuccessful()) {
                Log.d(TAG,"OK! Works fine!");
                startActivity(new Intent(Main3WelcomeActivity.this, Main3Activity.class));
                finish();
            } 
        }
    }).addOnFailureListener(new OnFailureListener() {
        @Override
        public void onFailure(@NonNull Exception e) {
            Log.e(TAG,"Ocurrio un error durante la eliminación del usuario", e);
        }
    });
}

1)当我执行该功能时,屏幕上会出现一条 Smart Lock 消息,并且用户再次登录。这是此消息的屏幕截图。

智能锁消息

2)其他情况下,当用户长时间登录时,函数会抛出这样的异常:

06-30 00:01:26.672 11152-11152/com.devpicon.android.firebasesamples E/Main3WelcomeActivity: Ocurrio un error durante la eliminación del usuario
com.google.firebase.FirebaseException: An internal error has occured. [ CREDENTIAL_TOO_OLD_LOGIN_AGAIN ]
at com.google.android.gms.internal.zzacq.zzbN(Unknown Source)
at com.google.android.gms.internal.zzacn$zzg.zza(Unknown Source)
at com.google.android.gms.internal.zzacy.zzbO(Unknown Source)
at com.google.android.gms.internal.zzacy$zza.onFailure(Unknown Source)
at com.google.android.gms.internal.zzact$zza.onTransact(Unknown Source)
at android.os.Binder.execTransact(Binder.java:453)

我读过我必须重新验证用户身份,但我不确定在使用 Google 登录时如何执行此操作。

4

9 回答 9

30

根据Firebase 文档,可以使用用户delete()方法从Firebase中删除用户

在删除用户之前,请重新验证用户。

示例代码

     final FirebaseUser user = FirebaseAuth.getInstance().getCurrentUser();

        // Get auth credentials from the user for re-authentication. The example below shows
        // email and password credentials but there are multiple possible providers,
        // such as GoogleAuthProvider or FacebookAuthProvider.
        AuthCredential credential = EmailAuthProvider
                .getCredential("user@example.com", "password1234");

        // Prompt the user to re-provide their sign-in credentials
        user.reauthenticate(credential)
                .addOnCompleteListener(new OnCompleteListener<Void>() {
                    @Override
                    public void onComplete(@NonNull Task<Void> task) {
           user.delete()
            .addOnCompleteListener(new OnCompleteListener<Void>() {
                @Override
                public void onComplete(@NonNull Task<Void> task) {
                    if (task.isSuccessful()) {
                        Log.d(TAG, "User account deleted.");
                    }
                }
            });

   }
});

更多详情:https ://firebase.google.com/docs/auth/android/manage-users#re-authenticate_a_user

如果您想使用其他 singin 提供者重新进行用户身份验证,只需更改Provider以下GoogleAuthProvider示例代码

GoogleAuthProvider.getCredential(googleIdToken,null);
于 2016-06-30T05:47:12.067 回答
5

Ansuita Jr. 提供的答案解释得非常漂亮,而且只有一个小问题是正确的。 即使没有成功重新认证,用户也会被删除。 这是因为我们使用

用户删除()

在始终执行的 onComplete() 方法中。因此,我们需要添加一个if检查来检查任务是否成功,如下所述

user.reauthenticate(credential)
          .addOnCompleteListener(new OnCompleteListener<Void>() {
             @Override
             public void onComplete(@NonNull Task<Void> task) {
                 if (task.isSuccessful()) {
                    Log.e("TAG", "onComplete: authentication complete");
                    user.delete()
                    .addOnCompleteListener (new OnCompleteListener<Void>() {
                           @Override
                           public void onComplete(@NonNull Task<Void> task) {
                                if (task.isSuccessful()) {
                                    Log.e("TAG", "User account deleted.");
                                } else {
                                    Log.e("TAG", "User account deletion unsucessful.");
                                }
                          }
                     });
                 } else {
                     Toast.makeText(UserProfileActivity.this, "Authentication failed", 
                               Toast.LENGTH_SHORT).show();
                 }
              }
         });
于 2018-10-08T11:22:45.137 回答
3

你的delete回调已经处理了失败的情况,为什么还要addOnFailureListener稍后添加?

尝试删除它,这样:

private void deleteAccount() {
    Log.d(TAG, "ingreso a deleteAccount");
    FirebaseAuth firebaseAuth = FirebaseAuth.getInstance();
    final FirebaseUser currentUser = firebaseAuth.getCurrentUser();
    currentUser.delete().addOnCompleteListener(new OnCompleteListener<Void>() {
        @Override
        public void onComplete(@NonNull Task<Void> task) {
            if (task.isSuccessful()) {
                Log.d(TAG,"OK! Works fine!");
                startActivity(new Intent(Main3WelcomeActivity.this, Main3Activity.class));
                finish();
            } else {
                Log.w(TAG,"Something is wrong!");
            }
        }
    });
}
于 2016-06-30T05:47:59.867 回答
3

首先,您需要存储用户登录时的身份验证令牌或密码。如果您的应用程序不提供谷歌登录、Facebook 登录等,您只需存储密码。

//If there's any, delete all stored content from this user on Real Time Database. 
yourDatabaseReferenceNode.removeValue();

//Getting the user instance.
final FirebaseUser user = FirebaseAuth.getInstance().getCurrentUser();

if (user != null) {
    //You need to get here the token you saved at logging-in time.
    String token = "userSavedToken";
    //You need to get here the password you saved at logging-in time.
    String password = "userSavedPassword";

    AuthCredential credential;

    //This means you didn't have the token because user used like Facebook Sign-in method.
    if (token == null) {
       credential = EmailAuthProvider.getCredential(user.getEmail(), password);
    } else {
       //Doesn't matter if it was Facebook Sign-in or others. It will always work using GoogleAuthProvider for whatever the provider.
       credential = GoogleAuthProvider.getCredential(token, null);
    }

    //We have to reauthenticate user because we don't know how long 
    //it was the sign-in. Calling reauthenticate, will update the 
    //user login and prevent FirebaseException (CREDENTIAL_TOO_OLD_LOGIN_AGAIN) on user.delete()
    user.reauthenticate(credential)
            .addOnCompleteListener(new OnCompleteListener<Void>() {
                    @Override
                    public void onComplete(@NonNull Task<Void> task) {
                        //Calling delete to remove the user and wait for a result.
                        user.delete().addOnCompleteListener(new OnCompleteListener<Void>() {
                            @Override
                            public void onComplete(@NonNull Task<Void> task) {
                                if (task.isSuccessful()) {
                                    //Ok, user remove
                                } else {
                                    //Handle the exception
                                    task.getException();
                                }
                            }
                        });
                    }
                });    
}        
于 2017-02-09T14:19:29.397 回答
1

@Android 开发者:

我遇到了一个问题,即卸载应用程序后 Firebase 身份验证信息持续存在于设备的磁盘中。经过实验和阅读,我发现在 Manifest 的标签中设置android:allowBackup="false"和将确保在卸载应用程序后不会保留身份信息。android:fullBackupContent="false"<application>

请注意,这种持久性并非在所有 Android 设备上都发生。实际上,它开始发生在我从未出现过此问题的设备之一上。

于 2019-01-24T11:06:22.590 回答
1

如果登录方法是“匿名”,你可以调用

FirebaseAuth.getInstance().getCurrentUser().delete().addOnCompleteListener(task -> {
                       if (task.isSuccessful()){
                           Log.d(TAG, "Deletion Success");
                       }
                    });

但如果是不同的方法,您将需要重新验证。 如何重新认证

于 2021-07-21T19:45:17.820 回答
0

使用这种方法:-

消除()

相当于调用set(null).

或者

删除用户()

removeUser(credentials, [onComplete])
于 2016-06-30T05:41:37.250 回答
0

仅获取当前用户并使用以下方法将其删除即可正常工作

user.delete();

你也可以通过 addinduser.delete().addOnCompleteListner(new OnCompleteListner)和更多的方式添加 Oncompletelistner

于 2019-04-19T11:01:52.287 回答
0

如果您使用的是AuthUIFirebaseAuth,您可以执行以下操作

 AuthUI.getInstance().delete(context).addOnSuccessListener {
           
 }.addOnFailureListener{

 }

或者

FirebaseAuth.getInstance().currentUser.delete().addOnSuccessListener...
于 2020-06-24T15:31:35.433 回答