UploadFileOperation.java 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496
  1. /**
  2. * ownCloud Android client application
  3. *
  4. * @author David A. Velasco
  5. * Copyright (C) 2015 ownCloud Inc.
  6. *
  7. * This program is free software: you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License version 2,
  9. * as published by the Free Software Foundation.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. *
  19. */
  20. package com.owncloud.android.operations;
  21. import java.io.File;
  22. import java.io.FileInputStream;
  23. import java.io.FileOutputStream;
  24. import java.io.IOException;
  25. import java.io.InputStream;
  26. import java.io.OutputStream;
  27. import java.nio.channels.FileChannel;
  28. import java.util.HashSet;
  29. import java.util.Iterator;
  30. import java.util.Set;
  31. import java.util.concurrent.atomic.AtomicBoolean;
  32. import org.apache.commons.httpclient.HttpStatus;
  33. import org.apache.commons.httpclient.methods.RequestEntity;
  34. import android.accounts.Account;
  35. import android.content.Context;
  36. import android.net.Uri;
  37. import com.owncloud.android.MainApp;
  38. import com.owncloud.android.datamodel.FileDataStorageManager;
  39. import com.owncloud.android.datamodel.OCFile;
  40. import com.owncloud.android.files.services.FileUploader;
  41. import com.owncloud.android.lib.common.OwnCloudClient;
  42. import com.owncloud.android.lib.common.network.OnDatatransferProgressListener;
  43. import com.owncloud.android.lib.common.network.ProgressiveDataTransferer;
  44. import com.owncloud.android.lib.common.operations.OperationCancelledException;
  45. import com.owncloud.android.lib.common.operations.RemoteOperation;
  46. import com.owncloud.android.lib.common.operations.RemoteOperationResult;
  47. import com.owncloud.android.lib.common.operations.RemoteOperationResult.ResultCode;
  48. import com.owncloud.android.lib.common.utils.Log_OC;
  49. import com.owncloud.android.lib.resources.files.ChunkedUploadRemoteFileOperation;
  50. import com.owncloud.android.lib.resources.files.ExistenceCheckRemoteOperation;
  51. import com.owncloud.android.lib.resources.files.UploadRemoteFileOperation;
  52. import com.owncloud.android.utils.FileStorageUtils;
  53. import com.owncloud.android.utils.UriUtils;
  54. /**
  55. * Remote operation performing the upload of a file to an ownCloud server
  56. */
  57. public class UploadFileOperation extends RemoteOperation {
  58. private static final String TAG = UploadFileOperation.class.getSimpleName();
  59. private Account mAccount;
  60. private OCFile mFile;
  61. private OCFile mOldFile;
  62. private String mRemotePath = null;
  63. private boolean mChunked = false;
  64. private boolean mIsInstant = false;
  65. private boolean mRemoteFolderToBeCreated = false;
  66. private boolean mForceOverwrite = false;
  67. private int mLocalBehaviour = FileUploader.LOCAL_BEHAVIOUR_COPY;
  68. private boolean mWasRenamed = false;
  69. private String mOriginalFileName = null;
  70. private String mOriginalStoragePath = null;
  71. private Set<OnDatatransferProgressListener> mDataTransferListeners = new HashSet<OnDatatransferProgressListener>();
  72. private AtomicBoolean mCancellationRequested = new AtomicBoolean(false);
  73. private Context mContext;
  74. private UploadRemoteFileOperation mUploadOperation;
  75. protected RequestEntity mEntity = null;
  76. public UploadFileOperation( Account account,
  77. OCFile file,
  78. boolean chunked,
  79. boolean isInstant,
  80. boolean forceOverwrite,
  81. int localBehaviour,
  82. Context context) {
  83. if (account == null)
  84. throw new IllegalArgumentException("Illegal NULL account in UploadFileOperation " +
  85. "creation");
  86. if (file == null)
  87. throw new IllegalArgumentException("Illegal NULL file in UploadFileOperation creation");
  88. if (file.getStoragePath() == null || file.getStoragePath().length() <= 0) {
  89. throw new IllegalArgumentException(
  90. "Illegal file in UploadFileOperation; storage path invalid: "
  91. + file.getStoragePath());
  92. }
  93. mAccount = account;
  94. mFile = file;
  95. mRemotePath = file.getRemotePath();
  96. mChunked = chunked;
  97. mIsInstant = isInstant;
  98. mForceOverwrite = forceOverwrite;
  99. mLocalBehaviour = localBehaviour;
  100. mOriginalStoragePath = mFile.getStoragePath();
  101. mOriginalFileName = mFile.getFileName();
  102. mContext = context;
  103. }
  104. public Account getAccount() {
  105. return mAccount;
  106. }
  107. public String getFileName() {
  108. return mOriginalFileName;
  109. }
  110. public OCFile getFile() {
  111. return mFile;
  112. }
  113. public OCFile getOldFile() {
  114. return mOldFile;
  115. }
  116. public String getOriginalStoragePath() {
  117. return mOriginalStoragePath;
  118. }
  119. public String getStoragePath() {
  120. return mFile.getStoragePath();
  121. }
  122. public String getRemotePath() {
  123. return mFile.getRemotePath();
  124. }
  125. public String getMimeType() {
  126. return mFile.getMimetype();
  127. }
  128. public boolean isInstant() {
  129. return mIsInstant;
  130. }
  131. public boolean isRemoteFolderToBeCreated() {
  132. return mRemoteFolderToBeCreated;
  133. }
  134. public void setRemoteFolderToBeCreated() {
  135. mRemoteFolderToBeCreated = true;
  136. }
  137. public boolean getForceOverwrite() {
  138. return mForceOverwrite;
  139. }
  140. public boolean wasRenamed() {
  141. return mWasRenamed;
  142. }
  143. public Set<OnDatatransferProgressListener> getDataTransferListeners() {
  144. return mDataTransferListeners;
  145. }
  146. public void addDatatransferProgressListener (OnDatatransferProgressListener listener) {
  147. synchronized (mDataTransferListeners) {
  148. mDataTransferListeners.add(listener);
  149. }
  150. if (mEntity != null) {
  151. ((ProgressiveDataTransferer)mEntity).addDatatransferProgressListener(listener);
  152. }
  153. }
  154. public void removeDatatransferProgressListener(OnDatatransferProgressListener listener) {
  155. synchronized (mDataTransferListeners) {
  156. mDataTransferListeners.remove(listener);
  157. }
  158. if (mEntity != null) {
  159. ((ProgressiveDataTransferer)mEntity).removeDatatransferProgressListener(listener);
  160. }
  161. }
  162. @Override
  163. protected RemoteOperationResult run(OwnCloudClient client) {
  164. RemoteOperationResult result = null;
  165. boolean localCopyPassed = false, nameCheckPassed = false;
  166. File temporalFile = null, originalFile = new File(mOriginalStoragePath), expectedFile = null;
  167. try {
  168. // / rename the file to upload, if necessary
  169. if (!mForceOverwrite) {
  170. String remotePath = getAvailableRemotePath(client, mRemotePath);
  171. mWasRenamed = !remotePath.equals(mRemotePath);
  172. if (mWasRenamed) {
  173. createNewOCFile(remotePath);
  174. }
  175. }
  176. nameCheckPassed = true;
  177. String expectedPath = FileStorageUtils.getDefaultSavePathFor(mAccount.name, mFile); // /
  178. // not
  179. // before
  180. // getAvailableRemotePath()
  181. // !!!
  182. expectedFile = new File(expectedPath);
  183. // check location of local file; if not the expected, copy to a
  184. // temporal file before upload (if COPY is the expected behaviour)
  185. if (!mOriginalStoragePath.equals(expectedPath) &&
  186. mLocalBehaviour == FileUploader.LOCAL_BEHAVIOUR_COPY) {
  187. if (FileStorageUtils.getUsableSpace(mAccount.name) < originalFile.length()) {
  188. result = new RemoteOperationResult(ResultCode.LOCAL_STORAGE_FULL);
  189. return result; // error condition when the file should be
  190. // copied
  191. } else {
  192. String temporalPath = FileStorageUtils.getTemporalPath(mAccount.name) +
  193. mFile.getRemotePath();
  194. mFile.setStoragePath(temporalPath);
  195. temporalFile = new File(temporalPath);
  196. File temporalParent = temporalFile.getParentFile();
  197. temporalParent.mkdirs();
  198. if (!temporalParent.isDirectory()) {
  199. throw new IOException("Unexpected error: parent directory could not be created");
  200. }
  201. temporalFile.createNewFile();
  202. if (!temporalFile.isFile()) {
  203. throw new IOException("Unexpected error: target file could not be created");
  204. }
  205. InputStream in = null;
  206. OutputStream out = null;
  207. try {
  208. // In case document provider schema as 'content://'
  209. if (mOriginalStoragePath.startsWith(UriUtils.URI_CONTENT_SCHEME)) {
  210. Uri uri = Uri.parse(mOriginalStoragePath);
  211. in = MainApp.getAppContext().getContentResolver().openInputStream(uri);
  212. out = new FileOutputStream(temporalFile);
  213. int nRead;
  214. byte[] data = new byte[16384];
  215. while (!mCancellationRequested.get() &&
  216. (nRead = in.read(data, 0, data.length)) != -1) {
  217. out.write(data, 0, nRead);
  218. }
  219. out.flush();
  220. } else {
  221. if (!mOriginalStoragePath.equals(temporalPath)) { // preventing
  222. // weird
  223. // but
  224. // possible
  225. // situation
  226. in = new FileInputStream(originalFile);
  227. out = new FileOutputStream(temporalFile);
  228. byte[] buf = new byte[1024];
  229. int len;
  230. while (!mCancellationRequested.get() && (len = in.read(buf)) > 0) {
  231. out.write(buf, 0, len);
  232. }
  233. }
  234. }
  235. if (mCancellationRequested.get()) {
  236. result = new RemoteOperationResult(new OperationCancelledException());
  237. }
  238. } catch (Exception e) {
  239. result = new RemoteOperationResult(ResultCode.LOCAL_STORAGE_NOT_COPIED);
  240. return result;
  241. } finally {
  242. try {
  243. if (in != null)
  244. in.close();
  245. } catch (Exception e) {
  246. Log_OC.d(TAG, "Weird exception while closing input stream for " +
  247. mOriginalStoragePath + " (ignoring)", e);
  248. }
  249. try {
  250. if (out != null)
  251. out.close();
  252. } catch (Exception e) {
  253. Log_OC.d(TAG, "Weird exception while closing output stream for " +
  254. expectedPath + " (ignoring)", e);
  255. }
  256. }
  257. }
  258. }
  259. localCopyPassed = (result == null);
  260. /// perform the upload
  261. if ( mChunked &&
  262. (new File(mFile.getStoragePath())).length() >
  263. ChunkedUploadRemoteFileOperation.CHUNK_SIZE ) {
  264. mUploadOperation = new ChunkedUploadRemoteFileOperation(mFile.getStoragePath(),
  265. mFile.getRemotePath(), mFile.getMimetype(), mFile.getEtagInConflict());
  266. } else {
  267. mUploadOperation = new UploadRemoteFileOperation(mFile.getStoragePath(),
  268. mFile.getRemotePath(), mFile.getMimetype(), mFile.getEtagInConflict());
  269. }
  270. Iterator <OnDatatransferProgressListener> listener = mDataTransferListeners.iterator();
  271. while (listener.hasNext()) {
  272. mUploadOperation.addDatatransferProgressListener(listener.next());
  273. }
  274. if (mCancellationRequested.get()) {
  275. throw new OperationCancelledException();
  276. }
  277. result = mUploadOperation.execute(client);
  278. /// move local temporal file or original file to its corresponding
  279. // location in the ownCloud local folder
  280. if (result.isSuccess()) {
  281. if (mLocalBehaviour == FileUploader.LOCAL_BEHAVIOUR_FORGET) {
  282. mFile.setStoragePath("");
  283. } else {
  284. mFile.setStoragePath(expectedPath);
  285. File fileToMove = null;
  286. if (temporalFile != null) { // FileUploader.LOCAL_BEHAVIOUR_COPY
  287. // ; see where temporalFile was
  288. // set
  289. fileToMove = temporalFile;
  290. } else { // FileUploader.LOCAL_BEHAVIOUR_MOVE
  291. fileToMove = originalFile;
  292. }
  293. if (!expectedFile.equals(fileToMove)) {
  294. File expectedFolder = expectedFile.getParentFile();
  295. expectedFolder.mkdirs();
  296. if (expectedFolder.isDirectory()){
  297. if (!fileToMove.renameTo(expectedFile)){
  298. // try to copy and then delete
  299. expectedFile.createNewFile();
  300. FileChannel inChannel = new FileInputStream(fileToMove).getChannel();
  301. FileChannel outChannel = new FileOutputStream(expectedFile).getChannel();
  302. try {
  303. inChannel.transferTo(0, inChannel.size(), outChannel);
  304. fileToMove.delete();
  305. } catch (Exception e){
  306. mFile.setStoragePath(null); // forget the local file
  307. // by now, treat this as a success; the file was
  308. // uploaded; the user won't like that the local file
  309. // is not linked, but this should be a very rare
  310. // fail;
  311. // the best option could be show a warning message
  312. // (but not a fail)
  313. // result = new
  314. // RemoteOperationResult(ResultCode.LOCAL_STORAGE_NOT_MOVED);
  315. // return result;
  316. }
  317. finally {
  318. if (inChannel != null) inChannel.close();
  319. if (outChannel != null) outChannel.close();
  320. }
  321. }
  322. } else {
  323. mFile.setStoragePath(null);
  324. }
  325. }
  326. }
  327. FileDataStorageManager.triggerMediaScan(originalFile.getAbsolutePath());
  328. FileDataStorageManager.triggerMediaScan(expectedFile.getAbsolutePath());
  329. } else if (result.getHttpCode() == HttpStatus.SC_PRECONDITION_FAILED ) {
  330. result = new RemoteOperationResult(ResultCode.SYNC_CONFLICT);
  331. }
  332. } catch (Exception e) {
  333. result = new RemoteOperationResult(e);
  334. } finally {
  335. if (temporalFile != null && !originalFile.equals(temporalFile)) {
  336. temporalFile.delete();
  337. }
  338. if (result == null){
  339. result = new RemoteOperationResult(ResultCode.UNKNOWN_ERROR);
  340. }
  341. if (result.isSuccess()) {
  342. Log_OC.i(TAG, "Upload of " + mOriginalStoragePath + " to " + mRemotePath + ": " +
  343. result.getLogMessage());
  344. } else {
  345. if (result.getException() != null) {
  346. String complement = "";
  347. if (!nameCheckPassed) {
  348. complement = " (while checking file existence in server)";
  349. } else if (!localCopyPassed) {
  350. complement = " (while copying local file to " +
  351. FileStorageUtils.getSavePath(mAccount.name)
  352. + ")";
  353. }
  354. Log_OC.e(TAG, "Upload of " + mOriginalStoragePath + " to " + mRemotePath +
  355. ": " + result.getLogMessage() + complement, result.getException());
  356. } else {
  357. Log_OC.e(TAG, "Upload of " + mOriginalStoragePath + " to " + mRemotePath +
  358. ": " + result.getLogMessage());
  359. }
  360. }
  361. }
  362. return result;
  363. }
  364. private void createNewOCFile(String newRemotePath) {
  365. // a new OCFile instance must be created for a new remote path
  366. OCFile newFile = new OCFile(newRemotePath);
  367. newFile.setCreationTimestamp(mFile.getCreationTimestamp());
  368. newFile.setFileLength(mFile.getFileLength());
  369. newFile.setMimetype(mFile.getMimetype());
  370. newFile.setModificationTimestamp(mFile.getModificationTimestamp());
  371. newFile.setModificationTimestampAtLastSyncForData(
  372. mFile.getModificationTimestampAtLastSyncForData());
  373. newFile.setEtag(mFile.getEtag());
  374. newFile.setFavorite(mFile.isFavorite());
  375. newFile.setLastSyncDateForProperties(mFile.getLastSyncDateForProperties());
  376. newFile.setLastSyncDateForData(mFile.getLastSyncDateForData());
  377. newFile.setStoragePath(mFile.getStoragePath());
  378. newFile.setParentId(mFile.getParentId());
  379. mOldFile = mFile;
  380. mFile = newFile;
  381. }
  382. /**
  383. * Checks if remotePath does not exist in the server and returns it, or adds
  384. * a suffix to it in order to avoid the server file is overwritten.
  385. *
  386. * @param wc
  387. * @param remotePath
  388. * @return
  389. */
  390. private String getAvailableRemotePath(OwnCloudClient wc, String remotePath) throws Exception {
  391. boolean check = existsFile(wc, remotePath);
  392. if (!check) {
  393. return remotePath;
  394. }
  395. int pos = remotePath.lastIndexOf(".");
  396. String suffix = "";
  397. String extension = "";
  398. if (pos >= 0) {
  399. extension = remotePath.substring(pos + 1);
  400. remotePath = remotePath.substring(0, pos);
  401. }
  402. int count = 2;
  403. do {
  404. suffix = " (" + count + ")";
  405. if (pos >= 0) {
  406. check = existsFile(wc, remotePath + suffix + "." + extension);
  407. }
  408. else {
  409. check = existsFile(wc, remotePath + suffix);
  410. }
  411. count++;
  412. } while (check);
  413. if (pos >= 0) {
  414. return remotePath + suffix + "." + extension;
  415. } else {
  416. return remotePath + suffix;
  417. }
  418. }
  419. private boolean existsFile(OwnCloudClient client, String remotePath){
  420. ExistenceCheckRemoteOperation existsOperation =
  421. new ExistenceCheckRemoteOperation(remotePath, mContext, false);
  422. RemoteOperationResult result = existsOperation.execute(client);
  423. return result.isSuccess();
  424. }
  425. public void cancel() {
  426. mCancellationRequested = new AtomicBoolean(true);
  427. if (mUploadOperation != null) {
  428. mUploadOperation.cancel();
  429. }
  430. }
  431. }