AccountManager.blockingGetAuthToken卡住了

时间:2017-12-03 22:08:32

标签: android retrofit2 accountmanager

我创建了一个AbstractThreadedSyncAdapter,它通过Retrofit2与服务器同步数据。为了处理身份验证,我创建了一个拦截器来添加令牌。

当拦截器需要通过调用accountManager.blockingGetAuthToken(...)来刷新令牌时,它会一直停留,直到android.accounts.OperationCanceledException被抛出。当我在AbstractAccountAuthenticator - >中设置断点时getAuthToken在拦截器完成之前它不会被击中。

public class TokenInterceptor implements Interceptor {

  @Override
  public Response intercept(Chain chain) throws IOException {
    String oldToken = accountManager.peekAuthToken(accounts[0], LoginActivity.AUTH_TOKEN_TYPE);

    if(newTokenRequired){
       accountManager.invalidateAuthToken(LoginActivity.ACCOUNT_TYPE, oldToken);
    }

    String token = accountManager.blockingGetAuthToken(accounts[0], LoginActivity.AUTH_TOKEN_TYPE, true); // <-- stucks here

    // add token to request and return response
  }

}

非常感谢先进的

更新

这是getAuthToken中的AccountAuthenticator方法:

@Override
public Bundle getAuthToken(AccountAuthenticatorResponse response, Account account, String authTokenType, Bundle options) {
   final AccountManager accountManager = AccountManager.get(context);
   final String refreshToken = accountManager.getPassword(account);
   final Bundle bundle = new Bundle();

   if (refreshToken != null) {
       try {
           Response<RefreshTokenResponse> refreshTokenResponse = apiService.refreshToken(refreshToken).execute();

           if (refreshTokenResponse.isSuccessful()) {
               bundle.putString(AccountManager.KEY_ACCOUNT_NAME, account.name);
               bundle.putString(AccountManager.KEY_ACCOUNT_TYPE, account.type);
               bundle.putString(AccountManager.KEY_AUTHTOKEN, refreshTokenResponse.body().getAccess_Token());
               accountManager.setPassword(account, refreshTokenResponse.body().getRefresh_token());
               return bundle;
           }
       } catch (Exception e) {
           Log.e(AccountAuthenticator.class.getName(), "getAuthToken: failed to get access token", e);
       }
   }

   // start login intent
   final Intent intent = new Intent(context, LoginActivity.class);
   intent.putExtra(LoginActivity.PARAM_USER_PASS, account.name);
   intent.putExtra(LoginActivity.AUTH_TOKEN_TYPE, authTokenType);
   intent.putExtra(AccountManager.KEY_ACCOUNT_AUTHENTICATOR_RESPONSE, response);
   bundle.putParcelable(AccountManager.KEY_INTENT, intent);
   return bundle;
}

我认为调用blockingGetAuthToken不是问题,因为如果我在拦截器中注释掉invalidateAuthToken(...),它会返回一个缓存的令牌。

因为我在拦截器中发出另一个HTTP请求会有问题吗? (apiService.getBlaBlaBla() - &gt; tokenInterceptor - &gt; getAuthToken - &gt; apiService.refreshToken())

1 个答案:

答案 0 :(得分:0)

问题是断点实际上已被击中,但是在不同的帧中。这样断点就阻止了身份验证器和拦截器超时。

要在帧之间切换,请转到Debug - &gt;调试器 - &gt;框架并从下拉列表中选择适当的框架。

更多信息:https://developer.android.com/studio/debug/index.html#stackFrames