Фрагмент входа в Google возвращает RESULT_CANCELED
Я пытаюсь интегрировать вход в Google в свое приложение для Android с помощью Firebase, но у меня возникают проблемы. Я следую этому уроку здесь, и я верю, что следовал этому слову. Я добавил Firebase в свое приложение, добавил свой отпечаток SHA-1, включил вход в Google и добавил зависимости в мои файлы оценок. Затем я скопировал код в проекте github, на который есть ссылка в руководстве. Однако, когда я запускаю приложение, и мой код ниже, я получаю сообщение об ошибке, когда возвращается фрагмент входа, и код результата RESULT_CANCELED
, Это ошибка вывода:
W/GoogleSignInActivity: Google sign in failed, resultCode: 0
com.google.android.gms.common.api.ApiException: 10:
at com.google.android.gms.common.internal.zzb.zzy(Unknown Source:14)
at com.google.android.gms.auth.api.signin.GoogleSignIn.getSignedInAccountFromIntent(Unknown Source:37)
at com.example.root.firebasesignin.LoginActivity.onActivityResult(LoginActivity.java:63)
at android.app.Activity.dispatchActivityResult(Activity.java:7267)
at android.app.ActivityThread.deliverResults(ActivityThread.java:4524)
at android.app.ActivityThread.handleSendResult(ActivityThread.java:4571)
at android.app.ActivityThread.-wrap19(Unknown Source:0)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1744)
at android.os.Handler.dispatchMessage(Handler.java:105)
at android.os.Looper.loop(Looper.java:164)
at android.app.ActivityThread.main(ActivityThread.java:6809)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:767)
Когда я создал отпечаток SHA-1, мне пришлось переделать хранилище ключей отладки в ~/.android/debug.keystore
используя команду
keytool -genkey -v -keystore ~/.android/debug.keystore -storepass android -alias androiddebugkey -keypass android -dname "CN=Android Debug,O=Android,C=US"
а затем я преобразовал хранилище ключей в pkcs12 с помощью команды
keytool -importkeystore -srckeystore ~/.android/debug.keystore -destkeystore ~/.android/debug.keystore -deststoretype pkcs12
Я никогда не использовал Firebase или Google для входа в систему, поэтому я очень растерялся. Я думаю, что хранилище ключей может быть проблемой, и когда я смотрю в File > Project Structure > Signings
ничего не отображается на левой панели, а в File > Project Structure > Build Types
поле Signing Config пусто. Опять же, я новичок в аутентификации Firebase и Google, поэтому извините, если я забыл что-то простое. Спасибо заранее за вашу помощь.
Это код для моей основной деятельности. Макет представляет собой просто кнопку входа в Google и панель действий.
package com.example.root.firebasesignin;
import android.content.Intent;
import android.os.Bundle;
import android.support.annotation.NonNull;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.Toast;
import com.google.android.gms.auth.api.signin.GoogleSignIn;
import com.google.android.gms.auth.api.signin.GoogleSignInAccount;
import com.google.android.gms.auth.api.signin.GoogleSignInClient;
import com.google.android.gms.auth.api.signin.GoogleSignInOptions;
import com.google.android.gms.common.api.ApiException;
import com.google.android.gms.tasks.OnCompleteListener;
import com.google.android.gms.tasks.Task;
import com.google.firebase.auth.AuthCredential;
import com.google.firebase.auth.AuthResult;
import com.google.firebase.auth.FirebaseAuth;
import com.google.firebase.auth.FirebaseUser;
import com.google.firebase.auth.GoogleAuthProvider;
public class LoginActivity extends AppCompatActivity implements View.OnClickListener {
private static final String TAG = "GoogleSignInActivity";
private static final int RC_SIGN_IN = 9001;
private FirebaseAuth mAuth;
private GoogleSignInClient mGoogleSignInClient;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_login);
// Button listeners
findViewById(R.id.sign_in_button).setOnClickListener(this);
// Configure Google Sign In
GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
.requestIdToken(getString(R.string.default_web_client_id))
.requestEmail()
.build();
mGoogleSignInClient = GoogleSignIn.getClient(this, gso);
mAuth = FirebaseAuth.getInstance();
}
@Override
public void onStart() {
super.onStart();
// Check if user is signed in (non-null) and update UI accordingly.
FirebaseUser currentUser = mAuth.getCurrentUser();
updateUI(currentUser);
}
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
// Result returned from launching the Intent from GoogleSignInApi.getSignInIntent(...);
if (requestCode == RC_SIGN_IN) {
Task<GoogleSignInAccount> task = GoogleSignIn.getSignedInAccountFromIntent(data);
try {
// Google Sign In was successful, authenticate with Firebase
GoogleSignInAccount account = task.getResult(ApiException.class);
firebaseAuthWithGoogle(account);
} catch (ApiException e) {
// Google Sign In failed, update UI appropriately
Log.w(TAG, "Google sign in failed, resultCode: " + resultCode, e);
updateUI(null);
}
}
}
private void firebaseAuthWithGoogle(GoogleSignInAccount acct) {
Log.d(TAG, "firebaseAuthWithGoogle:" + acct.getId());
AuthCredential credential = GoogleAuthProvider.getCredential(acct.getIdToken(), null);
mAuth.signInWithCredential(credential)
.addOnCompleteListener(this, new OnCompleteListener<AuthResult>() {
@Override
public void onComplete(@NonNull Task<AuthResult> task) {
if (task.isSuccessful()) {
// Sign in success, update UI with the signed-in user's information
Log.d(TAG, "signInWithCredential:success");
FirebaseUser user = mAuth.getCurrentUser();
updateUI(user);
} else {
// If sign in fails, display a message to the user.
Log.w(TAG, "signInWithCredential:failure", task.getException());
Toast.makeText(LoginActivity.this, "Authentiation failed", Toast.LENGTH_LONG).show();
updateUI(null);
}
}
});
}
private void signIn() {
Intent signInIntent = mGoogleSignInClient.getSignInIntent();
startActivityForResult(signInIntent, RC_SIGN_IN);
}
private void updateUI(FirebaseUser user) {
if (user != null)
findViewById(R.id.sign_in_button).setVisibility(View.GONE);
else
findViewById(R.id.sign_in_button).setVisibility(View.VISIBLE);
}
@Override
public void onClick(View v) {
if (v.getId() == R.id.sign_in_button)
signIn();
}
}
5 ответов
Другая причина, по которой может быть возвращено значение, заключается в том, что активность, из которой вы запускаете аутентификацию,
android:launchMode="singleTask"
в этом случае он всегда будет получать
RESULT_CANCELED
на его
onActivityResult()
метод.
Изменение его на
android:launchMode="singleTop"
исправил проблему.
Некоторая ссылка: /questions/11469166/onactivityresult-s-launchmodesingletask/11469180#11469180
com.google.android.gms.common.api.ApiException: 10
объясняется следующей цитатой.
ОШИБКА РАЗРАБОТЧИКА Приложение неправильно настроено. Эта ошибка не может быть исправлена и будет считаться фатальной. Разработчик должен просмотреть журналы после этого, чтобы определить более полезную информацию.
Это означает, что у вас может быть что-то не так с вашим отпечатком пальца SHA-1 или идентификатором клиента OAuth 2.0. Я сомневаюсь, что что-то не так с вашей конфигурацией SHA-1, так как вы следовали этому руководству. Осталось только получить правильный идентификатор клиента. Сначала вы переходите на https://console.cloud.google.com/apis/credentials.
- Ищите все, что связано с Android и находится под идентификаторами клиента OAuth 2.0.
Замените часть ваших кодов следующим текстом:
GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN) .requestIdToken(webClientId) .requestEmail() .build();
Переменная webClientId
содержит идентификатор клиента, который вы ранее получили с https://console.cloud.google.com/apis/credentials. Тогда вот и все.
У меня была аналогичная ситуация с использованием следующего кода:
private static final int PLAY_LOGIN = 1979;
GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN).build();
GoogleSignInClient client = GoogleSignIn.getClient(activity, gso);
activity.startActivityForResult(client.getSignInIntent(), PLAY_LOGIN);
И попытка получить результат входа в метод:
@Override
public boolean onActivityResult(@NonNull Context context, int requestCode, int resultCode, Intent data) {
if (requestCode == PLAY_LOGIN) {
if (resultCode == Activity.RESULT_OK) {
Task<GoogleSignInAccount> task = GoogleSignIn.getSignedInAccountFromIntent(data);
// process successful login data here...
} else {
Toast.makeText(context, "login error!", Toast.LENGTH_LONG).show();
}
return true;
}
return false;
}
Иногда (не всегда воспроизводимый) возвращаемый код будет неActivity.RESULT_OK
, ноActivity.RESULT_CANCELED
- хотя я всегда принимал вариант входа в диалог Google...
Итак, мой код отображал «ошибка входа в систему!» и это даже привело к тому, что мое приложение не прошло процесс проверки Google.
Чтобы исследовать проблему, я поместил точку останова отладчика в строку, гдеToast
отображается и просто продолжал переустанавливать мое приложение и принимать диалоговое окно входа в Google.
Затем иногда срабатывает точка останова, иIntent
имеет следующееmExtras
ценить:
Bundle[{googleSignInStatus=Status{statusCode=неизвестный код состояния: 12502, разрешение=null}}]
И, согласно документу Google в GoogleSignInStatusCodes, код GoogleSignInStatusCodes.SIGN_IN_CURRENTLY_IN_PROGRESS=12502 означает, что уже выполняется другой вход:
Итак, мой ответ — предупреждение для всех, у кого возникла эта проблема, несмотря на то, что в консоли Firebase все настроено правильно.
Возможно, вы захотите обработать этот случай в своем коде и отобразить пользователю другое сообщение, возможно, не ошибку, а просто предупреждение.
Убедитесь, что у вас правильно настроены предварительные условия:-
- Вы добавили ключ отладки SHA-1 в настройки конфигурации вашего проекта в аккаунте Google или Firebase.
- Загрузил и добавил google-services.json в свой проект Android после добавления ключа отладки SHA1 в свою учетную запись.
- Имя пакета (в учетной записи Firebase) и идентификатор приложения (в Android) должны совпадать. Имя пакета Android может отличаться, с этим нет проблем.
После этого просто отредактируйте существующую конфигурацию подписи отладки в настройках вашего проекта Android, как указано здесь /questions/25143632/gde-nahoditsya-debugkeystore-v-android-studio/25143645#25143645
УБЕДИТЕСЬ, ЧТО ВЫ ВЫПОЛНЯЕТЕ ОБЕ ШАГИ, УКАЗАННЫЕ ЗДЕСЬ
Это настроит конфигурацию подписи проекта по умолчанию для отладки с помощью SHA1 хранилища ключей отладки, и все готово!
- Создать новый проект с помощью учетной записи разработчика Firebase. И поместите ваш google-services.json в свое имя проекта / приложение.
- Затем перейдите в консоль разработчика Firebase -> выберите свой проект -> нажмите "Обзор проекта" (слева)-> выберите настройки проекта.
- Перейдите в раздел "Ваше приложение" ---> обновить отпечатки пальцев сертификата SHA Как получить ключ отладчика SHA1?
- Зайдите в андроид студию -> Нажмите Gradle в правой панели.
- Нажмите на название вашего проекта (root). ---> нажмите Задачи -> Отчет о подписи
- Нажмите на консоль Gradle в нижней панели.
- Тогда вы получили свой ключ SHA1.
Далее: 1. Зайдите в консоль разработчика Firebase и выберите свой проект. 2. Нажмите Аутентификация на левой боковой панели. 3. Нажмите МЕТОД ВХОДА и включите Google как true.