微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

android-自定义选择器活动:SecurityException UID n没有对content:// uri的许可

我正在构建一个Chooser应用程序,该应用程序将替换本地的Android Share对话框.效果很好,除非我尝试通过长按图像>从Chrome共享图像.分享图像.

我发现Google没有捕获到异常(崩溃),因此我可以通过Logcat进行查看:

>在Google上进行图片搜索.
>选择图像(这应该显示预览)
>长按图像
>选择“共享图像”
>我的选择器活动弹出
>选择Google
> Google因以下错误而崩溃:

java.lang.SecurityException: UID 10130 does not have permission to content://com.android.chrome.FileProvider/images/screenshot/15307295588677864462883877407218.jpg [user 0]

我的代码(简体):

@Override
public void onCreate() {
    handleIntent();
}

private void handleIntent() {

    // Get intent and payload
    mIntent = getIntent();
    mPayloadIntent = (Intent) mIntent.getParcelableExtra(Intent.EXTRA_INTENT);

    // Nullify some things for queryIntentActivities (or no results will be found)
    mPayloadIntent.setComponent(null);
    mPayloadIntent.setPackage(null);

    // Retrieve a list of targets we can send mPayloadIntent to..
    List<ResolveInfo> targets = context.getPackageManager().queryIntentActivities(mPayloadIntent, 0);
    // etc...

}

private void onClickTarget(ResolveInfo target) {

    // Prepare..
    ComponentName compName = new ComponentName(
                target.activityInfo.applicationInfo.packageName,
                target.activityInfo.name);

    // Build a 'new' shareIntent
    Intent shareIntent = new Intent(mPayloadIntent);
    shareIntent.addFlags(Intent.FLAG_ACTIVITY_FORWARD_RESULT | Intent.FLAG_ACTIVITY_PREVIoUS_IS_TOP);
    shareIntent.setComponent(compName);

    // Start the targeted activity with the shareIntent
    startActivity(shareIntent);
    finish();

}

AndroidManifest.xml:

<activity
    android:name=".ActShareReplace"
    android:label="Sharedr"
    android:theme="@style/AppTheme.TransparentActivity"
    >
    <intent-filter>
        <action android:name="android.intent.action.CHOOSER" />
        <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>
</activity>

如果我查看Intent.ACTION_CHOOSER的文档,它会说:

If you need to grant URI permissions through a chooser, you must specify the permissions to be granted on the ACTION_CHOOSER Intent in addition to the EXTRA_INTENT inside. This means using setClipData(ClipData) to specify the URIs to be granted as well as FLAG_GRANT_READ_URI_PERMISSION and/or FLAG_GRANT_WRITE_URI_PERMISSION as appropriate.

我不确定这是否是我的应用程序必须执行的操作,还是调用选择程序活动的应用程序的责任-但我认为是后者.我的应用无法为其接收的意图设置URI权限,可以吗?

无论如何,如果我检查mIntent和mPayloadIntent上的多余标记和标志,则会得到:

mIntent只有附加功能,没有标志(据我所知):

android.intent.extra.CHOSEN_COMPONENT_INTENT_SENDER
IntentSender{4fa3901: android.os.BinderProxy@3aec3a6} (android.content.IntentSender)

android.intent.extra.INTENT
Intent { act=android.intent.action.SEND typ=image/jpeg flg=0x80001 clip={image/jpeg U:content://com.android.chrome.FileProvider/images/screenshot/15307316967108618905323381238187.jpg} (has extras) } (android.content.Intent)

android.intent.extra.TITLE
Share via (java.lang.String)

mPayloadIntent:

android.intent.extra.STREAM
content://com.android.chrome.FileProvider/images/screenshot/1530731945132897653908815339041.jpg (android.net.Uri$HierarchicalUri)

> FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET
> FLAG_ACTIVITY_NEW_DOCUMENT
> FLAG_GRANT_READ_URI_PERMISSION

因此,mPayloadIntent确实具有FLAG_GRANT_READ_URI_PERMISSION,但mIntent没有.根据文档,它应该.

我已经读到我的应用有可能使用了URI权限,所以I tried caching the file myself,但是当我尝试通过ContentResolver访问URI时,我得到了-您猜到了-权限错误.

然后,我意识到我可能不必缓存文件,因为Android原生的Chooser Activity似乎也没有这样做.这就是我现在所在的位置.回到原点.

这是Chrome错误吗?一个Android错误?还是我做错了什么?

我会很乐意归咎于Chrome并提交错误报告,但是有人在从事类似项目的工作(遇到同样的问题),告诉我Whatsapp也有类似的问题.它也通过content:// uri共享图像.

为了完整起见,我正在使用Android 8.1的Pixel 2016进行测试.我不知道其他人(遇到WA的同一问题)正在使用什么.

解决方法:

Is this a Chrome bug? An Android bug? Or am I doing something wrong?

我的猜测是这是一个客户端错误,来自于直接创建ACTION_CHOOSER Intent对象而不是通过Intent.createChooser()的人们. Intent.createChooser()看起来像是从您所谓的mPayloadIntent中获取标志并将其添加到mIntent中.

您应该可以自己对此进行测试.创建一个剪贴簿应用,该应用创建带有ACTION_SEND Intent的EXTRA_STREAM指向某段内容(例如,由FileProvider提供).然后,尝试通过三种方式调用选择器:

>通过Intent.createChooser()包装Intent
>通过ACTION_CHOOSER Intent包装Intent,您可以按照文档所说的在两个Intent对象上放置标志
>通过ACTION_CHOOSER意图包装该意图,您可以在ACTION_CHOOSER意图上跳过标志

如果我是正确的,则#1和#2可以工作,而#3将以与您看到的相同的基本故障模式失败.

如果到目前为止我的理论仍然有效,请尝试再次运行这三个应用程序,但这一次使用系统选择器.我的猜测是,作为核心操作系统的一部分,系统选择器确实会获得一些特殊的好处,并且所有这三种都可以使用.否则,Chrome和WhatsApp的开发人员可能会在测试中遇到此问题并予以解决.

而且,如果所有这些理论都成立了……您就会有些困惑.我认为更多的人使用Intent.createChooser()而不是直接使用ACTION_CHOOSER,因为Intent.createChooser()更简单.而且,使用ACTION_CHOOSER的人员中的某些人可能实际上遵循了文档…

哈哈哈哈哈哈哈哈哈哈哈…哈哈哈哈哈哈哈哈哈哈哈!

…对于这些,您还可以.而且,某些使用ACTION_CHOOSER的人可能在EXTRA_STREAM中拥有一个Uri,该Uri可以被世界读取(这不是一个好主意,但在这里对您有用).仅适用于手动创建ACTION_CHOOSER,无法正确设置Intent标志,但是确实保护其内容错误客户端,因此您将无法正确处理Intent.

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐