AccountUtils.java 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  1. /**
  2. * ownCloud Android client application
  3. *
  4. * Copyright (C) 2012 Bartek Przybylski
  5. * Copyright (C) 2016 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 android.accounts.Account;
  22. import android.accounts.AccountManager;
  23. import android.content.Context;
  24. import android.content.SharedPreferences;
  25. import android.net.Uri;
  26. import android.preference.PreferenceManager;
  27. import com.owncloud.android.MainApp;
  28. import com.owncloud.android.datamodel.ArbitraryDataProvider;
  29. import com.owncloud.android.datamodel.FileDataStorageManager;
  30. import com.owncloud.android.lib.common.OwnCloudAccount;
  31. import com.owncloud.android.lib.common.OwnCloudClient;
  32. import com.owncloud.android.lib.common.OwnCloudClientManagerFactory;
  33. import com.owncloud.android.lib.common.UserInfo;
  34. import com.owncloud.android.lib.common.accounts.AccountTypeUtils;
  35. import com.owncloud.android.lib.common.accounts.AccountUtils.Constants;
  36. import com.owncloud.android.lib.common.operations.RemoteOperationResult;
  37. import com.owncloud.android.lib.common.utils.Log_OC;
  38. import com.owncloud.android.lib.resources.status.OwnCloudVersion;
  39. import com.owncloud.android.lib.resources.users.GetRemoteUserInfoOperation;
  40. import com.owncloud.android.operations.GetCapabilitiesOperarion;
  41. import com.owncloud.android.ui.activity.ManageAccountsActivity;
  42. import java.util.Locale;
  43. public class AccountUtils {
  44. private static final String TAG = AccountUtils.class.getSimpleName();
  45. public static final String WEBDAV_PATH_4_0_AND_LATER = "/remote.php/webdav";
  46. public static final String DAV_PATH = "/remote.php/dav";
  47. private static final String ODAV_PATH = "/remote.php/odav";
  48. private static final String SAML_SSO_PATH = "/remote.php/webdav";
  49. public static final String STATUS_PATH = "/status.php";
  50. public static final int ACCOUNT_VERSION = 1;
  51. public static final int ACCOUNT_VERSION_WITH_PROPER_ID = 2;
  52. /**
  53. * Can be used to get the currently selected ownCloud {@link Account} in the
  54. * application preferences.
  55. *
  56. * @param context The current application {@link Context}
  57. * @return The ownCloud {@link Account} currently saved in preferences, or the first
  58. * {@link Account} available, if valid (still registered in the system as ownCloud
  59. * account). If none is available and valid, returns null.
  60. */
  61. public static Account getCurrentOwnCloudAccount(Context context) {
  62. Account[] ocAccounts = getAccounts(context);
  63. Account defaultAccount = null;
  64. ArbitraryDataProvider arbitraryDataProvider = new ArbitraryDataProvider(context.getContentResolver());
  65. SharedPreferences appPreferences = PreferenceManager.getDefaultSharedPreferences(context);
  66. String accountName = appPreferences.getString("select_oc_account", null);
  67. // account validation: the saved account MUST be in the list of ownCloud Accounts known by the AccountManager
  68. if (accountName != null) {
  69. for (Account account : ocAccounts) {
  70. if (account.name.equals(accountName)) {
  71. defaultAccount = account;
  72. break;
  73. }
  74. }
  75. }
  76. if (defaultAccount == null && ocAccounts.length > 0) {
  77. // take first which is not pending for removal account as fallback
  78. for (Account account: ocAccounts) {
  79. boolean pendingForRemoval = arbitraryDataProvider.getBooleanValue(account,
  80. ManageAccountsActivity.PENDING_FOR_REMOVAL);
  81. if (!pendingForRemoval) {
  82. defaultAccount = account;
  83. break;
  84. }
  85. }
  86. }
  87. return defaultAccount;
  88. }
  89. public static Account[] getAccounts(Context context) {
  90. AccountManager accountManager = AccountManager.get(context);
  91. return accountManager.getAccountsByType(MainApp.getAccountType());
  92. }
  93. public static boolean exists(Account account, Context context) {
  94. Account[] ocAccounts = getAccounts(context);
  95. if (account != null && account.name != null) {
  96. int lastAtPos = account.name.lastIndexOf("@");
  97. String hostAndPort = account.name.substring(lastAtPos + 1);
  98. String username = account.name.substring(0, lastAtPos);
  99. String otherHostAndPort;
  100. String otherUsername;
  101. Locale currentLocale = context.getResources().getConfiguration().locale;
  102. for (Account otherAccount : ocAccounts) {
  103. lastAtPos = otherAccount.name.lastIndexOf("@");
  104. otherHostAndPort = otherAccount.name.substring(lastAtPos + 1);
  105. otherUsername = otherAccount.name.substring(0, lastAtPos);
  106. if (otherHostAndPort.equals(hostAndPort) &&
  107. otherUsername.toLowerCase(currentLocale).
  108. equals(username.toLowerCase(currentLocale))) {
  109. return true;
  110. }
  111. }
  112. }
  113. return false;
  114. }
  115. /**
  116. * returns the user's name based on the account name.
  117. *
  118. * @param accountName the account name
  119. * @return the user's name
  120. */
  121. public static String getAccountUsername(String accountName) {
  122. if (accountName != null) {
  123. return accountName.substring(0, accountName.lastIndexOf('@'));
  124. } else {
  125. return null;
  126. }
  127. }
  128. /**
  129. * Returns owncloud account identified by accountName or null if it does not exist.
  130. * @param context
  131. * @param accountName name of account to be returned
  132. * @return owncloud account named accountName
  133. */
  134. public static Account getOwnCloudAccountByName(Context context, String accountName) {
  135. Account[] ocAccounts = AccountManager.get(context).getAccountsByType(
  136. MainApp.getAccountType());
  137. for (Account account : ocAccounts) {
  138. if(account.name.equals(accountName)) {
  139. return account;
  140. }
  141. }
  142. return null;
  143. }
  144. public static boolean setCurrentOwnCloudAccount(final Context context, String accountName) {
  145. boolean result = false;
  146. if (accountName != null) {
  147. boolean found;
  148. for (final Account account : getAccounts(context)) {
  149. found = (account.name.equals(accountName));
  150. if (found) {
  151. SharedPreferences.Editor appPrefs = PreferenceManager.getDefaultSharedPreferences(context).edit();
  152. appPrefs.putString("select_oc_account", accountName);
  153. // update credentials
  154. Thread t = new Thread(new Runnable() {
  155. @Override
  156. public void run() {
  157. FileDataStorageManager storageManager = new FileDataStorageManager(account,
  158. context.getContentResolver());
  159. GetCapabilitiesOperarion getCapabilities = new GetCapabilitiesOperarion();
  160. RemoteOperationResult updateResult = getCapabilities.execute(storageManager, context);
  161. Log_OC.w(TAG, "Update Capabilities: " + updateResult.isSuccess());
  162. }
  163. });
  164. t.start();
  165. appPrefs.apply();
  166. result = true;
  167. break;
  168. }
  169. }
  170. }
  171. return result;
  172. }
  173. public static void resetOwnCloudAccount(Context context) {
  174. SharedPreferences.Editor appPrefs = PreferenceManager.getDefaultSharedPreferences(context).edit();
  175. appPrefs.putString("select_oc_account", null);
  176. appPrefs.apply();
  177. }
  178. /**
  179. * Returns the proper URL path to access the WebDAV interface of an ownCloud server,
  180. * according to its version and the authorization method used.
  181. *
  182. * @param version Version of ownCloud server.
  183. * @param authTokenType Authorization token type, matching some of the AUTH_TOKEN_TYPE_* constants in
  184. * {@link AccountAuthenticator}.
  185. * @return WebDAV path for given OC version and authorization method, null if OC version
  186. * is unknown; versions prior to ownCloud 4 are not supported anymore
  187. */
  188. public static String getWebdavPath(OwnCloudVersion version, String authTokenType) {
  189. if (version != null) {
  190. if (AccountTypeUtils.getAuthTokenTypeAccessToken(MainApp.getAccountType()).equals(authTokenType)) {
  191. return ODAV_PATH;
  192. }
  193. if (AccountTypeUtils.getAuthTokenTypeSamlSessionCookie(MainApp.getAccountType()).equals(authTokenType)) {
  194. return SAML_SSO_PATH;
  195. }
  196. return WEBDAV_PATH_4_0_AND_LATER;
  197. }
  198. return null;
  199. }
  200. /**
  201. * Update the accounts in AccountManager to meet the current version of accounts expected by the app, if needed.
  202. *
  203. * Introduced to handle a change in the structure of stored account names needed to allow different OC servers
  204. * in the same domain, but not in the same path.
  205. *
  206. * @param context Used to access the AccountManager.
  207. */
  208. public static void updateAccountVersion(Context context) {
  209. Account currentAccount = AccountUtils.getCurrentOwnCloudAccount(context);
  210. AccountManager accountMgr = AccountManager.get(context);
  211. if ( currentAccount != null ) {
  212. String currentAccountVersion = accountMgr.getUserData(currentAccount, Constants.KEY_OC_ACCOUNT_VERSION);
  213. if (!String.valueOf(ACCOUNT_VERSION_WITH_PROPER_ID).equalsIgnoreCase(currentAccountVersion)) {
  214. Log_OC.i(TAG, "Upgrading accounts to account version #" + ACCOUNT_VERSION_WITH_PROPER_ID);
  215. Account[] ocAccounts = accountMgr.getAccountsByType(MainApp.getAccountType());
  216. String serverUrl;
  217. String username;
  218. String newAccountName;
  219. String password;
  220. Account newAccount;
  221. for (Account account : ocAccounts) {
  222. // build new account name
  223. serverUrl = accountMgr.getUserData(account, Constants.KEY_OC_BASE_URL);
  224. // update user name
  225. try {
  226. OwnCloudAccount ocAccount = new OwnCloudAccount(account, context);
  227. OwnCloudClient client = OwnCloudClientManagerFactory.getDefaultSingleton()
  228. .getClientFor(ocAccount, context);
  229. GetRemoteUserInfoOperation remoteUserNameOperation = new GetRemoteUserInfoOperation();
  230. RemoteOperationResult result = remoteUserNameOperation.execute(client);
  231. if (result.isSuccess()) {
  232. UserInfo userInfo = (UserInfo) result.getData().get(0);
  233. username = userInfo.id;
  234. } else {
  235. // skip account, try it next time
  236. Log_OC.e(TAG, "Error while getting username for account: " + account.name);
  237. continue;
  238. }
  239. } catch (Exception e) {
  240. Log_OC.e(TAG, "Error while getting username: " + e.getMessage());
  241. continue;
  242. }
  243. newAccountName = com.owncloud.android.lib.common.accounts.AccountUtils.
  244. buildAccountName(Uri.parse(serverUrl), username);
  245. // migrate to a new account, if needed
  246. if (!newAccountName.equals(account.name)) {
  247. Log_OC.d(TAG, "Upgrading " + account.name + " to " + newAccountName);
  248. // create the new account
  249. newAccount = new Account(newAccountName, MainApp.getAccountType());
  250. password = accountMgr.getPassword(account);
  251. accountMgr.addAccountExplicitly(newAccount, (password != null) ? password : "", null);
  252. // copy base URL
  253. accountMgr.setUserData(newAccount, Constants.KEY_OC_BASE_URL, serverUrl);
  254. // copy server version
  255. accountMgr.setUserData(
  256. newAccount,
  257. Constants.KEY_OC_VERSION,
  258. accountMgr.getUserData(account, Constants.KEY_OC_VERSION)
  259. );
  260. // copy cookies
  261. accountMgr.setUserData(
  262. newAccount,
  263. Constants.KEY_COOKIES,
  264. accountMgr.getUserData(account, Constants.KEY_COOKIES)
  265. );
  266. // copy type of authentication
  267. final String isSamlStr = accountMgr.getUserData(account, Constants.KEY_SUPPORTS_SAML_WEB_SSO);
  268. if (Boolean.parseBoolean(isSamlStr)) {
  269. accountMgr.setUserData(newAccount, Constants.KEY_SUPPORTS_SAML_WEB_SSO, "TRUE");
  270. }
  271. final String isOauthStr = accountMgr.getUserData(account, Constants.KEY_SUPPORTS_OAUTH2);
  272. if (Boolean.parseBoolean(isOauthStr)) {
  273. accountMgr.setUserData(newAccount, Constants.KEY_SUPPORTS_OAUTH2, "TRUE");
  274. }
  275. /* TODO - study if it's possible to run this method in a background thread to copy the authToken
  276. if (isOAuth || isSaml) {
  277. accountMgr.setAuthToken(newAccount, mAuthTokenType, mAuthToken);
  278. }
  279. */
  280. // don't forget the account saved in preferences as the current one
  281. if (currentAccount.name.equals(account.name)) {
  282. AccountUtils.setCurrentOwnCloudAccount(context, newAccountName);
  283. }
  284. // remove the old account
  285. accountMgr.removeAccount(account, null, null);
  286. // will assume it succeeds, not a big deal otherwise
  287. } else {
  288. // servers which base URL is in the root of their domain need no change
  289. Log_OC.d(TAG, account.name + " needs no upgrade ");
  290. newAccount = account;
  291. }
  292. // at least, upgrade account version
  293. Log_OC.d(TAG, "Setting version " + ACCOUNT_VERSION_WITH_PROPER_ID + " to " + newAccountName);
  294. accountMgr.setUserData(newAccount,
  295. Constants.KEY_OC_ACCOUNT_VERSION, Integer.toString(ACCOUNT_VERSION_WITH_PROPER_ID));
  296. }
  297. }
  298. }
  299. }
  300. public static String trimWebdavSuffix(String url) {
  301. while(url.endsWith("/")) {
  302. url = url.substring(0, url.length() - 1);
  303. }
  304. int pos = url.lastIndexOf(WEBDAV_PATH_4_0_AND_LATER);
  305. if (pos >= 0) {
  306. url = url.substring(0, pos);
  307. } else {
  308. pos = url.lastIndexOf(ODAV_PATH);
  309. if (pos >= 0) {
  310. url = url.substring(0, pos);
  311. }
  312. }
  313. return url;
  314. }
  315. /**
  316. * Access the version of the OC server corresponding to an account SAVED IN THE ACCOUNTMANAGER
  317. *
  318. * @param account ownCloud account
  319. * @return Version of the OC server corresponding to account, according to the data saved
  320. * in the system AccountManager
  321. */
  322. public static OwnCloudVersion getServerVersion(Account account) {
  323. OwnCloudVersion serverVersion = null;
  324. if (account != null) {
  325. AccountManager accountMgr = AccountManager.get(MainApp.getAppContext());
  326. String serverVersionStr = accountMgr.getUserData(account, Constants.KEY_OC_VERSION);
  327. if (serverVersionStr != null) {
  328. serverVersion = new OwnCloudVersion(serverVersionStr);
  329. }
  330. }
  331. return serverVersion;
  332. }
  333. public static boolean hasSearchUsersSupport(Account account){
  334. OwnCloudVersion serverVersion = getServerVersion(account);
  335. return (serverVersion != null && serverVersion.isSearchUsersSupported());
  336. }
  337. public static boolean hasSearchSupport(Account account) {
  338. OwnCloudVersion serverVersion = getServerVersion(account);
  339. return (serverVersion != null && serverVersion.isSearchSupported());
  340. }
  341. }