/*
* Copyright (C) 2017 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.bluetooth.opp;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.content.pm.ProviderInfo;
import android.net.Uri;
import android.os.UserHandle;
import android.os.UserManager;
import android.support.v4.content.FileProvider;
import android.util.Log;
import java.io.File;
/**
* A FileProvider for files received by Bluetooth share
*/
public class BluetoothOppFileProvider extends FileProvider {
private static final String TAG = "BluetoothOppFileProvider";
private Context mContext = null;
private ProviderInfo mProviderInfo = null;
private boolean mRegisteredReceiver = false;
private boolean mInitialized = false;
/** Broadcast receiver that attach FileProvider info when user unlocks the phone for the
* first time after reboot and the credential-encrypted storage is available.
*/
private final BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() {
@Override
public void onReceive(final Context context, Intent intent) {
if (Intent.ACTION_USER_UNLOCKED.equals(intent.getAction())) {
attachInfo(mContext, mProviderInfo);
}
}
};
/**
* After the FileProvider is instantiated, this method is called to provide the system with
* information about the provider. The actual initialization is delayed until user unlock the
* device
*
* @param context A {@link Context} for the current component.
* @param info A {@link ProviderInfo} for the new provider.
*/
@Override
public void attachInfo(Context context, ProviderInfo info) {
synchronized (this) {
mContext = context;
mProviderInfo = info;
if (!mRegisteredReceiver) {
IntentFilter userFilter = new IntentFilter();
userFilter.addAction(Intent.ACTION_USER_UNLOCKED);
mContext.registerReceiverAsUser(
mBroadcastReceiver, UserHandle.CURRENT, userFilter, null, null);
mRegisteredReceiver = true;
}
UserManager userManager = (UserManager) mContext.getSystemService(Context.USER_SERVICE);
if (userManager.isUserUnlocked()) {
if (!mInitialized) {
if (Constants.DEBUG) Log.d(TAG, "Initialized");
super.attachInfo(mContext, mProviderInfo);
mInitialized = true;
}
if (mRegisteredReceiver) {
mContext.unregisterReceiver(mBroadcastReceiver);
mRegisteredReceiver = false;
}
}
}
}
/**
* Return a content URI for a given {@link File}. Specific temporary
* permissions for the content URI can be set with
* {@link Context#grantUriPermission(String, Uri, int)}, or added
* to an {@link Intent} by calling {@link Intent#setData(Uri) setData()} and then
* {@link Intent#setFlags(int) setFlags()}; in both cases, the applicable flags are
* {@link Intent#FLAG_GRANT_READ_URI_PERMISSION} and
* {@link Intent#FLAG_GRANT_WRITE_URI_PERMISSION}. A FileProvider can only return a
* content
{@link Uri} for file paths defined in their <paths>
* meta-data element. See the Class Overview for more information.
*
* @param context A {@link Context} for the current component.
* @param authority The authority of a {@link FileProvider} defined in a
* {@code } element in your app's manifest.
* @param file A {@link File} pointing to the filename for which you want a
* content
{@link Uri}.
* @return A content URI for the file. Null if the user hasn't unlock the phone
* @throws IllegalArgumentException When the given {@link File} is outside
* the paths supported by the provider.
*/
public static Uri getUriForFile(Context context, String authority, File file) {
UserManager userManager = (UserManager) context.getSystemService(Context.USER_SERVICE);
if (!userManager.isUserUnlocked()) {
return null;
}
context = context.createCredentialProtectedStorageContext();
return FileProvider.getUriForFile(context, authority, file);
}
}