与 FileProvider 共享文件时的权限拒绝

Paw*_*ęza 29 android android-intent android-permissions android-fileprovider

我正在尝试与 FileProvider 共享文件。我检查了该文件是否与 gmail、google drive 等应用程序正确共享。即使抛出以下异常:

2019-08-28 11:43:03.169 12573-12595/com.example.name E/DatabaseUtils: Writing exception to parcel
    java.lang.SecurityException: Permission Denial: reading androidx.core.content.FileProvider uri content://com.example.name.provider/external_files/Android/data/com.example.name/files/allergy_report.pdf from pid=6005, 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.query(ContentProvider.java:231)
        at android.content.ContentProviderNative.onTransact(ContentProviderNative.java:104)
        at android.os.Binder.execTransactInternal(Binder.java:1021)
        at android.os.Binder.execTransact(Binder.java:994)
Run Code Online (Sandbox Code Playgroud)

提供者

<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/file_provider_paths" />
</provider>
Run Code Online (Sandbox Code Playgroud)

file_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>
Run Code Online (Sandbox Code Playgroud)

分享意向

Intent intentShareFile = new Intent(Intent.ACTION_SEND);
        File fileWithinMyDir = new File(targetPdf);

        if(fileWithinMyDir.exists()) {
            intentShareFile.setType("application/pdf");
            Uri uri = FileProvider.getUriForFile(getActivity(), BuildConfig.APPLICATION_ID + ".provider", fileWithinMyDir);
            intentShareFile.putExtra(Intent.EXTRA_STREAM, uri);
            intentShareFile.putExtra(Intent.EXTRA_SUBJECT,
                    "Sharing File...");
            intentShareFile.putExtra(Intent.EXTRA_TEXT, "Sharing File...");
            intentShareFile.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            startActivity(Intent.createChooser(intentShareFile, "Share File"));
        }
Run Code Online (Sandbox Code Playgroud)

希望您能指出我的错误,为什么当应用程序被正确授予权限并且共享正常工作时会抛出此异常。

编辑:

我发现问题在于:

startActivity(Intent.createChooser(intentShareFile, "Share File"));
Run Code Online (Sandbox Code Playgroud)

当我把它简单地改成

startActivity(intentShareFile);
Run Code Online (Sandbox Code Playgroud)

然而,它显示了一些不同的选择应用程序的布局。但我仍然无法弄清楚为什么原始选择器不起作用。

Iak*_*vos 60

抱歉回复晚了。我是这样解决的:

Intent chooser = Intent.createChooser(intentShareFile, "Share File");

List<ResolveInfo> resInfoList = this.getPackageManager().queryIntentActivities(chooser, PackageManager.MATCH_DEFAULT_ONLY);

for (ResolveInfo resolveInfo : resInfoList) {
    String packageName = resolveInfo.activityInfo.packageName;
    this.grantUriPermission(packageName, uri, Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
}

startActivity(chooser);
Run Code Online (Sandbox Code Playgroud)

这对我有帮助:通过意图拒绝文件提供程序的权限

  • 惊人的!我最初错过的关键要点:传递选择器,而不是传递给 queryIntentActivities() 的意图 (4认同)
  • 天哪,我已经处理这个问题好几天了!这解决了我的问题,我不知道为什么我只是复制并粘贴它哈哈。我要读读刚刚救了我屁股的东西!谢谢! (3认同)
  • 那么“intent.putParcelableArrayListExtra(Intent.EXTRA_STREAM, uris);”又如何,其中 uris 是“ArrayList”? (3认同)
  • 这适用于目标 31 (2认同)

Har*_*men 8

除了上面Nit的回答之外,还可以ClipData直接设置要共享的,uri如下所示。

intent.setClipData(ClipData.newRawUri("", uri));
Run Code Online (Sandbox Code Playgroud)

这可以防止呈现意图选择器时出现安全异常。

如果要共享多个uri,可以将clipdata设置为多个uri,以防止安全异常。总结一下:


Intent intent = new Intent();
intent.setType(mimeType);
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);

if (uris.size() == 0) { 
   return;
}
else if (uris.size() == 1) {
   Uri uri = uris.get(0);
   intent.setAction(Intent.ACTION_SEND);
   intent.putExtra(Intent.EXTRA_STREAM, uri);
   intent.setClipData(ClipData.newRawUri("", uri));   
}
else {
  intent.setAction(Intent.ACTION_SEND_MULTIPLE);
  intent.putParcelableArrayListExtra(Intent.EXTRA_STREAM, uris);
  ClipData clipData = ClipData.newRawUri("", uris.get(0));

  for (int i = 1; i < uris.size(); i++) { 
      Uri uri = uris.get(i); 
      clipData.addItem(new ClipData.Item(uri));
  }
  intent.setClipData(clipData);

}

startActivity(Intent.createChooser(intent, title));

Run Code Online (Sandbox Code Playgroud)


Nit*_*Nit 5

还有一种方法可以通过 Intent 标志授予 URI 权限,而无需手动操作grantUriPermission()并保留Intent.createChooser().

Intent.createChooser()文档状态:

如果目标 Intent 已指定 FLAG_GRANT_READ_URI_PERMISSION 或 FLAG_GRANT_WRITE_URI_PERMISSION,则这些标志也将在返回的选择器 Intent 中设置,并适当设置其 ClipData:如果非空,则直接反映 getClipData(),或从构建的新 ClipData获取数据()。

因此,如果原始 Intent 在它的ClipData或 中设置了 Uri setData(),它应该可以正常工作。

在您的示例中,ACTION_SEND意图支持从setClipData()Jelly Bean (Android 4.1) 开始的 Uri 集。

长话短说,这是您的代码的一个工作示例:

Intent intentShareFile = new Intent(Intent.ACTION_SEND);
File fileWithinMyDir = new File(targetPdf);

if(fileWithinMyDir.exists()) {
    String mimeType = "application/pdf";
    String[] mimeTypeArray = new String[] { mimeType };
    
    intentShareFile.setType(mimeType);
    Uri uri = FileProvider.getUriForFile(getActivity(), BuildConfig.APPLICATION_ID + ".provider", fileWithinMyDir);
    
    // Add the uri as a ClipData
    intentShareFile.setClipData(new ClipData(
        "A label describing your file to the user",
        mimeTypeArray,
        new ClipData.Item(path)
    ));
    
    // EXTRA_STREAM is kept for compatibility with old applications
    intentShareFile.putExtra(Intent.EXTRA_STREAM, uri);
    
    intentShareFile.putExtra(Intent.EXTRA_SUBJECT,
            "Sharing File...");
    intentShareFile.putExtra(Intent.EXTRA_TEXT, "Sharing File...");
    intentShareFile.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
    startActivity(Intent.createChooser(intentShareFile, "Share File"));
}
Run Code Online (Sandbox Code Playgroud)