blob: e3b6abde3e38bdbb13eb1e7227ad72db4ba76868 [file] [log] [blame]
/*
* Copyright (C) 2013 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;
import static android.content.ContentResolver.wrap;
import android.content.ContentProviderClient;
import android.content.ContentResolver;
import android.content.Context;
import android.database.Cursor;
import android.net.Uri;
import android.os.ParcelFileDescriptor;
import android.os.RemoteException;
import android.provider.DocumentsContract;
import android.provider.DocumentsContract.Path;
import android.util.Log;
import androidx.annotation.Nullable;
import com.android.documentsui.archives.ArchivesProvider;
import com.android.documentsui.base.DocumentInfo;
import com.android.documentsui.base.RootInfo;
import com.android.documentsui.base.State;
import com.android.documentsui.base.UserId;
import java.io.FileNotFoundException;
import java.util.ArrayList;
import java.util.List;
/**
* Provides synchronous access to {@link DocumentInfo} instances given some identifying information
* and some documents API.
*/
public interface DocumentsAccess {
@Nullable DocumentInfo getRootDocument(RootInfo root);
@Nullable DocumentInfo getDocument(Uri uri, UserId userId);
@Nullable DocumentInfo getArchiveDocument(Uri uri, UserId userId);
boolean isDocumentUri(Uri uri);
@Nullable
Path findDocumentPath(Uri uri, UserId userId)
throws RemoteException, FileNotFoundException, CrossProfileNoPermissionException;
List<DocumentInfo> getDocuments(UserId userId, String authority, List<String> docIds)
throws RemoteException, CrossProfileNoPermissionException;
@Nullable Uri createDocument(DocumentInfo parentDoc, String mimeType, String displayName);
public static DocumentsAccess create(Context context, State state) {
return new RuntimeDocumentAccess(context, state);
}
public final class RuntimeDocumentAccess implements DocumentsAccess {
private static final String TAG = "DocumentAccess";
private final Context mContext;
private final State mState;
private RuntimeDocumentAccess(Context context, State state) {
mContext = context;
mState = state;
}
@Override
@Nullable
public DocumentInfo getRootDocument(RootInfo root) {
return getDocument(DocumentsContract.buildDocumentUri(root.authority, root.documentId),
root.userId);
}
@Override
public @Nullable DocumentInfo getDocument(Uri uri, UserId userId) {
try {
if (mState.canInteractWith(userId)) {
return DocumentInfo.fromUri(userId.getContentResolver(mContext), uri, userId);
}
} catch (FileNotFoundException e) {
Log.w(TAG, "Couldn't create DocumentInfo for uri: " + uri);
}
return null;
}
@Override
public List<DocumentInfo> getDocuments(UserId userId, String authority, List<String> docIds)
throws RemoteException, CrossProfileNoPermissionException {
if (!mState.canInteractWith(userId)) {
throw new CrossProfileNoPermissionException();
}
try (ContentProviderClient client = DocumentsApplication.acquireUnstableProviderOrThrow(
userId.getContentResolver(mContext), authority)) {
List<DocumentInfo> result = new ArrayList<>(docIds.size());
for (String docId : docIds) {
final Uri uri = DocumentsContract.buildDocumentUri(authority, docId);
try (final Cursor cursor = client.query(uri, null, null, null, null)) {
if (!cursor.moveToNext()) {
Log.e(TAG, "Couldn't create DocumentInfo for Uri: " + uri);
throw new RemoteException("Failed to move cursor.");
}
result.add(DocumentInfo.fromCursor(cursor, userId, authority));
}
}
return result;
}
}
@Override
public DocumentInfo getArchiveDocument(Uri uri, UserId userId) {
return getDocument(
ArchivesProvider.buildUriForArchive(uri, ParcelFileDescriptor.MODE_READ_ONLY),
userId);
}
@Override
public boolean isDocumentUri(Uri uri) {
return DocumentsContract.isDocumentUri(mContext, uri);
}
@Override
public Path findDocumentPath(Uri docUri, UserId userId)
throws RemoteException, FileNotFoundException, CrossProfileNoPermissionException {
if (!mState.canInteractWith(userId)) {
throw new CrossProfileNoPermissionException();
}
final ContentResolver resolver = userId.getContentResolver(mContext);
try (final ContentProviderClient client = DocumentsApplication
.acquireUnstableProviderOrThrow(resolver, docUri.getAuthority())) {
return DocumentsContract.findDocumentPath(wrap(client), docUri);
}
}
@Override
public Uri createDocument(DocumentInfo parentDoc, String mimeType, String displayName) {
final ContentResolver resolver = parentDoc.userId.getContentResolver(mContext);
try (ContentProviderClient client = DocumentsApplication.acquireUnstableProviderOrThrow(
resolver, parentDoc.derivedUri.getAuthority())) {
Uri createUri = DocumentsContract.createDocument(
wrap(client), parentDoc.derivedUri, mimeType, displayName);
// If the document info's user is the current user, we can simply return the uri.
// Otherwise, we need to create document with the content resolver from the other
// user. The uri returned from that content resolver does not contain the user
// info. Hence we need to append the other user info to the uri otherwise an app
// will think the uri is from the current user.
// The way to append a userInfo is to use the authority which contains user info
// obtained from the parentDoc.getDocumentUri().
return UserId.CURRENT_USER.equals(parentDoc.userId)
? createUri : appendEncodedParentAuthority(parentDoc, createUri);
} catch (Exception e) {
Log.w(TAG, "Failed to create document", e);
return null;
}
}
private Uri appendEncodedParentAuthority(DocumentInfo parentDoc, Uri uri) {
return uri.buildUpon().encodedAuthority(
parentDoc.getDocumentUri().getAuthority()).build();
}
}
}