FileStorageUtils.java 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. /**
  2. * ownCloud Android client application
  3. *
  4. * @author David A. Velasco
  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.utils;
  21. import java.io.File;
  22. import java.util.Collections;
  23. import java.util.Comparator;
  24. import java.util.Vector;
  25. import third_parties.daveKoeller.AlphanumComparator;
  26. import com.owncloud.android.MainApp;
  27. import com.owncloud.android.R;
  28. import com.owncloud.android.datamodel.OCFile;
  29. import com.owncloud.android.lib.resources.files.RemoteFile;
  30. import android.accounts.Account;
  31. import android.annotation.SuppressLint;
  32. import android.content.Context;
  33. import android.content.SharedPreferences;
  34. import android.preference.PreferenceManager;
  35. import android.net.Uri;
  36. import android.os.Environment;
  37. import android.os.StatFs;
  38. import android.webkit.MimeTypeMap;
  39. /**
  40. * Static methods to help in access to local file system.
  41. */
  42. public class FileStorageUtils {
  43. public static Integer mSortOrder;
  44. public static Boolean mSortAscending;
  45. public static final Integer SORT_NAME = 0;
  46. public static final Integer SORT_DATE = 1;
  47. public static final Integer SORT_SIZE = 2;
  48. //private static final String LOG_TAG = "FileStorageUtils";
  49. public static final String getSavePath(String accountName) {
  50. File sdCard = Environment.getExternalStorageDirectory();
  51. return sdCard.getAbsolutePath() + "/" + MainApp.getDataFolder() + "/" + Uri.encode(accountName, "@");
  52. // URL encoding is an 'easy fix' to overcome that NTFS and FAT32 don't allow ":" in file names, that can be in the accountName since 0.1.190B
  53. }
  54. public static final String getDefaultSavePathFor(String accountName, OCFile file) {
  55. return getSavePath(accountName) + file.getRemotePath();
  56. }
  57. public static final String getTemporalPath(String accountName) {
  58. File sdCard = Environment.getExternalStorageDirectory();
  59. return sdCard.getAbsolutePath() + "/" + MainApp.getDataFolder() + "/tmp/" + Uri.encode(accountName, "@");
  60. // URL encoding is an 'easy fix' to overcome that NTFS and FAT32 don't allow ":" in file names, that can be in the accountName since 0.1.190B
  61. }
  62. @SuppressLint("NewApi")
  63. public static final long getUsableSpace(String accountName) {
  64. File savePath = Environment.getExternalStorageDirectory();
  65. if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.GINGERBREAD) {
  66. return savePath.getUsableSpace();
  67. } else {
  68. StatFs stats = new StatFs(savePath.getAbsolutePath());
  69. return stats.getAvailableBlocks() * stats.getBlockSize();
  70. }
  71. }
  72. public static final String getLogPath() {
  73. return Environment.getExternalStorageDirectory() + File.separator + MainApp.getDataFolder() + File.separator + "log";
  74. }
  75. public static String getInstantUploadFilePath(Context context, String fileName) {
  76. SharedPreferences pref = PreferenceManager.getDefaultSharedPreferences(context);
  77. String uploadPathdef = context.getString(R.string.instant_upload_path);
  78. String uploadPath = pref.getString("instant_upload_path", uploadPathdef);
  79. String value = uploadPath + OCFile.PATH_SEPARATOR + (fileName == null ? "" : fileName);
  80. return value;
  81. }
  82. /**
  83. * Gets the composed path when video is or must be stored
  84. * @param context
  85. * @param fileName: video file name
  86. * @return String: video file path composed
  87. */
  88. public static String getInstantVideoUploadFilePath(Context context, String fileName) {
  89. SharedPreferences pref = PreferenceManager.getDefaultSharedPreferences(context);
  90. String uploadVideoPathdef = context.getString(R.string.instant_upload_path);
  91. String uploadVideoPath = pref.getString("instant_video_upload_path", uploadVideoPathdef);
  92. String value = uploadVideoPath + OCFile.PATH_SEPARATOR + (fileName == null ? "" : fileName);
  93. return value;
  94. }
  95. public static String getParentPath(String remotePath) {
  96. String parentPath = new File(remotePath).getParent();
  97. parentPath = parentPath.endsWith(OCFile.PATH_SEPARATOR) ? parentPath : parentPath + OCFile.PATH_SEPARATOR;
  98. return parentPath;
  99. }
  100. /**
  101. * Creates and populates a new {@link OCFile} object with the data read from the server.
  102. *
  103. * @param remote remote file read from the server (remote file or folder).
  104. * @return New OCFile instance representing the remote resource described by remote.
  105. */
  106. public static OCFile fillOCFile(RemoteFile remote) {
  107. OCFile file = new OCFile(remote.getRemotePath());
  108. file.setCreationTimestamp(remote.getCreationTimestamp());
  109. file.setFileLength(remote.getLength());
  110. file.setMimetype(remote.getMimeType());
  111. file.setModificationTimestamp(remote.getModifiedTimestamp());
  112. file.setEtag(remote.getEtag());
  113. file.setPermissions(remote.getPermissions());
  114. file.setRemoteId(remote.getRemoteId());
  115. return file;
  116. }
  117. /**
  118. * Creates and populates a new {@link RemoteFile} object with the data read from an {@link OCFile}.
  119. *
  120. * @param ocFile OCFile
  121. * @return New RemoteFile instance representing the resource described by ocFile.
  122. */
  123. public static RemoteFile fillRemoteFile(OCFile ocFile){
  124. RemoteFile file = new RemoteFile(ocFile.getRemotePath());
  125. file.setCreationTimestamp(ocFile.getCreationTimestamp());
  126. file.setLength(ocFile.getFileLength());
  127. file.setMimeType(ocFile.getMimetype());
  128. file.setModifiedTimestamp(ocFile.getModificationTimestamp());
  129. file.setEtag(ocFile.getEtag());
  130. file.setPermissions(ocFile.getPermissions());
  131. file.setRemoteId(ocFile.getRemoteId());
  132. return file;
  133. }
  134. /**
  135. * Sorts all filenames, regarding last user decision
  136. */
  137. public static Vector<OCFile> sortFolder(Vector<OCFile> files){
  138. switch (mSortOrder){
  139. case 0:
  140. files = FileStorageUtils.sortByName(files);
  141. break;
  142. case 1:
  143. files = FileStorageUtils.sortByDate(files);
  144. break;
  145. case 2:
  146. // mFiles = FileStorageUtils.sortBySize(mSortAscending);
  147. break;
  148. }
  149. return files;
  150. }
  151. /**
  152. * Sorts list by Date
  153. * @param files
  154. */
  155. public static Vector<OCFile> sortByDate(Vector<OCFile> files){
  156. final Integer val;
  157. if (mSortAscending){
  158. val = 1;
  159. } else {
  160. val = -1;
  161. }
  162. Collections.sort(files, new Comparator<OCFile>() {
  163. public int compare(OCFile o1, OCFile o2) {
  164. if (o1.isFolder() && o2.isFolder()) {
  165. Long obj1 = o1.getModificationTimestamp();
  166. return val * obj1.compareTo(o2.getModificationTimestamp());
  167. }
  168. else if (o1.isFolder()) {
  169. return -1;
  170. } else if (o2.isFolder()) {
  171. return 1;
  172. } else if (o1.getModificationTimestamp() == 0 || o2.getModificationTimestamp() == 0){
  173. return 0;
  174. } else {
  175. Long obj1 = o1.getModificationTimestamp();
  176. return val * obj1.compareTo(o2.getModificationTimestamp());
  177. }
  178. }
  179. });
  180. return files;
  181. }
  182. // /**
  183. // * Sorts list by Size
  184. // * @param sortAscending true: ascending, false: descending
  185. // */
  186. // public static Vector<OCFile> sortBySize(Vector<OCFile> files){
  187. // final Integer val;
  188. // if (mSortAscending){
  189. // val = 1;
  190. // } else {
  191. // val = -1;
  192. // }
  193. //
  194. // Collections.sort(files, new Comparator<OCFile>() {
  195. // public int compare(OCFile o1, OCFile o2) {
  196. // if (o1.isFolder() && o2.isFolder()) {
  197. // Long obj1 = getFolderSize(new File(FileStorageUtils.getDefaultSavePathFor(mAccount.name, o1)));
  198. // return val * obj1.compareTo(getFolderSize(new File(FileStorageUtils.getDefaultSavePathFor(mAccount.name, o2))));
  199. // }
  200. // else if (o1.isFolder()) {
  201. // return -1;
  202. // } else if (o2.isFolder()) {
  203. // return 1;
  204. // } else if (o1.getFileLength() == 0 || o2.getFileLength() == 0){
  205. // return 0;
  206. // } else {
  207. // Long obj1 = o1.getFileLength();
  208. // return val * obj1.compareTo(o2.getFileLength());
  209. // }
  210. // }
  211. // });
  212. //
  213. // return files;
  214. // }
  215. /**
  216. * Sorts list by Name
  217. * @param files files to sort
  218. */
  219. public static Vector<OCFile> sortByName(Vector<OCFile> files){
  220. final Integer val;
  221. if (mSortAscending){
  222. val = 1;
  223. } else {
  224. val = -1;
  225. }
  226. Collections.sort(files, new Comparator<OCFile>() {
  227. public int compare(OCFile o1, OCFile o2) {
  228. if (o1.isFolder() && o2.isFolder()) {
  229. return val * o1.getRemotePath().toLowerCase().compareTo(o2.getRemotePath().toLowerCase());
  230. } else if (o1.isFolder()) {
  231. return -1;
  232. } else if (o2.isFolder()) {
  233. return 1;
  234. }
  235. return val * new AlphanumComparator().compare(o1, o2);
  236. }
  237. });
  238. return files;
  239. }
  240. /**
  241. * Local Folder size
  242. * @param dir File
  243. * @return Size in bytes
  244. */
  245. public static long getFolderSize(File dir) {
  246. if (dir.exists()) {
  247. long result = 0;
  248. File[] fileList = dir.listFiles();
  249. for(int i = 0; i < fileList.length; i++) {
  250. if(fileList[i].isDirectory()) {
  251. result += getFolderSize(fileList[i]);
  252. } else {
  253. result += fileList[i].length();
  254. }
  255. }
  256. return result;
  257. }
  258. return 0;
  259. }
  260. /**
  261. * Mimetype String of a file
  262. * @param path
  263. * @return
  264. */
  265. public static String getMimeTypeFromName(String path) {
  266. String extension = "";
  267. int pos = path.lastIndexOf('.');
  268. if (pos >= 0) {
  269. extension = path.substring(pos + 1);
  270. }
  271. String result = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension.toLowerCase());
  272. return (result != null) ? result : "";
  273. }
  274. /**
  275. * Scans the default location for saving local copies of files searching for
  276. * a 'lost' file with the same full name as the {@link OCFile} received as
  277. * parameter.
  278. *
  279. * This method helps to keep linked local copies of the files when the app is uninstalled, and then
  280. * reinstalled in the device. OR after the cache of the app was deleted in system settings.
  281. *
  282. * The method is assuming that all the local changes in the file where synchronized in the past. This is dangerous,
  283. * but assuming the contrary could lead to massive unnecessary synchronizations of downloaded file after deleting
  284. * the app cache.
  285. *
  286. * This should be changed in the near future to avoid any chance of data loss, but we need to add some options
  287. * to limit hard automatic synchronizations to wifi, unless the user wants otherwise.
  288. *
  289. * @param file File to associate a possible 'lost' local file.
  290. * @param account Account holding file.
  291. */
  292. public static void searchForLocalFileInDefaultPath(OCFile file, Account account) {
  293. if (file.getStoragePath() == null && !file.isFolder()) {
  294. File f = new File(FileStorageUtils.getDefaultSavePathFor(account.name, file));
  295. if (f.exists()) {
  296. file.setStoragePath(f.getAbsolutePath());
  297. file.setLastSyncDateForData(f.lastModified());
  298. }
  299. }
  300. }
  301. }