Search code examples
javaandroidrestauthenticationaccountmanager

How can I make AccountManager handle multiple accounts per username?


I would like to use Android's built-in AccountManager to handle accounts for our native Android app. However, we have a peculiar need.

It appears that Android's concept of an Account is a combination of name (i.e. [email protected]) and type (i.e. com.example). Name is the username you login with. Type is associated with your application or company.

However, our REST backend allows a single user to have multiple accounts and each account must be accessed by its own unique hash tied to the combination of one username and one account number (in addition to type).

I already have the AccountAuthenticator, AuthenticationService, and AccountAuthenticatorActivity working for a single user with a single account number and single stored hash. Is there any way in which I could implement my AccountAuthenticator to handle users with multiple accounts under the same username, each requiring a hash? Would it be possible to attach a delimiter to the username and split it into username and account number every time I use it?

If I cannot figure out a way to handle this, how should I gracefully fall back during AbstractAccountAuthenticator.getAuthToken? Would it make sense to set the hash to a special flag value and use legacy login methods for that user? Or is that too much of a hack?

Thanks for your help!


Solution

  • I ended up serializing the data into the username using an at sign (@) as a delimiter. I chose the at sign because it's the only restricted character in an e-mail address that should only be used exactly one time.

    Here is the code from getAuthToken in my AccountAuthenticator which gets called only if I need to obtain a fresh token for that user and account id:

        /*
        *
        * The login params need to handle users with multiple accounts under the same username.
        *
        * Since Android's AccountManager does not allow multiple accounts per username, I had
        * to create a hack which joins and splits the username on a delimiter to serialize the
        * data and retrieve the account number for users with multiple accounts. I chose the @
        * sign as a delimiter because e-mail addresses have VERY few invalid characters in
        * the account name part of the address.
        *
        * If the user has multiple accounts, we need to create each one in AccountManager.
        *
        * */
    
        String[] accountParts = account.name.split("@");
        numParts = accountParts.length;
        if (numParts<2) {
            Log.wtf(Config.TAG, "Username split produced too few parts. WTF.");
            return null;
        }
        String email = accountParts[0] + "@" + accountParts[1];
    
        if (numParts==3) {
            String account_id = accountParts[2];
        } else if (numParts>3) {
            Log.wtf(Config.TAG, "Username split produced too many parts. WTF.");
            return null;
        }