3

我想在android上添加自定義帳戶AccountManager.addAccount()。我正在關注this tutorial。當我試圖用AccountManagerCallbackrun方法得到結果時,我得到AuthenticatorException,消息:android.accounts.AuthenticatorException: bind failureAndroid AccountManager拋出AuthenticatorException:添加帳戶時綁定失敗

經過一番研究,我發現了兩個潛在的解決方案,但我已經裏面的application標記和checked my account type。我還將清單權限與教程中的權限進行了比較。我正在使用android studio 1.4,並在幾個模擬器上和物理設備上嘗試過。

這裏是我的AndroidManifest.xml,也authenticator.xmlaccount_preferences.xml

<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
    package="com.test.myproject" > 

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

    <uses-permission android:name="android.permission.AUTHENTICATE_ACCOUNTS"/> 
    <uses-permission android:name="android.permission.MANAGE_ACCOUNTS"/> 

    <application 
     android:allowBackup="true" 
     android:icon="@mipmap/test_logo" 
     android:label="@string/app_name" 
     android:theme="@style/AppTheme" > 

     <activity 
      android:name=".view.MainActivity" 
      android:label="@string/app_name" > 
      <intent-filter> 
       <action android:name="android.intent.action.MAIN" /> 
       <category android:name="android.intent.category.LAUNCHER" /> 
      </intent-filter> 
     </activity> 
     <activity 
      android:name=".view.LoginActivity" 
      android:label="@string/app_name"> 
     </activity> 

     <service 
      android:name="com.test.myproject.model.utility.MyAuthenticatorService"> 
      <intent-filter> 
       <action android:name="android.accounts.AccountAuthenticator"/> 
      </intent-filter> 
      <meta-data 
       android:name="android.accounts.AccountAuthenticator" 
       android:resource="@xml/authenticator" /> 
     </service> 

     <meta-data 
      android:name="com.google.android.gms.version" 
      android:value="@integer/google_play_services_version" /> 

    </application> 
</manifest> 

我也試圖與任何效果命名服務.model.utility.MyAuthenticatorService

authenticator.xml:

<?xml version="1.0" encoding="utf-8"?> 
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android"> 
    <account-authenticator xmlns:android="http://schemas.android.com/apk/res/android" 
     android:accountType="com.test.myproject" 
     android:icon="@drawable/test_logo" 
     android:smallIcon="@drawable/test_logo" 
     android:label="@string/not_implemented" 
     android:accountPreferences="@xml/account_preferences" 
     /> 
</PreferenceScreen> 

account_preferences.xml:

<?xml version="1.0" encoding="utf-8"?> 
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android"> 
    <PreferenceCategory android:title="@string/not_implemented" /> 
    <CheckBoxPreference android:title="Use debug server" 
     android:key="isDebug" 
     android:summary="Connecting to a debug server instead of prod server"/> 
    <SwitchPreference android:title="Debug Logs" android:key="logsVerbose" android:summary="Show debug logs on LogCat"/> 
</PreferenceScreen> 

這裏是MyAuthenticatorService

public class MyAuthenticatorService extends Service { 
    @Nullable 
    @Override 
    public IBinder onBind(Intent intent) { 
     MyAuthenticator myAuthenticator = new MyAuthenticator(this); 
     return myAuthenticator.getIBinder(); 
    } 
} 

這裏是MyAuthenticator

public class MyAuthenticator extends AbstractAccountAuthenticator { 

    private Context context; 

    public MyAuthenticator(Context context) { 
     super(context); 
     this.context = context; 
    } 

    @Override 
    public Bundle addAccount(AccountAuthenticatorResponse response, String accountType, String authTokenType, String[] requiredFeatures, Bundle options) throws NetworkErrorException { 
     final Intent intent = new Intent(context, LoginActivity.class); 
     intent.putExtra(LoginActivity.ARG_ACCOUNT_TYPE, accountType); 
     intent.putExtra(LoginActivity.ARG_AUTH_TYPE, authTokenType); 
     intent.putExtra(LoginActivity.ARG_IS_ADDING_NEW_ACCOUNT, true); 
     intent.putExtra(AccountManager.KEY_ACCOUNT_AUTHENTICATOR_RESPONSE, response); 
     final Bundle bundle = new Bundle(); 
     bundle.putParcelable(AccountManager.KEY_INTENT, intent); 
     return bundle; 
    } 
    //other override methods (they all return null for now) 
} 

這裏是MainActivity

public class MainActivity extends AppCompatActivity { 

    private Toolbar toolbar; 
    private NavigationView navigationView; 
    private DrawerLayout drawerLayout; 
    private SharedPreferences sharedPreferences; 
    private AccountManager accountManager; 
    private static final String ACCOUNT_TYPE = "com.test.myproject"; 

    @Override 
    protected void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.activity_drawer); 

     this.accountManager = AccountManager.get(this); 

     signIn(ACCOUNT_TYPE, "access_token"); 

     //other stuff 

    } 

    private void signIn(String accountType, String authTokenType) { 

     final AccountManagerFuture<Bundle> future = accountManager.addAccount(ACCOUNT_TYPE, authTokenType, null, null, this, new AccountManagerCallback<Bundle>() { 
      @Override 
      public void run(AccountManagerFuture<Bundle> future) { 
       try { 
        Bundle bnd = future.getResult(); 
        showMessage("Account was created"); 
        Log.d("udinic", "AddNewAccount Bundle is " + bnd); 

       } catch (Exception e) { 
        e.printStackTrace(); 
        showMessage(e.getMessage()); 
       } 
      } 
     }, null); 
    } 
} 

debbuging時,錯誤會被拋出Bundle bnd = future.getResult();,以及未來有狀態3和成果android.accounts.AuthenticatorException: bind failure。艱難的執行永遠不會到LoginActivity,或至少斷點不會被觸發,在這裏它是:

public class LoginActivity extends AccountAuthenticatorActivity { 

    public final static String ARG_ACCOUNT_TYPE = "ACCOUNT_TYPE"; 
    public final static String ARG_ACCOUNT_NAME = "AUTH_TYPE"; 
    public final static String ARG_AUTH_TYPE = "ACCOUNT_NAME"; 
    public final static String PARAM_USER_PASS = "USER_PASS"; 

    private AccountManager accountManager; 

    @Override 
    protected void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.activity_login); 
     this.accountManager = AccountManager.get(getBaseContext()); 


    } 

    public void login(View view) { 
     EditText usernameEditText = (EditText)findViewById(R.id.textLoginUsername); 
     EditText passwordEditText = (EditText)findViewById(R.id.textLoginPassword); 

     String ownerUsername = usernameEditText.getText().toString(); 
     String ownerPassword = passwordEditText.getText().toString(); 
     String clientId = "test"; 
     String clientSecret = "test"; 

     AccountManager manager = AccountManager.get(this); 

     TokenRequestTask tokenRequestTask = new TokenRequestTask(); 
     tokenRequestTask.execute(ownerUsername, ownerPassword, clientId, clientSecret); 
    } 

    private class TokenRequestTask extends AsyncTask<String, Void, Intent> { 
     @Override 
     protected Intent doInBackground(String... params) { 

      final String accountType = getIntent().getStringExtra(ARG_ACCOUNT_TYPE); 

      String ownerUsername = params[0]; 
      String ownerSecret = params[1]; 
      String clientId = params[2]; 
      String clientSecret = params[3]; 

      String authToken = signIn(clientId, clientSecret, ownerUsername, ownerSecret, "password"); 

      Bundle resultData = new Bundle(); 
      resultData.putString(AccountManager.KEY_ACCOUNT_NAME, ownerUsername); 
      resultData.putString(AccountManager.KEY_ACCOUNT_TYPE, accountType); 
      resultData.putString(AccountManager.KEY_AUTHTOKEN, authToken); 
      resultData.putString(PARAM_USER_PASS, ownerSecret); 


      final Intent resultIntent = new Intent(); 
      resultIntent.putExtras(resultData); 
      return resultIntent; 
     } 
    } 

    private String signIn(String clientId, String clientSecret, String ownerUsername, String ownerSecret, String grantType) { 
     MyApi20ServiceImpl service = (MyApi20ServiceImpl)new ServiceBuilder().provider(MyApi20.class) 
       .apiKey(clientId) 
       .apiSecret(clientSecret) 
       .signatureType(SignatureType.QueryString) 
       .build(); 
     Token token = service.getAccessToken(ownerUsername, ownerSecret, grantType); 
     return token.getToken(); 
    } 
} 

下面是完整的堆棧跟蹤:

10-19 10:39:05.042 25931-25931/? I/art: Not late-enabling -Xcheck:jni (already on) 
10-19 10:39:05.042 25931-25931/? I/art: Late-enabling JIT 
10-19 10:39:05.068 25931-25931/? I/art: JIT created with code_cache_capacity=2MB compile_threshold=1000 
10-19 10:39:05.118 25931-25931/com.test.myproject W/System: ClassLoader referenced unknown path: /data/app/com.test.myproject-1/lib/x86 
10-19 10:39:05.355 25931-25960/com.test.myproject D/OpenGLRenderer: Use EGL_SWAP_BEHAVIOR_PRESERVED: true 
10-19 10:39:05.358 25931-25931/com.test.myproject D/: HostConnection::get() New Host Connection established 0xad974e50, tid 25931 
10-19 10:39:05.366 25931-25931/com.test.myproject W/System.err: android.accounts.AuthenticatorException: bind failure 
10-19 10:39:05.366 25931-25931/com.test.myproject W/System.err:  at android.accounts.AccountManager.convertErrorToException(AccountManager.java:2147) 
10-19 10:39:05.366 25931-25931/com.test.myproject W/System.err:  at android.accounts.AccountManager.-wrap0(AccountManager.java) 
10-19 10:39:05.366 25931-25931/com.test.myproject W/System.err:  at android.accounts.AccountManager$AmsTask$Response.onError(AccountManager.java:1990) 
10-19 10:39:05.366 25931-25931/com.test.myproject W/System.err:  at android.accounts.IAccountManagerResponse$Stub.onTransact(IAccountManagerResponse.java:69) 
10-19 10:39:05.366 25931-25931/com.test.myproject W/System.err:  at android.os.Binder.execTransact(Binder.java:453) 
10-19 10:39:05.423 25931-25960/com.test.myproject D/: HostConnection::get() New Host Connection established 0xac17e080, tid 25960 
10-19 10:39:05.433 25931-25960/com.test.myproject I/OpenGLRenderer: Initialized EGL, version 1.4 
10-19 10:39:05.538 25931-25960/com.test.myproject W/EGL_emulation: eglSurfaceAttrib not implemented 
10-19 10:39:05.538 25931-25960/com.test.myproject W/OpenGLRenderer: Failed to set EGL_SWAP_BEHAVIOR on surface 0xaf125cc0, error=EGL_SUCCESS 
10-19 10:39:05.903 25931-25931/com.test.myproject I/Choreographer: Skipped 31 frames! The application may be doing too much work on its main thread. 
10-19 10:39:05.975 25931-25960/com.test.myproject W/EGL_emulation: eglSurfaceAttrib not implemented 
10-19 10:39:05.975 25931-25960/com.test.myproject W/OpenGLRenderer: Failed to set EGL_SWAP_BEHAVIOR on surface 0xb3fe0c40, error=EGL_SUCCESS 
10-19 10:39:07.392 25931-25960/com.test.myproject E/Surface: getSlotFromBufferLocked: unknown buffer: 0xac027c50 

有人可以幫我解決這個問題呢?

+0

您是否嘗試將清單中的服務名稱更改爲: .model.utility。MyAuthenticatorService –

+0

我有,同樣的事情發生 – Miljac

+0

並添加了export =「false」到服務? –

回答

4

好使用,所以,這是一個真正的複製粘貼錯誤,這讓我瘋了。在authenticator.xml

<?xml version="1.0" encoding="utf-8"?> 
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android"> 
    <account-authenticator xmlns:android="http://schemas.android.com/apk/res/android" 
     android:accountType="com.test.myproject" 
     android:icon="@drawable/test_logo" 
     android:smallIcon="@drawable/test_logo" 
     android:label="@string/not_implemented" 
     android:accountPreferences="@xml/account_preferences" 
     /> 
</PreferenceScreen> 

</PreferenceScreen>不應該在這裏。

如果您收到此錯誤,請檢查您的xml文件。

1

可以addAccountExplicitly

+0

嗯,這不是我正在尋找的,這是更多的這種情況下的解決方法,我想使用'addAccount',而不是'addAccountExplicitly' 。 但是我確實嘗試過,並且得到了異常'java.lang.SecurityException:uid 10053無法顯式地添加類型爲com.test.myproject'的賬戶,但是我沒有再付出任何努力,因爲它不是我的正在尋找 – Miljac

0

我也有同樣的問題。

就像Miljac的,我的問題是關於AndroidManifest.xml文件。

相關問題