安全和许可Security and Permissions
Android是一个多进程系统,每个应用程序(以及系统的部分)运行在它自己的进程里。大多数程序和系统之间的安全性通过基础的Linux机制在进程级别进行支持,如分配给应用程序的用户和群组IDs。更多细化的安全特性通过“许可”机制来提供,它实施对一个进程能够执行的特定操作方面的限制,和对于每个URI的特定数据段的特许访问的授权。
安全架构Security Architecture
Android安全架构中的一个设计要点是在默认情况下应用程序没有权限执行对其它应用程序、操作系统或用户有害的操作。这些操作包括读/写用户的隐私数据(例如联系方式或e-mail),读/写其它应用程序的文件,执行网络访问,保持设备激活,等等。
应用程序的进程是一个安全的沙箱。它不能干扰其它应用程序,除非明确声明它需要额外的基本的沙箱不能提供的功能的许可权。这些许可权请求能够被不同方式的操作所处理,常见的是基于证书和用户提示的自动允许或禁止。应用程序的权限请求被声明为静态的,这样后面在安装时能够知道它们而且不会被改变。
应用程序签名Application Signing
所有的Android应用程序(.apk文件)必须用证书进行签名认证,而这个证书的私钥是由开发者保有的。该证书可以用以识别应用程序的作者。该证书也不需要被认证机构签名。Android应用程序完全允许而且一般也都是使用自签名(self-signed)证书。证书是用于在应用程序之间建立信任关系,而不是用于控制程序是否可以安装。签名影响安全性的最重要的方式是通过决定谁可以进入基于签名的许可,以及谁可以共享用户IDs。
用户IDs和文件访问User IDs and File Access
每一个Android应用程序(.apk文件)都会在安装时就分配一个独有的Linux用户ID,这就为它建立了一个沙盒,使其不能与其他应用程序进行接触(也不会让其它应用程序接触它)。这个用户ID会在安装时分配给它,并在该设备上一直保持同一个数值。
由于安全性限制措施是发生进程级,所以两个package中的代码不会运行在同一个进程当中,他们要作为不同的Linux用户出现。我们可以通过使用AndroidManifest.xml文件中的manifest标签中的sharedUserId属性,来使不同的package共用同一个用户ID。通过这种方式,这两个package就会被认为是同一个应用程序,拥有同一个用户ID(实际不一定),并且拥有同样的文件存取权限。注意:为了保持安全,只有当两个应用程序被同一个签名签署的时候(并且请求了同一个sharedUserId)才会被分配同样的用户ID.
所有存储在应用程序中的数据都会赋予一个属性-该应用程序的用户ID,这使得其他package无法访问这些数据。当通过这些方法getSharedPreferences(String, int), openFileOutput(String, int), 或者openOrCreateDatabase(String, int, SQLiteDatabase.CursorFactory)来创建一个新文件时,你可以通过使用MODE_WORLD_READABLE与/或MODE_WORLD_WRITEABLE标志位来设置是否允许其他package来访问读写这个文件。当设置这些标志位时,该文件仍然属于该应用程序,但是它的全局读写权限已经被设置,使得它对于其他任何应用程序都是可见的。
一个基本的Android程序通常是没有任何许可与之关联的,这就是说它不能做任何扰乱用户或破坏数据的勾当。那么为了使用设备被保护的特性,我们就必须在AndroidManifest.xml添加一个或多个<uses-permission>标签,用以声明你的应用程序需要的许可。
例如,一个想要监控接收短消息的应用程序需要指定:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.android.app.myapp" >
<uses-permission android:name="android.permission.RECEIVE_SMS" />
</manifest>
在应用程序安装时,该应用程序请求的权限许可是通过package installer来授予的。package installer是通过检查该应用程序的签名和/或用户的交换结果来确定是否给予该程序request的权限。在用户使用过程中不会去检查权限,也就是说要么在安装的时候就批准该权限,使其按照设计可以使用该权限;要么就不批准,这样用户也就根本无法使用该feature,也不会有任何提示告知用户尝试失败。
很多时候, 一个许可失败会导致一个SecurityException被抛回该应用程序. 但是Android并不保证这种情况会处处发生。例如,当数据被deliver到每一个receiver的时候,sendBroadcast(Intent) 方法会去检查permissions,在这个方法调用返回之后,你也不会收到任何exception。几乎绝大多数情况,一个permission failure都会打印到log当中。
Android系统定义的权限可以在Manifest.permission中找到。任何一个程序都可以定义并强制执行自己独有的permissions,因此Manifest.permission中定义的permissions并不是一个完整的列表(即有肯能有自定义的permissions)。
一个特定的许可可能会在你的程序操作过程中的很多地方都被实施:
² 当系统有来电的时候,用以阻止程序执行其它功能。
² 当启动一个活动(activity)的时候,会阻止应用程序启动其它应用程序的Acitivity。
² 在发送和接收广播的时候,去控制谁可以接收你的广播或谁可以发送广播给你。
² 当进入并操作一个内容提供器(content provider)的时候
² 当绑定或起动一个服务(service)的时候
声明和实施许可Declaring and Enforcing Permissions
为了实施你自己的permissions,你必须首先在AndroidManifest.xml文件中声明该permissions.通常我们通过使用一到多个<permission> tag来进行声明。
例如,一个应用程序想要控制谁能启动它的活动,可以为该操作声明许可如下:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.me.app.myapp" >
<permission android:name="com.me.app.myapp.permission.DEADLY_ACTIVITY"
android:label="@string/permlab_deadlyActivity"
android:description="@string/permdesc_deadlyActivity"
android:permissionGroup="android.permission-group.COST_MONEY"
android:protectionLevel="dangerous" />
</manifest>
这里<protectionLevel>属性是必需的,通过声明该属性,我们就可以告知系统如何去通知用户哪些应用程序需要这个许可,或者谁可以拥有该许可。具体请参看链接的文档。
<permissionGroup>属性是可选的,只是用于帮助系统显示许可permission给用户(实际是告知系统该许可是属于哪个许可组permission group的)。你通常会选择使用标准的system group来设定该属性,或者更为少见的用你自己定义的group。推荐使用一个已经存在的group,因为这样UI给用户显示许可的时候会更简单。
需要注意的是标签(label)和描述(description)都是需要为许可提供的。这些都是字符串资源,当用户去看许可列表(android:label)或者某个许可的详细信息(android:description)时,这些字符串资源就可以显示给用户。label应当尽量简短,之需要告知用户该许可是在保护什么功能就行。而description可以用于具体描述获取该许可的程序可以做哪些事情,实际上让用户可以知道如果他们同意程序获取该权限的话,该程序可以做什么。我们通常用两句话来描述许可,第一句描述该许可,第二句警告用户如果批准该权限会可能有什么不好的事情发生。下面是一个描述CALL_PHONE 许可的label和description的例子:
<string name="permlab_callPhone">directly call phone numbers</string>
<string name="permdesc_callPhone">Allows the application to call
phone numbers without your intervention. Malicious applications may
cause unexpected calls on your phone bill. Note that this does not
allow the application to call emergency numbers.</string>
你可以通过shell指令 adb shell pm list permissions 来查看目前系统已有的permissions. 特别的,"-s"选项会以一种用户会看到的基本相同的格式来显示这些permissions:
$ adb shell pm list permissions -s
All Permissions:
Network communication: view Wi-Fi state, create Bluetooth connections, full
Internet access, view network state
Your location: access extra location provider commands, fine (GPS) location,
mock location sources for testing, coarse (network-based) location
Services that cost you money: send SMS messages, directly call phone numbers
...
在清单文件里实施许可Enforcing Permissions in AndroidManifest.xml
用于限制进入系统或应用程序的Components的高级别许可可以再AndroidManifest.xml中实现.所有这些都可以通过在相应的component中包含 android:permission 属性,命名该permission以使其被用以控制进入的权限。
Activity许可(应用于<activity>标签)限制了谁才可以启动相应的活动。permission会在Context.startActivity()和Activity.startActivityForResult()的时候进行检查。如果caller没有所需的权限,则会抛出一个SecurityException。
Service许可(应用于<service>标签)用于限制谁才可以start或bind该service。在Context.startService() , Context.stopService()和Context.bindService()调用的时候会进行权限检查。如果caller没有所需的权限,则会抛出一个SecurityException。
BroadcastReceiver许可(应用于<receiver>标签)用于限制谁才可以向该receiver发送广播。权限检查会在Context.sendBroadcast() 返回时进行,由系统去发送已经提交的广播给相应的Receiver。最终,一个permission failure不会再返回给Caller一个exception;它只是不会去deliver该Intent而已。同样地,Context.registerReceiver()也可以有自己permission用于限制谁才可以向一个在程序中注册的receiver发送广播。另一种方式是,一个permission也可以提供给Context.sendBroadcast() 用以限制哪一个BroadcastReceiver才可以接收该广播。
ContentProvider许可(应用于<provider>标签)用于限制谁才可以访问ContentProvider提供的数据。(Content providers有一套而外的安全机制叫做URI permissions,这些在稍后讨论)不同于其它的Components,这里有两种不同的permission属性可以设置: android:readPermission 用于限制谁可以读取provider中的数据,而 android:writePermission 用于限制谁才可以向provider中写入数据。需要注意的是如果provider同时被读写许可,如果这时只有写许可并不意味着你就可以读取provider中的数据了。当你第一次获取provider的时候就要进行权限检查(如果你没有任何permission,则会抛出SecurityException), 并且这时你对provider执行了某些操作。当使用ContentResolver.query()时需要读权限,而当使用ContentResolver.insert(), ContentResolver.update(), ContentResolver.delete()时需要写权限。在所有这些情况下,没有所需的permission将会导致SecurityException被抛出。
发送广播时实施许可Enforcing Permissions when Sending Broadcasts
除了之前说过的Permission(用于限制谁才可以发送广播给相应的BroadcastReceiver),你还可以在发送广播的时候指定一个permission。在调用Context.sendBroadcast()的时候使用一个permission string,你就可以要求receiver的宿主程序必须有相应的permission。
值得注意的是Receiver和broadcaster都可以要求permission。当这种情况发生时,这两种permission检查都需要通过后才会将相应的intent发送给相关的目的地。
其他权限实施Other Permission Enforcement
在调用service的过程中可以设置任意细化的许可。这是通过Context.checkCallingPermission()方法来完成的。呼叫的时候使用一个想得到的permission string,并且当该权限获批的时候可以返回给呼叫方一个Integer(没有获批也会返回一个Integer)。需要注意的是这种情况只能发生在来自另一个进程的呼叫,通常是一个service发布的IDL接口或者是其他方式提供给其他的进程。
Android提供了很多其他的方式用于检查permissions。如果你有另一个进程的pid,你就可以通过Context.checkPermission(String, int, int)去针对那个pid去检查permission。如果你有另一个应用程序的package name,你可以直接用PackageManager的方法PackageManager.checkPermission(String, String)来确定该package是否已经拥有了相应的权限。
到目前为止我们讨论的标准的permission系统对于内容提供器(content provider)来说是不够的。一个内容提供器可能想保护它的读写权限,而同时与它对应的直属客户端也需要将特定的URI传递给其它应用程序,以便对该URI进行操作。一个典型的例子是邮件应用程序的附件。访问邮件需要使用permission来保护,因为这些是敏感的用户数据。然而,如果有一个指向图片附件的URI需要传递给图片浏览器,那个图片浏览器是不会有访问附件的权利的,因为它不可能拥有所有的邮件的访问权限。
针对这个问题的解决方案就是per-URI permission: 当启动一个activity或者给一个activity返回结果的时候,呼叫方可以设置Intent.FLAG_GRANT_READ_URI_PERMISSION和/或 Intent.FLAG_GRANT_WRITE_URI_PERMISSION。这赋予接收活动(activity)访问该意图(Intent)指定的URI的权限,而不论它是否有权限进入该意图对应的内容提供器。
这种机制允许一个通常的能力-风格(capability-style)模型,以用户交互(如打开一个附件, 从列表中选择一个联系人)来驱动细化的特别授权。这是一个很关键的能力,可以减少应用程序所需要的权限,只留下和程序行为直接相关的权限。
这些URI permission的获取需要内容提供器(包含那些URI)的配合。强烈推荐在内容提供器中实现这种能力,并通过android:grantUriPermissions或者<grant-uri-permissions>标签来声明支持。
更多的信息可以参考Context.grantUriPermission(), Context.revokeUriPermission()和Context.checkUriPermission()方法。
分享到:
相关推荐
8. **调试信息**:可能有logcat配置、ProGuard或R8混淆规则等,用于优化发布版本的代码大小和安全。 通过研究这个项目,开发者可以学习到如何实现即时通讯应用的基本架构,如何处理网络请求,以及如何优化Android...
它可以简化开发流程,使得在不同版本Android上的功能实现更加统一和高效。例如,使用Fragment可以让应用在多屏幕设备上更好地管理界面,Loader可以帮助实现异步数据加载,而ViewPager则提供了优雅的页面滑动效果。 ...
的 Android SDK 提供了在 Android 平台上使用 JaVa 语言进行 Android 应用开发必须的工具和 API 接口。 特性 • 应用程序框架 支持组件的重用与替换 • Dalvik Dalvik Dalvik Dalvik 虚拟机 专为移动设备优化 • ...
在这个项目中,衰减脉冲的实现可能涉及到自定义动画和绘图技术,包括使用Canvas进行像素级操作,以及利用Android的Animation框架来控制视图的动态变化。 【标签】"开源项目" 表明"android-pulse-indicator" 的源...
Bootstrap通常指的是Web开发中的一个前端框架,用于快速构建美观、响应式的网站,但在这里,它似乎被用于Android应用程序的开发,可能是为了提供类似的快速开发和一致性设计的便利。 Android Bootstrap是一个开源...
### Google+Android 开发入门指南(第二版) 知识点总结 #### 一、书籍基本信息 - **书名**:《Google+Android 开发入门指南(第二版)》 - **作者**:Mark L. Murphy - **出版年份**:2010年 - **ISBN-13(平装版)...
【Android-nRF-Toolbox-master.zip】是一个包含与Android设备配对使用的nRF(Nordic Semiconductor)蓝牙设备相关的...对这些知识点的深入理解和实践将有助于开发出高效、稳定且用户体验良好的Android-nRF应用程序。
《Android开发指南中文版》是一本专为Android开发者编写的详尽教程,旨在帮助初学者和有经验的开发者深入理解Android平台的各个方面。本指南涵盖了从基础概念到高级特性的全面内容,包括环境搭建、应用结构、用户...
Kotlin是一种现代、类型安全、面向表达式的编程语言,它被广泛用于Android应用开发,因为它提供了简洁的语法、可空安全性以及与Java的无缝集成。 描述中提到“这是一个存储库”,通常这意味着它包含了一个或多个...
1. **应用程序框架**:检查Android应用程序接口(API)的正确性和一致性,确保应用在不同设备上运行无误。 2. **库测试**:验证核心库函数的兼容性,确保跨设备的数据交换和处理一致。 3. **系统服务测试**:测试如...
通过这个项目,开发者可以深入理解Android应用的开发流程,从设计UI到实现功能,再到调试和测试,全方位提升Android开发技能。同时,这也是一个很好的实践平台,可以在此基础上添加新功能或优化现有功能,提高编程...
Murphy撰写的一本关于Android开发的专业书籍,重点介绍了Android Jetpack框架的使用和理解。该书适用于已经有一定Android基础的开发者,旨在帮助他们掌握Jetpack组件库,提升应用开发效率和质量。 Android Jetpack...
8. **许可证和协议**:关于软件使用的法律文件,如Apache 2.0许可证等。 使用这个CTS测试套件,开发者和设备制造商可以验证其应用程序或设备是否符合Android 9.0的兼容性标准,确保用户能在各种设备上顺畅地运行...
Uni-App 是一种跨平台的移动应用开发框架,它允许开发者使用 Vue.js 语法编写一次代码,然后发布到iOS、Android以及多端小程序等多个平台。2.9.8版本的 Uni-App SDK 提供了该版本下完整的功能集,包括对最新特性和...
从这里开始 安装SDK 更新SDK 开发与调试 Hello Android 应用程序解析 记事本教程 开发工具 应用程序模型 应用程序生命周期 开发应用程序 用户界面工具 应用构成部分 数据存储与取回 安全模型 资源管理与多国版本 ...
通过研究这个开源项目,开发者不仅可以学习到如何在Android上实现面部滤镜,还能了解计算机视觉、图像处理和图形编程的相关知识,为开发自己的AR应用打下坚实的基础。此外,参与开源社区,与其他开发者交流,也是...
VLC for Android、Android TV和ChromeOS是流行的多媒体播放器VLC在移动设备和平板电脑上的版本,由VideoLAN项目开发。这个“vlc-android-master.zip”压缩包包含了VLC安卓应用的主要源代码和资源,供开发者研究、...
通过阅读和理解这个框架的源码,开发者不仅可以快速集成到自己的项目中,还能学习到Android UI组件的封装技巧和最佳实践,提高自己的开发技能。同时,对于想要深入研究Android自定义View或者对话框实现的开发者来说...
学习和使用Xposed框架需要一定的Android开发基础,包括熟悉Java或Kotlin编程语言、理解Android系统架构以及对JNI(Java Native Interface)有一定的了解。通过Xposed,开发者可以实现如修改系统UI、拦截系统事件、...
总的来说,ArcGIS Runtime SDK for Android 100.5.0 提供了一套完整的地图开发框架,涵盖了从地图显示、空间分析到用户交互的各个方面,是Android开发者构建专业GIS应用的理想选择。通过深入学习和利用这个SDK,...