Android P 静默安装

此前低版本静默安装一般分为两种套路:

  1. shell 调用 pm 命令
  2. 反射调用 PackageManager 的 install 方法

但是在 9.0 上都失效了

分析 PackageInstaller 的源码,和 PackageManager 的源码。发现 PackageManager 多了一个getPackageInstaller 的接口,返回了 PackageInstaller 对象,再来看一看 PackageInstaller的接口

初步猜测在 Android P 上采用类似 socket 的方式与 server 端通信完成安装。

PackageInstaller

查阅官方文档后得知,PackageInstaller 提供了安装、更新以及卸载等功能,其中包括单 APK 和多 APK 安装。

具体的安装行为是通过 PackageInstaller 内部的 Session 完成的。所有的应用都有权限创建这个 Session,但是可能会需要用户的确认才能完成安装(权限不足)。

Session

创建 Session 可以为其指定参数 SessionParams,其中一个作用就是要全部替换还是局部替换 MODE_FULL_INSTALLMODE_INHERIT_EXISTING

如何安装?

通过 IO 流的方式向 Session 内输送 apk 数据。具体代码可以看下文。需要注意的是,PackageInsatller 对于安装结果回调没有采用普通的函数回调,而是采用 Intent 的方式完成回调,比如 广播。

如何在回调中判断是否成功

以广播为例,收到的 Intent 中带有信息,通过PackageInstaller.EXTRA_STATUS key 可以获取到安装结果,通常会有 STATUS_PENDING_USER_ACTIONSTATUS_SUCCESSSTATUS_FAILURESTATUS_FAILURE_ABORTEDSTATUS_FAILURE_BLOCKEDSTATUS_FAILURE_CONFLICTSTATUS_FAILURE_INCOMPATIBLE STATUS_FAILURE_INVALID,和 STATUS_FAILURE_STORAGE

具体代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
/**
* 函数入口
**/
fun installApkInP(apkFilePath: String) {
val apkFile = File(apkFilePath)
val packageInstaller = packageManager.packageInstaller
val sessionParams = PackageInstaller.SessionParams(
PackageInstaller
.SessionParams.MODE_FULL_INSTALL
)
sessionParams.setSize(apkFile.length())
val sessionId = createSession(packageInstaller, sessionParams)
if (sessionId != -1) {
val copySuccess = copyApkFile(packageInstaller, sessionId, apkFilePath)
if (copySuccess) {
install(packageInstaller, sessionId)
}
}
}

/**
* 根据 sessionParams 创建 Session
**/
private fun createSession(
packageInstaller: PackageInstaller,
sessionParams: PackageInstaller.SessionParams
): Int {
var sessionId = -1
try {
sessionId = packageInstaller.createSession(sessionParams)
} catch (e: IOException) {
e.printStackTrace()
}
return sessionId
}

/**
* 将 apk 文件输入 session
**/
private fun copyApkFile(
packageInstaller: PackageInstaller,
sessionId: Int, apkFilePath: String
): Boolean {
var success = false
val apkFile = File(apkFilePath)
try {
packageInstaller.openSession(sessionId).use { session ->
session.openWrite("app.apk", 0, apkFile.length()).use { out ->
FileInputStream(apkFile).use { input ->
var read: Int
val buffer = ByteArray(65536)
while (input.read(buffer).also { read = it } != -1) {
out.write(buffer, 0, read)
}
session.fsync(out)
success = true
}
}
}
} catch (e: IOException) {
e.printStackTrace()
}
return success
}

/**
* 最后提交 session,并且设置回调
**/
private fun install(packageInstaller: PackageInstaller, sessionId: Int) {
try {
packageInstaller.openSession(sessionId).use { session ->
val intent = Intent(this, InstallResultReceiver::class.java)
val pendingIntent = PendingIntent.getBroadcast(
this,
1, intent,
PendingIntent.FLAG_UPDATE_CURRENT
)
session.commit(pendingIntent.intentSender)
}
} catch (e: IOException) {
e.printStackTrace()
}
}