UploadFileOperation.java 19 KB

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