隐私合规:检测第三方SDK调用的隐私权限

隐私合规:检测第三方SDK调用的隐私权限

原文地址

隐私合规:检测第三方SDK调用的隐私权限

前言

看了一圈各大网站目前关于隐私合规检测的分享,发现大家几乎都是自己写一套动态代理、Hook或者ASM来实现代码拦截从而检测不合规的代码堆栈,虽然这也是一个很好的方法,但是对于一些很紧急的任务又或者是刚入门的开发者来说,无疑是很耗时间的,所以笔者在这里提供一种目前 Android 11 支持的API思路。官方文档传送门

最近被工信部下架的应用太多了,隐私合规检测特别严,明明把所有能看得到的隐私合规问题都解决了,但是工信部就是说你没解决。Ok,fine! 你说没解决就没解决吧(欲哭无泪😭)

虽然我们是把“看得到”的隐私问题都解决了,但是那些看不到的呢?比如说依赖的第三方 SDK,它们会不会在你不知道的情况下偷偷调用了一些隐私权限呢?相信一般的公司项目都是比较大型的,依赖了各种各样的第三方SDK吧(如果是大佬的话就当我没说,毕竟大佬们都喜欢自己造轮子~)

检测方法

为了让应用及其依赖项访问用户私密数据的过程更加透明,Android 11 引入了数据访问审核功能。

笔者使用的这种方式,是针对Android 11及以上的,因为这是Android 11的新特性,具体可以看官方文档。想要检测应用使用的隐私权限的话,需要先暂时把targetSdkVersion升到30,然后等检测完不合规的地方后再还原回原本使用的 targetSdkVersion 版本。

其实当 Android 11 的新特性出来之后,开发者们肯定或多或少的对AppOpsManager.OnOpNotedCallback有印象,但是真到了需要用的时候,可能就想不起这个回调了,在 App 中注册了这个回调之后,当应用每次发生以下任一事件时都执行相应操作:

  • 应用的代码访问私密数据
  • 依赖库或 SDK 中的代码访问私密数据

注:此博客所讲的例子是基于定位权限的,其他隐私权限同理。

简单使用

以下代码段是用于数据访问时的AppOpsManager.OnOpNotedCallback回调:

override fun onCreate(savedInstanceState: Bundle?) {val appOpsCallback = object : AppOpsManager.OnOpNotedCallback() {private fun logPrivateDataAccess(opCode: String, trace: String) {Log.i("youzi", "Private data accessed. Operation: $opCode\n Stack Trace:\n $trace")}override fun onNoted(syncNotedAppOp: SyncNotedAppOp) {logPrivateDataAccess(syncNotedAppOp.op, Throwable().stackTrace.toString())}override fun onSelfNoted(syncNotedAppOp: SyncNotedAppOp) {logPrivateDataAccess(syncNotedAppOp.op, Throwable().stackTrace.toString())}override fun onAsyncNoted(asyncNotedAppOp: AsyncNotedAppOp) {logPrivateDataAccess(asyncNotedAppOp.op, asyncNotedAppOp.message)}}// 创建AppOpsManager实例并添加上面定义的回调val appOpsManager = getSystemService(AppOpsManager::class.java) as AppOpsManagerappOpsManager.setOnOpNotedCallback(mainExecutor, appOpsCallback)
}

对应Java代码:

        if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {AppOpsManager appOpsManager = getSystemService(AppOpsManager.class);Log.e("ZG", "AppOpsManager: ");appOpsManager.setOnOpNotedCallback(getMainExecutor(), new AppOpsManager.OnOpNotedCallback() {private void logPrivateDataAccess(String opCode, String trace) {Log.e("ZG", "Private data accessed. Operation: " + opCode +"\n Stack Trace:\n " + trace);}@Overridepublic void onNoted(@NonNull SyncNotedAppOp op) {logPrivateDataAccess(op.getOp(), Arrays.toString((new Throwable()).getStackTrace()));}@Overridepublic void onSelfNoted(@NonNull SyncNotedAppOp op) {logPrivateDataAccess(op.getOp(), Arrays.toString((new Throwable()).getStackTrace()));}@Overridepublic void onAsyncNoted(@NonNull AsyncNotedAppOp asyncOp) {logPrivateDataAccess(asyncOp.getOp(), Arrays.toString((new Throwable()).getStackTrace()));}});}

回调里需要重写的几个方法笔者就不多赘述了,官方文档里讲得已经很具体了。这里的logPrivateDataAccess方法是自己写的,名字可随意定义,用于打印使用到的权限名字,还有使用到的代码堆栈,方便大家看到在哪里调用了某个权限。

这里给个笔者检测时打印出来的日志示例:

pic

可以看到,打印出的日志当中会有使用到的定位权限名字fine_location,还有Stack Trace下面的堆栈信息,此堆栈表示在DebugDB.initialize()中调用了定位权限,此时我们可以点击后面括号中的的信息跳转到该方法,然后就可以看得到此方法是属于哪个SDK的了。

(笔者悲催的发现这个 SDK 是没有地方使用的,属于被废弃掉的了,如果要一个个去查所有使用到的 SDK 里是否调用了权限,那将是非常大的工作量😭,大家也可以趁机看一下自己项目里哪些不再需要的 SDK ,早删早轻松)

按归因标记使用

  1. 当只需要检测某个页面中是否调用了隐私权限的话,只需要在需要检测的Activity中的onCreate注册该回调。

  2. 当需要在应用中检测所有页面是否调用了隐私权限的话,需要在应用的Application中的onCreate中注册该回调。

  3. 当需要检测特定的权限时,可以使用按归因标记审核数据访问,通俗点来说就是创建专属的Tag标记。

这里的场景一跟场景二上面已经说明了,区别在于在不同的地方注册而已,现在主要是要讲一下场景三,引用一下官方文档的一句话:

如果您在某个 Activity 中访问数据(例如请求位置信息或访问用户的联系人列表),请在该 Activity 的 onCreate()方法中调用 createAttributionContext(),并传入您希望与应用的一部分相关联的归因标记。

举个例子,也就是说如果你需要在某个Activity中调用LocationManager,你需要先创建一个与之关联的 Tag 标记,如下代码段所示:

    private lateinit var attributionContext: Contextprivate lateinit var locationManager: LocationManageroverride fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)// 创建归因标记,简称TagattributionContext = createAttributionContext("findLocation")// 通过上面的attributionContext来创建locationManager实例,关联归因标记locationManager = attributionContext.getSystemService(LocationManager::class.java) as LocationManager}

上面可以看到我们通过createAttributionContext("findLocation")创建了一个 Tag,然后根据这个 context 来创建LocationManager,在以后需要调用定位的地方就可以直接使用这个与 Tag 相关联的 locationManager 了,接下来我们只需要稍微修改一下刚刚所使用的AppOpsManager.OnOpNotedCallback即可:

    private lateinit var attributionContext: Contextprivate lateinit var locationManager: LocationManageroverride fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)// 创建归因标记,简称TagattributionContext = createAttributionContext("findLocation")// 通过上面的attributionContext来创建locationManager实例,关联归因标记locationManager = attributionContext.getSystemService(LocationManager::class.java) as LocationManagerval appOpsCallback = object : AppOpsManager.OnOpNotedCallback() {// 新增attributionTag参数private fun logPrivateDataAccess(opCode: String, attributionTag: String, trace: String) {Log.i("youzi", "Private data accessed. Operation: $opCode\n Attribution Tag:$attributionTag\n Stack Trace:\n $trace")}override fun onNoted(syncNotedAppOp: SyncNotedAppOp) {syncNotedAppOp.attributionTag?.let {logPrivateDataAccess(syncNotedAppOp.op, it, Throwable().stackTrace.toString())}}override fun onSelfNoted(syncNotedAppOp: SyncNotedAppOp) {syncNotedAppOp.attributionTag?.let {logPrivateDataAccess(syncNotedAppOp.op, it, Throwable().stackTrace.toString())}}override fun onAsyncNoted(asyncNotedAppOp: AsyncNotedAppOp) {asyncNotedAppOp.attributionTag?.let {logPrivateDataAccess(syncNotedAppOp.op, it, Throwable().stackTrace.toString())}}}// 创建AppOpsManager实例并添加上面定义的回调val appOpsManager = getSystemService(AppOpsManager::class.java) as AppOpsManagerappOpsManager.setOnOpNotedCallback(mainExecutor, appOpsCallback)}

我们通过为刚刚的logPrivateDataAccess新增了一个需要传的参数attributionTag,然后在重写AppOpsManager.OnOpNotedCallback()中的三个方法时,判断一下 attributionTag 是否为 null ,如果为 null 的话则表示与我们需要检测的权限无关联,此时就不需要打印。

如果在应用中给多种隐私权限设置了Tag,则可以在回调中拿到attributionTag的时候,判断一下是否是自己想要检测的那个Tag~

这个例子是写在 Activity 中的,如果需要全局使用的话,可以自己定义一个 LocationManager 的单例,然后统一设置 Tag,以便以后需要检测的时候可以使用归因标记方式。

总结

顺便总结一下,在使用下面这段代码的时候,竟然不知道wifiManager.connectionInfo这个方法会调用定位,现在“深刻”的记住了。

    val wifiManager = applicationContext.getSystemService(WIFI_SERVICE) as WifiManagerval wifiInfo = wifiManager.connectionInfo

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/73089.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

讲解:微信内直接打开第三方app

现在很多公司app运营会要求app可以通过微信分享的链接点击直接打开第三方app,这样就可以直接跳转到店铺,或是对应的商品详情,免去用户自己寻找的过程,提升app使用体验。 就比如拼多多,当你的朋友发来一条拼多多砍价消息…

微信支付成功后,不回调接口问题

微信支付已经成功了,但是传的那个notify_url支付回调地址没进入? 按网上说的原因检查,好像都没问题。最后想起来这个接口没有设置不拦截,微信回调时,没传token,被拦截了。在过滤器中,添加回调接…

微信支付付款成功,回调函数不执行

最近用laravel框架做个微商城接微信支付,明明付款成功回调函数就是不执行,数据库存没变化 网上也查了好久,发现是因为回调路由放到了登录中间件里边... //微信支付回调接口 Route::post(wxpay_notify, Front\PayControllerwxpay_notify); …

app调用微信支付接口第一次成功,后面就调取不成功

原因1.可能由于微信缓存问题引起,将微信数据清除或者重装 原因2.微信签名不正确 图中红框部分,Android应用签名要全部小写,并且不能有冒号 在微信开放平台修改应用签名 转载自:https://blog.csdn.net/niuzimao/article/details/50685442 微信…

请教微信支付回调不成功的问题

我是微信端授权登陆的网站, 在里面写的微信支付, 但是回调一直不成功,有大佬可以看看吗? 接收方法 用的这位大佬的代码https://blog.csdn.net/kolinhu/article/details/81061722#commentsedit 原来是外网无法访问的问题

如何用postman验证微信支付V3普通商户的接口参数

为了帮助商户开发者快速便捷的上手V3支付的接口参数设置格式,同时验证各个参数的可用性和正确性,微信支付发布了一个预支付的脚本,并将脚本部署到 Postman 云工作台 WeChat Pay Public Workspace。只需要将集合《微信支付 APIv3》 fork 到自己…

chatgpt赋能python:Python拆分九位数:一种快速高效的处理方法

Python拆分九位数:一种快速高效的处理方法 介绍 Python是一种广泛应用于各种领域的高级编程语言,它的简单易用性、灵活性和庞大的库使其成为许多程序员的首选。在实际应用中,我们常常会遇到需要对较大数字进行处理的情况,而如何…

chatgpt赋能python:Python小数点位数:了解Python中的浮点精度问题

Python 小数点位数:了解 Python 中的浮点精度问题 Python 作为一门常用的编程语言,使用广泛,涉及到的应用场景也越来越多。但是,在实际使用过程中,我们经常会遇到一些浮点数精度问题。本文将介绍 Python 中浮点数的相…

解密大模型与文心一言

近日,百度集团副总裁袁佛玉受邀出席数字纵深论坛并进行“解密大模型与文心一言“主题分享。 数字纵深论坛由北京大学光华管理学院数字教育中心主办,专注于分享科技前沿与商业洞见。 以下为分享精华回顾: 大家都能感受到,ChatGPT和…

chatgpt赋能Python-python3_7如何下载

Python3.7如何下载?详细步骤分享! Python是一门当今最热门、最常用、最易学的编程语言之一,且拥有庞大的社区和强大的库支持。在这篇文章中,我们将会详细介绍如何下载Python3.7版本,让大家能够轻松上手Python编程。 …

【历史上的今天】7 月 19 日:IMAP 协议之父出生;Project Kotlin 公开亮相;CT 成像实现新突破

整理 | 王启隆 透过「历史上的今天」,从过去看未来,从现在亦可以改变未来。 今天是 2023 年 7 月 19 日,在 2010 年的今天,亚马逊发布了一份新闻稿,提到它现在销售的 Kindle 书籍比精装书还多。亚马逊公司旗下 Lab126…

摆平Win2008与VPC2007的诡异冲突

摆平 Win2008 与 VPC2007 的冲突 话说上个周末,参加完51CTO组织的虚拟化技术沙龙,俺和老方,老胡,小废物等人杀奔西直门的红顶老汉烧鸡公,席间觥筹交错,一顿狂磋。说句题外话,这个鸡的味道真是不…

NeurIPS 2020 | 基于协同集成与分发的协同显著性目标检测网络

论文题目:CoADNet: Collaborative Aggregation-and-Distribution Networks for Co-Salient Object Detection 论文链接:https://proceedings.neurips.cc/paper/2020/file/4dc3ed26a29c9c3df3ec373524377a5b-Paper.pdf 论文代码:https://githu…

别再对 Istio 一脸懵了,万众期待的第一本 Istio 著作现!已!上!市

过去一年里,Service Mesh作为新兴的微服务架构,在国内大热。其被誉为是下一代微服务,是云原生技术栈的关键组件之一。而Istio作为第2代Service Mesh的头牌, 在 GitHub 上疯狂收获近 15000 颗星,成为 2018 年最热门的项…

每周送新书:Android软件安全、深入浅出Istio、软件架构设计

每周送新书:Android软件安全、深入浅出Istio、软件架构设计 本栏目每周会给大家推荐三个不同领域的新书,每周关注,总有适合你的好书。 每周送新书活动规则: 在文末评论里回复你对本周推荐图书或相关技术的看法,再或者…

(附源码)基于spring boot 青少年公共卫生教育平台的设计与实现 毕业设计 643214

Springboot青少年公共卫生教育平台 摘 要 信息化社会内需要与之针对性的信息获取途径,但是途径的扩展基本上为人们所努力的方向,由于站在的角度存在偏差,人们经常能够获得不同类型信息,这也是技术最为难以攻克的课题。针对青少年公…

IT服务管理的指挥与智慧

IT服务管理的指挥与智慧 在当今动荡不宁、不断变化的 IT 环境中,IT服务管理(ITSM)的重要性不言而喻,它通过保证关键业务系统和服务的可用性和可靠性,为整个企业提供至关重要的支持。随着技术变得越来越复杂&#xff0c…

php比特教务选排课系统的设计与实现毕业设计源码301826

比特教务选排课系统的设计与实现 摘 要 随着人类向信息社会的不断迈进,风起云涌的信息时代正掀起一次新的革命,同时计算机网络技术高速发展,网络管理运用也变得越来越广泛。因此,建立一个B/S结构的比特教务选排课系统来管理选课信…

Springboot爱护大自然的设计与实现 毕业设计-附源码231643

爱护大自然 摘 要 热爱大自然,保护自然环境,是人们处理与自然关系过程中应当遵守的基本行为准则。第一,热爱大自然,是人类社会赖以生存和发展的物质条件。自然界一切对人类社会生活有益的存在物,如山川草木、飞禽走兽、…

python面向对象案例步骤分析_Python面向对象之继承原理与用法案例分析

本文实例讲述了Python面向对象之继承原理与用法。分享给大家供大家参考,具体如下: 目标 单继承 多继承 面向对象三大特性 封装 根据 职责 将 属性 和 方法 封装 到一个抽象的 类 中 继承 实现代码的重用,相同的代码不需要重复的编写 多态 不同…