权限拒绝:无法附加文件,文件需要导出提供程序,或 grantUriPermission(),API 29
Posted
技术标签:
【中文标题】权限拒绝:无法附加文件,文件需要导出提供程序,或 grantUriPermission(),API 29【英文标题】:Permission Denial: Couln't attach file, File requires the provider be exported, or grantUriPermission(), API 29 【发布时间】:2020-10-01 13:32:28 【问题描述】:我一直在尝试从外部存储共享一个图像,但它在 API 29 中不起作用。错误消息是:权限被拒绝,文件需要导出提供程序,或 grantUriPermission()。当我谷歌它似乎是旧版本的问题。我不知道为什么我在使用 29 API 时遇到这个问题。 我已经使用了所有与grantUriPermission() 相关的解决方案,您可以在我的方法中看到:
private fun shareInEmail()
val filename = "Avoir+$timeStamp.jpg"
val filelocation =
File(Environment.getExternalStorageDirectory().getAbsolutePath(), filename)
//val path = Uri.fromFile(filelocation)
val path: Uri = FileProvider.getUriForFile(
requireContext(),
context!!.applicationContext.packageName + ".provider",
filelocation
)
val emailIntent = Intent(Intent.ACTION_SEND)
//if current version is greater than 21 version
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP )
emailIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
else if(Build.VERSION.SDK_INT == Build.VERSION_CODES.Q) // if current version is equal to 29 version
//grant permision for app with package to all
val resInfoList: List<ResolveInfo> = requireContext().packageManager
.queryIntentActivities(emailIntent, PackageManager.MATCH_DEFAULT_ONLY)
for (resolveInfo in resInfoList)
val packageName: String = resolveInfo.activityInfo.packageName
requireContext().grantUriPermission(
packageName,
path,
Intent.FLAG_GRANT_WRITE_URI_PERMISSION or Intent.FLAG_GRANT_READ_URI_PERMISSION
)
else //lower versions
val resInfoList: List<ResolveInfo> = requireContext().packageManager
.queryIntentActivities(emailIntent, PackageManager.MATCH_DEFAULT_ONLY)
for (resolveInfo in resInfoList)
val packageName: String = resolveInfo.activityInfo.packageName
requireContext().grantUriPermission(
packageName,
path,
Intent.FLAG_GRANT_WRITE_URI_PERMISSION or Intent.FLAG_GRANT_READ_URI_PERMISSION
)
//emailIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
// set the type to 'email'
emailIntent.type = "vnd.android.cursor.dir/email"
val to = arrayOf("contact@mjclambres.fr")
emailIntent.putExtra(Intent.EXTRA_EMAIL, to)
// the attachment
emailIntent.putExtra(Intent.EXTRA_STREAM, path)
// the mail subject
emailIntent.putExtra(Intent.EXTRA_SUBJECT, "Transaction")
if (emailIntent.resolveActivity(context!!.getPackageManager()) != null)
startActivity(Intent.createChooser(emailIntent, "Transaction"))
我指定,如果是 API 是 29,授予权限。但还是不行
为了分享内容,我使用了 FileProvider。
这是我的清单:
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="$applicationId.provider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/provider_paths"/>
</provider>
这是我的 provider_paths.xml 文件
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path
name="external_files"
path="." />
</paths>
这是日志:
E/DatabaseUtils: Writing exception to parcel
java.lang.SecurityException: Permission Denial: reading androidx.core.content.FileProvider uri content://com.ideasfactory.mjcprojet.provider/.provider/Avoir%2B2020-06-11%2019%3A16.jpg from pid=15561, uid=1000 requires the provider be exported, or grantUriPermission()
at android.content.ContentProvider.enforceReadPermissionInner(ContentProvider.java:729)
at android.content.ContentProvider$Transport.enforceReadPermission(ContentProvider.java:602)
at android.content.ContentProvider$Transport.enforceFilePermission(ContentProvider.java:593)
at android.content.ContentProvider$Transport.openTypedAssetFile(ContentProvider.java:507)
at android.content.ContentProviderNative.onTransact(ContentProviderNative.java:307)
at android.os.Binder.execTransactInternal(Binder.java:1021)
at android.os.Binder.execTransact(Binder.java:994)
我一直在这个 wyle,我不知道我错过了什么。感谢您的帮助。
【问题讨论】:
【参考方案1】:经过几个小时的搜索,我终于找到了基于enter link description here的解决方案。我必须在清单中添加这一行:android:requestLegacyExternalStorage="true"
以使内容在版本 10、Q API 29 和一切正常工作中可重新使用。我可以附加该文件。
【讨论】:
是的,该请求将使其工作,但 Android 11 呢?这完全忽略了这个权限。您找到解决方法了吗?以上是关于权限拒绝:无法附加文件,文件需要导出提供程序,或 grantUriPermission(),API 29的主要内容,如果未能解决你的问题,请参考以下文章
Laravel 无法以附加模式打开流或文件“storage/laravel.log”:无法打开流:RHEL8 中的权限被拒绝
流或文件 \"/var/www/html/storage/logs/laravel.log\" 无法以附加模式打开...权限被拒绝。 gcp“云运行”
SQL Server附加数据库失败:无法打开物理文件,操作系统错误5:拒绝访问”解决方法
无法打开物理文件 XXX.mdf",操作系统错误 5.5(拒绝访问) 的解决办法
信息:无法打开物理文件“路径”。操作错误5:“5(拒绝访问。)”(MICROSOFT SQL Server,错误:5120)