FileDataStorageManager.java 58 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328
  1. /* ownCloud Android client application
  2. * Copyright (C) 2012 Bartek Przybylski
  3. * Copyright (C) 2012-2013 ownCloud Inc.
  4. *
  5. * This program is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License version 2,
  7. * as published by the Free Software Foundation.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. *
  17. */
  18. package com.owncloud.android.datamodel;
  19. import java.io.File;
  20. import java.util.ArrayList;
  21. import java.util.Collection;
  22. import java.util.Collections;
  23. import java.util.Iterator;
  24. import java.util.Vector;
  25. import com.owncloud.android.MainApp;
  26. import com.owncloud.android.db.ProviderMeta.ProviderTableMeta;
  27. import com.owncloud.android.lib.resources.shares.OCShare;
  28. import com.owncloud.android.lib.resources.shares.ShareType;
  29. import com.owncloud.android.lib.resources.files.FileUtils;
  30. import com.owncloud.android.utils.FileStorageUtils;
  31. import com.owncloud.android.utils.Log_OC;
  32. import android.accounts.Account;
  33. import android.content.ContentProviderClient;
  34. import android.content.ContentProviderOperation;
  35. import android.content.ContentProviderResult;
  36. import android.content.ContentResolver;
  37. import android.content.ContentUris;
  38. import android.content.ContentValues;
  39. import android.content.OperationApplicationException;
  40. import android.database.Cursor;
  41. import android.net.Uri;
  42. import android.os.RemoteException;
  43. public class FileDataStorageManager {
  44. public static final int ROOT_PARENT_ID = 0;
  45. private ContentResolver mContentResolver;
  46. private ContentProviderClient mContentProviderClient;
  47. private Account mAccount;
  48. private static String TAG = FileDataStorageManager.class.getSimpleName();
  49. public FileDataStorageManager(Account account, ContentResolver cr) {
  50. mContentProviderClient = null;
  51. mContentResolver = cr;
  52. mAccount = account;
  53. }
  54. public FileDataStorageManager(Account account, ContentProviderClient cp) {
  55. mContentProviderClient = cp;
  56. mContentResolver = null;
  57. mAccount = account;
  58. }
  59. public void setAccount(Account account) {
  60. mAccount = account;
  61. }
  62. public Account getAccount() {
  63. return mAccount;
  64. }
  65. public void setContentResolver(ContentResolver cr) {
  66. mContentResolver = cr;
  67. }
  68. public ContentResolver getContentResolver() {
  69. return mContentResolver;
  70. }
  71. public void setContentProviderClient(ContentProviderClient cp) {
  72. mContentProviderClient = cp;
  73. }
  74. public ContentProviderClient getContentProviderClient() {
  75. return mContentProviderClient;
  76. }
  77. public OCFile getFileByPath(String path) {
  78. Cursor c = getCursorForValue(ProviderTableMeta.FILE_PATH, path);
  79. OCFile file = null;
  80. if (c.moveToFirst()) {
  81. file = createFileInstance(c);
  82. }
  83. c.close();
  84. if (file == null && OCFile.ROOT_PATH.equals(path)) {
  85. return createRootDir(); // root should always exist
  86. }
  87. return file;
  88. }
  89. public OCFile getFileById(long id) {
  90. Cursor c = getCursorForValue(ProviderTableMeta._ID, String.valueOf(id));
  91. OCFile file = null;
  92. if (c.moveToFirst()) {
  93. file = createFileInstance(c);
  94. }
  95. c.close();
  96. return file;
  97. }
  98. public OCFile getFileByLocalPath(String path) {
  99. Cursor c = getCursorForValue(ProviderTableMeta.FILE_STORAGE_PATH, path);
  100. OCFile file = null;
  101. if (c.moveToFirst()) {
  102. file = createFileInstance(c);
  103. }
  104. c.close();
  105. return file;
  106. }
  107. public boolean fileExists(long id) {
  108. return fileExists(ProviderTableMeta._ID, String.valueOf(id));
  109. }
  110. public boolean fileExists(String path) {
  111. return fileExists(ProviderTableMeta.FILE_PATH, path);
  112. }
  113. public Vector<OCFile> getFolderContent(OCFile f) {
  114. if (f != null && f.isFolder() && f.getFileId() != -1) {
  115. return getFolderContent(f.getFileId());
  116. } else {
  117. return new Vector<OCFile>();
  118. }
  119. }
  120. public Vector<OCFile> getFolderImages(OCFile folder) {
  121. Vector<OCFile> ret = new Vector<OCFile>();
  122. if (folder != null) {
  123. // TODO better implementation, filtering in the access to database (if possible) instead of here
  124. Vector<OCFile> tmp = getFolderContent(folder);
  125. OCFile current = null;
  126. for (int i=0; i<tmp.size(); i++) {
  127. current = tmp.get(i);
  128. if (current.isImage()) {
  129. ret.add(current);
  130. }
  131. }
  132. }
  133. return ret;
  134. }
  135. public boolean saveFile(OCFile file) {
  136. boolean overriden = false;
  137. ContentValues cv = new ContentValues();
  138. cv.put(ProviderTableMeta.FILE_MODIFIED, file.getModificationTimestamp());
  139. cv.put(ProviderTableMeta.FILE_MODIFIED_AT_LAST_SYNC_FOR_DATA, file.getModificationTimestampAtLastSyncForData());
  140. cv.put(ProviderTableMeta.FILE_CREATION, file.getCreationTimestamp());
  141. cv.put(ProviderTableMeta.FILE_CONTENT_LENGTH, file.getFileLength());
  142. cv.put(ProviderTableMeta.FILE_CONTENT_TYPE, file.getMimetype());
  143. cv.put(ProviderTableMeta.FILE_NAME, file.getFileName());
  144. //if (file.getParentId() != DataStorageManager.ROOT_PARENT_ID)
  145. cv.put(ProviderTableMeta.FILE_PARENT, file.getParentId());
  146. cv.put(ProviderTableMeta.FILE_PATH, file.getRemotePath());
  147. if (!file.isFolder())
  148. cv.put(ProviderTableMeta.FILE_STORAGE_PATH, file.getStoragePath());
  149. cv.put(ProviderTableMeta.FILE_ACCOUNT_OWNER, mAccount.name);
  150. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE, file.getLastSyncDateForProperties());
  151. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE_FOR_DATA, file.getLastSyncDateForData());
  152. cv.put(ProviderTableMeta.FILE_KEEP_IN_SYNC, file.keepInSync() ? 1 : 0);
  153. cv.put(ProviderTableMeta.FILE_ETAG, file.getEtag());
  154. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, file.isShareByLink() ? 1 : 0);
  155. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, file.getPublicLink());
  156. boolean sameRemotePath = fileExists(file.getRemotePath());
  157. if (sameRemotePath ||
  158. fileExists(file.getFileId()) ) { // for renamed files; no more delete and create
  159. OCFile oldFile = null;
  160. if (sameRemotePath) {
  161. oldFile = getFileByPath(file.getRemotePath());
  162. file.setFileId(oldFile.getFileId());
  163. } else {
  164. oldFile = getFileById(file.getFileId());
  165. }
  166. overriden = true;
  167. if (getContentResolver() != null) {
  168. getContentResolver().update(ProviderTableMeta.CONTENT_URI, cv,
  169. ProviderTableMeta._ID + "=?",
  170. new String[] { String.valueOf(file.getFileId()) });
  171. } else {
  172. try {
  173. getContentProviderClient().update(ProviderTableMeta.CONTENT_URI,
  174. cv, ProviderTableMeta._ID + "=?",
  175. new String[] { String.valueOf(file.getFileId()) });
  176. } catch (RemoteException e) {
  177. Log_OC.e(TAG,
  178. "Fail to insert insert file to database "
  179. + e.getMessage());
  180. }
  181. }
  182. } else {
  183. Uri result_uri = null;
  184. if (getContentResolver() != null) {
  185. result_uri = getContentResolver().insert(
  186. ProviderTableMeta.CONTENT_URI_FILE, cv);
  187. } else {
  188. try {
  189. result_uri = getContentProviderClient().insert(
  190. ProviderTableMeta.CONTENT_URI_FILE, cv);
  191. } catch (RemoteException e) {
  192. Log_OC.e(TAG,
  193. "Fail to insert insert file to database "
  194. + e.getMessage());
  195. }
  196. }
  197. if (result_uri != null) {
  198. long new_id = Long.parseLong(result_uri.getPathSegments()
  199. .get(1));
  200. file.setFileId(new_id);
  201. }
  202. }
  203. // if (file.isFolder()) {
  204. // updateFolderSize(file.getFileId());
  205. // } else {
  206. // updateFolderSize(file.getParentId());
  207. // }
  208. return overriden;
  209. }
  210. /**
  211. * Inserts or updates the list of files contained in a given folder.
  212. *
  213. * CALLER IS THE RESPONSIBLE FOR GRANTING RIGHT UPDATE OF INFORMATION, NOT THIS METHOD.
  214. * HERE ONLY DATA CONSISTENCY SHOULD BE GRANTED
  215. *
  216. * @param folder
  217. * @param files
  218. * @param removeNotUpdated
  219. */
  220. public void saveFolder(OCFile folder, Collection<OCFile> updatedFiles, Collection<OCFile> filesToRemove) {
  221. Log_OC.d(TAG, "Saving folder " + folder.getRemotePath() + " with " + updatedFiles.size() + " children and " + filesToRemove.size() + " files to remove");
  222. ArrayList<ContentProviderOperation> operations = new ArrayList<ContentProviderOperation>(updatedFiles.size());
  223. // prepare operations to insert or update files to save in the given folder
  224. for (OCFile file : updatedFiles) {
  225. ContentValues cv = new ContentValues();
  226. cv.put(ProviderTableMeta.FILE_MODIFIED, file.getModificationTimestamp());
  227. cv.put(ProviderTableMeta.FILE_MODIFIED_AT_LAST_SYNC_FOR_DATA, file.getModificationTimestampAtLastSyncForData());
  228. cv.put(ProviderTableMeta.FILE_CREATION, file.getCreationTimestamp());
  229. cv.put(ProviderTableMeta.FILE_CONTENT_LENGTH, file.getFileLength());
  230. cv.put(ProviderTableMeta.FILE_CONTENT_TYPE, file.getMimetype());
  231. cv.put(ProviderTableMeta.FILE_NAME, file.getFileName());
  232. //cv.put(ProviderTableMeta.FILE_PARENT, file.getParentId());
  233. cv.put(ProviderTableMeta.FILE_PARENT, folder.getFileId());
  234. cv.put(ProviderTableMeta.FILE_PATH, file.getRemotePath());
  235. if (!file.isFolder()) {
  236. cv.put(ProviderTableMeta.FILE_STORAGE_PATH, file.getStoragePath());
  237. }
  238. cv.put(ProviderTableMeta.FILE_ACCOUNT_OWNER, mAccount.name);
  239. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE, file.getLastSyncDateForProperties());
  240. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE_FOR_DATA, file.getLastSyncDateForData());
  241. cv.put(ProviderTableMeta.FILE_KEEP_IN_SYNC, file.keepInSync() ? 1 : 0);
  242. cv.put(ProviderTableMeta.FILE_ETAG, file.getEtag());
  243. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, file.isShareByLink() ? 1 : 0);
  244. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, file.getPublicLink());
  245. boolean existsByPath = fileExists(file.getRemotePath());
  246. if (existsByPath || fileExists(file.getFileId())) {
  247. // updating an existing file
  248. operations.add(ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI).
  249. withValues(cv).
  250. withSelection( ProviderTableMeta._ID + "=?",
  251. new String[] { String.valueOf(file.getFileId()) })
  252. .build());
  253. } else {
  254. // adding a new file
  255. operations.add(ContentProviderOperation.newInsert(ProviderTableMeta.CONTENT_URI).withValues(cv).build());
  256. }
  257. }
  258. // prepare operations to remove files in the given folder
  259. String where = ProviderTableMeta.FILE_ACCOUNT_OWNER + "=?" + " AND " + ProviderTableMeta.FILE_PATH + "=?";
  260. String [] whereArgs = null;
  261. for (OCFile file : filesToRemove) {
  262. if (file.getParentId() == folder.getFileId()) {
  263. whereArgs = new String[]{mAccount.name, file.getRemotePath()};
  264. //Uri.withAppendedPath(ProviderTableMeta.CONTENT_URI_FILE, "" + file.getFileId());
  265. if (file.isFolder()) {
  266. operations.add(ContentProviderOperation
  267. .newDelete(ContentUris.withAppendedId(ProviderTableMeta.CONTENT_URI_DIR, file.getFileId())).withSelection(where, whereArgs)
  268. .build());
  269. // TODO remove local folder
  270. } else {
  271. operations.add(ContentProviderOperation
  272. .newDelete(ContentUris.withAppendedId(ProviderTableMeta.CONTENT_URI_FILE, file.getFileId())).withSelection(where, whereArgs)
  273. .build());
  274. if (file.isDown()) {
  275. new File(file.getStoragePath()).delete();
  276. // TODO move the deletion of local contents after success of deletions
  277. }
  278. }
  279. }
  280. }
  281. // update metadata of folder
  282. ContentValues cv = new ContentValues();
  283. cv.put(ProviderTableMeta.FILE_MODIFIED, folder.getModificationTimestamp());
  284. cv.put(ProviderTableMeta.FILE_MODIFIED_AT_LAST_SYNC_FOR_DATA, folder.getModificationTimestampAtLastSyncForData());
  285. cv.put(ProviderTableMeta.FILE_CREATION, folder.getCreationTimestamp());
  286. cv.put(ProviderTableMeta.FILE_CONTENT_LENGTH, 0); // FileContentProvider calculates the right size
  287. cv.put(ProviderTableMeta.FILE_CONTENT_TYPE, folder.getMimetype());
  288. cv.put(ProviderTableMeta.FILE_NAME, folder.getFileName());
  289. cv.put(ProviderTableMeta.FILE_PARENT, folder.getParentId());
  290. cv.put(ProviderTableMeta.FILE_PATH, folder.getRemotePath());
  291. cv.put(ProviderTableMeta.FILE_ACCOUNT_OWNER, mAccount.name);
  292. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE, folder.getLastSyncDateForProperties());
  293. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE_FOR_DATA, folder.getLastSyncDateForData());
  294. cv.put(ProviderTableMeta.FILE_KEEP_IN_SYNC, folder.keepInSync() ? 1 : 0);
  295. cv.put(ProviderTableMeta.FILE_ETAG, folder.getEtag());
  296. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, folder.isShareByLink() ? 1 : 0);
  297. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, folder.getPublicLink());
  298. operations.add(ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI).
  299. withValues(cv).
  300. withSelection( ProviderTableMeta._ID + "=?",
  301. new String[] { String.valueOf(folder.getFileId()) })
  302. .build());
  303. // apply operations in batch
  304. ContentProviderResult[] results = null;
  305. Log_OC.d(TAG, "Sending " + operations.size() + " operations to FileContentProvider");
  306. try {
  307. if (getContentResolver() != null) {
  308. results = getContentResolver().applyBatch(MainApp.getAuthority(), operations);
  309. } else {
  310. results = getContentProviderClient().applyBatch(operations);
  311. }
  312. } catch (OperationApplicationException e) {
  313. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  314. } catch (RemoteException e) {
  315. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  316. }
  317. // update new id in file objects for insertions
  318. if (results != null) {
  319. long newId;
  320. Iterator<OCFile> filesIt = updatedFiles.iterator();
  321. OCFile file = null;
  322. for (int i=0; i<results.length; i++) {
  323. if (filesIt.hasNext()) {
  324. file = filesIt.next();
  325. } else {
  326. file = null;
  327. }
  328. if (results[i].uri != null) {
  329. newId = Long.parseLong(results[i].uri.getPathSegments().get(1));
  330. //updatedFiles.get(i).setFileId(newId);
  331. if (file != null) {
  332. file.setFileId(newId);
  333. }
  334. }
  335. }
  336. }
  337. //updateFolderSize(folder.getFileId());
  338. }
  339. // /**
  340. // *
  341. // * @param id
  342. // */
  343. // private void updateFolderSize(long id) {
  344. // if (id > FileDataStorageManager.ROOT_PARENT_ID) {
  345. // Log_OC.d(TAG, "Updating size of " + id);
  346. // if (getContentResolver() != null) {
  347. // getContentResolver().update(ProviderTableMeta.CONTENT_URI_DIR,
  348. // new ContentValues(), // won't be used, but cannot be null; crashes in KLP
  349. // ProviderTableMeta._ID + "=?",
  350. // new String[] { String.valueOf(id) });
  351. // } else {
  352. // try {
  353. // getContentProviderClient().update(ProviderTableMeta.CONTENT_URI_DIR,
  354. // new ContentValues(), // won't be used, but cannot be null; crashes in KLP
  355. // ProviderTableMeta._ID + "=?",
  356. // new String[] { String.valueOf(id) });
  357. //
  358. // } catch (RemoteException e) {
  359. // Log_OC.e(TAG, "Exception in update of folder size through compatibility patch " + e.getMessage());
  360. // }
  361. // }
  362. // } else {
  363. // Log_OC.e(TAG, "not updating size for folder " + id);
  364. // }
  365. // }
  366. public void removeFile(OCFile file, boolean removeDBData, boolean removeLocalCopy) {
  367. if (file != null) {
  368. if (file.isFolder()) {
  369. removeFolder(file, removeDBData, removeLocalCopy);
  370. } else {
  371. if (removeDBData) {
  372. //Uri file_uri = Uri.withAppendedPath(ProviderTableMeta.CONTENT_URI_FILE, ""+file.getFileId());
  373. Uri file_uri = ContentUris.withAppendedId(ProviderTableMeta.CONTENT_URI_FILE, file.getFileId());
  374. String where = ProviderTableMeta.FILE_ACCOUNT_OWNER + "=?" + " AND " + ProviderTableMeta.FILE_PATH + "=?";
  375. String [] whereArgs = new String[]{mAccount.name, file.getRemotePath()};
  376. if (getContentProviderClient() != null) {
  377. try {
  378. getContentProviderClient().delete(file_uri, where, whereArgs);
  379. } catch (RemoteException e) {
  380. e.printStackTrace();
  381. }
  382. } else {
  383. getContentResolver().delete(file_uri, where, whereArgs);
  384. }
  385. //updateFolderSize(file.getParentId());
  386. }
  387. if (removeLocalCopy && file.isDown() && file.getStoragePath() != null) {
  388. boolean success = new File(file.getStoragePath()).delete();
  389. if (!removeDBData && success) {
  390. // maybe unnecessary, but should be checked TODO remove if unnecessary
  391. file.setStoragePath(null);
  392. saveFile(file);
  393. }
  394. }
  395. }
  396. }
  397. }
  398. public void removeFolder(OCFile folder, boolean removeDBData, boolean removeLocalContent) {
  399. if (folder != null && folder.isFolder()) {
  400. if (removeDBData && folder.getFileId() != -1) {
  401. removeFolderInDb(folder);
  402. }
  403. if (removeLocalContent) {
  404. File localFolder = new File(FileStorageUtils.getDefaultSavePathFor(mAccount.name, folder));
  405. removeLocalFolder(localFolder);
  406. }
  407. }
  408. }
  409. private void removeFolderInDb(OCFile folder) {
  410. Uri folder_uri = Uri.withAppendedPath(ProviderTableMeta.CONTENT_URI_DIR, ""+ folder.getFileId()); // URI for recursive deletion
  411. String where = ProviderTableMeta.FILE_ACCOUNT_OWNER + "=?" + " AND " + ProviderTableMeta.FILE_PATH + "=?";
  412. String [] whereArgs = new String[]{mAccount.name, folder.getRemotePath()};
  413. if (getContentProviderClient() != null) {
  414. try {
  415. getContentProviderClient().delete(folder_uri, where, whereArgs);
  416. } catch (RemoteException e) {
  417. e.printStackTrace();
  418. }
  419. } else {
  420. getContentResolver().delete(folder_uri, where, whereArgs);
  421. }
  422. //updateFolderSize(folder.getParentId());
  423. }
  424. private void removeLocalFolder(File folder) {
  425. if (folder.exists()) {
  426. File[] files = folder.listFiles();
  427. if (files != null) {
  428. for (File file : files) {
  429. if (file.isDirectory()) {
  430. removeLocalFolder(file);
  431. } else {
  432. file.delete();
  433. }
  434. }
  435. }
  436. folder.delete();
  437. }
  438. }
  439. /**
  440. * Updates database for a folder that was moved to a different location.
  441. *
  442. * TODO explore better (faster) implementations
  443. * TODO throw exceptions up !
  444. */
  445. public void moveFolder(OCFile folder, String newPath) {
  446. // TODO check newPath
  447. if (folder != null && folder.isFolder() && folder.fileExists() && !OCFile.ROOT_PATH.equals(folder.getFileName())) {
  448. /// 1. get all the descendants of 'dir' in a single QUERY (including 'dir')
  449. Cursor c = null;
  450. if (getContentProviderClient() != null) {
  451. try {
  452. c = getContentProviderClient().query(ProviderTableMeta.CONTENT_URI,
  453. null,
  454. ProviderTableMeta.FILE_ACCOUNT_OWNER + "=? AND " + ProviderTableMeta.FILE_PATH + " LIKE ? ",
  455. new String[] { mAccount.name, folder.getRemotePath() + "%" }, ProviderTableMeta.FILE_PATH + " ASC ");
  456. } catch (RemoteException e) {
  457. Log_OC.e(TAG, e.getMessage());
  458. }
  459. } else {
  460. c = getContentResolver().query(ProviderTableMeta.CONTENT_URI,
  461. null,
  462. ProviderTableMeta.FILE_ACCOUNT_OWNER + "=? AND " + ProviderTableMeta.FILE_PATH + " LIKE ? ",
  463. new String[] { mAccount.name, folder.getRemotePath() + "%" }, ProviderTableMeta.FILE_PATH + " ASC ");
  464. }
  465. /// 2. prepare a batch of update operations to change all the descendants
  466. ArrayList<ContentProviderOperation> operations = new ArrayList<ContentProviderOperation>(c.getCount());
  467. int lengthOfOldPath = folder.getRemotePath().length();
  468. String defaultSavePath = FileStorageUtils.getSavePath(mAccount.name);
  469. int lengthOfOldStoragePath = defaultSavePath.length() + lengthOfOldPath;
  470. if (c.moveToFirst()) {
  471. do {
  472. ContentValues cv = new ContentValues(); // don't take the constructor out of the loop and clear the object
  473. OCFile child = createFileInstance(c);
  474. cv.put(ProviderTableMeta.FILE_PATH, newPath + child.getRemotePath().substring(lengthOfOldPath));
  475. if (child.getStoragePath() != null && child.getStoragePath().startsWith(defaultSavePath)) {
  476. cv.put(ProviderTableMeta.FILE_STORAGE_PATH, defaultSavePath + newPath + child.getStoragePath().substring(lengthOfOldStoragePath));
  477. }
  478. operations.add(ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI).
  479. withValues(cv).
  480. withSelection( ProviderTableMeta._ID + "=?",
  481. new String[] { String.valueOf(child.getFileId()) })
  482. .build());
  483. } while (c.moveToNext());
  484. }
  485. c.close();
  486. /// 3. apply updates in batch
  487. try {
  488. if (getContentResolver() != null) {
  489. getContentResolver().applyBatch(MainApp.getAuthority(), operations);
  490. } else {
  491. getContentProviderClient().applyBatch(operations);
  492. }
  493. } catch (OperationApplicationException e) {
  494. Log_OC.e(TAG, "Fail to update descendants of " + folder.getFileId() + " in database", e);
  495. } catch (RemoteException e) {
  496. Log_OC.e(TAG, "Fail to update desendants of " + folder.getFileId() + " in database", e);
  497. }
  498. }
  499. }
  500. private Vector<OCFile> getFolderContent(long parentId) {
  501. Vector<OCFile> ret = new Vector<OCFile>();
  502. Uri req_uri = Uri.withAppendedPath(
  503. ProviderTableMeta.CONTENT_URI_DIR,
  504. String.valueOf(parentId));
  505. Cursor c = null;
  506. if (getContentProviderClient() != null) {
  507. try {
  508. c = getContentProviderClient().query(req_uri, null,
  509. ProviderTableMeta.FILE_PARENT + "=?" ,
  510. new String[] { String.valueOf(parentId)}, null);
  511. } catch (RemoteException e) {
  512. Log_OC.e(TAG, e.getMessage());
  513. return ret;
  514. }
  515. } else {
  516. c = getContentResolver().query(req_uri, null,
  517. ProviderTableMeta.FILE_PARENT + "=?" ,
  518. new String[] { String.valueOf(parentId)}, null);
  519. }
  520. if (c.moveToFirst()) {
  521. do {
  522. OCFile child = createFileInstance(c);
  523. ret.add(child);
  524. } while (c.moveToNext());
  525. }
  526. c.close();
  527. Collections.sort(ret);
  528. return ret;
  529. }
  530. public Cursor getContent(long parentId) {
  531. Uri req_uri = Uri.withAppendedPath(
  532. ProviderTableMeta.CONTENT_URI_DIR,
  533. String.valueOf(parentId));
  534. Cursor c = null;
  535. if (getContentProviderClient() != null) {
  536. try {
  537. c = getContentProviderClient().query(req_uri, null,
  538. ProviderTableMeta.FILE_PARENT + "=?" ,
  539. new String[] { String.valueOf(parentId)}, null);
  540. } catch (RemoteException e) {
  541. Log_OC.e(TAG, e.getMessage());
  542. return c;
  543. }
  544. } else {
  545. c = getContentResolver().query(req_uri, null,
  546. ProviderTableMeta.FILE_PARENT + "=?" ,
  547. new String[] { String.valueOf(parentId)}, null);
  548. }
  549. return c;
  550. }
  551. private OCFile createRootDir() {
  552. OCFile file = new OCFile(OCFile.ROOT_PATH);
  553. file.setMimetype("DIR");
  554. file.setParentId(FileDataStorageManager.ROOT_PARENT_ID);
  555. saveFile(file);
  556. return file;
  557. }
  558. private boolean fileExists(String cmp_key, String value) {
  559. Cursor c;
  560. if (getContentResolver() != null) {
  561. c = getContentResolver()
  562. .query(ProviderTableMeta.CONTENT_URI,
  563. null,
  564. cmp_key + "=? AND "
  565. + ProviderTableMeta.FILE_ACCOUNT_OWNER
  566. + "=?",
  567. new String[] { value, mAccount.name }, null);
  568. } else {
  569. try {
  570. c = getContentProviderClient().query(
  571. ProviderTableMeta.CONTENT_URI,
  572. null,
  573. cmp_key + "=? AND "
  574. + ProviderTableMeta.FILE_ACCOUNT_OWNER + "=?",
  575. new String[] { value, mAccount.name }, null);
  576. } catch (RemoteException e) {
  577. Log_OC.e(TAG,
  578. "Couldn't determine file existance, assuming non existance: "
  579. + e.getMessage());
  580. return false;
  581. }
  582. }
  583. boolean retval = c.moveToFirst();
  584. c.close();
  585. return retval;
  586. }
  587. private Cursor getCursorForValue(String key, String value) {
  588. Cursor c = null;
  589. if (getContentResolver() != null) {
  590. c = getContentResolver()
  591. .query(ProviderTableMeta.CONTENT_URI,
  592. null,
  593. key + "=? AND "
  594. + ProviderTableMeta.FILE_ACCOUNT_OWNER
  595. + "=?",
  596. new String[] { value, mAccount.name }, null);
  597. } else {
  598. try {
  599. c = getContentProviderClient().query(
  600. ProviderTableMeta.CONTENT_URI,
  601. null,
  602. key + "=? AND " + ProviderTableMeta.FILE_ACCOUNT_OWNER
  603. + "=?", new String[] { value, mAccount.name },
  604. null);
  605. } catch (RemoteException e) {
  606. Log_OC.e(TAG, "Could not get file details: " + e.getMessage());
  607. c = null;
  608. }
  609. }
  610. return c;
  611. }
  612. // private Cursor getShareCursorForValue(String key, String value) {
  613. // Cursor c = null;
  614. // if (getContentResolver() != null) {
  615. // c = getContentResolver()
  616. // .query(ProviderTableMeta.CONTENT_URI_SHARE,
  617. // null,
  618. // key + "=? AND "
  619. // + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER
  620. // + "=?",
  621. // new String[] { value, mAccount.name }, null);
  622. // } else {
  623. // try {
  624. // c = getContentProviderClient().query(
  625. // ProviderTableMeta.CONTENT_URI_SHARE,
  626. // null,
  627. // key + "=? AND " + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER
  628. // + "=?", new String[] { value, mAccount.name },
  629. // null);
  630. // } catch (RemoteException e) {
  631. // Log_OC.e(TAG, "Could not get file details: " + e.getMessage());
  632. // c = null;
  633. // }
  634. // }
  635. // return c;
  636. // }
  637. public OCFile createFileInstance(Cursor c) {
  638. OCFile file = null;
  639. if (c != null) {
  640. file = new OCFile(c.getString(c
  641. .getColumnIndex(ProviderTableMeta.FILE_PATH)));
  642. file.setFileId(c.getLong(c.getColumnIndex(ProviderTableMeta._ID)));
  643. file.setParentId(c.getLong(c
  644. .getColumnIndex(ProviderTableMeta.FILE_PARENT)));
  645. file.setMimetype(c.getString(c
  646. .getColumnIndex(ProviderTableMeta.FILE_CONTENT_TYPE)));
  647. if (!file.isFolder()) {
  648. file.setStoragePath(c.getString(c
  649. .getColumnIndex(ProviderTableMeta.FILE_STORAGE_PATH)));
  650. if (file.getStoragePath() == null) {
  651. // try to find existing file and bind it with current account; - with the current update of SynchronizeFolderOperation, this won't be necessary anymore after a full synchronization of the account
  652. File f = new File(FileStorageUtils.getDefaultSavePathFor(mAccount.name, file));
  653. if (f.exists()) {
  654. file.setStoragePath(f.getAbsolutePath());
  655. file.setLastSyncDateForData(f.lastModified());
  656. }
  657. }
  658. }
  659. file.setFileLength(c.getLong(c
  660. .getColumnIndex(ProviderTableMeta.FILE_CONTENT_LENGTH)));
  661. file.setCreationTimestamp(c.getLong(c
  662. .getColumnIndex(ProviderTableMeta.FILE_CREATION)));
  663. file.setModificationTimestamp(c.getLong(c
  664. .getColumnIndex(ProviderTableMeta.FILE_MODIFIED)));
  665. file.setModificationTimestampAtLastSyncForData(c.getLong(c
  666. .getColumnIndex(ProviderTableMeta.FILE_MODIFIED_AT_LAST_SYNC_FOR_DATA)));
  667. file.setLastSyncDateForProperties(c.getLong(c
  668. .getColumnIndex(ProviderTableMeta.FILE_LAST_SYNC_DATE)));
  669. file.setLastSyncDateForData(c.getLong(c.
  670. getColumnIndex(ProviderTableMeta.FILE_LAST_SYNC_DATE_FOR_DATA)));
  671. file.setKeepInSync(c.getInt(
  672. c.getColumnIndex(ProviderTableMeta.FILE_KEEP_IN_SYNC)) == 1 ? true : false);
  673. file.setEtag(c.getString(c.getColumnIndex(ProviderTableMeta.FILE_ETAG)));
  674. file.setShareByLink(c.getInt(
  675. c.getColumnIndex(ProviderTableMeta.FILE_SHARE_BY_LINK)) == 1 ? true : false);
  676. file.setPublicLink(c.getString(c.getColumnIndex(ProviderTableMeta.FILE_PUBLIC_LINK)));
  677. }
  678. return file;
  679. }
  680. /**
  681. * Returns if the file/folder is shared by link or not
  682. * @param path Path of the file/folder
  683. * @return
  684. */
  685. public boolean isShareByLink(String path) {
  686. Cursor c = getCursorForValue(ProviderTableMeta.FILE_STORAGE_PATH, path);
  687. OCFile file = null;
  688. if (c.moveToFirst()) {
  689. file = createFileInstance(c);
  690. }
  691. c.close();
  692. return file.isShareByLink();
  693. }
  694. /**
  695. * Returns the public link of the file/folder
  696. * @param path Path of the file/folder
  697. * @return
  698. */
  699. public String getPublicLink(String path) {
  700. Cursor c = getCursorForValue(ProviderTableMeta.FILE_STORAGE_PATH, path);
  701. OCFile file = null;
  702. if (c.moveToFirst()) {
  703. file = createFileInstance(c);
  704. }
  705. c.close();
  706. return file.getPublicLink();
  707. }
  708. // Methods for Shares
  709. public boolean saveShare(OCShare share) {
  710. boolean overriden = false;
  711. ContentValues cv = new ContentValues();
  712. cv.put(ProviderTableMeta.OCSHARES_FILE_SOURCE, share.getFileSource());
  713. cv.put(ProviderTableMeta.OCSHARES_ITEM_SOURCE, share.getItemSource());
  714. cv.put(ProviderTableMeta.OCSHARES_SHARE_TYPE, share.getShareType().getValue());
  715. cv.put(ProviderTableMeta.OCSHARES_SHARE_WITH, share.getShareWith());
  716. cv.put(ProviderTableMeta.OCSHARES_PATH, share.getPath());
  717. cv.put(ProviderTableMeta.OCSHARES_PERMISSIONS, share.getPermissions());
  718. cv.put(ProviderTableMeta.OCSHARES_SHARED_DATE, share.getSharedDate());
  719. cv.put(ProviderTableMeta.OCSHARES_EXPIRATION_DATE, share.getExpirationDate());
  720. cv.put(ProviderTableMeta.OCSHARES_TOKEN, share.getToken());
  721. cv.put(ProviderTableMeta.OCSHARES_SHARE_WITH_DISPLAY_NAME, share.getSharedWithDisplayName());
  722. cv.put(ProviderTableMeta.OCSHARES_IS_DIRECTORY, share.isFolder() ? 1 : 0);
  723. cv.put(ProviderTableMeta.OCSHARES_USER_ID, share.getUserId());
  724. cv.put(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED, share.getIdRemoteShared());
  725. cv.put(ProviderTableMeta.OCSHARES_ACCOUNT_OWNER, mAccount.name);
  726. if (shareExists(share.getIdRemoteShared())) { // for renamed files; no more delete and create
  727. overriden = true;
  728. if (getContentResolver() != null) {
  729. getContentResolver().update(ProviderTableMeta.CONTENT_URI_SHARE, cv,
  730. ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED + "=?",
  731. new String[] { String.valueOf(share.getIdRemoteShared()) });
  732. } else {
  733. try {
  734. getContentProviderClient().update(ProviderTableMeta.CONTENT_URI_SHARE,
  735. cv, ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED + "=?",
  736. new String[] { String.valueOf(share.getIdRemoteShared()) });
  737. } catch (RemoteException e) {
  738. Log_OC.e(TAG,
  739. "Fail to insert insert file to database "
  740. + e.getMessage());
  741. }
  742. }
  743. } else {
  744. Uri result_uri = null;
  745. if (getContentResolver() != null) {
  746. result_uri = getContentResolver().insert(
  747. ProviderTableMeta.CONTENT_URI_SHARE, cv);
  748. } else {
  749. try {
  750. result_uri = getContentProviderClient().insert(
  751. ProviderTableMeta.CONTENT_URI_SHARE, cv);
  752. } catch (RemoteException e) {
  753. Log_OC.e(TAG,
  754. "Fail to insert insert file to database "
  755. + e.getMessage());
  756. }
  757. }
  758. if (result_uri != null) {
  759. long new_id = Long.parseLong(result_uri.getPathSegments()
  760. .get(1));
  761. share.setId(new_id);
  762. }
  763. }
  764. return overriden;
  765. }
  766. // private OCShare getShareById(long id) {
  767. // Cursor c = getShareCursorForValue(ProviderTableMeta._ID, String.valueOf(id));
  768. // OCShare share = null;
  769. // if (c.moveToFirst()) {
  770. // share = createShareInstance(c);
  771. // }
  772. // c.close();
  773. // return share;
  774. // }
  775. //
  776. // private OCShare getShareByRemoteId(long remoteId) {
  777. // Cursor c = getShareCursorForValue(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED, String.valueOf(remoteId));
  778. // OCShare share = null;
  779. // if (c.moveToFirst()) {
  780. // share = createShareInstance(c);
  781. // }
  782. // c.close();
  783. // return share;
  784. // }
  785. public OCShare getFirstShareByPathAndType(String path, ShareType type) {
  786. Cursor c = null;
  787. if (getContentResolver() != null) {
  788. c = getContentResolver().query(
  789. ProviderTableMeta.CONTENT_URI_SHARE,
  790. null,
  791. ProviderTableMeta.OCSHARES_PATH + "=? AND "
  792. + ProviderTableMeta.OCSHARES_SHARE_TYPE + "=? AND "
  793. + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?",
  794. new String[] { path, Integer.toString(type.getValue()), mAccount.name },
  795. null);
  796. } else {
  797. try {
  798. c = getContentProviderClient().query(
  799. ProviderTableMeta.CONTENT_URI_SHARE,
  800. null,
  801. ProviderTableMeta.OCSHARES_PATH + "=? AND "
  802. + ProviderTableMeta.OCSHARES_SHARE_TYPE + "=? AND "
  803. + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?",
  804. new String[] { path, Integer.toString(type.getValue()), mAccount.name },
  805. null);
  806. } catch (RemoteException e) {
  807. Log_OC.e(TAG, "Could not get file details: " + e.getMessage());
  808. c = null;
  809. }
  810. }
  811. OCShare share = null;
  812. if (c.moveToFirst()) {
  813. share = createShareInstance(c);
  814. }
  815. c.close();
  816. return share;
  817. }
  818. private OCShare createShareInstance(Cursor c) {
  819. OCShare share = null;
  820. if (c != null) {
  821. share = new OCShare(c.getString(c
  822. .getColumnIndex(ProviderTableMeta.OCSHARES_PATH)));
  823. share.setId(c.getLong(c.getColumnIndex(ProviderTableMeta._ID)));
  824. share.setFileSource(c.getLong(c
  825. .getColumnIndex(ProviderTableMeta.OCSHARES_ITEM_SOURCE)));
  826. share.setShareType(ShareType.fromValue(c.getInt(c
  827. .getColumnIndex(ProviderTableMeta.OCSHARES_SHARE_TYPE))));
  828. share.setPermissions(c.getInt(c
  829. .getColumnIndex(ProviderTableMeta.OCSHARES_PERMISSIONS)));
  830. share.setSharedDate(c.getLong(c
  831. .getColumnIndex(ProviderTableMeta.OCSHARES_SHARED_DATE)));
  832. share.setExpirationDate(c.getLong(c
  833. .getColumnIndex(ProviderTableMeta.OCSHARES_EXPIRATION_DATE)));
  834. share.setToken(c.getString(c
  835. .getColumnIndex(ProviderTableMeta.OCSHARES_TOKEN)));
  836. share.setSharedWithDisplayName(c.getString(c
  837. .getColumnIndex(ProviderTableMeta.OCSHARES_SHARE_WITH_DISPLAY_NAME)));
  838. share.setIsFolder(c.getInt(
  839. c.getColumnIndex(ProviderTableMeta.OCSHARES_IS_DIRECTORY)) == 1 ? true : false);
  840. share.setUserId(c.getLong(c.getColumnIndex(ProviderTableMeta.OCSHARES_USER_ID)));
  841. share.setIdRemoteShared(c.getLong(c.getColumnIndex(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED)));
  842. }
  843. return share;
  844. }
  845. private boolean shareExists(String cmp_key, String value) {
  846. Cursor c;
  847. if (getContentResolver() != null) {
  848. c = getContentResolver()
  849. .query(ProviderTableMeta.CONTENT_URI_SHARE,
  850. null,
  851. cmp_key + "=? AND "
  852. + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER
  853. + "=?",
  854. new String[] { value, mAccount.name }, null);
  855. } else {
  856. try {
  857. c = getContentProviderClient().query(
  858. ProviderTableMeta.CONTENT_URI_SHARE,
  859. null,
  860. cmp_key + "=? AND "
  861. + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?",
  862. new String[] { value, mAccount.name }, null);
  863. } catch (RemoteException e) {
  864. Log_OC.e(TAG,
  865. "Couldn't determine file existance, assuming non existance: "
  866. + e.getMessage());
  867. return false;
  868. }
  869. }
  870. boolean retval = c.moveToFirst();
  871. c.close();
  872. return retval;
  873. }
  874. private boolean shareExists(long remoteId) {
  875. return shareExists(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED, String.valueOf(remoteId));
  876. }
  877. private void cleanSharedFiles() {
  878. ContentValues cv = new ContentValues();
  879. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, false);
  880. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, "");
  881. String where = ProviderTableMeta.FILE_ACCOUNT_OWNER + "=?";
  882. String [] whereArgs = new String[]{mAccount.name};
  883. if (getContentResolver() != null) {
  884. getContentResolver().update(ProviderTableMeta.CONTENT_URI, cv, where, whereArgs);
  885. } else {
  886. try {
  887. getContentProviderClient().update(ProviderTableMeta.CONTENT_URI, cv, where, whereArgs);
  888. } catch (RemoteException e) {
  889. Log_OC.e(TAG, "Exception in cleanSharedFiles" + e.getMessage());
  890. }
  891. }
  892. }
  893. private void cleanSharedFilesInFolder(OCFile folder) {
  894. ContentValues cv = new ContentValues();
  895. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, false);
  896. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, "");
  897. String where = ProviderTableMeta.FILE_ACCOUNT_OWNER + "=? AND " + ProviderTableMeta.FILE_PARENT + "=?";
  898. String [] whereArgs = new String[] { mAccount.name , String.valueOf(folder.getFileId()) };
  899. if (getContentResolver() != null) {
  900. getContentResolver().update(ProviderTableMeta.CONTENT_URI, cv, where, whereArgs);
  901. } else {
  902. try {
  903. getContentProviderClient().update(ProviderTableMeta.CONTENT_URI, cv, where, whereArgs);
  904. } catch (RemoteException e) {
  905. Log_OC.e(TAG, "Exception in cleanSharedFilesInFolder " + e.getMessage());
  906. }
  907. }
  908. }
  909. private void cleanShares() {
  910. String where = ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?";
  911. String [] whereArgs = new String[]{mAccount.name};
  912. if (getContentResolver() != null) {
  913. getContentResolver().delete(ProviderTableMeta.CONTENT_URI_SHARE, where, whereArgs);
  914. } else {
  915. try {
  916. getContentProviderClient().delete(ProviderTableMeta.CONTENT_URI_SHARE, where, whereArgs);
  917. } catch (RemoteException e) {
  918. Log_OC.e(TAG, "Exception in cleanShares" + e.getMessage());
  919. }
  920. }
  921. }
  922. public void saveShares(Collection<OCShare> shares) {
  923. cleanShares();
  924. if (shares != null) {
  925. ArrayList<ContentProviderOperation> operations = new ArrayList<ContentProviderOperation>(shares.size());
  926. // prepare operations to insert or update files to save in the given folder
  927. for (OCShare share : shares) {
  928. ContentValues cv = new ContentValues();
  929. cv.put(ProviderTableMeta.OCSHARES_FILE_SOURCE, share.getFileSource());
  930. cv.put(ProviderTableMeta.OCSHARES_ITEM_SOURCE, share.getItemSource());
  931. cv.put(ProviderTableMeta.OCSHARES_SHARE_TYPE, share.getShareType().getValue());
  932. cv.put(ProviderTableMeta.OCSHARES_SHARE_WITH, share.getShareWith());
  933. cv.put(ProviderTableMeta.OCSHARES_PATH, share.getPath());
  934. cv.put(ProviderTableMeta.OCSHARES_PERMISSIONS, share.getPermissions());
  935. cv.put(ProviderTableMeta.OCSHARES_SHARED_DATE, share.getSharedDate());
  936. cv.put(ProviderTableMeta.OCSHARES_EXPIRATION_DATE, share.getExpirationDate());
  937. cv.put(ProviderTableMeta.OCSHARES_TOKEN, share.getToken());
  938. cv.put(ProviderTableMeta.OCSHARES_SHARE_WITH_DISPLAY_NAME, share.getSharedWithDisplayName());
  939. cv.put(ProviderTableMeta.OCSHARES_IS_DIRECTORY, share.isFolder() ? 1 : 0);
  940. cv.put(ProviderTableMeta.OCSHARES_USER_ID, share.getUserId());
  941. cv.put(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED, share.getIdRemoteShared());
  942. cv.put(ProviderTableMeta.OCSHARES_ACCOUNT_OWNER, mAccount.name);
  943. if (shareExists(share.getIdRemoteShared())) {
  944. // updating an existing file
  945. operations.add(ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI_SHARE).
  946. withValues(cv).
  947. withSelection( ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED + "=?",
  948. new String[] { String.valueOf(share.getIdRemoteShared()) })
  949. .build());
  950. } else {
  951. // adding a new file
  952. operations.add(ContentProviderOperation.newInsert(ProviderTableMeta.CONTENT_URI_SHARE).withValues(cv).build());
  953. }
  954. }
  955. // apply operations in batch
  956. if (operations.size() > 0) {
  957. @SuppressWarnings("unused")
  958. ContentProviderResult[] results = null;
  959. Log_OC.d(TAG, "Sending " + operations.size() + " operations to FileContentProvider");
  960. try {
  961. if (getContentResolver() != null) {
  962. results = getContentResolver().applyBatch(MainApp.getAuthority(), operations);
  963. } else {
  964. results = getContentProviderClient().applyBatch(operations);
  965. }
  966. } catch (OperationApplicationException e) {
  967. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  968. } catch (RemoteException e) {
  969. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  970. }
  971. }
  972. }
  973. }
  974. public void updateSharedFiles(Collection<OCFile> sharedFiles) {
  975. cleanSharedFiles();
  976. if (sharedFiles != null) {
  977. ArrayList<ContentProviderOperation> operations = new ArrayList<ContentProviderOperation>(sharedFiles.size());
  978. // prepare operations to insert or update files to save in the given folder
  979. for (OCFile file : sharedFiles) {
  980. ContentValues cv = new ContentValues();
  981. cv.put(ProviderTableMeta.FILE_MODIFIED, file.getModificationTimestamp());
  982. cv.put(ProviderTableMeta.FILE_MODIFIED_AT_LAST_SYNC_FOR_DATA, file.getModificationTimestampAtLastSyncForData());
  983. cv.put(ProviderTableMeta.FILE_CREATION, file.getCreationTimestamp());
  984. cv.put(ProviderTableMeta.FILE_CONTENT_LENGTH, file.getFileLength());
  985. cv.put(ProviderTableMeta.FILE_CONTENT_TYPE, file.getMimetype());
  986. cv.put(ProviderTableMeta.FILE_NAME, file.getFileName());
  987. cv.put(ProviderTableMeta.FILE_PARENT, file.getParentId());
  988. cv.put(ProviderTableMeta.FILE_PATH, file.getRemotePath());
  989. if (!file.isFolder()) {
  990. cv.put(ProviderTableMeta.FILE_STORAGE_PATH, file.getStoragePath());
  991. }
  992. cv.put(ProviderTableMeta.FILE_ACCOUNT_OWNER, mAccount.name);
  993. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE, file.getLastSyncDateForProperties());
  994. cv.put(ProviderTableMeta.FILE_LAST_SYNC_DATE_FOR_DATA, file.getLastSyncDateForData());
  995. cv.put(ProviderTableMeta.FILE_KEEP_IN_SYNC, file.keepInSync() ? 1 : 0);
  996. cv.put(ProviderTableMeta.FILE_ETAG, file.getEtag());
  997. cv.put(ProviderTableMeta.FILE_SHARE_BY_LINK, file.isShareByLink() ? 1 : 0);
  998. cv.put(ProviderTableMeta.FILE_PUBLIC_LINK, file.getPublicLink());
  999. boolean existsByPath = fileExists(file.getRemotePath());
  1000. if (existsByPath || fileExists(file.getFileId())) {
  1001. // updating an existing file
  1002. operations.add(ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI).
  1003. withValues(cv).
  1004. withSelection( ProviderTableMeta._ID + "=?",
  1005. new String[] { String.valueOf(file.getFileId()) })
  1006. .build());
  1007. } else {
  1008. // adding a new file
  1009. operations.add(ContentProviderOperation.newInsert(ProviderTableMeta.CONTENT_URI).withValues(cv).build());
  1010. }
  1011. }
  1012. // apply operations in batch
  1013. if (operations.size() > 0) {
  1014. @SuppressWarnings("unused")
  1015. ContentProviderResult[] results = null;
  1016. Log_OC.d(TAG, "Sending " + operations.size() + " operations to FileContentProvider");
  1017. try {
  1018. if (getContentResolver() != null) {
  1019. results = getContentResolver().applyBatch(MainApp.getAuthority(), operations);
  1020. } else {
  1021. results = getContentProviderClient().applyBatch(operations);
  1022. }
  1023. } catch (OperationApplicationException e) {
  1024. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1025. } catch (RemoteException e) {
  1026. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1027. }
  1028. }
  1029. }
  1030. }
  1031. public void removeShare(OCShare share){
  1032. Uri share_uri = ProviderTableMeta.CONTENT_URI_SHARE;
  1033. String where = ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?" + " AND " + ProviderTableMeta.FILE_PATH + "=?";
  1034. String [] whereArgs = new String[]{mAccount.name, share.getPath()};
  1035. if (getContentProviderClient() != null) {
  1036. try {
  1037. getContentProviderClient().delete(share_uri, where, whereArgs);
  1038. } catch (RemoteException e) {
  1039. e.printStackTrace();
  1040. }
  1041. } else {
  1042. getContentResolver().delete(share_uri, where, whereArgs);
  1043. }
  1044. }
  1045. public void saveSharesDB(ArrayList<OCShare> shares) {
  1046. saveShares(shares);
  1047. ArrayList<OCFile> sharedFiles = new ArrayList<OCFile>();
  1048. for (OCShare share : shares) {
  1049. // Get the path
  1050. String path = share.getPath();
  1051. if (share.isFolder()) {
  1052. path = path + FileUtils.PATH_SEPARATOR;
  1053. }
  1054. // Update OCFile with data from share: ShareByLink ¿and publicLink?
  1055. OCFile file = getFileByPath(path);
  1056. if (file != null) {
  1057. if (share.getShareType().equals(ShareType.PUBLIC_LINK)) {
  1058. file.setShareByLink(true);
  1059. sharedFiles.add(file);
  1060. }
  1061. }
  1062. }
  1063. updateSharedFiles(sharedFiles);
  1064. }
  1065. public void saveSharesInFolder(ArrayList<OCShare> shares, OCFile folder) {
  1066. cleanSharedFilesInFolder(folder);
  1067. ArrayList<ContentProviderOperation> operations = new ArrayList<ContentProviderOperation>();
  1068. operations = prepareRemoveSharesInFolder(folder, operations);
  1069. if (shares != null) {
  1070. // prepare operations to insert or update files to save in the given folder
  1071. for (OCShare share : shares) {
  1072. ContentValues cv = new ContentValues();
  1073. cv.put(ProviderTableMeta.OCSHARES_FILE_SOURCE, share.getFileSource());
  1074. cv.put(ProviderTableMeta.OCSHARES_ITEM_SOURCE, share.getItemSource());
  1075. cv.put(ProviderTableMeta.OCSHARES_SHARE_TYPE, share.getShareType().getValue());
  1076. cv.put(ProviderTableMeta.OCSHARES_SHARE_WITH, share.getShareWith());
  1077. cv.put(ProviderTableMeta.OCSHARES_PATH, share.getPath());
  1078. cv.put(ProviderTableMeta.OCSHARES_PERMISSIONS, share.getPermissions());
  1079. cv.put(ProviderTableMeta.OCSHARES_SHARED_DATE, share.getSharedDate());
  1080. cv.put(ProviderTableMeta.OCSHARES_EXPIRATION_DATE, share.getExpirationDate());
  1081. cv.put(ProviderTableMeta.OCSHARES_TOKEN, share.getToken());
  1082. cv.put(ProviderTableMeta.OCSHARES_SHARE_WITH_DISPLAY_NAME, share.getSharedWithDisplayName());
  1083. cv.put(ProviderTableMeta.OCSHARES_IS_DIRECTORY, share.isFolder() ? 1 : 0);
  1084. cv.put(ProviderTableMeta.OCSHARES_USER_ID, share.getUserId());
  1085. cv.put(ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED, share.getIdRemoteShared());
  1086. cv.put(ProviderTableMeta.OCSHARES_ACCOUNT_OWNER, mAccount.name);
  1087. /*
  1088. if (shareExists(share.getIdRemoteShared())) {
  1089. // updating an existing share resource
  1090. operations.add(ContentProviderOperation.newUpdate(ProviderTableMeta.CONTENT_URI_SHARE).
  1091. withValues(cv).
  1092. withSelection( ProviderTableMeta.OCSHARES_ID_REMOTE_SHARED + "=?",
  1093. new String[] { String.valueOf(share.getIdRemoteShared()) })
  1094. .build());
  1095. } else {
  1096. */
  1097. // adding a new share resource
  1098. operations.add(ContentProviderOperation.newInsert(ProviderTableMeta.CONTENT_URI_SHARE).withValues(cv).build());
  1099. //}
  1100. }
  1101. }
  1102. // apply operations in batch
  1103. if (operations.size() > 0) {
  1104. @SuppressWarnings("unused")
  1105. ContentProviderResult[] results = null;
  1106. Log_OC.d(TAG, "Sending " + operations.size() + " operations to FileContentProvider");
  1107. try {
  1108. if (getContentResolver() != null) {
  1109. results = getContentResolver().applyBatch(MainApp.getAuthority(), operations);
  1110. } else {
  1111. results = getContentProviderClient().applyBatch(operations);
  1112. }
  1113. } catch (OperationApplicationException e) {
  1114. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1115. } catch (RemoteException e) {
  1116. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1117. }
  1118. }
  1119. //}
  1120. }
  1121. private ArrayList<ContentProviderOperation> prepareRemoveSharesInFolder(OCFile folder, ArrayList<ContentProviderOperation> preparedOperations) {
  1122. if (folder != null) {
  1123. String where = ProviderTableMeta.OCSHARES_PATH + "=?" + " AND " + ProviderTableMeta.OCSHARES_ACCOUNT_OWNER + "=?";
  1124. String [] whereArgs = new String[]{ "", mAccount.name };
  1125. Vector<OCFile> files = getFolderContent(folder);
  1126. for (OCFile file : files) {
  1127. whereArgs[0] = file.getRemotePath();
  1128. preparedOperations.add(ContentProviderOperation.newDelete(ProviderTableMeta.CONTENT_URI_SHARE)
  1129. .withSelection(where, whereArgs)
  1130. .build());
  1131. }
  1132. }
  1133. return preparedOperations;
  1134. /*
  1135. if (operations.size() > 0) {
  1136. try {
  1137. if (getContentResolver() != null) {
  1138. getContentResolver().applyBatch(MainApp.getAuthority(), operations);
  1139. } else {
  1140. getContentProviderClient().applyBatch(operations);
  1141. }
  1142. } catch (OperationApplicationException e) {
  1143. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1144. } catch (RemoteException e) {
  1145. Log_OC.e(TAG, "Exception in batch of operations " + e.getMessage());
  1146. }
  1147. }
  1148. */
  1149. /*
  1150. if (getContentResolver() != null) {
  1151. getContentResolver().delete(ProviderTableMeta.CONTENT_URI_SHARE,
  1152. where,
  1153. whereArgs);
  1154. } else {
  1155. try {
  1156. getContentProviderClient().delete( ProviderTableMeta.CONTENT_URI_SHARE,
  1157. where,
  1158. whereArgs);
  1159. } catch (RemoteException e) {
  1160. Log_OC.e(TAG, "Exception deleting shares in a folder " + e.getMessage());
  1161. }
  1162. }
  1163. */
  1164. //}
  1165. }
  1166. }