FileDataStorageManager.java 67 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610
  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.datamodel;
  21. import java.io.File;
  22. import java.util.ArrayList;
  23. import java.util.Collection;
  24. import java.util.Collections;
  25. import java.util.Iterator;
  26. import java.util.List;
  27. import java.util.Vector;
  28. import android.accounts.Account;
  29. import android.content.ContentProviderClient;
  30. import android.content.ContentProviderOperation;
  31. import android.content.ContentProviderResult;
  32. import android.content.ContentResolver;
  33. import android.content.ContentUris;
  34. import android.content.ContentValues;
  35. import android.content.Intent;
  36. import android.content.OperationApplicationException;
  37. import android.database.Cursor;
  38. import android.net.Uri;
  39. import android.os.RemoteException;
  40. import android.provider.MediaStore;
  41. import com.owncloud.android.MainApp;
  42. import com.owncloud.android.db.ProviderMeta.ProviderTableMeta;
  43. import com.owncloud.android.lib.common.utils.Log_OC;
  44. import com.owncloud.android.lib.resources.files.FileUtils;
  45. import com.owncloud.android.lib.resources.shares.OCShare;
  46. import com.owncloud.android.lib.resources.shares.ShareType;
  47. import com.owncloud.android.utils.FileStorageUtils;
  48. import java.io.FileInputStream;
  49. import java.io.FileOutputStream;
  50. import java.io.IOException;
  51. import java.io.InputStream;
  52. import java.io.OutputStream;
  53. public class FileDataStorageManager {
  54. public static final int ROOT_PARENT_ID = 0;
  55. private ContentResolver mContentResolver;
  56. private ContentProviderClient mContentProviderClient;
  57. private Account mAccount;
  58. private static String TAG = FileDataStorageManager.class.getSimpleName();
  59. public FileDataStorageManager(Account account, ContentResolver cr) {
  60. mContentProviderClient = null;
  61. mContentResolver = cr;
  62. mAccount = account;
  63. }
  64. public FileDataStorageManager(Account account, ContentProviderClient cp) {
  65. mContentProviderClient = cp;
  66. mContentResolver = null;
  67. mAccount = account;
  68. }
  69. public void setAccount(Account account) {
  70. mAccount = account;
  71. }
  72. public Account getAccount() {
  73. return mAccount;
  74. }
  75. public void setContentResolver(ContentResolver cr) {
  76. mContentResolver = cr;
  77. }
  78. public ContentResolver getContentResolver() {
  79. return mContentResolver;
  80. }
  81. public void setContentProviderClient(ContentProviderClient cp) {
  82. mContentProviderClient = cp;
  83. }
  84. public ContentProviderClient getContentProviderClient() {
  85. return mContentProviderClient;
  86. }
  87. public OCFile getFileByPath(String path) {
  88. Cursor c = getCursorForValue(ProviderTableMeta.FILE_PATH, path);
  89. OCFile file = null;
  90. if (c.moveToFirst()) {
  91. file = createFileInstance(c);
  92. }
  93. c.close();
  94. if (file == null && OCFile.ROOT_PATH.equals(path)) {
  95. return createRootDir(); // root should always exist
  96. }
  97. return file;
  98. }
  99. public OCFile getFileById(long id) {
  100. Cursor c = getCursorForValue(ProviderTableMeta._ID, String.valueOf(id));
  101. OCFile file = null;
  102. if (c.moveToFirst()) {
  103. file = createFileInstance(c);
  104. }
  105. c.close();
  106. return file;
  107. }
  108. public OCFile getFileByLocalPath(String path) {
  109. Cursor c = getCursorForValue(ProviderTableMeta.FILE_STORAGE_PATH, path);
  110. OCFile file = null;
  111. if (c.moveToFirst()) {
  112. file = createFileInstance(c);
  113. }
  114. c.close();
  115. return file;
  116. }
  117. public boolean fileExists(long id) {
  118. return fileExists(ProviderTableMeta._ID, String.valueOf(id));
  119. }
  120. public boolean fileExists(String path) {
  121. return fileExists(ProviderTableMeta.FILE_PATH, path);
  122. }
  123. public Vector<OCFile> getFolderContent(OCFile f/*, boolean onlyOnDevice*/) {
  124. if (f != null && f.isFolder() && f.getFileId() != -1) {
  125. // TODO Enable when "On Device" is recovered ?
  126. return getFolderContent(f.getFileId()/*, onlyOnDevice*/);
  127. } else {
  128. return new Vector<OCFile>();
  129. }
  130. }
  131. public Vector<OCFile> getFolderImages(OCFile folder/*, boolean onlyOnDevice*/) {
  132. Vector<OCFile> ret = new Vector<OCFile>();
  133. if (folder != null) {
  134. // TODO better implementation, filtering in the access to database instead of here
  135. // TODO Enable when "On Device" is recovered ?
  136. Vector<OCFile> tmp = getFolderContent(folder/*, onlyOnDevice*/);
  137. OCFile current = null;
  138. for (int i=0; i<tmp.size(); i++) {
  139. current = tmp.get(i);
  140. if (current.isImage()) {
  141. ret.add(current);
  142. }
  143. }
  144. }
  145. return ret;
  146. }
  147. public boolean saveFile(OCFile file) {
  148. boolean overriden = false;
  149. ContentValues cv = new ContentValues();
  150. cv.put(ProviderTableMeta.FILE_MODIFIED, file.getModificationTimestamp());
  151. cv.put(
  152. ProviderTableMeta.FILE_MODIFIED_AT_LAST_SYNC_FOR_DATA,
  153. file.getModificationTimestampAtLastSyncForData()
  154. );
  155. cv.put(ProviderTableMeta.FILE_CREATION, file.getCreationTimestamp());
  156. cv.put(ProviderTableMeta.FILE_CONTENT_LENGTH, file.getFileLength());
  157. cv.put(ProviderTableMeta.FILE_CONTENT_TYPE, file.getMimetype());
  158. cv.put(ProviderTableMeta.FILE_NAME, file.getFileName());
  159. //if (file.getParentId() != DataStorageManager.ROOT_PARENT_ID)
  160. cv.put(ProviderTableMeta.FILE_PARENT, file.getParentId());
  161. cv.put(ProviderTableMeta.FILE_PATH, file.getRemotePath());
  162. if (!file.isFolder())
  163. cv.put(ProviderTableMeta.FILE_STORAGE_PATH, file.getStoragePath());
  164. cv.put(ProviderTableMeta.FILE_ACCOUNT_OWNER, mAccount.name);
  165. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE, file.getLastSyncDateForProperties());
  166. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE_FOR_DATA, file.getLastSyncDateForData());
  167. cv.put(ProviderTableMeta.FILE_KEEP_IN_SYNC, file.isFavorite() ? 1 : 0);
  168. cv.put(ProviderTableMeta.FILE_ETAG, file.getEtag());
  169. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, file.isShareByLink() ? 1 : 0);
  170. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, file.getPublicLink());
  171. cv.put(ProviderTableMeta.FILE_PERMISSIONS, file.getPermissions());
  172. cv.put(ProviderTableMeta.FILE_REMOTE_ID, file.getRemoteId());
  173. cv.put(ProviderTableMeta.FILE_UPDATE_THUMBNAIL, file.needsUpdateThumbnail());
  174. cv.put(ProviderTableMeta.FILE_IS_DOWNLOADING, file.isDownloading());
  175. boolean sameRemotePath = fileExists(file.getRemotePath());
  176. if (sameRemotePath ||
  177. fileExists(file.getFileId()) ) { // for renamed files
  178. OCFile oldFile = null;
  179. if (sameRemotePath) {
  180. oldFile = getFileByPath(file.getRemotePath());
  181. file.setFileId(oldFile.getFileId());
  182. } else {
  183. oldFile = getFileById(file.getFileId());
  184. }
  185. overriden = true;
  186. if (getContentResolver() != null) {
  187. getContentResolver().update(ProviderTableMeta.CONTENT_URI, cv,
  188. ProviderTableMeta._ID + "=?",
  189. new String[]{String.valueOf(file.getFileId())});
  190. } else {
  191. try {
  192. getContentProviderClient().update(ProviderTableMeta.CONTENT_URI,
  193. cv, ProviderTableMeta._ID + "=?",
  194. new String[]{String.valueOf(file.getFileId())});
  195. } catch (RemoteException e) {
  196. Log_OC.e(TAG,
  197. "Fail to insert insert file to database "
  198. + e.getMessage());
  199. }
  200. }
  201. } else {
  202. Uri result_uri = null;
  203. if (getContentResolver() != null) {
  204. result_uri = getContentResolver().insert(
  205. ProviderTableMeta.CONTENT_URI_FILE, cv);
  206. } else {
  207. try {
  208. result_uri = getContentProviderClient().insert(
  209. ProviderTableMeta.CONTENT_URI_FILE, cv);
  210. } catch (RemoteException e) {
  211. Log_OC.e(TAG,
  212. "Fail to insert insert file to database "
  213. + e.getMessage());
  214. }
  215. }
  216. if (result_uri != null) {
  217. long new_id = Long.parseLong(result_uri.getPathSegments()
  218. .get(1));
  219. file.setFileId(new_id);
  220. }
  221. }
  222. // if (file.isFolder()) {
  223. // updateFolderSize(file.getFileId());
  224. // } else {
  225. // updateFolderSize(file.getParentId());
  226. // }
  227. return overriden;
  228. }
  229. /**
  230. * Inserts or updates the list of files contained in a given folder.
  231. * <p/>
  232. * CALLER IS THE RESPONSIBLE FOR GRANTING RIGHT UPDATE OF INFORMATION, NOT THIS METHOD.
  233. * HERE ONLY DATA CONSISTENCY SHOULD BE GRANTED
  234. *
  235. * @param folder
  236. * @param updatedFiles
  237. * @param filesToRemove
  238. */
  239. public void saveFolder(
  240. OCFile folder, Collection<OCFile> updatedFiles, Collection<OCFile> filesToRemove
  241. ) {
  242. Log_OC.d(TAG, "Saving folder " + folder.getRemotePath() + " with " + updatedFiles.size()
  243. + " children and " + filesToRemove.size() + " files to remove");
  244. ArrayList<ContentProviderOperation> operations =
  245. new ArrayList<ContentProviderOperation>(updatedFiles.size());
  246. // prepare operations to insert or update files to save in the given folder
  247. for (OCFile file : updatedFiles) {
  248. ContentValues cv = new ContentValues();
  249. cv.put(ProviderTableMeta.FILE_MODIFIED, file.getModificationTimestamp());
  250. cv.put(
  251. ProviderTableMeta.FILE_MODIFIED_AT_LAST_SYNC_FOR_DATA,
  252. file.getModificationTimestampAtLastSyncForData()
  253. );
  254. cv.put(ProviderTableMeta.FILE_CREATION, file.getCreationTimestamp());
  255. cv.put(ProviderTableMeta.FILE_CONTENT_LENGTH, file.getFileLength());
  256. cv.put(ProviderTableMeta.FILE_CONTENT_TYPE, file.getMimetype());
  257. cv.put(ProviderTableMeta.FILE_NAME, file.getFileName());
  258. //cv.put(ProviderTableMeta.FILE_PARENT, file.getParentId());
  259. cv.put(ProviderTableMeta.FILE_PARENT, folder.getFileId());
  260. cv.put(ProviderTableMeta.FILE_PATH, file.getRemotePath());
  261. if (!file.isFolder()) {
  262. cv.put(ProviderTableMeta.FILE_STORAGE_PATH, file.getStoragePath());
  263. }
  264. cv.put(ProviderTableMeta.FILE_ACCOUNT_OWNER, mAccount.name);
  265. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE, file.getLastSyncDateForProperties());
  266. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE_FOR_DATA, file.getLastSyncDateForData());
  267. cv.put(ProviderTableMeta.FILE_KEEP_IN_SYNC, file.isFavorite() ? 1 : 0);
  268. cv.put(ProviderTableMeta.FILE_ETAG, file.getEtag());
  269. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, file.isShareByLink() ? 1 : 0);
  270. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, file.getPublicLink());
  271. cv.put(ProviderTableMeta.FILE_PERMISSIONS, file.getPermissions());
  272. cv.put(ProviderTableMeta.FILE_REMOTE_ID, file.getRemoteId());
  273. cv.put(ProviderTableMeta.FILE_UPDATE_THUMBNAIL, file.needsUpdateThumbnail());
  274. cv.put(ProviderTableMeta.FILE_IS_DOWNLOADING, file.isDownloading());
  275. boolean existsByPath = fileExists(file.getRemotePath());
  276. if (existsByPath || fileExists(file.getFileId())) {
  277. // updating an existing file
  278. operations.add(ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI).
  279. withValues(cv).
  280. withSelection(ProviderTableMeta._ID + "=?",
  281. new String[]{String.valueOf(file.getFileId())})
  282. .build());
  283. } else {
  284. // adding a new file
  285. operations.add(ContentProviderOperation.newInsert(ProviderTableMeta.CONTENT_URI).
  286. withValues(cv).build());
  287. }
  288. }
  289. // prepare operations to remove files in the given folder
  290. String where = ProviderTableMeta.FILE_ACCOUNT_OWNER + "=?" + " AND " +
  291. ProviderTableMeta.FILE_PATH + "=?";
  292. String [] whereArgs = null;
  293. for (OCFile file : filesToRemove) {
  294. if (file.getParentId() == folder.getFileId()) {
  295. whereArgs = new String[]{mAccount.name, file.getRemotePath()};
  296. //Uri.withAppendedPath(ProviderTableMeta.CONTENT_URI_FILE, "" + file.getFileId());
  297. if (file.isFolder()) {
  298. operations.add(ContentProviderOperation.newDelete(
  299. ContentUris.withAppendedId(
  300. ProviderTableMeta.CONTENT_URI_DIR, file.getFileId()
  301. )
  302. ).withSelection(where, whereArgs).build());
  303. File localFolder =
  304. new File(FileStorageUtils.getDefaultSavePathFor(mAccount.name, file));
  305. if (localFolder.exists()) {
  306. removeLocalFolder(localFolder);
  307. }
  308. } else {
  309. operations.add(ContentProviderOperation.newDelete(
  310. ContentUris.withAppendedId(
  311. ProviderTableMeta.CONTENT_URI_FILE, file.getFileId()
  312. )
  313. ).withSelection(where, whereArgs).build());
  314. if (file.isDown()) {
  315. String path = file.getStoragePath();
  316. new File(path).delete();
  317. triggerMediaScan(path); // notify MediaScanner about removed file
  318. }
  319. }
  320. }
  321. }
  322. // update metadata of folder
  323. ContentValues cv = new ContentValues();
  324. cv.put(ProviderTableMeta.FILE_MODIFIED, folder.getModificationTimestamp());
  325. cv.put(
  326. ProviderTableMeta.FILE_MODIFIED_AT_LAST_SYNC_FOR_DATA,
  327. folder.getModificationTimestampAtLastSyncForData()
  328. );
  329. cv.put(ProviderTableMeta.FILE_CREATION, folder.getCreationTimestamp());
  330. cv.put(ProviderTableMeta.FILE_CONTENT_LENGTH, 0);
  331. cv.put(ProviderTableMeta.FILE_CONTENT_TYPE, folder.getMimetype());
  332. cv.put(ProviderTableMeta.FILE_NAME, folder.getFileName());
  333. cv.put(ProviderTableMeta.FILE_PARENT, folder.getParentId());
  334. cv.put(ProviderTableMeta.FILE_PATH, folder.getRemotePath());
  335. cv.put(ProviderTableMeta.FILE_ACCOUNT_OWNER, mAccount.name);
  336. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE, folder.getLastSyncDateForProperties());
  337. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE_FOR_DATA, folder.getLastSyncDateForData());
  338. cv.put(ProviderTableMeta.FILE_KEEP_IN_SYNC, folder.isFavorite() ? 1 : 0);
  339. cv.put(ProviderTableMeta.FILE_ETAG, folder.getEtag());
  340. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, folder.isShareByLink() ? 1 : 0);
  341. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, folder.getPublicLink());
  342. cv.put(ProviderTableMeta.FILE_PERMISSIONS, folder.getPermissions());
  343. cv.put(ProviderTableMeta.FILE_REMOTE_ID, folder.getRemoteId());
  344. operations.add(ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI).
  345. withValues(cv).
  346. withSelection(ProviderTableMeta._ID + "=?",
  347. new String[]{String.valueOf(folder.getFileId())})
  348. .build());
  349. // apply operations in batch
  350. ContentProviderResult[] results = null;
  351. Log_OC.d(TAG, "Sending " + operations.size() + " operations to FileContentProvider");
  352. try {
  353. if (getContentResolver() != null) {
  354. results = getContentResolver().applyBatch(MainApp.getAuthority(), operations);
  355. } else {
  356. results = getContentProviderClient().applyBatch(operations);
  357. }
  358. } catch (OperationApplicationException e) {
  359. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  360. } catch (RemoteException e) {
  361. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  362. }
  363. // update new id in file objects for insertions
  364. if (results != null) {
  365. long newId;
  366. Iterator<OCFile> filesIt = updatedFiles.iterator();
  367. OCFile file = null;
  368. for (int i = 0; i < results.length; i++) {
  369. if (filesIt.hasNext()) {
  370. file = filesIt.next();
  371. } else {
  372. file = null;
  373. }
  374. if (results[i].uri != null) {
  375. newId = Long.parseLong(results[i].uri.getPathSegments().get(1));
  376. //updatedFiles.get(i).setFileId(newId);
  377. if (file != null) {
  378. file.setFileId(newId);
  379. }
  380. }
  381. }
  382. }
  383. //updateFolderSize(folder.getFileId());
  384. }
  385. // /**
  386. // *
  387. // * @param id
  388. // */
  389. // private void updateFolderSize(long id) {
  390. // if (id > FileDataStorageManager.ROOT_PARENT_ID) {
  391. // Log_OC.d(TAG, "Updating size of " + id);
  392. // if (getContentResolver() != null) {
  393. // getContentResolver().update(ProviderTableMeta.CONTENT_URI_DIR,
  394. // new ContentValues(),
  395. // won't be used, but cannot be null; crashes in KLP
  396. // ProviderTableMeta._ID + "=?",
  397. // new String[] { String.valueOf(id) });
  398. // } else {
  399. // try {
  400. // getContentProviderClient().update(ProviderTableMeta.CONTENT_URI_DIR,
  401. // new ContentValues(),
  402. // won't be used, but cannot be null; crashes in KLP
  403. // ProviderTableMeta._ID + "=?",
  404. // new String[] { String.valueOf(id) });
  405. //
  406. // } catch (RemoteException e) {
  407. // Log_OC.e(
  408. // TAG, "Exception in update of folder size through compatibility patch " + e.getMessage());
  409. // }
  410. // }
  411. // } else {
  412. // Log_OC.e(TAG, "not updating size for folder " + id);
  413. // }
  414. // }
  415. public boolean removeFile(OCFile file, boolean removeDBData, boolean removeLocalCopy) {
  416. boolean success = true;
  417. if (file != null) {
  418. if (file.isFolder()) {
  419. success = removeFolder(file, removeDBData, removeLocalCopy);
  420. } else {
  421. if (removeDBData) {
  422. Uri file_uri = ContentUris.withAppendedId(
  423. ProviderTableMeta.CONTENT_URI_FILE,
  424. file.getFileId()
  425. );
  426. String where = ProviderTableMeta.FILE_ACCOUNT_OWNER + "=?" + " AND " +
  427. ProviderTableMeta.FILE_PATH + "=?";
  428. String [] whereArgs = new String[]{mAccount.name, file.getRemotePath()};
  429. int deleted = 0;
  430. if (getContentProviderClient() != null) {
  431. try {
  432. deleted = getContentProviderClient().delete(file_uri, where, whereArgs);
  433. } catch (RemoteException e) {
  434. e.printStackTrace();
  435. }
  436. } else {
  437. deleted = getContentResolver().delete(file_uri, where, whereArgs);
  438. }
  439. success &= (deleted > 0);
  440. }
  441. String localPath = file.getStoragePath();
  442. if (removeLocalCopy && file.isDown() && localPath != null && success) {
  443. success = new File(localPath).delete();
  444. if (success) {
  445. deleteFileInMediaScan(localPath);
  446. }
  447. if (!removeDBData && success) {
  448. // maybe unnecessary, but should be checked TODO remove if unnecessary
  449. file.setStoragePath(null);
  450. saveFile(file);
  451. }
  452. }
  453. }
  454. }
  455. return success;
  456. }
  457. public boolean removeFolder(OCFile folder, boolean removeDBData, boolean removeLocalContent) {
  458. boolean success = true;
  459. if (folder != null && folder.isFolder()) {
  460. if (removeDBData && folder.getFileId() != -1) {
  461. success = removeFolderInDb(folder);
  462. }
  463. if (removeLocalContent && success) {
  464. success = removeLocalFolder(folder);
  465. }
  466. }
  467. return success;
  468. }
  469. private boolean removeFolderInDb(OCFile folder) {
  470. Uri folder_uri = Uri.withAppendedPath(ProviderTableMeta.CONTENT_URI_DIR, "" +
  471. folder.getFileId()); // URI for recursive deletion
  472. String where = ProviderTableMeta.FILE_ACCOUNT_OWNER + "=?" + " AND " +
  473. ProviderTableMeta.FILE_PATH + "=?";
  474. String [] whereArgs = new String[]{mAccount.name, folder.getRemotePath()};
  475. int deleted = 0;
  476. if (getContentProviderClient() != null) {
  477. try {
  478. deleted = getContentProviderClient().delete(folder_uri, where, whereArgs);
  479. } catch (RemoteException e) {
  480. e.printStackTrace();
  481. }
  482. } else {
  483. deleted = getContentResolver().delete(folder_uri, where, whereArgs);
  484. }
  485. return deleted > 0;
  486. }
  487. private boolean removeLocalFolder(OCFile folder) {
  488. boolean success = true;
  489. String localFolderPath = FileStorageUtils.getDefaultSavePathFor(mAccount.name, folder);
  490. File localFolder = new File(localFolderPath);
  491. if (localFolder.exists()) {
  492. // stage 1: remove the local files already registered in the files database
  493. // TODO Enable when "On Device" is recovered ?
  494. Vector<OCFile> files = getFolderContent(folder.getFileId()/*, false*/);
  495. if (files != null) {
  496. for (OCFile file : files) {
  497. if (file.isFolder()) {
  498. success &= removeLocalFolder(file);
  499. } else {
  500. if (file.isDown()) {
  501. File localFile = new File(file.getStoragePath());
  502. success &= localFile.delete();
  503. if (success) {
  504. // notify MediaScanner about removed file
  505. deleteFileInMediaScan(file.getStoragePath());
  506. file.setStoragePath(null);
  507. saveFile(file);
  508. }
  509. }
  510. }
  511. }
  512. }
  513. // stage 2: remove the folder itself and any local file inside out of sync;
  514. // for instance, after clearing the app cache or reinstalling
  515. success &= removeLocalFolder(localFolder);
  516. }
  517. return success;
  518. }
  519. private boolean removeLocalFolder(File localFolder) {
  520. boolean success = true;
  521. File[] localFiles = localFolder.listFiles();
  522. if (localFiles != null) {
  523. for (File localFile : localFiles) {
  524. if (localFile.isDirectory()) {
  525. success &= removeLocalFolder(localFile);
  526. } else {
  527. String path = localFile.getAbsolutePath();
  528. success &= localFile.delete();
  529. }
  530. }
  531. }
  532. success &= localFolder.delete();
  533. return success;
  534. }
  535. /**
  536. * Updates database and file system for a file or folder that was moved to a different location.
  537. *
  538. * TODO explore better (faster) implementations
  539. * TODO throw exceptions up !
  540. */
  541. public void moveLocalFile(OCFile file, String targetPath, String targetParentPath) {
  542. if (file != null && file.fileExists() && !OCFile.ROOT_PATH.equals(file.getFileName())) {
  543. OCFile targetParent = getFileByPath(targetParentPath);
  544. if (targetParent == null) {
  545. throw new IllegalStateException("Parent folder of the target path does not exist!!");
  546. }
  547. /// 1. get all the descendants of the moved element in a single QUERY
  548. Cursor c = null;
  549. if (getContentProviderClient() != null) {
  550. try {
  551. c = getContentProviderClient().query(
  552. ProviderTableMeta.CONTENT_URI,
  553. null,
  554. ProviderTableMeta.FILE_ACCOUNT_OWNER + "=? AND " +
  555. ProviderTableMeta.FILE_PATH + " LIKE ? ",
  556. new String[]{
  557. mAccount.name,
  558. file.getRemotePath() + "%"
  559. },
  560. ProviderTableMeta.FILE_PATH + " ASC "
  561. );
  562. } catch (RemoteException e) {
  563. Log_OC.e(TAG, e.getMessage());
  564. }
  565. } else {
  566. c = getContentResolver().query(
  567. ProviderTableMeta.CONTENT_URI,
  568. null,
  569. ProviderTableMeta.FILE_ACCOUNT_OWNER + "=? AND " +
  570. ProviderTableMeta.FILE_PATH + " LIKE ? ",
  571. new String[]{
  572. mAccount.name,
  573. file.getRemotePath() + "%"
  574. },
  575. ProviderTableMeta.FILE_PATH + " ASC "
  576. );
  577. }
  578. /// 2. prepare a batch of update operations to change all the descendants
  579. ArrayList<ContentProviderOperation> operations =
  580. new ArrayList<ContentProviderOperation>(c.getCount());
  581. String defaultSavePath = FileStorageUtils.getSavePath(mAccount.name);
  582. List<String> originalPathsToTriggerMediaScan = new ArrayList<String>();
  583. List<String> newPathsToTriggerMediaScan = new ArrayList<String>();
  584. if (c.moveToFirst()) {
  585. int lengthOfOldPath = file.getRemotePath().length();
  586. int lengthOfOldStoragePath = defaultSavePath.length() + lengthOfOldPath;
  587. do {
  588. ContentValues cv = new ContentValues(); // keep construction in the loop
  589. OCFile child = createFileInstance(c);
  590. cv.put(
  591. ProviderTableMeta.FILE_PATH,
  592. targetPath + child.getRemotePath().substring(lengthOfOldPath)
  593. );
  594. if (child.getStoragePath() != null &&
  595. child.getStoragePath().startsWith(defaultSavePath)) {
  596. // update link to downloaded content - but local move is not done here!
  597. String targetLocalPath = defaultSavePath + targetPath +
  598. child.getStoragePath().substring(lengthOfOldStoragePath);
  599. cv.put(ProviderTableMeta.FILE_STORAGE_PATH, targetLocalPath);
  600. originalPathsToTriggerMediaScan.add(child.getStoragePath());
  601. newPathsToTriggerMediaScan.add(targetLocalPath);
  602. }
  603. if (child.getRemotePath().equals(file.getRemotePath())) {
  604. cv.put(
  605. ProviderTableMeta.FILE_PARENT,
  606. targetParent.getFileId()
  607. );
  608. }
  609. operations.add(
  610. ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI).
  611. withValues(cv).
  612. withSelection(
  613. ProviderTableMeta._ID + "=?",
  614. new String[]{String.valueOf(child.getFileId())}
  615. )
  616. .build());
  617. } while (c.moveToNext());
  618. }
  619. c.close();
  620. /// 3. apply updates in batch
  621. try {
  622. if (getContentResolver() != null) {
  623. getContentResolver().applyBatch(MainApp.getAuthority(), operations);
  624. } else {
  625. getContentProviderClient().applyBatch(operations);
  626. }
  627. } catch (Exception e) {
  628. Log_OC.e(TAG, "Fail to update " + file.getFileId() + " and descendants in database", e);
  629. }
  630. /// 4. move in local file system
  631. String originalLocalPath = FileStorageUtils.getDefaultSavePathFor(mAccount.name, file);
  632. String targetLocalPath = defaultSavePath + targetPath;
  633. File localFile = new File(originalLocalPath);
  634. boolean renamed = false;
  635. if (localFile.exists()) {
  636. File targetFile = new File(targetLocalPath);
  637. File targetFolder = targetFile.getParentFile();
  638. if (!targetFolder.exists()) {
  639. targetFolder.mkdirs();
  640. }
  641. renamed = localFile.renameTo(targetFile);
  642. }
  643. if (renamed) {
  644. Iterator<String> it = originalPathsToTriggerMediaScan.iterator();
  645. while (it.hasNext()) {
  646. // Notify MediaScanner about removed file
  647. deleteFileInMediaScan(it.next());
  648. }
  649. it = newPathsToTriggerMediaScan.iterator();
  650. while (it.hasNext()) {
  651. // Notify MediaScanner about new file/folder
  652. triggerMediaScan(it.next());
  653. }
  654. }
  655. }
  656. }
  657. public void copyLocalFile(OCFile file, String targetPath) {
  658. if (file != null && file.fileExists() && !OCFile.ROOT_PATH.equals(file.getFileName())) {
  659. String localPath = FileStorageUtils.getDefaultSavePathFor(mAccount.name, file);
  660. File localFile = new File(localPath);
  661. boolean copied = false;
  662. String defaultSavePath = FileStorageUtils.getSavePath(mAccount.name);
  663. if (localFile.exists()) {
  664. File targetFile = new File(defaultSavePath + targetPath);
  665. File targetFolder = targetFile.getParentFile();
  666. if (!targetFolder.exists()) {
  667. targetFolder.mkdirs();
  668. }
  669. copied = copyFile(localFile, targetFile);
  670. }
  671. Log_OC.d(TAG, "Local file COPIED : " + copied);
  672. }
  673. }
  674. private static boolean copyFile(File src, File target) {
  675. boolean ret = true;
  676. InputStream in = null;
  677. OutputStream out = null;
  678. try {
  679. in = new FileInputStream(src);
  680. out = new FileOutputStream(target);
  681. byte[] buf = new byte[1024];
  682. int len;
  683. while ((len = in.read(buf)) > 0) {
  684. out.write(buf, 0, len);
  685. }
  686. } catch (IOException ex) {
  687. ret = false;
  688. } finally {
  689. if (in != null) try {
  690. in.close();
  691. } catch (IOException e) {
  692. e.printStackTrace(System.err);
  693. }
  694. if (out != null) try {
  695. out.close();
  696. } catch (IOException e) {
  697. e.printStackTrace(System.err);
  698. }
  699. }
  700. return ret;
  701. }
  702. private Vector<OCFile> getFolderContent(long parentId/*, boolean onlyOnDevice*/) {
  703. Vector<OCFile> ret = new Vector<OCFile>();
  704. Uri req_uri = Uri.withAppendedPath(
  705. ProviderTableMeta.CONTENT_URI_DIR,
  706. String.valueOf(parentId));
  707. Cursor c = null;
  708. if (getContentProviderClient() != null) {
  709. try {
  710. c = getContentProviderClient().query(req_uri, null,
  711. ProviderTableMeta.FILE_PARENT + "=?",
  712. new String[]{String.valueOf(parentId)}, null);
  713. } catch (RemoteException e) {
  714. Log_OC.e(TAG, e.getMessage());
  715. return ret;
  716. }
  717. } else {
  718. c = getContentResolver().query(req_uri, null,
  719. ProviderTableMeta.FILE_PARENT + "=?",
  720. new String[]{String.valueOf(parentId)}, null);
  721. }
  722. if (c.moveToFirst()) {
  723. do {
  724. OCFile child = createFileInstance(c);
  725. // TODO Enable when "On Device" is recovered ?
  726. // if (child.isFolder() || !onlyOnDevice || onlyOnDevice && child.isDown()){
  727. ret.add(child);
  728. // }
  729. } while (c.moveToNext());
  730. }
  731. c.close();
  732. Collections.sort(ret);
  733. return ret;
  734. }
  735. private OCFile createRootDir() {
  736. OCFile file = new OCFile(OCFile.ROOT_PATH);
  737. file.setMimetype("DIR");
  738. file.setParentId(FileDataStorageManager.ROOT_PARENT_ID);
  739. saveFile(file);
  740. return file;
  741. }
  742. private boolean fileExists(String cmp_key, String value) {
  743. Cursor c;
  744. if (getContentResolver() != null) {
  745. c = getContentResolver()
  746. .query(ProviderTableMeta.CONTENT_URI,
  747. null,
  748. cmp_key + "=? AND "
  749. + ProviderTableMeta.FILE_ACCOUNT_OWNER
  750. + "=?",
  751. new String[]{value, mAccount.name}, null);
  752. } else {
  753. try {
  754. c = getContentProviderClient().query(
  755. ProviderTableMeta.CONTENT_URI,
  756. null,
  757. cmp_key + "=? AND "
  758. + ProviderTableMeta.FILE_ACCOUNT_OWNER + "=?",
  759. new String[]{value, mAccount.name}, null);
  760. } catch (RemoteException e) {
  761. Log_OC.e(TAG,
  762. "Couldn't determine file existance, assuming non existance: "
  763. + e.getMessage());
  764. return false;
  765. }
  766. }
  767. boolean retval = c.moveToFirst();
  768. c.close();
  769. return retval;
  770. }
  771. private Cursor getCursorForValue(String key, String value) {
  772. Cursor c = null;
  773. if (getContentResolver() != null) {
  774. c = getContentResolver()
  775. .query(ProviderTableMeta.CONTENT_URI,
  776. null,
  777. key + "=? AND "
  778. + ProviderTableMeta.FILE_ACCOUNT_OWNER
  779. + "=?",
  780. new String[]{value, mAccount.name}, null);
  781. } else {
  782. try {
  783. c = getContentProviderClient().query(
  784. ProviderTableMeta.CONTENT_URI,
  785. null,
  786. key + "=? AND " + ProviderTableMeta.FILE_ACCOUNT_OWNER
  787. + "=?", new String[]{value, mAccount.name},
  788. null);
  789. } catch (RemoteException e) {
  790. Log_OC.e(TAG, "Could not get file details: " + e.getMessage());
  791. c = null;
  792. }
  793. }
  794. return c;
  795. }
  796. private OCFile createFileInstance(Cursor c) {
  797. OCFile file = null;
  798. if (c != null) {
  799. file = new OCFile(c.getString(c
  800. .getColumnIndex(ProviderTableMeta.FILE_PATH)));
  801. file.setFileId(c.getLong(c.getColumnIndex(ProviderTableMeta._ID)));
  802. file.setParentId(c.getLong(c
  803. .getColumnIndex(ProviderTableMeta.FILE_PARENT)));
  804. file.setMimetype(c.getString(c
  805. .getColumnIndex(ProviderTableMeta.FILE_CONTENT_TYPE)));
  806. if (!file.isFolder()) {
  807. file.setStoragePath(c.getString(c
  808. .getColumnIndex(ProviderTableMeta.FILE_STORAGE_PATH)));
  809. if (file.getStoragePath() == null) {
  810. // try to find existing file and bind it with current account;
  811. // with the current update of SynchronizeFolderOperation, this won't be
  812. // necessary anymore after a full synchronization of the account
  813. File f = new File(FileStorageUtils.getDefaultSavePathFor(mAccount.name, file));
  814. if (f.exists()) {
  815. file.setStoragePath(f.getAbsolutePath());
  816. file.setLastSyncDateForData(f.lastModified());
  817. }
  818. }
  819. }
  820. file.setFileLength(c.getLong(c
  821. .getColumnIndex(ProviderTableMeta.FILE_CONTENT_LENGTH)));
  822. file.setCreationTimestamp(c.getLong(c
  823. .getColumnIndex(ProviderTableMeta.FILE_CREATION)));
  824. file.setModificationTimestamp(c.getLong(c
  825. .getColumnIndex(ProviderTableMeta.FILE_MODIFIED)));
  826. file.setModificationTimestampAtLastSyncForData(c.getLong(c
  827. .getColumnIndex(ProviderTableMeta.FILE_MODIFIED_AT_LAST_SYNC_FOR_DATA)));
  828. file.setLastSyncDateForProperties(c.getLong(c
  829. .getColumnIndex(ProviderTableMeta.FILE_LAST_SYNC_DATE)));
  830. file.setLastSyncDateForData(c.getLong(c.
  831. getColumnIndex(ProviderTableMeta.FILE_LAST_SYNC_DATE_FOR_DATA)));
  832. file.setFavorite(c.getInt(
  833. c.getColumnIndex(ProviderTableMeta.FILE_KEEP_IN_SYNC)) == 1 ? true : false);
  834. file.setEtag(c.getString(c.getColumnIndex(ProviderTableMeta.FILE_ETAG)));
  835. file.setShareByLink(c.getInt(
  836. c.getColumnIndex(ProviderTableMeta.FILE_SHARE_BY_LINK)) == 1 ? true : false);
  837. file.setPublicLink(c.getString(c.getColumnIndex(ProviderTableMeta.FILE_PUBLIC_LINK)));
  838. file.setPermissions(c.getString(c.getColumnIndex(ProviderTableMeta.FILE_PERMISSIONS)));
  839. file.setRemoteId(c.getString(c.getColumnIndex(ProviderTableMeta.FILE_REMOTE_ID)));
  840. file.setNeedsUpdateThumbnail(c.getInt(
  841. c.getColumnIndex(ProviderTableMeta.FILE_UPDATE_THUMBNAIL)) == 1 ? true : false);
  842. file.setDownloading(c.getInt(
  843. c.getColumnIndex(ProviderTableMeta.FILE_IS_DOWNLOADING)) == 1 ? true : false);
  844. }
  845. return file;
  846. }
  847. /**
  848. * Returns if the file/folder is shared by link or not
  849. *
  850. * @param path Path of the file/folder
  851. * @return
  852. */
  853. public boolean isShareByLink(String path) {
  854. Cursor c = getCursorForValue(ProviderTableMeta.FILE_STORAGE_PATH, path);
  855. OCFile file = null;
  856. if (c.moveToFirst()) {
  857. file = createFileInstance(c);
  858. }
  859. c.close();
  860. return file.isShareByLink();
  861. }
  862. /**
  863. * Returns the public link of the file/folder
  864. *
  865. * @param path Path of the file/folder
  866. * @return
  867. */
  868. public String getPublicLink(String path) {
  869. Cursor c = getCursorForValue(ProviderTableMeta.FILE_STORAGE_PATH, path);
  870. OCFile file = null;
  871. if (c.moveToFirst()) {
  872. file = createFileInstance(c);
  873. }
  874. c.close();
  875. return file.getPublicLink();
  876. }
  877. // Methods for Shares
  878. public boolean saveShare(OCShare share) {
  879. boolean overriden = false;
  880. ContentValues cv = new ContentValues();
  881. cv.put(ProviderTableMeta.OCSHARES_FILE_SOURCE, share.getFileSource());
  882. cv.put(ProviderTableMeta.OCSHARES_ITEM_SOURCE, share.getItemSource());
  883. cv.put(ProviderTableMeta.OCSHARES_SHARE_TYPE, share.getShareType().getValue());
  884. cv.put(ProviderTableMeta.OCSHARES_SHARE_WITH, share.getShareWith());
  885. cv.put(ProviderTableMeta.OCSHARES_PATH, share.getPath());
  886. cv.put(ProviderTableMeta.OCSHARES_PERMISSIONS, share.getPermissions());
  887. cv.put(ProviderTableMeta.OCSHARES_SHARED_DATE, share.getSharedDate());
  888. cv.put(ProviderTableMeta.OCSHARES_EXPIRATION_DATE, share.getExpirationDate());
  889. cv.put(ProviderTableMeta.OCSHARES_TOKEN, share.getToken());
  890. cv.put(
  891. ProviderTableMeta.OCSHARES_SHARE_WITH_DISPLAY_NAME,
  892. share.getSharedWithDisplayName()
  893. );
  894. cv.put(ProviderTableMeta.OCSHARES_IS_DIRECTORY, share.isFolder() ? 1 : 0);
  895. cv.put(ProviderTableMeta.OCSHARES_USER_ID, share.getUserId());
  896. cv.put(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED, share.getIdRemoteShared());
  897. cv.put(ProviderTableMeta.OCSHARES_ACCOUNT_OWNER, mAccount.name);
  898. if (shareExists(share.getIdRemoteShared())) { // for renamed files
  899. overriden = true;
  900. if (getContentResolver() != null) {
  901. getContentResolver().update(ProviderTableMeta.CONTENT_URI_SHARE, cv,
  902. ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED + "=?",
  903. new String[]{String.valueOf(share.getIdRemoteShared())});
  904. } else {
  905. try {
  906. getContentProviderClient().update(ProviderTableMeta.CONTENT_URI_SHARE,
  907. cv, ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED + "=?",
  908. new String[]{String.valueOf(share.getIdRemoteShared())});
  909. } catch (RemoteException e) {
  910. Log_OC.e(TAG,
  911. "Fail to insert insert file to database "
  912. + e.getMessage());
  913. }
  914. }
  915. } else {
  916. Uri result_uri = null;
  917. if (getContentResolver() != null) {
  918. result_uri = getContentResolver().insert(
  919. ProviderTableMeta.CONTENT_URI_SHARE, cv);
  920. } else {
  921. try {
  922. result_uri = getContentProviderClient().insert(
  923. ProviderTableMeta.CONTENT_URI_SHARE, cv);
  924. } catch (RemoteException e) {
  925. Log_OC.e(TAG,
  926. "Fail to insert insert file to database "
  927. + e.getMessage());
  928. }
  929. }
  930. if (result_uri != null) {
  931. long new_id = Long.parseLong(result_uri.getPathSegments()
  932. .get(1));
  933. share.setId(new_id);
  934. }
  935. }
  936. return overriden;
  937. }
  938. public OCShare getFirstShareByPathAndType(String path, ShareType type) {
  939. Cursor c = null;
  940. if (getContentResolver() != null) {
  941. c = getContentResolver().query(
  942. ProviderTableMeta.CONTENT_URI_SHARE,
  943. null,
  944. ProviderTableMeta.OCSHARES_PATH + "=? AND "
  945. + ProviderTableMeta.OCSHARES_SHARE_TYPE + "=? AND "
  946. + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?",
  947. new String[]{path, Integer.toString(type.getValue()), mAccount.name},
  948. null);
  949. } else {
  950. try {
  951. c = getContentProviderClient().query(
  952. ProviderTableMeta.CONTENT_URI_SHARE,
  953. null,
  954. ProviderTableMeta.OCSHARES_PATH + "=? AND "
  955. + ProviderTableMeta.OCSHARES_SHARE_TYPE + "=? AND "
  956. + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?",
  957. new String[]{path, Integer.toString(type.getValue()), mAccount.name},
  958. null);
  959. } catch (RemoteException e) {
  960. Log_OC.e(TAG, "Could not get file details: " + e.getMessage());
  961. c = null;
  962. }
  963. }
  964. OCShare share = null;
  965. if (c.moveToFirst()) {
  966. share = createShareInstance(c);
  967. }
  968. c.close();
  969. return share;
  970. }
  971. private OCShare createShareInstance(Cursor c) {
  972. OCShare share = null;
  973. if (c != null) {
  974. share = new OCShare(c.getString(c
  975. .getColumnIndex(ProviderTableMeta.OCSHARES_PATH)));
  976. share.setId(c.getLong(c.getColumnIndex(ProviderTableMeta._ID)));
  977. share.setFileSource(c.getLong(c
  978. .getColumnIndex(ProviderTableMeta.OCSHARES_ITEM_SOURCE)));
  979. share.setShareType(ShareType.fromValue(c.getInt(c
  980. .getColumnIndex(ProviderTableMeta.OCSHARES_SHARE_TYPE))));
  981. share.setPermissions(c.getInt(c
  982. .getColumnIndex(ProviderTableMeta.OCSHARES_PERMISSIONS)));
  983. share.setSharedDate(c.getLong(c
  984. .getColumnIndex(ProviderTableMeta.OCSHARES_SHARED_DATE)));
  985. share.setExpirationDate(c.getLong(c
  986. .getColumnIndex(ProviderTableMeta.OCSHARES_EXPIRATION_DATE)));
  987. share.setToken(c.getString(c
  988. .getColumnIndex(ProviderTableMeta.OCSHARES_TOKEN)));
  989. share.setSharedWithDisplayName(c.getString(c
  990. .getColumnIndex(ProviderTableMeta.OCSHARES_SHARE_WITH_DISPLAY_NAME)));
  991. share.setIsFolder(c.getInt(
  992. c.getColumnIndex(ProviderTableMeta.OCSHARES_IS_DIRECTORY)) == 1 ? true : false);
  993. share.setUserId(c.getLong(c.getColumnIndex(ProviderTableMeta.OCSHARES_USER_ID)));
  994. share.setIdRemoteShared(
  995. c.getLong(c.getColumnIndex(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED))
  996. );
  997. }
  998. return share;
  999. }
  1000. private boolean shareExists(String cmp_key, String value) {
  1001. Cursor c;
  1002. if (getContentResolver() != null) {
  1003. c = getContentResolver()
  1004. .query(ProviderTableMeta.CONTENT_URI_SHARE,
  1005. null,
  1006. cmp_key + "=? AND "
  1007. + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER
  1008. + "=?",
  1009. new String[]{value, mAccount.name}, null);
  1010. } else {
  1011. try {
  1012. c = getContentProviderClient().query(
  1013. ProviderTableMeta.CONTENT_URI_SHARE,
  1014. null,
  1015. cmp_key + "=? AND "
  1016. + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?",
  1017. new String[]{value, mAccount.name}, null);
  1018. } catch (RemoteException e) {
  1019. Log_OC.e(TAG,
  1020. "Couldn't determine file existance, assuming non existance: "
  1021. + e.getMessage());
  1022. return false;
  1023. }
  1024. }
  1025. boolean retval = c.moveToFirst();
  1026. c.close();
  1027. return retval;
  1028. }
  1029. private boolean shareExists(long remoteId) {
  1030. return shareExists(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED, String.valueOf(remoteId));
  1031. }
  1032. private void cleanSharedFiles() {
  1033. ContentValues cv = new ContentValues();
  1034. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, false);
  1035. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, "");
  1036. String where = ProviderTableMeta.FILE_ACCOUNT_OWNER + "=?";
  1037. String[] whereArgs = new String[]{mAccount.name};
  1038. if (getContentResolver() != null) {
  1039. getContentResolver().update(ProviderTableMeta.CONTENT_URI, cv, where, whereArgs);
  1040. } else {
  1041. try {
  1042. getContentProviderClient().update(
  1043. ProviderTableMeta.CONTENT_URI, cv, where, whereArgs
  1044. );
  1045. } catch (RemoteException e) {
  1046. Log_OC.e(TAG, "Exception in cleanSharedFiles" + e.getMessage());
  1047. }
  1048. }
  1049. }
  1050. private void cleanSharedFilesInFolder(OCFile folder) {
  1051. ContentValues cv = new ContentValues();
  1052. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, false);
  1053. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, "");
  1054. String where = ProviderTableMeta.FILE_ACCOUNT_OWNER + "=? AND " +
  1055. ProviderTableMeta.FILE_PARENT + "=?";
  1056. String [] whereArgs = new String[] { mAccount.name , String.valueOf(folder.getFileId()) };
  1057. if (getContentResolver() != null) {
  1058. getContentResolver().update(ProviderTableMeta.CONTENT_URI, cv, where, whereArgs);
  1059. } else {
  1060. try {
  1061. getContentProviderClient().update(
  1062. ProviderTableMeta.CONTENT_URI, cv, where, whereArgs
  1063. );
  1064. } catch (RemoteException e) {
  1065. Log_OC.e(TAG, "Exception in cleanSharedFilesInFolder " + e.getMessage());
  1066. }
  1067. }
  1068. }
  1069. private void cleanShares() {
  1070. String where = ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?";
  1071. String[] whereArgs = new String[]{mAccount.name};
  1072. if (getContentResolver() != null) {
  1073. getContentResolver().delete(ProviderTableMeta.CONTENT_URI_SHARE, where, whereArgs);
  1074. } else {
  1075. try {
  1076. getContentProviderClient().delete(
  1077. ProviderTableMeta.CONTENT_URI_SHARE, where, whereArgs
  1078. );
  1079. } catch (RemoteException e) {
  1080. Log_OC.e(TAG, "Exception in cleanShares" + e.getMessage());
  1081. }
  1082. }
  1083. }
  1084. public void saveShares(Collection<OCShare> shares) {
  1085. cleanShares();
  1086. if (shares != null) {
  1087. ArrayList<ContentProviderOperation> operations =
  1088. new ArrayList<ContentProviderOperation>(shares.size());
  1089. // prepare operations to insert or update files to save in the given folder
  1090. for (OCShare share : shares) {
  1091. ContentValues cv = new ContentValues();
  1092. cv.put(ProviderTableMeta.OCSHARES_FILE_SOURCE, share.getFileSource());
  1093. cv.put(ProviderTableMeta.OCSHARES_ITEM_SOURCE, share.getItemSource());
  1094. cv.put(ProviderTableMeta.OCSHARES_SHARE_TYPE, share.getShareType().getValue());
  1095. cv.put(ProviderTableMeta.OCSHARES_SHARE_WITH, share.getShareWith());
  1096. cv.put(ProviderTableMeta.OCSHARES_PATH, share.getPath());
  1097. cv.put(ProviderTableMeta.OCSHARES_PERMISSIONS, share.getPermissions());
  1098. cv.put(ProviderTableMeta.OCSHARES_SHARED_DATE, share.getSharedDate());
  1099. cv.put(ProviderTableMeta.OCSHARES_EXPIRATION_DATE, share.getExpirationDate());
  1100. cv.put(ProviderTableMeta.OCSHARES_TOKEN, share.getToken());
  1101. cv.put(
  1102. ProviderTableMeta.OCSHARES_SHARE_WITH_DISPLAY_NAME,
  1103. share.getSharedWithDisplayName()
  1104. );
  1105. cv.put(ProviderTableMeta.OCSHARES_IS_DIRECTORY, share.isFolder() ? 1 : 0);
  1106. cv.put(ProviderTableMeta.OCSHARES_USER_ID, share.getUserId());
  1107. cv.put(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED, share.getIdRemoteShared());
  1108. cv.put(ProviderTableMeta.OCSHARES_ACCOUNT_OWNER, mAccount.name);
  1109. if (shareExists(share.getIdRemoteShared())) {
  1110. // updating an existing file
  1111. operations.add(
  1112. ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI_SHARE).
  1113. withValues(cv).
  1114. withSelection(
  1115. ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED + "=?",
  1116. new String[] { String.valueOf(share.getIdRemoteShared()) }
  1117. ).
  1118. build()
  1119. );
  1120. } else {
  1121. // adding a new file
  1122. operations.add(
  1123. ContentProviderOperation.newInsert(ProviderTableMeta.CONTENT_URI_SHARE).
  1124. withValues(cv).
  1125. build()
  1126. );
  1127. }
  1128. }
  1129. // apply operations in batch
  1130. if (operations.size() > 0) {
  1131. @SuppressWarnings("unused")
  1132. ContentProviderResult[] results = null;
  1133. Log_OC.d(TAG, "Sending " + operations.size() +
  1134. " operations to FileContentProvider");
  1135. try {
  1136. if (getContentResolver() != null) {
  1137. results = getContentResolver().applyBatch(
  1138. MainApp.getAuthority(), operations
  1139. );
  1140. } else {
  1141. results = getContentProviderClient().applyBatch(operations);
  1142. }
  1143. } catch (OperationApplicationException e) {
  1144. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1145. } catch (RemoteException e) {
  1146. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1147. }
  1148. }
  1149. }
  1150. }
  1151. public void updateSharedFiles(Collection<OCFile> sharedFiles) {
  1152. cleanSharedFiles();
  1153. if (sharedFiles != null) {
  1154. ArrayList<ContentProviderOperation> operations =
  1155. new ArrayList<ContentProviderOperation>(sharedFiles.size());
  1156. // prepare operations to insert or update files to save in the given folder
  1157. for (OCFile file : sharedFiles) {
  1158. ContentValues cv = new ContentValues();
  1159. cv.put(ProviderTableMeta.FILE_MODIFIED, file.getModificationTimestamp());
  1160. cv.put(
  1161. ProviderTableMeta.FILE_MODIFIED_AT_LAST_SYNC_FOR_DATA,
  1162. file.getModificationTimestampAtLastSyncForData()
  1163. );
  1164. cv.put(ProviderTableMeta.FILE_CREATION, file.getCreationTimestamp());
  1165. cv.put(ProviderTableMeta.FILE_CONTENT_LENGTH, file.getFileLength());
  1166. cv.put(ProviderTableMeta.FILE_CONTENT_TYPE, file.getMimetype());
  1167. cv.put(ProviderTableMeta.FILE_NAME, file.getFileName());
  1168. cv.put(ProviderTableMeta.FILE_PARENT, file.getParentId());
  1169. cv.put(ProviderTableMeta.FILE_PATH, file.getRemotePath());
  1170. if (!file.isFolder()) {
  1171. cv.put(ProviderTableMeta.FILE_STORAGE_PATH, file.getStoragePath());
  1172. }
  1173. cv.put(ProviderTableMeta.FILE_ACCOUNT_OWNER, mAccount.name);
  1174. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE, file.getLastSyncDateForProperties());
  1175. cv.put(
  1176. ProviderTableMeta.FILE_LAST_SYNC_DATE_FOR_DATA,
  1177. file.getLastSyncDateForData()
  1178. );
  1179. cv.put(ProviderTableMeta.FILE_KEEP_IN_SYNC, file.isFavorite() ? 1 : 0);
  1180. cv.put(ProviderTableMeta.FILE_ETAG, file.getEtag());
  1181. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, file.isShareByLink() ? 1 : 0);
  1182. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, file.getPublicLink());
  1183. cv.put(ProviderTableMeta.FILE_PERMISSIONS, file.getPermissions());
  1184. cv.put(ProviderTableMeta.FILE_REMOTE_ID, file.getRemoteId());
  1185. cv.put(
  1186. ProviderTableMeta.FILE_UPDATE_THUMBNAIL,
  1187. file.needsUpdateThumbnail() ? 1 : 0
  1188. );
  1189. cv.put(
  1190. ProviderTableMeta.FILE_IS_DOWNLOADING,
  1191. file.isDownloading() ? 1 : 0
  1192. );
  1193. boolean existsByPath = fileExists(file.getRemotePath());
  1194. if (existsByPath || fileExists(file.getFileId())) {
  1195. // updating an existing file
  1196. operations.add(
  1197. ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI).
  1198. withValues(cv).
  1199. withSelection(
  1200. ProviderTableMeta._ID + "=?",
  1201. new String[] { String.valueOf(file.getFileId()) }
  1202. ).build()
  1203. );
  1204. } else {
  1205. // adding a new file
  1206. operations.add(
  1207. ContentProviderOperation.newInsert(ProviderTableMeta.CONTENT_URI).
  1208. withValues(cv).
  1209. build()
  1210. );
  1211. }
  1212. }
  1213. // apply operations in batch
  1214. if (operations.size() > 0) {
  1215. @SuppressWarnings("unused")
  1216. ContentProviderResult[] results = null;
  1217. Log_OC.d(TAG, "Sending " + operations.size() +
  1218. " operations to FileContentProvider");
  1219. try {
  1220. if (getContentResolver() != null) {
  1221. results = getContentResolver().applyBatch(
  1222. MainApp.getAuthority(), operations
  1223. );
  1224. } else {
  1225. results = getContentProviderClient().applyBatch(operations);
  1226. }
  1227. } catch (OperationApplicationException e) {
  1228. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1229. } catch (RemoteException e) {
  1230. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1231. }
  1232. }
  1233. }
  1234. }
  1235. public void removeShare(OCShare share) {
  1236. Uri share_uri = ProviderTableMeta.CONTENT_URI_SHARE;
  1237. String where = ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?" + " AND " +
  1238. ProviderTableMeta.FILE_PATH + "=?";
  1239. String [] whereArgs = new String[]{mAccount.name, share.getPath()};
  1240. if (getContentProviderClient() != null) {
  1241. try {
  1242. getContentProviderClient().delete(share_uri, where, whereArgs);
  1243. } catch (RemoteException e) {
  1244. e.printStackTrace();
  1245. }
  1246. } else {
  1247. getContentResolver().delete(share_uri, where, whereArgs);
  1248. }
  1249. }
  1250. public void saveSharesDB(ArrayList<OCShare> shares) {
  1251. saveShares(shares);
  1252. ArrayList<OCFile> sharedFiles = new ArrayList<OCFile>();
  1253. for (OCShare share : shares) {
  1254. // Get the path
  1255. String path = share.getPath();
  1256. if (share.isFolder()) {
  1257. path = path + FileUtils.PATH_SEPARATOR;
  1258. }
  1259. // Update OCFile with data from share: ShareByLink and publicLink
  1260. OCFile file = getFileByPath(path);
  1261. if (file != null) {
  1262. if (share.getShareType().equals(ShareType.PUBLIC_LINK)) {
  1263. file.setShareByLink(true);
  1264. sharedFiles.add(file);
  1265. }
  1266. }
  1267. }
  1268. updateSharedFiles(sharedFiles);
  1269. }
  1270. public void saveSharesInFolder(ArrayList<OCShare> shares, OCFile folder) {
  1271. cleanSharedFilesInFolder(folder);
  1272. ArrayList<ContentProviderOperation> operations = new ArrayList<ContentProviderOperation>();
  1273. operations = prepareRemoveSharesInFolder(folder, operations);
  1274. if (shares != null) {
  1275. // prepare operations to insert or update files to save in the given folder
  1276. for (OCShare share : shares) {
  1277. ContentValues cv = new ContentValues();
  1278. cv.put(ProviderTableMeta.OCSHARES_FILE_SOURCE, share.getFileSource());
  1279. cv.put(ProviderTableMeta.OCSHARES_ITEM_SOURCE, share.getItemSource());
  1280. cv.put(ProviderTableMeta.OCSHARES_SHARE_TYPE, share.getShareType().getValue());
  1281. cv.put(ProviderTableMeta.OCSHARES_SHARE_WITH, share.getShareWith());
  1282. cv.put(ProviderTableMeta.OCSHARES_PATH, share.getPath());
  1283. cv.put(ProviderTableMeta.OCSHARES_PERMISSIONS, share.getPermissions());
  1284. cv.put(ProviderTableMeta.OCSHARES_SHARED_DATE, share.getSharedDate());
  1285. cv.put(ProviderTableMeta.OCSHARES_EXPIRATION_DATE, share.getExpirationDate());
  1286. cv.put(ProviderTableMeta.OCSHARES_TOKEN, share.getToken());
  1287. cv.put(
  1288. ProviderTableMeta.OCSHARES_SHARE_WITH_DISPLAY_NAME,
  1289. share.getSharedWithDisplayName()
  1290. );
  1291. cv.put(ProviderTableMeta.OCSHARES_IS_DIRECTORY, share.isFolder() ? 1 : 0);
  1292. cv.put(ProviderTableMeta.OCSHARES_USER_ID, share.getUserId());
  1293. cv.put(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED, share.getIdRemoteShared());
  1294. cv.put(ProviderTableMeta.OCSHARES_ACCOUNT_OWNER, mAccount.name);
  1295. /*
  1296. if (shareExists(share.getIdRemoteShared())) {
  1297. // updating an existing share resource
  1298. operations.add(
  1299. ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI_SHARE).
  1300. withValues(cv).
  1301. withSelection( ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED + "=?",
  1302. new String[] { String.valueOf(share.getIdRemoteShared()) })
  1303. .build());
  1304. } else {
  1305. */
  1306. // adding a new share resource
  1307. operations.add(
  1308. ContentProviderOperation.newInsert(ProviderTableMeta.CONTENT_URI_SHARE).
  1309. withValues(cv).
  1310. build()
  1311. );
  1312. //}
  1313. }
  1314. }
  1315. // apply operations in batch
  1316. if (operations.size() > 0) {
  1317. @SuppressWarnings("unused")
  1318. ContentProviderResult[] results = null;
  1319. Log_OC.d(TAG, "Sending " + operations.size() + " operations to FileContentProvider");
  1320. try {
  1321. if (getContentResolver() != null) {
  1322. results = getContentResolver().applyBatch(MainApp.getAuthority(), operations);
  1323. } else {
  1324. results = getContentProviderClient().applyBatch(operations);
  1325. }
  1326. } catch (OperationApplicationException e) {
  1327. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1328. } catch (RemoteException e) {
  1329. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1330. }
  1331. }
  1332. //}
  1333. }
  1334. private ArrayList<ContentProviderOperation> prepareRemoveSharesInFolder(
  1335. OCFile folder, ArrayList<ContentProviderOperation> preparedOperations) {
  1336. if (folder != null) {
  1337. String where = ProviderTableMeta.OCSHARES_PATH + "=?" + " AND "
  1338. + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?";
  1339. String [] whereArgs = new String[]{ "", mAccount.name };
  1340. // TODO Enable when "On Device" is recovered ?
  1341. Vector<OCFile> files = getFolderContent(folder /*, false*/);
  1342. for (OCFile file : files) {
  1343. whereArgs[0] = file.getRemotePath();
  1344. preparedOperations.add(
  1345. ContentProviderOperation.newDelete(ProviderTableMeta.CONTENT_URI_SHARE).
  1346. withSelection(where, whereArgs).
  1347. build()
  1348. );
  1349. }
  1350. }
  1351. return preparedOperations;
  1352. /*
  1353. if (operations.size() > 0) {
  1354. try {
  1355. if (getContentResolver() != null) {
  1356. getContentResolver().applyBatch(MainApp.getAuthority(), operations);
  1357. } else {
  1358. getContentProviderClient().applyBatch(operations);
  1359. }
  1360. } catch (OperationApplicationException e) {
  1361. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1362. } catch (RemoteException e) {
  1363. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1364. }
  1365. }
  1366. */
  1367. /*
  1368. if (getContentResolver() != null) {
  1369. getContentResolver().delete(ProviderTableMeta.CONTENT_URI_SHARE,
  1370. where,
  1371. whereArgs);
  1372. } else {
  1373. try {
  1374. getContentProviderClient().delete( ProviderTableMeta.CONTENT_URI_SHARE,
  1375. where,
  1376. whereArgs);
  1377. } catch (RemoteException e) {
  1378. Log_OC.e(TAG, "Exception deleting shares in a folder " + e.getMessage());
  1379. }
  1380. }
  1381. */
  1382. //}
  1383. }
  1384. public void triggerMediaScan(String path) {
  1385. Intent intent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
  1386. intent.setData(Uri.fromFile(new File(path)));
  1387. MainApp.getAppContext().sendBroadcast(intent);
  1388. }
  1389. public void deleteFileInMediaScan(String path) {
  1390. String mimetypeString = FileStorageUtils.getMimeTypeFromName(path);
  1391. ContentResolver contentResolver = getContentResolver();
  1392. if (contentResolver != null) {
  1393. if (mimetypeString.startsWith("image/")) {
  1394. // Images
  1395. contentResolver.delete(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
  1396. MediaStore.Images.Media.DATA + "=?", new String[]{path});
  1397. } else if (mimetypeString.startsWith("audio/")) {
  1398. // Audio
  1399. contentResolver.delete(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI,
  1400. MediaStore.Audio.Media.DATA + "=?", new String[]{path});
  1401. } else if (mimetypeString.startsWith("video/")) {
  1402. // Video
  1403. contentResolver.delete(MediaStore.Video.Media.EXTERNAL_CONTENT_URI,
  1404. MediaStore.Video.Media.DATA + "=?", new String[]{path});
  1405. }
  1406. } else {
  1407. ContentProviderClient contentProviderClient = getContentProviderClient();
  1408. try {
  1409. if (mimetypeString.startsWith("image/")) {
  1410. // Images
  1411. contentProviderClient.delete(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
  1412. MediaStore.Images.Media.DATA + "=?", new String[]{path});
  1413. } else if (mimetypeString.startsWith("audio/")) {
  1414. // Audio
  1415. contentProviderClient.delete(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI,
  1416. MediaStore.Audio.Media.DATA + "=?", new String[]{path});
  1417. } else if (mimetypeString.startsWith("video/")) {
  1418. // Video
  1419. contentProviderClient.delete(MediaStore.Video.Media.EXTERNAL_CONTENT_URI,
  1420. MediaStore.Video.Media.DATA + "=?", new String[]{path});
  1421. }
  1422. } catch (RemoteException e) {
  1423. Log_OC.e(TAG, "Exception deleting media file in MediaStore " + e.getMessage());
  1424. }
  1425. }
  1426. }
  1427. }