33

チュートリアルに従おうとしました:https ://developers.google.com/android/guides/http-auth 。

コード:

token = GoogleAuthUtil.getToken(getApplicationContext(),
                        mEmail, mScope);

マニフェスト:

<uses-permission android:name="android.permission.GET_ACCOUNTS"/>
<uses-permission android:name="android.permission.NETWORK"/>
<uses-permission android:name="android.permission.USE_CREDENTIALS"/>
<uses-permission android:name="android.permission.INTERNET"/>

エラー:

01-17 18:37:38.230: W/System.err(3689): com.google.android.gms.auth.UserRecoverableAuthException: NeedPermission
01-17 18:37:38.230: W/System.err(3689):     at com.google.android.gms.auth.GoogleAuthUtil.getToken(Unknown Source)
01-17 18:37:38.230: W/System.err(3689):     at com.google.android.gms.auth.GoogleAuthUtil.getToken(Unknown Source)
01-17 18:37:38.230: W/System.err(3689):     at com.example.mgoogleauth.MainActivity$GetIOStreamTask.doInBackground(MainActivity.java:39)
01-17 18:37:38.230: W/System.err(3689):     at com.example.mgoogleauth.MainActivity$GetIOStreamTask.doInBackground(MainActivity.java:1)
01-17 18:37:38.230: W/System.err(3689):     at android.os.AsyncTask$2.call(AsyncTask.java:287)
01-17 18:37:38.230: W/System.err(3689):     at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
01-17 18:37:38.230: W/System.err(3689):     at java.util.concurrent.FutureTask.run(FutureTask.java:137)
01-17 18:37:38.230: W/System.err(3689):     at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1076)
01-17 18:37:38.230: W/System.err(3689):     at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:569)
01-17 18:37:38.230: W/System.err(3689):     at java.lang.Thread.run(Thread.java:856)
4

6 に答える 6

75

Android用のドライブクイックスタートに従ってみてください。これは、ファイルを認証してドライブにアップロードする方法を示すステップバイステップガイドです:https ://developers.google.com/drive/quickstart-android

具体的には、UserRecoverableExceptionをキャッチしておらず、ユーザーにアプリを承認させるインテントをトリガーしていないようです。これは、クイックスタートサンプルでリンクして処理したGooglePlayサービスのドキュメントに次のように記載されています。

...
} catch (UserRecoverableAuthIOException e) {
  startActivityForResult(e.getIntent(), REQUEST_AUTHORIZATION);
}
... 
于 2013-01-17T19:05:14.290 に答える
12

公式のGoogleAuthUtilチュートリアルのメソッドgetAndUseAuthTokenBlocking()は、例外を処理する方法をかなりよく説明しています。

// Example of how to use the GoogleAuthUtil in a blocking, non-main thread context
   void getAndUseAuthTokenBlocking() {
       try {
          // Retrieve a token for the given account and scope. It will always return either
          // a non-empty String or throw an exception.
          final String token = GoogleAuthUtil.getToken(Context, String, String)(context, email, scope);
          // Do work with token.
          ...
          if (server indicates token is invalid) {
              // invalidate the token that we found is bad so that GoogleAuthUtil won't
              // return it next time (it may have cached it)
              GoogleAuthUtil.invalidateToken(Context, String)(context, token);
              // consider retrying getAndUseTokenBlocking() once more
              return;
          }
          return;
       } catch (GooglePlayServicesAvailabilityException playEx) {
         Dialog alert = GooglePlayServicesUtil.getErrorDialog(
             playEx.getConnectionStatusCode(),
             this,
             MY_ACTIVITYS_AUTH_REQUEST_CODE);
         ...
       } catch (UserRecoverableAuthException userAuthEx) {
          // Start the user recoverable action using the intent returned by
          // getIntent()
          myActivity.startActivityForResult(
                  userAuthEx.getIntent(),
                  MY_ACTIVITYS_AUTH_REQUEST_CODE);
          return;
       } catch (IOException transientEx) {
          // network or server error, the call is expected to succeed if you try again later.
          // Don't attempt to call again immediately - the request is likely to
          // fail, you'll hit quotas or back-off.
          ...
          return;
       } catch (GoogleAuthException authEx) {
          // Failure. The call is not expected to ever succeed so it should not be
          // retried.
          ...
          return;
       }
   }
于 2013-04-29T19:41:25.813 に答える
7

同じエラーが発生しました。私の場合、間違ったスコープを使用していました。変更するだけです。

https://www.googleapis.com/auth/plus.login

にとって

https://www.googleapis.com/auth/userinfo.profile
于 2014-11-10T20:27:36.257 に答える
3

このドキュメントページ https://developers.google.com/+/mobile/android/sign-inの例には、この例外の適切な説明があります。

特に、次の行に注意する必要があるようです。

認証コードをリクエストすると、GoogleAuthUtil.getTokenへの最初の呼び出しで常にUserRecoverableAuthExceptionがスローされます

catch (UserRecoverableAuthException e) {
  // Requesting an authorization code will always throw
  // UserRecoverableAuthException on the first call to GoogleAuthUtil.getToken
  // because the user must consent to offline access to their data.  After
  // consent is granted control is returned to your activity in onActivityResult
  // and the second call to GoogleAuthUtil.getToken will succeed.
  startActivityForResult(e.getIntent(), AUTH_CODE_REQUEST_CODE);
  return;
}
于 2015-04-22T18:13:46.790 に答える
2

ここでの答えは受動的な解決策であり、予防的なものではないことがわかりました。
私の非常に短い経験によると、UserRecoverableAuthException:NeedPermissionは次の3つの場合にスローされます。

#1Googleにサインインするときに適切なスコープは要求されません

要求しているスコープが正しいことを確認してください。Androidは、認証プロセス後に許可要求ダイアログを介して許可を許可するようにユーザーに要求する必要があります。これにより、APIを呼び出すときにUserRecoverableAuthExceptionが防止されます。

GoogleSignInOptions o = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
            .requestScopes(new Scope(DriveScopes.DRIVE)) // request permission for Drive API
            .requestScopes(new Scope(SheetsScopes.SPREADSHEETS)) // request permission for Sheets API
            .requestEmail()
            .build();

#2ユーザーが許可を拒否しました

ユーザーが許可要求ダイアログの「拒否」ボタンを押しました。

#3ユーザーがアプリからの権限を拒否しました

Android 8.1.xには、個々のアプリの権限を拒否できるメニューがあります。ただし、他のバージョンについてはわかりません。

Settings > Google > Connected apps

#2と#3によってスローされるUserRecoverableAuthExceptionは、ユーザーアクティビティの結果であるため、避けられません。しかし、ユーザーが拒否したにもかかわらず、以下のコードで許可要求ダイアログを再度表示することは無意味ではありませんか?

} catch (UserRecoverableAuthIOException e) {
  startActivityForResult(e.getIntent(), REQUEST_AUTHORIZATION);
}
于 2019-01-15T08:20:48.080 に答える
1

ドキュメントは最近更新され、SDK M(権限のリクエスト)をサポートし、OAuthダイアログも表示するようになりました。

Googleドキュメントは最新ではないことがよくありますが、問題を報告するときに注意を払っているようです。この例は、私が1週間フィードバックを送信して更新されました。したがって、機能しない例を見つけた場合は、フィードバックを送信してください。

https://developers.google.com/drive/v3/web/quickstart/android

于 2016-04-06T19:22:26.133 に答える