AccountManagerFuture.getResults 在连接处于活动状态时抛出 IOException

Posted

技术标签:

【中文标题】AccountManagerFuture.getResults 在连接处于活动状态时抛出 IOException【英文标题】:IOException thrown by AccountManagerFuture.getResults whereas connection is alive 【发布时间】:2012-06-01 06:18:28 【问题描述】:

以下方法在我的 android 设备上调用时系统地抛出 IOException 而互联网连接是活动的(我可以使用我的 Android 设备检索电子邮件或连接到 gmail)。

有人可以帮忙吗?

private void performAuthentication() 
    Log.d("GAWidget", "performAuthentication");
    GoogleCredential credential = new GoogleCredential();
    GoogleAccountManager accountManager = new GoogleAccountManager(this);
    Log.d("GAWidget", "after getting accountManager");
    Account account = accountManager.getAccountByName("balteo@gmail.com");
    Log.d("GAWidget", "after getting account"+"account.name: "+account.name);
    accountManager.getAccountManager().getAuthToken(account, "oauth2:https://www.googleapis.com/auth/‌​analytics.readonly",
            true, new AccountManagerCallback<Bundle>() 

                public void run(AccountManagerFuture<Bundle> future) 
                    try 
                        String token = future.getResult(15, TimeUnit.SECONDS).getString(AccountManager.KEY_AUTHTOKEN);
                        Log.d("GAWidget", "token: "+token);
                        useToken(token);
                     catch (OperationCanceledException e) 
                        Log.e("GAWidget", "OperationCanceledException", e);
                     catch (AuthenticatorException e) 
                        Log.e("GAWidget", "AuthenticatorException", e);
                     catch (IOException e) 
                        Log.e("GAWidget", "IOException", e);
                    
                

            , null);

编辑:这是堆栈跟踪:

05-27 19:09:04.319: E/GAWidget(12487): IOException
05-27 19:09:04.319: E/GAWidget(12487): java.io.IOException
05-27 19:09:04.319: E/GAWidget(12487):  at android.accounts.AccountManager.convertErrorToException(AccountManager.java:1440)
05-27 19:09:04.319: E/GAWidget(12487):  at android.accounts.AccountManager.access$400(AccountManager.java:138)
05-27 19:09:04.319: E/GAWidget(12487):  at android.accounts.AccountManager$AmsTask$Response.onError(AccountManager.java:1301)
05-27 19:09:04.319: E/GAWidget(12487):  at android.accounts.IAccountManagerResponse$Stub.onTransact(IAccountManagerResponse.java:69)
05-27 19:09:04.319: E/GAWidget(12487):  at android.os.Binder.execTransact(Binder.java:320)
05-27 19:09:04.319: E/GAWidget(12487):  at dalvik.system.NativeStart.run(Native Method)

装运的罐子:

google-http-client-1.9.0-beta.jar
google-http-client-android2-1.9.0-beta.jar (only for SDK >= 2.1)
google-http-client-android3-1.9.0-beta.jar (only for SDK >= 3.0)
gson-2.1.jar
guava-11.0.1.jar
jackson-core-asl-1.9.4.jar
jsr305-1.3.9.jar
protobuf-java-2.2.0.jar 

【问题讨论】:

【参考方案1】:

AccountManager 正在将任何网络错误转换为IOException。网络错误可能是某种意外的 HTTP 状态,因此它可能与网络连接没有直接关系。请注意,AccountManager 支持一些但不是所有的 'oauth2:' 类型标记,因此这可能是相关的。尝试使用已知受支持的令牌。另请查看 logcat 以获取提示,那里可能有一些警告。这是完整的堆栈跟踪吗?

这适用于 GN 4.0.4(注意它使用的是系统 AccountManager,而不是 GoogleAccountManager):

    AccountManager am = AccountManager.get(this);
    Account[] accounts = accountManager.getAccountsByType("com.google");
    String AUTH_TOKEN_TYPE = "oauth2:https://www.googleapis.com/auth/analytics.readonly";
    am.getAuthToken(accounts[0], AUTH_TOKEN_TYPE, null,
                   this, new AccountManagerCallback<Bundle>() 
                        public void run(AccountManagerFuture<Bundle> future) 
                           try 
                             String token = 
future.getResult().getString(AccountManager.KEY_AUTHTOKEN);
                            System.out.println("token " + token);
                             catch (...) 

                            
                        , null);

编辑:这是我从您的帖子中复制令牌类型时得到的:

0000000: 226f 6175 7468 323a 6874 7470 733a 2f2f  "oauth2:https://
0000010: 7777 772e 676f 6f67 6c65 6170 6973 2e63  www.googleapis.c
0000020: 6f6d 2f61 7574 682f 3f3f 616e 616c 7974  om/auth/??analyt
0000030: 6963 732e 7265 6164 6f6e 6c79 220a       ics.readonly".

再次,这可能是我的浏览器或 smth,通过检查您的字符串(尤其是最后一部分)

【讨论】:

顺便说一句,这在我的 Galaxy Nexus (4.0.4) 上运行良好,但您的令牌类型中似乎有一些奇怪的字符。可能只是复制粘贴错误,但请重新键入字符串并确保它是 ASCII 以防万一。 感谢您的回复 Nikolay,我试图删除 url 中的“oauth2:”部分,我得到一个空令牌。能否请您详细说明令牌类型中的奇怪字符?我不明白你的意思。关于警告我也得到这个:“GoogleLoginService:状态 200 但响应不包括身份验证令牌”。有什么想法吗? 我有时也会收到以下警告:“IInputConnectionWrapper showStatusIcon on inactive InputConnection” 你能告诉我你是如何设法让它与你的 Nexus 一起工作的吗?尤其是您如何更改代码以使其成功运行? 哦,我现在明白了。我使用的是系统 AccountManager,而不是 GoogleAccountManager。 (再次)查看更新后的帖子。

以上是关于AccountManagerFuture.getResults 在连接处于活动状态时抛出 IOException的主要内容,如果未能解决你的问题,请参考以下文章