Google(严重)plans to ruin storage permission,这样应用程序将无法使用标准File API(和文件路径)访问文件系统。 against it中的许多功能都是use a flag,因为它改变了应用访问存储的方式,并且在许多方面都是受限制的API。
因此,如果我们需要在将来的某些Android版本上完全使用SAF(存储访问框架)(在Android Q上,我们至少可以暂时here来使用常规存储权限),希望处理各种存储量并访问那里的所有文件。
例如,假设您要创建一个文件管理器并显示设备的所有存储卷,以显示用户可以授予访问权限的内容,并且如果您已经可以访问每个文件,则只需输入该文件即可。这样的事情看似很合理,但由于我找不到解决办法。
从API 24(here)开始,我们终于可以列出所有存储卷,如下所示:
val storageManager = getSystemService(Context.STORAGE_SERVICE) as StorageManager
val storageVolumes = storageManager.storageVolumes
而且,这是有史以来第一次,我们可以使用Intent请求访问storageVolume(here)。因此,例如,如果我们要请求用户授予对主数据库的访问权限(实际上是从那里开始,实际上并没有真正询问任何内容),我们可以使用以下方法:
startActivityForResult(storageManager.primaryStorageVolume.createOpenDocumentTreeIntent(), REQUEST_CODE__DIRECTORTY_PERMISSION)
代替startActivityForResult(Intent(Intent.ACTION_OPEN_DOCUMENT_TREE), REQUEST_CODE__DIRECTORTY_PERMISSION)
,希望用户在那里选择正确的东西。
要最终获得用户选择的内容,我们需要这样做:
@TargetApi(Build.VERSION_CODES.KITKAT)
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
if (requestCode == REQUEST_CODE__DIRECTORTY_PERMISSION && resultCode == Activity.RESULT_OK && data != null) {
val treeUri = data.data ?: return
contentResolver.takePersistableUriPermission(treeUri, Intent.FLAG_GRANT_READ_URI_PERMISSION or Intent.FLAG_GRANT_WRITE_URI_PERMISSION)
val pickedDir = DocumentFile.fromTreeUri(this, treeUri)
...
到目前为止,我们可以请求对各种存储卷的许可...
但是,如果您想知道您获得了哪些权限,哪些人没有,就会出现问题。
有一个有关Google(uuid)的“范围目录访问”的视频,他们专门谈论StorageVolume类。他们甚至提供有关侦听StorageVolume挂载事件的信息,但他们并没有提供任何有关识别我们可以访问的内容的信息。
StorageVolume类的唯一ID是this,但甚至不能保证它返回任何东西。实际上,它在各种情况下都返回null。例如,主存储的情况。
使用createOpenDocumentTreeIntent
函数时,我注意到其中隐藏了一个Uri,可能会告诉您以哪个开头。它位于Extras内部,位于名为“ android.provider.extra.INITIAL_URI”的键中。例如,当在主存储上检查其值时,我得到了:
content://com.android.externalstorage.documents/root/primary
当我查看Uri时,我在onActivityResult中得到了回报,我得到的东西与#2有点相似,但是对于我展示的treeUri
变量却有所不同:
content://com.android.externalstorage.documents/tree/primary%3A
要获取到目前为止可以访问的内容的列表,可以使用UriPermission:
valsistenceedUriPermissions = contentResolver.persistedUriPermissions
这将返回一个{{3}}列表,每个列表都有一个Uri。可悲的是,当我使用它时,我得到了与#3相同的东西,我无法真正与从StorageVolume获得的东西进行比较:
content://com.android.externalstorage.documents/tree/primary%3A
如您所见,我在存储卷列表和用户授予的内容之间找不到任何类型的映射。
我什至不知道用户是否选择了一个存储卷,因为createOpenDocumentTreeIntent
的功能仅会将用户发送到StorageVolume,但是仍然可以选择一个文件夹。
我唯一拥有的是我在其他问题上发现的许多解决方法功能,而且我认为它们并不可靠,尤其是由于我们实际上无法访问File API和文件路径。
如果您认为它们有用,我会在这里写下它们:
@TargetApi(VERSION_CODES.LOLLIPOP)
private static String getVolumeIdFromTreeUri(final Uri treeUri) {
final String docId = DocumentsContract.getTreeDocumentId(treeUri);
final int end = docId.indexOf(':');
String result = end == -1 ? null : docId.substring(0, end);
return result;
}
private static String getDocumentPathFromTreeUri(final Uri treeUri) {
final String docId = DocumentsContract.getTreeDocumentId(treeUri);
//TODO avoid using spliting of a string (because it uses extra strings creation)
final String[] split = docId.split(":");
if ((split.length >= 2) && (split[1] != null))
return split[1];
else
return File.separator;
}
public static String getFullPathOfDocumentFile(Context context, DocumentFile documentFile) {
String volumePath = getVolumePath(context, getVolumeIdFromTreeUri(documentFile.getUri()));
if (volumePath == null)
return null;
DocumentFile parent = documentFile.getParentFile();
if (parent == null)
return volumePath;
final LinkedList<String> fileHierarchy = new LinkedList<>();
while (true) {
fileHierarchy.add(0, documentFile.getName());
documentFile = parent;
parent = documentFile.getParentFile();
if (parent == null)
break;
}
final StringBuilder sb = new StringBuilder(volumePath).append(File.separator);
for (String fileName : fileHierarchy)
sb.append(fileName).append(File.separator);
return sb.toString();
}
/**
* Get the full path of a document from its tree URI.
*
* @param treeUri The tree RI.
* @return The path (without trailing file separator).
*/
public static String getFullPathFromTreeUri(Context context, final Uri treeUri) {
if (treeUri == null)
return null;
String volumePath = getVolumePath(context, getVolumeIdFromTreeUri(treeUri));
if (volumePath == null)
return File.separator;
if (volumePath.endsWith(File.separator))
volumePath = volumePath.substring(0, volumePath.length() - 1);
String documentPath = getDocumentPathFromTreeUri(treeUri);
if (documentPath.endsWith(File.separator))
documentPath = documentPath.substring(0, documentPath.length() - 1);
if (documentPath.length() > 0)
if (documentPath.startsWith(File.separator))
return volumePath + documentPath;
else return volumePath + File.separator + documentPath;
return volumePath;
}
/**
* Get the path of a certain volume.
*
* @param volumeId The volume id.
* @return The path.
*/
private static String getVolumePath(Context context, final String volumeId) {
if (VERSION.SDK_INT < VERSION_CODES.LOLLIPOP)
return null;
try {
final StorageManager storageManager = (StorageManager) context.getSystemService(Context.STORAGE_SERVICE);
if (VERSION.SDK_INT >= VERSION_CODES.N) {
final Class<?> storageVolumeClazz = StorageVolume.class;
final Method getPath = storageVolumeClazz.getMethod("getPath");
final List<StorageVolume> storageVolumes = storageManager.getStorageVolumes();
for (final StorageVolume storageVolume : storageVolumes) {
final String uuid = storageVolume.getUuid();
final boolean primary = storageVolume.isPrimary();
// primary volume?
if (primary && PRIMARY_VOLUME_NAME.equals(volumeId)) {
return (String) getPath.invoke(storageVolume);
}
// other volumes?
if (uuid != null && uuid.equals(volumeId))
return (String) getPath.invoke(storageVolume);
}
return null;
}
final Class<?> storageVolumeClazz = Class.forName("android.os.storage.StorageVolume");
final Method getVolumeList = storageManager.getClass().getMethod("getVolumeList");
final Method getUuid = storageVolumeClazz.getMethod("getUuid");
//noinspection JavaReflectionMemberAccess
final Method getPath = storageVolumeClazz.getMethod("getPath");
final Method isPrimary = storageVolumeClazz.getMethod("isPrimary");
final Object result = getVolumeList.invoke(storageManager);
final int length = Array.getLength(result);
for (int i = 0; i < length; i++) {
final Object storageVolumeElement = Array.get(result, i);
final String uuid = (String) getUuid.invoke(storageVolumeElement);
final Boolean primary = (Boolean) isPrimary.invoke(storageVolumeElement);
// primary volume?
if (primary && PRIMARY_VOLUME_NAME.equals(volumeId)) {
return (String) getPath.invoke(storageVolumeElement);
}
// other volumes?
if (uuid != null && uuid.equals(volumeId))
return (String) getPath.invoke(storageVolumeElement);
}
// not found.
return null;
} catch (Exception ex) {
return null;
}
}
如何在StorageVolume列表和已授予UriPermission列表之间进行映射?
换句话说,给定一个StorageVolume列表,我怎么知道我可以访问哪些内容,而我没有访问权限,如果我可以访问的话,如何打开它并查看其中的内容呢?
答案 0 :(得分:1)
这里是获得所需东西的另一种方法。这是一种变通方法,就像您在不使用反射或文件路径的情况下发布的那样。
在模拟器上,我看到以下允许访问的项目。
persistedUriPermissions数组内容(仅URI的值):
0 uri = content://com.android.externalstorage.documents/tree/primary%3A
1 uri = content://com.android.externalstorage.documents/tree/1D03-2E0E%3ADownload
2 uri = content://com.android.externalstorage.documents/tree/1D03-2E0E%3A
3 uri = content://com.android.externalstorage.documents/tree/primary%3ADCIM
4 uri = content://com.android.externalstorage.documents/tree/primary%3AAlarms
“%3A”是冒号(“:”)。因此,对于一个卷来说,似乎URI的构造如下,其中“
uri =“ content://com.android.externalstorage.documents/tree/
:”
如果uri是直接在卷下的目录,则结构为:
uri =“ content://com.android.externalstorage.documents/tree/ <卷>:<目录>”
对于结构更深的目录,格式为:
uri =“ content://com.android.externalstorage.documents/tree/ <卷>:<目录> / <目录> / <目录> ...”
因此,这只是从这些格式的URI中提取卷的问题。提取的音量可用作StorageManager.storageVolumes
的键。下面的代码就是这样做的。
在我看来,应该有一种更简单的方法来解决此问题。 API中存储卷和URI之间必须缺少链接。我不能说这种技术可以涵盖所有情况。
我还质疑storageVolume.uuid
返回的UUID,它似乎是32位值。我以为UUID的长度为128位。这是UUID的替代格式还是从UUID衍生而来?有趣的是,一切都将下降! :(
MainActivity.kt
class MainActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
val storageManager = getSystemService(Context.STORAGE_SERVICE) as StorageManager
var storageVolumes = storageManager.storageVolumes
val storageVolumePathsWeHaveAccessTo = HashSet<String>()
checkAccessButton.setOnClickListener {
checkAccessToStorageVolumes()
}
requestAccessButton.setOnClickListener {
storageVolumes = storageManager.storageVolumes
val primaryVolume = storageManager.primaryStorageVolume
val intent = primaryVolume.createOpenDocumentTreeIntent()
startActivityForResult(intent, 1)
}
}
private fun checkAccessToStorageVolumes() {
val storageVolumePathsWeHaveAccessTo = HashSet<String>()
val persistedUriPermissions = contentResolver.persistedUriPermissions
persistedUriPermissions.forEach {
storageVolumePathsWeHaveAccessTo.add(it.uri.toString())
}
val storageManager = getSystemService(Context.STORAGE_SERVICE) as StorageManager
val storageVolumes = storageManager.storageVolumes
for (storageVolume in storageVolumes) {
val uuid = if (storageVolume.isPrimary) {
// Primary storage doesn't get a UUID here.
"primary"
} else {
storageVolume.uuid
}
val volumeUri = uuid?.let { buildVolumeUriFromUuid(it) }
when {
uuid == null ->
Log.d("AppLog", "UUID is null for ${storageVolume.getDescription(this)}!")
storageVolumePathsWeHaveAccessTo.contains(volumeUri) ->
Log.d("AppLog", "Have access to $uuid")
else -> Log.d("AppLog", "Don't have access to $uuid")
}
}
}
private fun buildVolumeUriFromUuid(uuid: String): String {
return DocumentsContract.buildTreeDocumentUri(
"com.android.externalstorage.documents",
"$uuid:"
).toString()
}
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
Log.d("AppLog", "resultCode:$resultCode")
val uri = data?.data ?: return
val takeFlags =
Intent.FLAG_GRANT_READ_URI_PERMISSION or Intent.FLAG_GRANT_WRITE_URI_PERMISSION
contentResolver.takePersistableUriPermission(uri, takeFlags)
Log.d("AppLog", "granted uri: ${uri.path}")
}
}
答案 1 :(得分:0)
编辑:找到了一种解决方法,但是有一天可能无法正常工作。
它使用反射来获取StorageVolume实例的真实路径,并使用我之前获得的路径来获取persistedUriPermissions的路径。如果它们之间有交叉点,则意味着我可以访问storageVolume。
似乎可以在模拟器上工作,该模拟器最终同时具有内部存储和SD卡。
希望我们将获得适当的API,而无需使用反射。
如果没有更好的方法,还有更好的方法,请告诉我。
所以,这是一个例子:
MainActivity.kt
class MainActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
val storageManager = getSystemService(Context.STORAGE_SERVICE) as StorageManager
val storageVolumes = storageManager.storageVolumes
val primaryVolume = storageManager.primaryStorageVolume
checkAccessButton.setOnClickListener {
val persistedUriPermissions = contentResolver.persistedUriPermissions
val storageVolumePathsWeHaveAccessTo = HashSet<String>()
Log.d("AppLog", "got access to paths:")
for (persistedUriPermission in persistedUriPermissions) {
val path = FileUtilEx.getFullPathFromTreeUri(this, persistedUriPermission.uri)
?: continue
Log.d("AppLog", "path: $path")
storageVolumePathsWeHaveAccessTo.add(path)
}
Log.d("AppLog", "storage volumes:")
for (storageVolume in storageVolumes) {
val volumePath = FileUtilEx.getVolumePath(storageVolume)
if (volumePath == null) {
Log.d("AppLog", "storageVolume \"${storageVolume.getDescription(this)}\" - failed to get volumePath")
} else {
val hasAccess = storageVolumePathsWeHaveAccessTo.contains(volumePath)
Log.d("AppLog", "storageVolume \"${storageVolume.getDescription(this)}\" - volumePath:$volumePath - gotAccess? $hasAccess")
}
}
}
requestAccessButton.setOnClickListener {
val intent = primaryVolume.createOpenDocumentTreeIntent()
startActivityForResult(intent, 1)
}
}
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
Log.d("AppLog", "resultCode:$resultCode")
val uri = data?.data ?: return
val takeFlags = Intent.FLAG_GRANT_READ_URI_PERMISSION or Intent.FLAG_GRANT_WRITE_URI_PERMISSION
contentResolver.takePersistableUriPermission(uri, takeFlags)
val fullPathFromTreeUri = FileUtilEx.getFullPathFromTreeUri(this, uri)
Log.d("AppLog", "granted uri:$uri $fullPathFromTreeUri")
}
}
FileUtilEx.java
/**
* Get the full path of a document from its tree URI.
*
* @param treeUri The tree RI.
* @return The path (without trailing file separator).
*/
public static String getFullPathFromTreeUri(Context context, final Uri treeUri) {
if (treeUri == null)
return null;
String volumePath = getVolumePath(context, getVolumeIdFromTreeUri(treeUri));
if (volumePath == null)
return File.separator;
if (volumePath.endsWith(File.separator))
volumePath = volumePath.substring(0, volumePath.length() - 1);
String documentPath = getDocumentPathFromTreeUri(treeUri);
if (documentPath.endsWith(File.separator))
documentPath = documentPath.substring(0, documentPath.length() - 1);
if (documentPath.length() > 0)
if (documentPath.startsWith(File.separator))
return volumePath + documentPath;
else return volumePath + File.separator + documentPath;
return volumePath;
}
public static String getVolumePath(StorageVolume storageVolume){
if (VERSION.SDK_INT < VERSION_CODES.LOLLIPOP)
return null;
try{
final Class<?> storageVolumeClazz = StorageVolume.class;
final Method getPath = storageVolumeClazz.getMethod("getPath");
return (String) getPath.invoke(storageVolume);
} catch (NoSuchMethodException | IllegalAccessException | InvocationTargetException e) {
e.printStackTrace();
}
return null;
}
/**
* Get the path of a certain volume.
*
* @param volumeId The volume id.
* @return The path.
*/
@SuppressLint("ObsoleteSdkInt")
private static String getVolumePath(Context context, final String volumeId) {
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP)
return null;
try {
final StorageManager storageManager = (StorageManager) context.getSystemService(Context.STORAGE_SERVICE);
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) {
final Class<?> storageVolumeClazz = StorageVolume.class;
//noinspection JavaReflectionMemberAccess
final Method getPath = storageVolumeClazz.getMethod("getPath");
final List<StorageVolume> storageVolumes = storageManager.getStorageVolumes();
for (final StorageVolume storageVolume : storageVolumes) {
final String uuid = storageVolume.getUuid();
final boolean primary = storageVolume.isPrimary();
// primary volume?
if (primary && PRIMARY_VOLUME_NAME.equals(volumeId)) {
return (String) getPath.invoke(storageVolume);
}
// other volumes?
if (uuid != null && uuid.equals(volumeId))
return (String) getPath.invoke(storageVolume);
}
return null;
}
final Class<?> storageVolumeClazz = Class.forName("android.os.storage.StorageVolume");
final Method getVolumeList = storageManager.getClass().getMethod("getVolumeList");
final Method getUuid = storageVolumeClazz.getMethod("getUuid");
//noinspection JavaReflectionMemberAccess
final Method getPath = storageVolumeClazz.getMethod("getPath");
final Method isPrimary = storageVolumeClazz.getMethod("isPrimary");
final Object result = getVolumeList.invoke(storageManager);
final int length = Array.getLength(result);
for (int i = 0; i < length; i++) {
final Object storageVolumeElement = Array.get(result, i);
final String uuid = (String) getUuid.invoke(storageVolumeElement);
final Boolean primary = (Boolean) isPrimary.invoke(storageVolumeElement);
// primary volume?
if (primary && PRIMARY_VOLUME_NAME.equals(volumeId)) {
return (String) getPath.invoke(storageVolumeElement);
}
// other volumes?
if (uuid != null && uuid.equals(volumeId))
return (String) getPath.invoke(storageVolumeElement);
}
// not found.
return null;
} catch (Exception ex) {
return null;
}
}
/**
* Get the document path (relative to volume name) for a tree URI (LOLLIPOP).
*
* @param treeUri The tree URI.
* @return the document path.
*/
@TargetApi(VERSION_CODES.LOLLIPOP)
private static String getDocumentPathFromTreeUri(final Uri treeUri) {
final String docId = DocumentsContract.getTreeDocumentId(treeUri);
//TODO avoid using spliting of a string (because it uses extra strings creation)
final String[] split = docId.split(":");
if ((split.length >= 2) && (split[1] != null))
return split[1];
else
return File.separator;
}
/**
* Get the volume ID from the tree URI.
*
* @param treeUri The tree URI.
* @return The volume ID.
*/
@TargetApi(VERSION_CODES.LOLLIPOP)
private static String getVolumeIdFromTreeUri(final Uri treeUri) {
final String docId = DocumentsContract.getTreeDocumentId(treeUri);
final int end = docId.indexOf(':');
String result = end == -1 ? null : docId.substring(0, end);
return result;
}
activity_main.xml
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent"
android:gravity="center" android:orientation="vertical" tools:context=".MainActivity">
<Button
android:id="@+id/checkAccessButton" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="checkAccess"/>
<Button
android:id="@+id/requestAccessButton" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="requestAccess"/>
</LinearLayout>
将其放在一个简单的函数中,在这里:
/** for each storageVolume, tells if we have access or not, via a HashMap (true for each iff we identified it has access*/
fun getStorageVolumesAccessState(context: Context): HashMap<StorageVolume, Boolean> {
val storageManager = context.getSystemService(Context.STORAGE_SERVICE) as StorageManager
val storageVolumes = storageManager.storageVolumes
val persistedUriPermissions = context.contentResolver.persistedUriPermissions
val storageVolumePathsWeHaveAccessTo = HashSet<String>()
// Log.d("AppLog", "got access to paths:")
for (persistedUriPermission in persistedUriPermissions) {
val path = FileUtilEx.getFullPathFromTreeUri(context, persistedUriPermission.uri)
?: continue
// Log.d("AppLog", "path: $path")
storageVolumePathsWeHaveAccessTo.add(path)
}
// Log.d("AppLog", "storage volumes:")
val result = HashMap<StorageVolume, Boolean>(storageVolumes.size)
for (storageVolume in storageVolumes) {
val volumePath = FileUtilEx.getVolumePath(storageVolume)
val hasAccess = volumePath != null && storageVolumePathsWeHaveAccessTo.contains(volumePath)
result[storageVolume] = hasAccess
}
return result
}
答案 2 :(得分:0)
对于 API 30 (Android 11)
@TargetApi(30)
private fun getVolumePathApi30(context:Context, uuid: String): String{
// /storage/emulated/0/Android/data/{packageName}/files
// /storage/0222-9FE1/Android/data/{packageName}/files
val list = ContextCompat.getExternalFilesDirs(context, null)
.map{ it.canonicalPath.replace(reAndroidDataFolder, "") }
// /storage/emulated/0
// /storage/0222-9FE1
val path = if( uuid == "primary") {
list.firstOrNull()
}else {
list.find { it.contains(uuid, ignoreCase = true) }
}
return path ?: error("can't find volume for uuid $uuid")
}