AccountUtils.java 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. /**
  2. * ownCloud Android client application
  3. *
  4. * Copyright (C) 2012 Bartek Przybylski
  5. * Copyright (C) 2015 ownCloud Inc.
  6. *
  7. * This program is free software: you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License version 2,
  9. * as published by the Free Software Foundation.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. *
  19. */
  20. package com.owncloud.android.authentication;
  21. import java.util.Locale;
  22. import com.owncloud.android.MainApp;
  23. import com.owncloud.android.lib.common.accounts.AccountTypeUtils;
  24. import com.owncloud.android.lib.common.accounts.AccountUtils.Constants;
  25. import com.owncloud.android.lib.common.utils.Log_OC;
  26. import com.owncloud.android.lib.resources.status.OwnCloudVersion;
  27. import android.accounts.Account;
  28. import android.accounts.AccountManager;
  29. import android.content.Context;
  30. import android.content.SharedPreferences;
  31. import android.net.Uri;
  32. import android.preference.PreferenceManager;
  33. public class AccountUtils {
  34. private static final String TAG = AccountUtils.class.getSimpleName();
  35. public static final String WEBDAV_PATH_4_0_AND_LATER = "/remote.php/webdav";
  36. private static final String ODAV_PATH = "/remote.php/odav";
  37. private static final String SAML_SSO_PATH = "/remote.php/webdav";
  38. public static final String STATUS_PATH = "/status.php";
  39. public static final int ACCOUNT_VERSION = 1;
  40. /**
  41. * Can be used to get the currently selected ownCloud {@link Account} in the
  42. * application preferences.
  43. *
  44. * @param context The current application {@link Context}
  45. * @return The ownCloud {@link Account} currently saved in preferences, or the first
  46. * {@link Account} available, if valid (still registered in the system as ownCloud
  47. * account). If none is available and valid, returns null.
  48. */
  49. public static Account getCurrentOwnCloudAccount(Context context) {
  50. Account[] ocAccounts = AccountManager.get(context).getAccountsByType(
  51. MainApp.getAccountType());
  52. Account defaultAccount = null;
  53. SharedPreferences appPreferences = PreferenceManager
  54. .getDefaultSharedPreferences(context);
  55. String accountName = appPreferences
  56. .getString("select_oc_account", null);
  57. // account validation: the saved account MUST be in the list of ownCloud Accounts known by the AccountManager
  58. if (accountName != null) {
  59. for (Account account : ocAccounts) {
  60. if (account.name.equals(accountName)) {
  61. defaultAccount = account;
  62. break;
  63. }
  64. }
  65. }
  66. if (defaultAccount == null && ocAccounts.length != 0) {
  67. // take first account as fallback
  68. defaultAccount = ocAccounts[0];
  69. }
  70. return defaultAccount;
  71. }
  72. public static boolean exists(Account account, Context context) {
  73. Account[] ocAccounts = AccountManager.get(context).getAccountsByType(
  74. MainApp.getAccountType());
  75. if (account != null && account.name != null) {
  76. int lastAtPos = account.name.lastIndexOf("@");
  77. String hostAndPort = account.name.substring(lastAtPos + 1);
  78. String username = account.name.substring(0, lastAtPos);
  79. String otherHostAndPort, otherUsername;
  80. Locale currentLocale = context.getResources().getConfiguration().locale;
  81. for (Account otherAccount : ocAccounts) {
  82. lastAtPos = otherAccount.name.lastIndexOf("@");
  83. otherHostAndPort = otherAccount.name.substring(lastAtPos + 1);
  84. otherUsername = otherAccount.name.substring(0, lastAtPos);
  85. if (otherHostAndPort.equals(hostAndPort) &&
  86. otherUsername.toLowerCase(currentLocale).
  87. equals(username.toLowerCase(currentLocale))) {
  88. return true;
  89. }
  90. }
  91. }
  92. return false;
  93. }
  94. public static boolean setCurrentOwnCloudAccount(Context context, String accountName) {
  95. boolean result = false;
  96. if (accountName != null) {
  97. Account[] ocAccounts = AccountManager.get(context).getAccountsByType(
  98. MainApp.getAccountType());
  99. boolean found;
  100. for (Account account : ocAccounts) {
  101. found = (account.name.equals(accountName));
  102. if (found) {
  103. SharedPreferences.Editor appPrefs = PreferenceManager
  104. .getDefaultSharedPreferences(context).edit();
  105. appPrefs.putString("select_oc_account", accountName);
  106. appPrefs.commit();
  107. result = true;
  108. break;
  109. }
  110. }
  111. }
  112. return result;
  113. }
  114. /**
  115. * Returns the proper URL path to access the WebDAV interface of an ownCloud server,
  116. * according to its version and the authorization method used.
  117. *
  118. * @param version Version of ownCloud server.
  119. * @param authTokenType Authorization token type, matching some of the AUTH_TOKEN_TYPE_* constants in
  120. * {@link AccountAuthenticator}.
  121. * @return WebDAV path for given OC version and authorization method, null if OC version
  122. * is unknown; versions prior to ownCloud 4 are not supported anymore
  123. */
  124. public static String getWebdavPath(OwnCloudVersion version, String authTokenType) {
  125. if (version != null) {
  126. if (AccountTypeUtils.getAuthTokenTypeAccessToken(MainApp.getAccountType()).equals(authTokenType)) {
  127. return ODAV_PATH;
  128. }
  129. if (AccountTypeUtils.getAuthTokenTypeSamlSessionCookie(MainApp.getAccountType()).equals(authTokenType)) {
  130. return SAML_SSO_PATH;
  131. }
  132. return WEBDAV_PATH_4_0_AND_LATER;
  133. }
  134. return null;
  135. }
  136. /**
  137. * Update the accounts in AccountManager to meet the current version of accounts expected by the app, if needed.
  138. *
  139. * Introduced to handle a change in the structure of stored account names needed to allow different OC servers
  140. * in the same domain, but not in the same path.
  141. *
  142. * @param context Used to access the AccountManager.
  143. */
  144. public static void updateAccountVersion(Context context) {
  145. Account currentAccount = AccountUtils.getCurrentOwnCloudAccount(context);
  146. AccountManager accountMgr = AccountManager.get(context);
  147. if ( currentAccount != null ) {
  148. String currentAccountVersion = accountMgr.getUserData(currentAccount, Constants.KEY_OC_ACCOUNT_VERSION);
  149. if (currentAccountVersion == null) {
  150. Log_OC.i(TAG, "Upgrading accounts to account version #" + ACCOUNT_VERSION);
  151. Account[] ocAccounts = accountMgr.getAccountsByType(MainApp.getAccountType());
  152. String serverUrl, username, newAccountName, password;
  153. Account newAccount;
  154. for (Account account : ocAccounts) {
  155. // build new account name
  156. serverUrl = accountMgr.getUserData(account, Constants.KEY_OC_BASE_URL);
  157. username = account.name.substring(0, account.name.lastIndexOf('@'));
  158. newAccountName = com.owncloud.android.lib.common.accounts.AccountUtils.
  159. buildAccountName(Uri.parse(serverUrl), username);
  160. // migrate to a new account, if needed
  161. if (!newAccountName.equals(account.name)) {
  162. Log_OC.d(TAG, "Upgrading " + account.name + " to " + newAccountName);
  163. // create the new account
  164. newAccount = new Account(newAccountName, MainApp.getAccountType());
  165. password = accountMgr.getPassword(account);
  166. accountMgr.addAccountExplicitly(newAccount, (password != null) ? password : "", null);
  167. // copy base URL
  168. accountMgr.setUserData(newAccount, Constants.KEY_OC_BASE_URL, serverUrl);
  169. // copy server version
  170. accountMgr.setUserData(
  171. newAccount,
  172. Constants.KEY_OC_VERSION,
  173. accountMgr.getUserData(account, Constants.KEY_OC_VERSION)
  174. );
  175. // copy cookies
  176. accountMgr.setUserData(
  177. newAccount,
  178. Constants.KEY_COOKIES,
  179. accountMgr.getUserData(account, Constants.KEY_COOKIES)
  180. );
  181. // copy type of authentication
  182. String isSamlStr = accountMgr.getUserData(account, Constants.KEY_SUPPORTS_SAML_WEB_SSO);
  183. boolean isSaml = "TRUE".equals(isSamlStr);
  184. if (isSaml) {
  185. accountMgr.setUserData(newAccount, Constants.KEY_SUPPORTS_SAML_WEB_SSO, "TRUE");
  186. }
  187. String isOauthStr = accountMgr.getUserData(account, Constants.KEY_SUPPORTS_OAUTH2);
  188. boolean isOAuth = "TRUE".equals(isOauthStr);
  189. if (isOAuth) {
  190. accountMgr.setUserData(newAccount, Constants.KEY_SUPPORTS_OAUTH2, "TRUE");
  191. }
  192. /* TODO - study if it's possible to run this method in a background thread to copy the authToken
  193. if (isOAuth || isSaml) {
  194. accountMgr.setAuthToken(newAccount, mAuthTokenType, mAuthToken);
  195. }
  196. */
  197. // don't forget the account saved in preferences as the current one
  198. if (currentAccount.name.equals(account.name)) {
  199. AccountUtils.setCurrentOwnCloudAccount(context, newAccountName);
  200. }
  201. // remove the old account
  202. accountMgr.removeAccount(account, null, null);
  203. // will assume it succeeds, not a big deal otherwise
  204. } else {
  205. // servers which base URL is in the root of their domain need no change
  206. Log_OC.d(TAG, account.name + " needs no upgrade ");
  207. newAccount = account;
  208. }
  209. // at least, upgrade account version
  210. Log_OC.d(TAG, "Setting version " + ACCOUNT_VERSION + " to " + newAccountName);
  211. accountMgr.setUserData(
  212. newAccount, Constants.KEY_OC_ACCOUNT_VERSION, Integer.toString(ACCOUNT_VERSION)
  213. );
  214. }
  215. }
  216. }
  217. }
  218. public static String trimWebdavSuffix(String url) {
  219. while(url.endsWith("/")) {
  220. url = url.substring(0, url.length() - 1);
  221. }
  222. int pos = url.lastIndexOf(WEBDAV_PATH_4_0_AND_LATER);
  223. if (pos >= 0) {
  224. url = url.substring(0, pos);
  225. } else {
  226. pos = url.lastIndexOf(ODAV_PATH);
  227. if (pos >= 0) {
  228. url = url.substring(0, pos);
  229. }
  230. }
  231. return url;
  232. }
  233. /**
  234. * Access the version of the OC server corresponding to an account SAVED IN THE ACCOUNTMANAGER
  235. *
  236. * @param account ownCloud account
  237. * @return Version of the OC server corresponding to account, according to the data saved
  238. * in the system AccountManager
  239. */
  240. public static OwnCloudVersion getServerVersion(Account account) {
  241. OwnCloudVersion serverVersion = null;
  242. if (account != null) {
  243. AccountManager accountMgr = AccountManager.get(MainApp.getAppContext());
  244. String serverVersionStr = accountMgr.getUserData(account, Constants.KEY_OC_VERSION);
  245. if (serverVersionStr != null) {
  246. serverVersion = new OwnCloudVersion(serverVersionStr);
  247. }
  248. }
  249. return serverVersion;
  250. }
  251. }