| /* |
| * Copyright (C) 2016 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package com.android.documentsui.services; |
| |
| import static android.content.ContentResolver.wrap; |
| |
| import static com.android.documentsui.base.SharedMinimal.DEBUG; |
| import static com.android.documentsui.services.FileOperationService.OPERATION_MOVE; |
| |
| import android.app.Notification; |
| import android.app.Notification.Builder; |
| import android.content.Context; |
| import android.net.Uri; |
| import android.os.DeadObjectException; |
| import android.os.Messenger; |
| import android.os.RemoteException; |
| import android.provider.DocumentsContract; |
| import android.provider.DocumentsContract.Document; |
| import android.util.Log; |
| |
| import com.android.documentsui.MetricConsts; |
| import com.android.documentsui.Metrics; |
| import com.android.documentsui.R; |
| import com.android.documentsui.base.DocumentInfo; |
| import com.android.documentsui.base.DocumentStack; |
| import com.android.documentsui.base.Features; |
| import com.android.documentsui.base.UserId; |
| import com.android.documentsui.clipping.UrisSupplier; |
| |
| import java.io.FileNotFoundException; |
| |
| import javax.annotation.Nullable; |
| |
| // TODO: Stop extending CopyJob. |
| final class MoveJob extends CopyJob { |
| |
| private static final String TAG = "MoveJob"; |
| |
| private final @Nullable Uri mSrcParentUri; |
| |
| // mSrcParent may be populated during setup. |
| private @Nullable DocumentInfo mSrcParent; |
| |
| /** |
| * Moves files to a destination identified by {@code destination}. |
| * Performs most work by delegating to CopyJob, then deleting |
| * a file after it has been copied. |
| * |
| * @see @link {@link Job} constructor for most param descriptions. |
| */ |
| MoveJob(Context service, Listener listener, String id, DocumentStack destination, |
| UrisSupplier srcs, @Nullable Uri srcParent, Messenger messenger, Features features) { |
| super(service, listener, id, OPERATION_MOVE, destination, srcs, messenger, features); |
| mSrcParentUri = srcParent; |
| } |
| |
| @Override |
| Builder createProgressBuilder() { |
| return super.createProgressBuilder( |
| service.getString(R.string.move_notification_title), |
| R.drawable.ic_menu_copy, |
| service.getString(android.R.string.cancel), |
| R.drawable.ic_cab_cancel); |
| } |
| |
| @Override |
| public Notification getSetupNotification() { |
| return getSetupNotification(service.getString(R.string.move_preparing)); |
| } |
| |
| @Override |
| public Notification getProgressNotification() { |
| return getProgressNotification(R.string.copy_remaining); |
| } |
| |
| @Override |
| Notification getFailureNotification() { |
| return getFailureNotification( |
| R.plurals.move_error_notification_title, R.drawable.ic_menu_copy); |
| } |
| |
| @Override |
| public boolean setUp() { |
| if (mSrcParentUri != null) { |
| try { |
| mSrcParent = DocumentInfo.fromUri(appContext.getContentResolver(), mSrcParentUri, |
| UserId.DEFAULT_USER); |
| } catch (FileNotFoundException e) { |
| Log.e(TAG, "Failed to create srcParent.", e); |
| failureCount = mResourceUris.getItemCount(); |
| return false; |
| } |
| } |
| |
| return super.setUp(); |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * Only check space for moves across authorities. For now we don't know if the doc in |
| * {@link #mSrcs} is in the same root of destination, and if it's optimized move in the same |
| * root it should succeed regardless of free space, but it's for sure a failure if there is no |
| * enough free space if docs are moved from another authority. |
| */ |
| @Override |
| boolean checkSpace() { |
| long size = 0; |
| for (DocumentInfo src : mResolvedDocs) { |
| if (!src.authority.equals(stack.getRoot().authority)) { |
| if (src.isDirectory()) { |
| try { |
| size += calculateFileSizesRecursively(getClient(src), src.derivedUri); |
| } catch (RemoteException|ResourceException e) { |
| Log.w(TAG, "Failed to obtain client for %s" + src.derivedUri + ".", e); |
| |
| // Failed to calculate size, but move may still succeed. |
| return true; |
| } |
| } else { |
| size += src.size; |
| } |
| } |
| } |
| |
| return verifySpaceAvailable(size); |
| } |
| |
| void processDocument(DocumentInfo src, DocumentInfo srcParent, DocumentInfo dest) |
| throws ResourceException { |
| // When moving within the same provider, try to use optimized moving. |
| // If not supported, then fallback to byte-by-byte copy/move. |
| if (src.authority.equals(dest.authority) && (srcParent != null || mSrcParent != null)) { |
| if ((src.flags & Document.FLAG_SUPPORTS_MOVE) != 0) { |
| try { |
| if (DocumentsContract.moveDocument(wrap(getClient(src)), src.derivedUri, |
| srcParent != null ? srcParent.derivedUri : mSrcParent.derivedUri, |
| dest.derivedUri) != null) { |
| Metrics.logFileOperated(operationType, MetricConsts.OPMODE_PROVIDER); |
| makeOptimizedCopyProgress(src); |
| return; |
| } |
| } catch (FileNotFoundException | RemoteException | RuntimeException e) { |
| if (e instanceof DeadObjectException) { |
| releaseClient(src); |
| } |
| Metrics.logFileOperationFailure( |
| appContext, MetricConsts.SUBFILEOP_QUICK_MOVE, src.derivedUri); |
| Log.e(TAG, "Provider side move failed for: " + src.derivedUri |
| + " due to an exception: ", e); |
| } |
| // If optimized move fails, then fallback to byte-by-byte copy. |
| if (DEBUG) { |
| Log.d(TAG, "Fallback to byte-by-byte move for: " + src.derivedUri); |
| } |
| } |
| } |
| |
| // Moving virtual files by bytes is not supported. This is because, it would involve |
| // conversion, and the source file should not be deleted in such case (as it's a different |
| // file). |
| if (src.isVirtual()) { |
| throw new ResourceException("Cannot move virtual file %s byte by byte.", |
| src.derivedUri); |
| } |
| |
| // If we couldn't do an optimized copy...we fall back to vanilla byte copy. |
| byteCopyDocument(src, dest); |
| |
| // Remove the source document. |
| if(!isCanceled()) { |
| deleteDocument(src, srcParent); |
| } |
| } |
| |
| @Override |
| public String toString() { |
| return new StringBuilder() |
| .append("MoveJob") |
| .append("{") |
| .append("id=" + id) |
| .append(", uris=" + mResourceUris) |
| .append(", docs=" + mResolvedDocs) |
| .append(", srcParent=" + mSrcParent) |
| .append(", destination=" + stack) |
| .append("}") |
| .toString(); |
| } |
| } |