权限拒绝:无法附加文件,文件需要导出提供程序,或 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)