UploadFileOperation.java 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819
  1. /**
  2. * ownCloud Android client application
  3. *
  4. * @author David A. Velasco
  5. * Copyright (C) 2016 ownCloud Inc.
  6. *
  7. * This program is free software: you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License version 2,
  9. * as published by the Free Software Foundation.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. *
  19. */
  20. package com.owncloud.android.operations;
  21. import android.accounts.Account;
  22. import android.content.Context;
  23. import android.net.Uri;
  24. import com.owncloud.android.datamodel.FileDataStorageManager;
  25. import com.owncloud.android.datamodel.OCFile;
  26. import com.owncloud.android.datamodel.ThumbnailsCacheManager;
  27. import com.owncloud.android.db.OCUpload;
  28. import com.owncloud.android.db.PreferenceManager;
  29. import com.owncloud.android.files.services.FileUploader;
  30. import com.owncloud.android.lib.common.OwnCloudClient;
  31. import com.owncloud.android.lib.common.network.OnDatatransferProgressListener;
  32. import com.owncloud.android.lib.common.network.ProgressiveDataTransferer;
  33. import com.owncloud.android.lib.common.operations.OperationCancelledException;
  34. import com.owncloud.android.lib.common.operations.RemoteOperation;
  35. import com.owncloud.android.lib.common.operations.RemoteOperationResult;
  36. import com.owncloud.android.lib.common.operations.RemoteOperationResult.ResultCode;
  37. import com.owncloud.android.lib.common.utils.Log_OC;
  38. import com.owncloud.android.lib.resources.files.ChunkedUploadRemoteFileOperation;
  39. import com.owncloud.android.lib.resources.files.ExistenceCheckRemoteOperation;
  40. import com.owncloud.android.lib.resources.files.ReadRemoteFileOperation;
  41. import com.owncloud.android.lib.resources.files.RemoteFile;
  42. import com.owncloud.android.lib.resources.files.UploadRemoteFileOperation;
  43. import com.owncloud.android.operations.common.SyncOperation;
  44. import com.owncloud.android.utils.ConnectivityUtils;
  45. import com.owncloud.android.utils.FileStorageUtils;
  46. import com.owncloud.android.utils.MimeType;
  47. import com.owncloud.android.utils.MimetypeIconUtil;
  48. import com.owncloud.android.utils.UriUtils;
  49. import org.apache.commons.httpclient.HttpStatus;
  50. import org.apache.commons.httpclient.methods.RequestEntity;
  51. import java.io.File;
  52. import java.io.FileInputStream;
  53. import java.io.FileOutputStream;
  54. import java.io.IOException;
  55. import java.io.InputStream;
  56. import java.io.OutputStream;
  57. import java.nio.channels.FileChannel;
  58. import java.util.HashSet;
  59. import java.util.Iterator;
  60. import java.util.Set;
  61. import java.util.concurrent.atomic.AtomicBoolean;
  62. /**
  63. * Operation performing the update in the ownCloud server
  64. * of a file that was modified locally.
  65. */
  66. public class UploadFileOperation extends SyncOperation {
  67. public static final int CREATED_BY_USER = 0;
  68. public static final int CREATED_AS_INSTANT_PICTURE = 1;
  69. public static final int CREATED_AS_INSTANT_VIDEO = 2;
  70. public static OCFile obtainNewOCFileToUpload(String remotePath, String localPath, String mimeType) {
  71. // MIME type
  72. if (mimeType == null || mimeType.length() <= 0) {
  73. mimeType = MimetypeIconUtil.getBestMimeTypeByFilename(localPath);
  74. }
  75. OCFile newFile = new OCFile(remotePath);
  76. newFile.setStoragePath(localPath);
  77. newFile.setLastSyncDateForProperties(0);
  78. newFile.setLastSyncDateForData(0);
  79. // size
  80. if (localPath != null && localPath.length() > 0) {
  81. File localFile = new File(localPath);
  82. newFile.setFileLength(localFile.length());
  83. newFile.setLastSyncDateForData(localFile.lastModified());
  84. } // don't worry about not assigning size, the problems with localPath
  85. // are checked when the UploadFileOperation instance is created
  86. newFile.setMimetype(mimeType);
  87. return newFile;
  88. }
  89. private static final String TAG = UploadFileOperation.class.getSimpleName();
  90. private Account mAccount;
  91. /**
  92. * OCFile which is to be uploaded.
  93. */
  94. private OCFile mFile;
  95. /**
  96. * Original OCFile which is to be uploaded in case file had to be renamed
  97. * (if forceOverwrite==false and remote file already exists).
  98. */
  99. private OCFile mOldFile;
  100. private String mRemotePath = null;
  101. private boolean mChunked = false;
  102. private boolean mRemoteFolderToBeCreated = false;
  103. private boolean mForceOverwrite = false;
  104. private int mLocalBehaviour = FileUploader.LOCAL_BEHAVIOUR_COPY;
  105. private int mCreatedBy = CREATED_BY_USER;
  106. private boolean mWasRenamed = false;
  107. private long mOCUploadId = -1;
  108. /**
  109. * Local path to file which is to be uploaded (before any possible renaming or moving).
  110. */
  111. private String mOriginalStoragePath = null;
  112. private Set<OnDatatransferProgressListener> mDataTransferListeners = new HashSet<OnDatatransferProgressListener>();
  113. private OnRenameListener mRenameUploadListener;
  114. private final AtomicBoolean mCancellationRequested = new AtomicBoolean(false);
  115. private final AtomicBoolean mUploadStarted = new AtomicBoolean(false);
  116. private Context mContext;
  117. private UploadRemoteFileOperation mUploadOperation;
  118. protected RequestEntity mEntity = null;
  119. public UploadFileOperation(Account account,
  120. OCFile file,
  121. OCUpload upload,
  122. boolean chunked,
  123. boolean forceOverwrite,
  124. int localBehaviour,
  125. Context context
  126. ) {
  127. if (account == null)
  128. throw new IllegalArgumentException("Illegal NULL account in UploadFileOperation " +
  129. "creation");
  130. if (upload == null)
  131. throw new IllegalArgumentException("Illegal NULL file in UploadFileOperation creation");
  132. if (upload.getLocalPath() == null || upload.getLocalPath().length() <= 0) {
  133. throw new IllegalArgumentException(
  134. "Illegal file in UploadFileOperation; storage path invalid: "
  135. + upload.getLocalPath());
  136. }
  137. mAccount = account;
  138. if (file == null) {
  139. mFile = obtainNewOCFileToUpload(
  140. upload.getRemotePath(),
  141. upload.getLocalPath(),
  142. upload.getMimeType()
  143. );
  144. } else {
  145. mFile = file;
  146. }
  147. mRemotePath = upload.getRemotePath();
  148. mChunked = chunked;
  149. mForceOverwrite = forceOverwrite;
  150. mLocalBehaviour = localBehaviour;
  151. mOriginalStoragePath = mFile.getStoragePath();
  152. mContext = context;
  153. mOCUploadId = upload.getUploadId();
  154. mCreatedBy = upload.getCreadtedBy();
  155. mRemoteFolderToBeCreated = upload.isCreateRemoteFolder();
  156. }
  157. public Account getAccount() {
  158. return mAccount;
  159. }
  160. public String getFileName() {
  161. return (mFile != null) ? mFile.getFileName() : null;
  162. }
  163. public OCFile getFile() {
  164. return mFile;
  165. }
  166. /**
  167. * If remote file was renamed, return original OCFile which was uploaded. Is
  168. * null is file was not renamed.
  169. */
  170. public OCFile getOldFile() {
  171. return mOldFile;
  172. }
  173. public String getOriginalStoragePath() {
  174. return mOriginalStoragePath;
  175. }
  176. public String getStoragePath() {
  177. return mFile.getStoragePath();
  178. }
  179. public String getRemotePath() {
  180. return mFile.getRemotePath();
  181. }
  182. public String getMimeType() {
  183. return mFile.getMimetype();
  184. }
  185. public int getLocalBehaviour() {
  186. return mLocalBehaviour;
  187. }
  188. public void setRemoteFolderToBeCreated() {
  189. mRemoteFolderToBeCreated = true;
  190. }
  191. public boolean wasRenamed() {
  192. return mWasRenamed;
  193. }
  194. public void setCreatedBy(int createdBy) {
  195. mCreatedBy = createdBy;
  196. if (createdBy < CREATED_BY_USER || CREATED_AS_INSTANT_VIDEO < createdBy) {
  197. mCreatedBy = CREATED_BY_USER;
  198. }
  199. }
  200. public int getCreatedBy () {
  201. return mCreatedBy;
  202. }
  203. public boolean isInstantPicture() {
  204. return mCreatedBy == CREATED_AS_INSTANT_PICTURE;
  205. }
  206. public boolean isInstantVideo() {
  207. return mCreatedBy == CREATED_AS_INSTANT_VIDEO;
  208. }
  209. public void setOCUploadId(long id){
  210. mOCUploadId = id;
  211. }
  212. public long getOCUploadId() {
  213. return mOCUploadId;
  214. }
  215. public Set<OnDatatransferProgressListener> getDataTransferListeners() {
  216. return mDataTransferListeners;
  217. }
  218. public void addDatatransferProgressListener (OnDatatransferProgressListener listener) {
  219. synchronized (mDataTransferListeners) {
  220. mDataTransferListeners.add(listener);
  221. }
  222. if (mEntity != null) {
  223. ((ProgressiveDataTransferer)mEntity).addDatatransferProgressListener(listener);
  224. }
  225. if(mUploadOperation != null){
  226. mUploadOperation.addDatatransferProgressListener(listener);
  227. }
  228. }
  229. public void removeDatatransferProgressListener(OnDatatransferProgressListener listener) {
  230. synchronized (mDataTransferListeners) {
  231. mDataTransferListeners.remove(listener);
  232. }
  233. if (mEntity != null) {
  234. ((ProgressiveDataTransferer)mEntity).removeDatatransferProgressListener(listener);
  235. }
  236. if(mUploadOperation != null){
  237. mUploadOperation.removeDatatransferProgressListener(listener);
  238. }
  239. }
  240. public void addRenameUploadListener (OnRenameListener listener) {
  241. mRenameUploadListener = listener;
  242. }
  243. @Override
  244. protected RemoteOperationResult run(OwnCloudClient client) {
  245. mCancellationRequested.set(false);
  246. mUploadStarted.set(true);
  247. RemoteOperationResult result = null;
  248. File temporalFile = null, originalFile = new File(mOriginalStoragePath), expectedFile = null;
  249. try {
  250. /// Check that connectivity conditions are met and delays the upload otherwise
  251. if (delayForWifi()) {
  252. Log_OC.d(TAG, "Upload delayed until WiFi is available: " + getRemotePath());
  253. return new RemoteOperationResult(ResultCode.DELAYED_FOR_WIFI);
  254. }
  255. /// check if the file continues existing before schedule the operation
  256. if (!originalFile.exists()) {
  257. Log_OC.d(TAG, mOriginalStoragePath.toString() + " not exists anymore");
  258. return new RemoteOperationResult(ResultCode.LOCAL_FILE_NOT_FOUND);
  259. }
  260. /// check the existence of the parent folder for the file to upload
  261. String remoteParentPath = new File(getRemotePath()).getParent();
  262. remoteParentPath = remoteParentPath.endsWith(OCFile.PATH_SEPARATOR) ?
  263. remoteParentPath : remoteParentPath + OCFile.PATH_SEPARATOR;
  264. result = grantFolderExistence(remoteParentPath, client);
  265. if (!result.isSuccess()) {
  266. return result;
  267. }
  268. /// set parent local id in uploading file
  269. OCFile parent = getStorageManager().getFileByPath(remoteParentPath);
  270. mFile.setParentId(parent.getFileId());
  271. /// automatic rename of file to upload in case of name collision in server
  272. Log_OC.d(TAG, "Checking name collision in server");
  273. if (!mForceOverwrite) {
  274. String remotePath = getAvailableRemotePath(client, mRemotePath);
  275. mWasRenamed = !remotePath.equals(mRemotePath);
  276. if (mWasRenamed) {
  277. createNewOCFile(remotePath);
  278. Log_OC.d(TAG, "File renamed as " + remotePath);
  279. }
  280. mRemotePath = remotePath;
  281. mRenameUploadListener.onRenameUpload();
  282. }
  283. if (mCancellationRequested.get()) {
  284. throw new OperationCancelledException();
  285. }
  286. String expectedPath = FileStorageUtils.getDefaultSavePathFor(mAccount.name, mFile);
  287. expectedFile = new File(expectedPath);
  288. /// copy the file locally before uploading
  289. if (mLocalBehaviour == FileUploader.LOCAL_BEHAVIOUR_COPY &&
  290. !mOriginalStoragePath.equals(expectedPath)) {
  291. String temporalPath = FileStorageUtils.getTemporalPath(mAccount.name) + mFile.getRemotePath();
  292. mFile.setStoragePath(temporalPath);
  293. temporalFile = new File(temporalPath);
  294. result = copy(originalFile, temporalFile);
  295. if (result != null) {
  296. return result;
  297. }
  298. }
  299. if (mCancellationRequested.get()) {
  300. throw new OperationCancelledException();
  301. }
  302. /// perform the upload
  303. if ( mChunked &&
  304. (new File(mFile.getStoragePath())).length() >
  305. ChunkedUploadRemoteFileOperation.CHUNK_SIZE ) {
  306. mUploadOperation = new ChunkedUploadRemoteFileOperation(mFile.getStoragePath(),
  307. mFile.getRemotePath(), mFile.getMimetype(), mFile.getEtagInConflict());
  308. } else {
  309. mUploadOperation = new UploadRemoteFileOperation(mFile.getStoragePath(),
  310. mFile.getRemotePath(), mFile.getMimetype(), mFile.getEtagInConflict());
  311. }
  312. Iterator <OnDatatransferProgressListener> listener = mDataTransferListeners.iterator();
  313. while (listener.hasNext()) {
  314. mUploadOperation.addDatatransferProgressListener(listener.next());
  315. }
  316. if (mCancellationRequested.get()) {
  317. throw new OperationCancelledException();
  318. }
  319. result = mUploadOperation.execute(client);
  320. /// move local temporal file or original file to its corresponding
  321. // location in the ownCloud local folder
  322. if (result.isSuccess()) {
  323. if (mLocalBehaviour == FileUploader.LOCAL_BEHAVIOUR_FORGET) {
  324. String temporalPath = FileStorageUtils.getTemporalPath(mAccount.name) + mFile.getRemotePath();
  325. if (mOriginalStoragePath.equals(temporalPath)) {
  326. // delete local file is was pre-copied in temporary folder (see .ui.helpers.UriUploader)
  327. temporalFile = new File(temporalPath);
  328. temporalFile.delete();
  329. }
  330. mFile.setStoragePath("");
  331. } else {
  332. mFile.setStoragePath(expectedPath);
  333. if (temporalFile != null) { // FileUploader.LOCAL_BEHAVIOUR_COPY
  334. move(temporalFile, expectedFile);
  335. } else { // FileUploader.LOCAL_BEHAVIOUR_MOVE
  336. move(originalFile, expectedFile);
  337. getStorageManager().deleteFileInMediaScan(originalFile.getAbsolutePath());
  338. }
  339. FileDataStorageManager.triggerMediaScan(expectedFile.getAbsolutePath());
  340. }
  341. } else if (result.getHttpCode() == HttpStatus.SC_PRECONDITION_FAILED ) {
  342. result = new RemoteOperationResult(ResultCode.SYNC_CONFLICT);
  343. }
  344. } catch (Exception e) {
  345. result = new RemoteOperationResult(e);
  346. } finally {
  347. mUploadStarted.set(false);
  348. if (temporalFile != null && !originalFile.equals(temporalFile)) {
  349. temporalFile.delete();
  350. }
  351. if (result == null){
  352. result = new RemoteOperationResult(ResultCode.UNKNOWN_ERROR);
  353. }
  354. if (result.isSuccess()) {
  355. Log_OC.i(TAG, "Upload of " + mOriginalStoragePath + " to " + mRemotePath + ": " +
  356. result.getLogMessage());
  357. } else {
  358. if (result.getException() != null) {
  359. if(result.isCancelled()){
  360. Log_OC.w(TAG, "Upload of " + mOriginalStoragePath + " to " + mRemotePath +
  361. ": " + result.getLogMessage());
  362. } else {
  363. Log_OC.e(TAG, "Upload of " + mOriginalStoragePath + " to " + mRemotePath +
  364. ": " + result.getLogMessage(), result.getException());
  365. }
  366. } else {
  367. Log_OC.e(TAG, "Upload of " + mOriginalStoragePath + " to " + mRemotePath +
  368. ": " + result.getLogMessage());
  369. }
  370. }
  371. }
  372. if (result.isSuccess()) {
  373. saveUploadedFile(client);
  374. } else if (result.getCode() == ResultCode.SYNC_CONFLICT) {
  375. getStorageManager().saveConflict(mFile, mFile.getEtagInConflict());
  376. }
  377. return result;
  378. }
  379. /**
  380. * Checks origin of current upload and network type to decide if should be delayed, according to
  381. * current user preferences.
  382. *
  383. * @return 'True' if the upload was delayed until WiFi connectivity is available, 'false' otherwise.
  384. */
  385. private boolean delayForWifi() {
  386. boolean delayInstantPicture = (
  387. isInstantPicture() && PreferenceManager.instantPictureUploadViaWiFiOnly(mContext)
  388. );
  389. boolean delayInstantVideo = (
  390. isInstantVideo() && PreferenceManager.instantVideoUploadViaWiFiOnly(mContext)
  391. );
  392. return (
  393. (delayInstantPicture || delayInstantVideo) &&
  394. !ConnectivityUtils.isAppConnectedViaUnmeteredWiFi(mContext)
  395. );
  396. }
  397. /**
  398. * Checks the existence of the folder where the current file will be uploaded both
  399. * in the remote server and in the local database.
  400. * <p/>
  401. * If the upload is set to enforce the creation of the folder, the method tries to
  402. * create it both remote and locally.
  403. *
  404. * @param pathToGrant Full remote path whose existence will be granted.
  405. * @return An {@link OCFile} instance corresponding to the folder where the file
  406. * will be uploaded.
  407. */
  408. private RemoteOperationResult grantFolderExistence(String pathToGrant, OwnCloudClient client) {
  409. RemoteOperation operation = new ExistenceCheckRemoteOperation(pathToGrant, mContext, false);
  410. RemoteOperationResult result = operation.execute(client);
  411. if (!result.isSuccess() && result.getCode() == ResultCode.FILE_NOT_FOUND && mRemoteFolderToBeCreated) {
  412. SyncOperation syncOp = new CreateFolderOperation(pathToGrant, true);
  413. result = syncOp.execute(client, getStorageManager());
  414. }
  415. if (result.isSuccess()) {
  416. OCFile parentDir = getStorageManager().getFileByPath(pathToGrant);
  417. if (parentDir == null) {
  418. parentDir = createLocalFolder(pathToGrant);
  419. }
  420. if (parentDir != null) {
  421. result = new RemoteOperationResult(ResultCode.OK);
  422. } else {
  423. result = new RemoteOperationResult(ResultCode.UNKNOWN_ERROR);
  424. }
  425. }
  426. return result;
  427. }
  428. private OCFile createLocalFolder(String remotePath) {
  429. String parentPath = new File(remotePath).getParent();
  430. parentPath = parentPath.endsWith(OCFile.PATH_SEPARATOR) ?
  431. parentPath : parentPath + OCFile.PATH_SEPARATOR;
  432. OCFile parent = getStorageManager().getFileByPath(parentPath);
  433. if (parent == null) {
  434. parent = createLocalFolder(parentPath);
  435. }
  436. if (parent != null) {
  437. OCFile createdFolder = new OCFile(remotePath);
  438. createdFolder.setMimetype(MimeType.DIRECTORY);
  439. createdFolder.setParentId(parent.getFileId());
  440. getStorageManager().saveFile(createdFolder);
  441. return createdFolder;
  442. }
  443. return null;
  444. }
  445. /**
  446. * Create a new OCFile mFile with new remote path. This is required if forceOverwrite==false.
  447. * New file is stored as mFile, original as mOldFile.
  448. * @param newRemotePath new remote path
  449. */
  450. private void createNewOCFile(String newRemotePath) {
  451. // a new OCFile instance must be created for a new remote path
  452. OCFile newFile = new OCFile(newRemotePath);
  453. newFile.setCreationTimestamp(mFile.getCreationTimestamp());
  454. newFile.setFileLength(mFile.getFileLength());
  455. newFile.setMimetype(mFile.getMimetype());
  456. newFile.setModificationTimestamp(mFile.getModificationTimestamp());
  457. newFile.setModificationTimestampAtLastSyncForData(
  458. mFile.getModificationTimestampAtLastSyncForData()
  459. );
  460. newFile.setEtag(mFile.getEtag());
  461. newFile.setFavorite(mFile.isFavorite());
  462. newFile.setLastSyncDateForProperties(mFile.getLastSyncDateForProperties());
  463. newFile.setLastSyncDateForData(mFile.getLastSyncDateForData());
  464. newFile.setStoragePath(mFile.getStoragePath());
  465. newFile.setParentId(mFile.getParentId());
  466. mOldFile = mFile;
  467. mFile = newFile;
  468. }
  469. /**
  470. * Checks if remotePath does not exist in the server and returns it, or adds
  471. * a suffix to it in order to avoid the server file is overwritten.
  472. *
  473. * @param wc
  474. * @param remotePath
  475. * @return
  476. */
  477. private String getAvailableRemotePath(OwnCloudClient wc, String remotePath) {
  478. boolean check = existsFile(wc, remotePath);
  479. if (!check) {
  480. return remotePath;
  481. }
  482. int pos = remotePath.lastIndexOf(".");
  483. String suffix = "";
  484. String extension = "";
  485. if (pos >= 0) {
  486. extension = remotePath.substring(pos + 1);
  487. remotePath = remotePath.substring(0, pos);
  488. }
  489. int count = 2;
  490. do {
  491. suffix = " (" + count + ")";
  492. if (pos >= 0) {
  493. check = existsFile(wc, remotePath + suffix + "." + extension);
  494. }
  495. else {
  496. check = existsFile(wc, remotePath + suffix);
  497. }
  498. count++;
  499. } while (check);
  500. if (pos >= 0) {
  501. return remotePath + suffix + "." + extension;
  502. } else {
  503. return remotePath + suffix;
  504. }
  505. }
  506. private boolean existsFile(OwnCloudClient client, String remotePath){
  507. ExistenceCheckRemoteOperation existsOperation =
  508. new ExistenceCheckRemoteOperation(remotePath, mContext, false);
  509. RemoteOperationResult result = existsOperation.execute(client);
  510. return result.isSuccess();
  511. }
  512. /**
  513. * Allows to cancel the actual upload operation. If actual upload operating
  514. * is in progress it is cancelled, if upload preparation is being performed
  515. * upload will not take place.
  516. */
  517. public void cancel() {
  518. if (mUploadOperation == null) {
  519. if (mUploadStarted.get()) {
  520. Log_OC.d(TAG, "Cancelling upload during upload preparations.");
  521. mCancellationRequested.set(true);
  522. } else {
  523. Log_OC.e(TAG, "No upload in progress. This should not happen.");
  524. }
  525. } else {
  526. Log_OC.d(TAG, "Cancelling upload during actual upload operation.");
  527. mUploadOperation.cancel();
  528. }
  529. }
  530. /**
  531. * As soon as this method return true, upload can be cancel via cancel().
  532. */
  533. public boolean isUploadInProgress() {
  534. return mUploadStarted.get();
  535. }
  536. /**
  537. * TODO rewrite with homogeneous fail handling, remove dependency on {@link RemoteOperationResult},
  538. * TODO use Exceptions instead
  539. *
  540. * @param sourceFile Source file to copy.
  541. * @param targetFile Target location to copy the file.
  542. * @return {@link RemoteOperationResult}
  543. * @throws IOException
  544. */
  545. private RemoteOperationResult copy(File sourceFile, File targetFile) throws IOException {
  546. Log_OC.d(TAG, "Copying local file");
  547. RemoteOperationResult result = null;
  548. if (FileStorageUtils.getUsableSpace(mAccount.name) < sourceFile.length()) {
  549. result = new RemoteOperationResult(ResultCode.LOCAL_STORAGE_FULL);
  550. return result; // error condition when the file should be copied
  551. } else {
  552. Log_OC.d(TAG, "Creating temporal folder");
  553. File temporalParent = targetFile.getParentFile();
  554. temporalParent.mkdirs();
  555. if (!temporalParent.isDirectory()) {
  556. throw new IOException(
  557. "Unexpected error: parent directory could not be created");
  558. }
  559. Log_OC.d(TAG, "Creating temporal file");
  560. targetFile.createNewFile();
  561. if (!targetFile.isFile()) {
  562. throw new IOException(
  563. "Unexpected error: target file could not be created");
  564. }
  565. Log_OC.d(TAG, "Copying file contents");
  566. InputStream in = null;
  567. OutputStream out = null;
  568. try {
  569. if (!mOriginalStoragePath.equals(targetFile.getAbsolutePath())) {
  570. // In case document provider schema as 'content://'
  571. if (mOriginalStoragePath.startsWith(UriUtils.URI_CONTENT_SCHEME)) {
  572. Uri uri = Uri.parse(mOriginalStoragePath);
  573. in = mContext.getContentResolver().openInputStream(uri);
  574. } else {
  575. in = new FileInputStream(sourceFile);
  576. }
  577. out = new FileOutputStream(targetFile);
  578. int nRead;
  579. byte[] buf = new byte[4096];
  580. while (!mCancellationRequested.get() &&
  581. (nRead = in.read(buf)) > -1) {
  582. out.write(buf, 0, nRead);
  583. }
  584. out.flush();
  585. } // else: weird but possible situation, nothing to copy
  586. if (mCancellationRequested.get()) {
  587. result = new RemoteOperationResult(new OperationCancelledException());
  588. return result;
  589. }
  590. } catch (Exception e) {
  591. result = new RemoteOperationResult(ResultCode.LOCAL_STORAGE_NOT_COPIED);
  592. return result;
  593. } finally {
  594. try {
  595. if (in != null)
  596. in.close();
  597. } catch (Exception e) {
  598. Log_OC.d(TAG, "Weird exception while closing input stream for " +
  599. mOriginalStoragePath + " (ignoring)", e);
  600. }
  601. try {
  602. if (out != null)
  603. out.close();
  604. } catch (Exception e) {
  605. Log_OC.d(TAG, "Weird exception while closing output stream for " +
  606. targetFile.getAbsolutePath() + " (ignoring)", e);
  607. }
  608. }
  609. }
  610. return result;
  611. }
  612. /**
  613. * TODO rewrite with homogeneous fail handling, remove dependency on {@link RemoteOperationResult},
  614. * TODO use Exceptions instead
  615. *
  616. * TODO refactor both this and 'copy' in a single method
  617. *
  618. * @param sourceFile Source file to move.
  619. * @param targetFile Target location to move the file.
  620. * @return {@link RemoteOperationResult}
  621. * @throws IOException
  622. */
  623. private void move(File sourceFile, File targetFile) throws IOException {
  624. if (!targetFile.equals(sourceFile)) {
  625. File expectedFolder = targetFile.getParentFile();
  626. expectedFolder.mkdirs();
  627. if (expectedFolder.isDirectory()){
  628. if (!sourceFile.renameTo(targetFile)){
  629. // try to copy and then delete
  630. targetFile.createNewFile();
  631. FileChannel inChannel = new FileInputStream(sourceFile).getChannel();
  632. FileChannel outChannel = new FileOutputStream(targetFile).getChannel();
  633. try {
  634. inChannel.transferTo(0, inChannel.size(), outChannel);
  635. sourceFile.delete();
  636. } catch (Exception e){
  637. mFile.setStoragePath(""); // forget the local file
  638. // by now, treat this as a success; the file was uploaded
  639. // the best option could be show a warning message
  640. }
  641. finally {
  642. if (inChannel != null) inChannel.close();
  643. if (outChannel != null) outChannel.close();
  644. }
  645. }
  646. } else {
  647. mFile.setStoragePath("");
  648. }
  649. }
  650. }
  651. /**
  652. * Saves a OC File after a successful upload.
  653. * <p/>
  654. * A PROPFIND is necessary to keep the props in the local database
  655. * synchronized with the server, specially the modification time and Etag
  656. * (where available)
  657. * <p/>
  658. */
  659. private void saveUploadedFile(OwnCloudClient client) {
  660. OCFile file = mFile;
  661. if (file.fileExists()) {
  662. file = getStorageManager().getFileById(file.getFileId());
  663. }
  664. long syncDate = System.currentTimeMillis();
  665. file.setLastSyncDateForData(syncDate);
  666. // new PROPFIND to keep data consistent with server
  667. // in theory, should return the same we already have
  668. // TODO from the appropriate OC server version, get data from last PUT response headers, instead
  669. // TODO of a new PROPFIND; the latter may fail, specially for chunked uploads
  670. ReadRemoteFileOperation operation = new ReadRemoteFileOperation(getRemotePath());
  671. RemoteOperationResult result = operation.execute(client);
  672. if (result.isSuccess()) {
  673. updateOCFile(file, (RemoteFile) result.getData().get(0));
  674. file.setLastSyncDateForProperties(syncDate);
  675. } else {
  676. Log_OC.e(TAG, "Error reading properties of file after successful upload; this is gonna hurt...");
  677. }
  678. if (mWasRenamed) {
  679. OCFile oldFile = getStorageManager().getFileByPath(mOldFile.getRemotePath());
  680. if (oldFile != null) {
  681. oldFile.setStoragePath(null);
  682. getStorageManager().saveFile(oldFile);
  683. getStorageManager().saveConflict(oldFile, null);
  684. }
  685. // else: it was just an automatic renaming due to a name
  686. // coincidence; nothing else is needed, the storagePath is right
  687. // in the instance returned by mCurrentUpload.getFile()
  688. }
  689. file.setNeedsUpdateThumbnail(true);
  690. getStorageManager().saveFile(file);
  691. getStorageManager().saveConflict(file, null);
  692. FileDataStorageManager.triggerMediaScan(file.getStoragePath());
  693. // generate new Thumbnail
  694. final ThumbnailsCacheManager.ThumbnailGenerationTask task =
  695. new ThumbnailsCacheManager.ThumbnailGenerationTask(getStorageManager(), mAccount);
  696. task.execute(file);
  697. }
  698. private void updateOCFile(OCFile file, RemoteFile remoteFile) {
  699. file.setCreationTimestamp(remoteFile.getCreationTimestamp());
  700. file.setFileLength(remoteFile.getLength());
  701. file.setMimetype(remoteFile.getMimeType());
  702. file.setModificationTimestamp(remoteFile.getModifiedTimestamp());
  703. file.setModificationTimestampAtLastSyncForData(remoteFile.getModifiedTimestamp());
  704. file.setEtag(remoteFile.getEtag());
  705. file.setRemoteId(remoteFile.getRemoteId());
  706. }
  707. public interface OnRenameListener {
  708. void onRenameUpload();
  709. }
  710. }