当前位置:   article > 正文

Android包管理机制(一)PackageInstaller的初始化

packageinstaller installer = getsystemservice

前言

包管理机制是Android中的重要机制,是应用开发和系统开发需要掌握的知识点之一。 包指的是Apk、jar和so文件等等,它们被加载到Android内存中,由一个包转变成可执行的代码,这就需要一个机制来进行包的加载、解析、管理等操作,这就是包管理机制。包管理机制由许多类一起组成,其中核心为PackageManagerService(PMS),它负责对包进行管理,如果直接讲PMS会比较难以理解,因此我们需要一个切入点,这个切入点就是常见的APK的安装。 讲到APK的安装之前,先了解下PackageManager、APK文件结构和安装方式。

1.PackageManager简介

与ActivityManager和AMS的关系类似,PMS也有一个对应的管理类PackageManager,用于向应用程序进程提供一些功能。PackageManager是一个抽象类,它的具体实现类为ApplicationPackageManager,ApplicationPackageManager中的方法会通过IPackageManager与AMS进行进程间通信,因此PackageManager所提供的功能最终是由PMS来实现的,这么设计的主要用意是为了避免系统服务PMS直接被访问。PackageManager提供了一些功能,主要有以下几点:

  1. 获取一个应用程序的所有信息(ApplicationInfo)。
  2. 获取四大组件的信息。
  3. 查询permission相关信息。
  4. 获取包的信息。
  5. 安装、卸载APK.

2.APK文件结构和安装方式

APK是AndroidPackage的缩写,即Android安装包,它实际上是zip格式的压缩文件,一般情况下,解压后的文件结构如下表所示。

APK的安装方式主要有以下2种:

  • 通过adb命令安装:adb 命令包括adb push/install
  • 通过系统安装器packageinstaller进行安装:packageinstaller是系统内置的应用程序,用于安装和卸载应用程序。

这两种方式最终都会调用PMS的scanPackageDirtyLI方法用来解析包,在此之前的调用链是不同的,本篇文章会介绍第二种方式,对于用户来说,这是最常用的安装方式;对于开发者来说,这是调用链最长的安装方式,能学到的更多。

3.寻找PackageInstaller入口

在Android7.0之前我们可以通过如下代码安装指定路径中的APK。

  1. Intent intent = new Intent(Intent.ACTION_VIEW);
  2. intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  3. intent.setDataAndType(Uri.parse("file://" + path),"application/vnd.android.package-archive");
  4. context.startActivity(intent);
  5. 复制代码

但是Android7.0或更高版本再这么做,就会报FileUriExposedException异常。这是因为StrictMode API 政策禁止应用程序将file:// Uri暴露给另一个应用程序,如果包含file:// Uri的 intent 离开你的应用,就会报FileUriExposedException 异常。为了解决这个问题,谷歌提供了FileProvider,FileProvider继承自ContentProvider ,使用它可以将file://Uri替换为content://Uri,具体怎么使用FileProvider并不是本文的重点,只要知道无论是Android7.0之前还是Android7.0以及更高版本,都会调用如下代码:

  1. Intent intent = new Intent(Intent.ACTION_VIEW);
  2. intent.setDataAndType(xxxxx, "application/vnd.android.package-archive");
  3. 复制代码

Intent的Action属性为ACTION_VIEW,Type属性指定Intent的数据类型为application/vnd.android.package-archive。 能隐式匹配的Activity为InstallStart,需要注意的是,这里分析的源码基于Android8.0,7.0能隐式匹配的Activity为PackageInstallerActivity。 packages/apps/PackageInstaller/AndroidManifest.xml

  1. <activity android:name=".InstallStart"
  2. android:exported="true"
  3. android:excludeFromRecents="true">
  4. <intent-filter android:priority="1">
  5. <action android:name="android.intent.action.VIEW" />
  6. <action android:name="android.intent.action.INSTALL_PACKAGE" />
  7. <category android:name="android.intent.category.DEFAULT" />
  8. <data android:scheme="file" />
  9. <data android:scheme="content" />
  10. <data android:mimeType="application/vnd.android.package-archive" />
  11. </intent-filter>
  12. ...
  13. </activity>
  14. 复制代码

InstallStart是PackageInstaller中的入口Activity,其中PackageInstaller是系统内置的应用程序,用于安装和卸载应用。当我们调用PackageInstaller来安装应用时会跳转到InstallStart,并调用它的onCreate方法: packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStart.java

  1. @Override
  2. protected void onCreate(@Nullable Bundle savedInstanceState) {
  3. if (PackageInstaller.ACTION_CONFIRM_PERMISSIONS.equals(intent.getAction())) {//1
  4. nextActivity.setClass(this, PackageInstallerActivity.class);
  5. } else {
  6. Uri packageUri = intent.getData();
  7. if (packageUri == null) {//2
  8. Intent result = new Intent();
  9. result.putExtra(Intent.EXTRA_INSTALL_RESULT,
  10. PackageManager.INSTALL_FAILED_INVALID_URI);
  11. setResult(RESULT_FIRST_USER, result);
  12. nextActivity = null;
  13. } else {
  14. if (packageUri.getScheme().equals(SCHEME_CONTENT)) {//3
  15. nextActivity.setClass(this, InstallStaging.class);
  16. } else {
  17. nextActivity.setClass(this, PackageInstallerActivity.class);
  18. }
  19. }
  20. }
  21. if (nextActivity != null) {
  22. startActivity(nextActivity);
  23. }
  24. finish();
  25. }
  26. 复制代码

注释1处判断Intent的Action是否为CONFIRM_PERMISSIONS,根据本文的应用情景显然不是,接着往下看,注释2处判断packageUri 是否为空也不成立,注释3处,判断Uri的Scheme协议是否是content,如果是就跳转到InstallStaging,如果不是就跳转到PackageInstallerActivity。本文的应用情景中,Android7.0以及更高版本我们会使用FileProvider来处理URI ,FileProvider会隐藏共享文件的真实路径,将路径转换成content://Uri路径,这样就会跳转到InstallStaging。InstallStaging的onResume方法如下所示。

packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStaging.java

  1. @Override
  2. protected void onResume() {
  3. super.onResume();
  4. if (mStagingTask == null) {
  5. if (mStagedFile == null) {
  6. try {
  7. mStagedFile = TemporaryFileManager.getStagedFile(this);//1
  8. } catch (IOException e) {
  9. showError();
  10. return;
  11. }
  12. }
  13. mStagingTask = new StagingAsyncTask();
  14. mStagingTask.execute(getIntent().getData());//2
  15. }
  16. }
  17. 复制代码

注释1处如果File类型的mStagedFile 为null,则创建mStagedFile ,mStagedFile用于存储临时数据。 注释2处启动StagingAsyncTask,并传入了content协议的Uri,如下所示。 packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStaging.java

  1. private final class StagingAsyncTask extends AsyncTask<Uri, Void, Boolean> {
  2. @Override
  3. protected Boolean doInBackground(Uri... params) {
  4. if (params == null || params.length <= 0) {
  5. return false;
  6. }
  7. Uri packageUri = params[0];
  8. try (InputStream in = getContentResolver().openInputStream(packageUri)) {
  9. if (in == null) {
  10. return false;
  11. }
  12. try (OutputStream out = new FileOutputStream(mStagedFile)) {
  13. byte[] buffer = new byte[4096];
  14. int bytesRead;
  15. while ((bytesRead = in.read(buffer)) >= 0) {
  16. if (isCancelled()) {
  17. return false;
  18. }
  19. out.write(buffer, 0, bytesRead);
  20. }
  21. }
  22. } catch (IOException | SecurityException e) {
  23. Log.w(LOG_TAG, "Error staging apk from content URI", e);
  24. return false;
  25. }
  26. return true;
  27. }
  28. @Override
  29. protected void onPostExecute(Boolean success) {
  30. if (success) {
  31. Intent installIntent = new Intent(getIntent());
  32. installIntent.setClass(InstallStaging.this, PackageInstallerActivity.class);
  33. installIntent.setData(Uri.fromFile(mStagedFile));
  34. installIntent
  35. .setFlags(installIntent.getFlags() & ~Intent.FLAG_ACTIVITY_FORWARD_RESULT);
  36. installIntent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION);
  37. startActivityForResult(installIntent, 0);
  38. } else {
  39. showError();
  40. }
  41. }
  42. }
  43. }
  44. 复制代码

doInBackground方法中将packageUri(content协议的Uri)的内容写入到mStagedFile中,如果写入成功,onPostExecute方法中会跳转到PackageInstallerActivity中,并将mStagedFile传进去。绕了一圈又回到了PackageInstallerActivity,这里可以看出InstallStaging主要起了转换的作用,将content协议的Uri转换为File协议,然后跳转到PackageInstallerActivity,这样就可以像此前版本(Android7.0之前)一样启动安装流程了。

4.PackageInstallerActivity解析

从功能上来说,PackageInstallerActivity才是应用安装器PackageInstaller真正的入口Activity,PackageInstallerActivity的onCreate方法如下所示。 packages/apps/PackageInstaller/src/com/android/packageinstaller/PackageInstallerActivity.java

  1. @Override
  2. protected void onCreate(Bundle icicle) {
  3. super.onCreate(icicle);
  4. if (icicle != null) {
  5. mAllowUnknownSources = icicle.getBoolean(ALLOW_UNKNOWN_SOURCES_KEY);
  6. }
  7. mPm = getPackageManager();
  8. mIpm = AppGlobals.getPackageManager();
  9. mAppOpsManager = (AppOpsManager) getSystemService(Context.APP_OPS_SERVICE);
  10. mInstaller = mPm.getPackageInstaller();
  11. mUserManager = (UserManager) getSystemService(Context.USER_SERVICE);
  12. ...
  13. //根据Uri的Scheme进行预处理
  14. boolean wasSetUp = processPackageUri(packageUri);//1
  15. if (!wasSetUp) {
  16. return;
  17. }
  18. bindUi(R.layout.install_confirm, false);
  19. //判断是否是未知来源的应用,如果开启允许安装未知来源选项则直接初始化安装
  20. checkIfAllowedAndInitiateInstall();//2
  21. }
  22. 复制代码

首先初始话安装所需要的各种对象,比如PackageManager、IPackageManager、AppOpsManager和UserManager等等,它们的描述如下表所示。

类名描述
PackageManager用于向应用程序进程提供一些功能,最终的功能是由PMS来实现的
IPackageManager一个AIDL的接口,用于和PMS进行进程间通信
AppOpsManager用于权限动态检测,在Android4.3中被引入
PackageInstaller提供安装、升级和删除应用程序功能
UserManager用于多用户管理

注释1处的processPackageUri方法如下所示。 packages/apps/PackageInstaller/src/com/android/packageinstaller/PackageInstallerActivity.java

  1. private boolean processPackageUri(final Uri packageUri) {
  2. mPackageURI = packageUri;
  3. final String scheme = packageUri.getScheme();//1
  4. switch (scheme) {
  5. case SCHEME_PACKAGE: {
  6. try {
  7. ...
  8. } break;
  9. case SCHEME_FILE: {
  10. File sourceFile = new File(packageUri.getPath());//1
  11. //得到sourceFile的包信息
  12. PackageParser.Package parsed = PackageUtil.getPackageInfo(this, sourceFile);//2
  13. if (parsed == null) {
  14. Log.w(TAG, "Parse error when parsing manifest. Discontinuing installation");
  15. showDialogInner(DLG_PACKAGE_ERROR);
  16. setPmResult(PackageManager.INSTALL_FAILED_INVALID_APK);
  17. return false;
  18. }
  19. //对parsed进行进一步处理得到包信息PackageInfo
  20. mPkgInfo = PackageParser.generatePackageInfo(parsed, null,
  21. PackageManager.GET_PERMISSIONS, 0, 0, null,
  22. new PackageUserState());//3
  23. mAppSnippet = PackageUtil.getAppSnippet(this, mPkgInfo.applicationInfo, sourceFile);
  24. } break;
  25. default: {
  26. Log.w(TAG, "Unsupported scheme " + scheme);
  27. setPmResult(PackageManager.INSTALL_FAILED_INVALID_URI);
  28. finish();
  29. return false;
  30. }
  31. }
  32. return true;
  33. }
  34. 复制代码

首先在注释1处得到packageUri的Scheme协议,接着根据这个Scheme协议分别对package协议和file协议进行处理,如果不是这两个协议就会关闭PackageInstallerActivity并return false。我们主要来看file协议的处理,注释1处根据packageUri创建一个新的File。注释2处的内部会用PackageParser的parsePackage方法解析这个File(这个File其实是APK文件),得到APK的包信息Package ,Package包含了该APK的所有信息。注释3处会将Package根据uid、用户状态信息和PackageManager的配置等变量对包信息Package做进一步处理得到PackageInfo。 回到PackageInstallerActivity的onCreate方法的注释2处,checkIfAllowedAndInitiateInstall方法如下所示。 packages/apps/PackageInstaller/src/com/android/packageinstaller/PackageInstallerActivity.java

  1. private void checkIfAllowedAndInitiateInstall() {
  2. //判断如果允许安装未知来源或者根据Intent判断得出该APK不是未知来源
  3. if (mAllowUnknownSources || !isInstallRequestFromUnknownSource(getIntent())) {//1
  4. //初始化安装
  5. initiateInstall();//2
  6. return;
  7. }
  8. // 如果管理员限制来自未知源的安装, 就弹出提示Dialog或者跳转到设置界面
  9. if (isUnknownSourcesDisallowed()) {
  10. if ((mUserManager.getUserRestrictionSource(UserManager.DISALLOW_INSTALL_UNKNOWN_SOURCES,
  11. Process.myUserHandle()) & UserManager.RESTRICTION_SOURCE_SYSTEM) != 0) {
  12. showDialogInner(DLG_UNKNOWN_SOURCES_RESTRICTED_FOR_USER);
  13. return;
  14. } else {
  15. startActivity(new Intent(Settings.ACTION_SHOW_ADMIN_SUPPORT_DETAILS));
  16. finish();
  17. }
  18. } else {
  19. handleUnknownSources();//3
  20. }
  21. }
  22. 复制代码

注释1处判断允许安装未知来源或者根据Intent判断得出该APK不是未知来源,就调用注释2处的initiateInstall方法来初始化安装。如果管理员限制来自未知源的安装, 就弹出提示Dialog或者跳转到设置界面,否则就调用注释3处的handleUnknownSources方法来处理未知来源的APK。注释2处的initiateInstall方法如下所示。 packages/apps/PackageInstaller/src/com/android/packageinstaller/PackageInstallerActivity.java

  1. private void initiateInstall() {
  2. String pkgName = mPkgInfo.packageName;//1
  3. String[] oldName = mPm.canonicalToCurrentPackageNames(new String[] { pkgName });
  4. if (oldName != null && oldName.length > 0 && oldName[0] != null) {
  5. pkgName = oldName[0];
  6. mPkgInfo.packageName = pkgName;
  7. mPkgInfo.applicationInfo.packageName = pkgName;
  8. }
  9. try {
  10. //根据包名获取应用程序信息
  11. mAppInfo = mPm.getApplicationInfo(pkgName,
  12. PackageManager.MATCH_UNINSTALLED_PACKAGES);//2
  13. if ((mAppInfo.flags&ApplicationInfo.FLAG_INSTALLED) == 0) {
  14. mAppInfo = null;
  15. }
  16. } catch (NameNotFoundException e) {
  17. mAppInfo = null;
  18. }
  19. //初始化安装确认界面
  20. startInstallConfirm();//3
  21. }
  22. 复制代码

注释1处得到包名,注释2处根据包名获取获取应用程序信息ApplicationInfo。注释3处的startInstallConfirm方法如下所示。 packages/apps/PackageInstaller/src/com/android/packageinstaller/PackageInstallerActivity.java

  1. private void startInstallConfirm() {
  2. //省略初始化界面代码
  3. ...
  4. AppSecurityPermissions perms = new AppSecurityPermissions(this, mPkgInfo);//1
  5. final int N = perms.getPermissionCount(AppSecurityPermissions.WHICH_ALL);
  6. if (mAppInfo != null) {
  7. msg = (mAppInfo.flags & ApplicationInfo.FLAG_SYSTEM) != 0
  8. ? R.string.install_confirm_question_update_system
  9. : R.string.install_confirm_question_update;
  10. mScrollView = new CaffeinatedScrollView(this);
  11. mScrollView.setFillViewport(true);
  12. boolean newPermissionsFound = false;
  13. if (!supportsRuntimePermissions) {
  14. newPermissionsFound =
  15. (perms.getPermissionCount(AppSecurityPermissions.WHICH_NEW) > 0);
  16. if (newPermissionsFound) {
  17. permVisible = true;
  18. mScrollView.addView(perms.getPermissionsView(
  19. AppSecurityPermissions.WHICH_NEW));//2
  20. }
  21. }
  22. ...
  23. }
  24. 复制代码

startInstallConfirm方法中首先初始化安装确认界面,就是我们平常安装APK时出现的界面,界面上有确认和取消按钮并会列出安装该APK需要访问的系统权限。需要注意的是,不同厂商定制的Android系统会有不同的安装确认界面。 注释1处会创建AppSecurityPermissions,它会提取出APK中权限信息并展示出来,这个负责展示的View是AppSecurityPermissions的内部类PermissionItemView。注释2处调用AppSecurityPermissions的getPermissionsView方法来获取PermissionItemView,并将PermissionItemView添加到CaffeinatedScrollView中,这样安装该APK需要访问的系统权限就可以全部的展示出来了,PackageInstaller的初始化工作就完成了。

5.总结

现在来总结下PackageInstaller初始化的过程:

  1. 根据Uri的Scheme协议不同,跳转到不同的界面,content协议跳转到InstallStart,其他的跳转到PackageInstallerActivity。本文应用场景中,如果是Android7.0以及更高版本会跳转到InstallStart。
  2. InstallStart将content协议的Uri转换为File协议,然后跳转到PackageInstallerActivity。
  3. PackageInstallerActivity会分别对package协议和file协议的Uri进行处理,如果是file协议会解析APK文件得到包信息PackageInfo。
  4. PackageInstallerActivity中会对未知来源进行处理,如果允许安装未知来源或者根据Intent判断得出该APK不是未知来源,就会初始化安装确认界面,如果管理员限制来自未知源的安装, 就弹出提示Dialog或者跳转到设置界面。

PackageInstaller的初始化就讲到这,关于PackageInstaller的安装APK的过程会在本系列的下一篇文章进行讲解。

声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop】
推荐阅读
相关标签
  

闽ICP备14008679号