【干货】Android系统定制基础篇:第三部分(Android静默安装、Android支持usb打印机)

这篇具有很好参考价值的文章主要介绍了【干货】Android系统定制基础篇:第三部分(Android静默安装、Android支持usb打印机)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1、Android静默安装

一些产品要求APP在升级时能够实现静默安装,而无需弹出安装界面让用户确认。这里提出两种实现方案:

方案一 APP调用『pm』命令实现静默安装,此方案无须修改Android源码,但需要root权限。
方案二 修改Android PackageInstaller 源码,增加Intent参数来指定是否要静默安装,同时支持通过属性配置需要静默安装的APP包名,只要是属性配置的包名就走静默安装,其它APP进默认安装。

方案一 通过pm命令安装

APP调用『pm』命令实现静默安装,此方案无须修改Android源码,但需要root权限。实现如下:

/**
 * Silent install
 *
 * @param path Package
 * @return true: success false: failed
 */
public static boolean installSilent(String path) {
    boolean result = false;
    BufferedReader es = null;
    DataOutputStream os = null;

    try {
        Process process = Runtime.getRuntime().exec("su");
        os = new DataOutputStream(process.getOutputStream());

        String command = "pm install -r " + path + "\n";
        os.write(command.getBytes(Charset.forName("utf-8")));
        os.flush();
        os.writeBytes("exit\n");
        os.flush();

        process.waitFor();
        es = new BufferedReader(new InputStreamReader(process.getErrorStream()));

        String line;
        StringBuilder builder = new StringBuilder();
        while ((line = es.readLine()) != null) {
            builder.append(line);
        }
        Log.d(TAG, "install msg is " + builder.toString());

        /* Installation is considered a Failure if the result contains
            the Failure character, or a success if it is not.
             */
        if (!builder.toString().contains("Failure")) {
            result = true;
        }
    } catch (Exception e) {
        Log.e(TAG, e.getMessage(), e);
    } finally {
        try {
            if (os != null) {
                os.close();
            }
            if (es != null) {
                es.close();
            }
        } catch (IOException e) {
            Log.e(TAG, e.getMessage(), e);
        }
    }

    return result;
}

方案二 修改PackageInstaller源码

如果没有root权限,方案一将无法实现,因此我们通过定制 PackageInstaller 来实现指定包名可以静默安装,并增加Intent参数来指定静默安装还是默认安装。具体修改如下:

diff --git a/packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStart.java b/packages/apps/Pac
old mode 100644
new mode 100755
index 12441b5..cbf8c41
--- a/packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStart.java
+++ b/packages/apps/PackageInstaller/src/com/android/packageinstaller/InstallStart.java
@@ -22,17 +22,30 @@ import android.app.ActivityManager;
 import android.app.AppGlobals;
 import android.app.IActivityManager;
 import android.content.ContentResolver;
+import android.content.Context;
 import android.content.Intent;
 import android.content.pm.ApplicationInfo;
 import android.content.pm.IPackageManager;
 import android.content.pm.PackageInstaller;
 import android.content.pm.PackageManager;
+import android.content.pm.PackageInfo;
+import android.content.pm.PackageParser;
+import android.content.pm.PackageUserState;
+import android.content.pm.ProviderInfo;
 import android.net.Uri;
 import android.os.Build;
 import android.os.Bundle;
 import android.os.RemoteException;
+import android.os.SystemProperties;
 import android.support.annotation.Nullable;
+import android.text.TextUtils;
 import android.util.Log;
+import android.content.pm.IPackageInstallObserver;
+import android.support.v4.content.FileProvider;
+
+import java.io.File;
+import java.lang.reflect.Method;
+import java.util.List;
 
 import com.android.internal.annotations.VisibleForTesting;
 
@@ -43,6 +56,8 @@ import com.android.internal.annotations.VisibleForTesting;
 public class InstallStart extends Activity {
     private static final String LOG_TAG = InstallStart.class.getSimpleName();
 
+    private static final String EXTRA_SILENT_INSTALL = "silent_install";
+
     private static final String DOWNLOADS_AUTHORITY = "downloads";
     private IActivityManager mIActivityManager;
     private IPackageManager mIPackageManager;
@@ -91,40 +106,57 @@ public class InstallStart extends Activity {
             return;
         }
 
-        Intent nextActivity = new Intent(intent);
-        nextActivity.setFlags(Intent.FLAG_ACTIVITY_FORWARD_RESULT);
-
-        // The the installation source as the nextActivity thinks this activity is the source, hence
-        // set the originating UID and sourceInfo explicitly
-        nextActivity.putExtra(PackageInstallerActivity.EXTRA_CALLING_PACKAGE, callingPackage);
-        nextActivity.putExtra(PackageInstallerActivity.EXTRA_ORIGINAL_SOURCE_INFO, sourceInfo);
-        nextActivity.putExtra(Intent.EXTRA_ORIGINATING_UID, originatingUid);
+        Uri pkgUri = intent.getData();
+        String path = "";
+        if (pkgUri != null) {
+            if (pkgUri.getScheme().equals(ContentResolver.SCHEME_FILE)) {
+                path = pkgUri.getPath();
+            } else if (pkgUri.getScheme().equals(ContentResolver.SCHEME_CONTENT)) {
+                path = providerUri2Path(this, pkgUri);
+            }
+        }
 
-        if (PackageInstaller.ACTION_CONFIRM_PERMISSIONS.equals(intent.getAction())) {
-            nextActivity.setClass(this, PackageInstallerActivity.class);
+        if (isSilentInstall(intent, path)) {
+            Log.i(LOG_TAG, "silent install path: " + path);
+            getPackageManager().installPackage(Uri.fromFile(new File(path)),
+                    new PackageInstallObserver(), 2, null);
         } else {
-            Uri packageUri = intent.getData();
-
-            if (packageUri != null && (packageUri.getScheme().equals(ContentResolver.SCHEME_FILE)
-                    || packageUri.getScheme().equals(ContentResolver.SCHEME_CONTENT))) {
-                // Copy file to prevent it from being changed underneath this process
-                nextActivity.setClass(this, InstallStaging.class);
-            } else if (packageUri != null && packageUri.getScheme().equals(
-                    PackageInstallerActivity.SCHEME_PACKAGE)) {
+            Intent nextActivity = new Intent(intent);
+            nextActivity.setFlags(Intent.FLAG_ACTIVITY_FORWARD_RESULT);
+
+            // The the installation source as the nextActivity thinks this activity is the source, hence
+            // set the originating UID and sourceInfo explicitly
+            nextActivity.putExtra(PackageInstallerActivity.EXTRA_CALLING_PACKAGE, callingPackage);
+            nextActivity.putExtra(PackageInstallerActivity.EXTRA_ORIGINAL_SOURCE_INFO, sourceInfo);
+            nextActivity.putExtra(Intent.EXTRA_ORIGINATING_UID, originatingUid);
+
+            if (PackageInstaller.ACTION_CONFIRM_PERMISSIONS.equals(intent.getAction())) {
                 nextActivity.setClass(this, PackageInstallerActivity.class);
             } else {
-                Intent result = new Intent();
-                result.putExtra(Intent.EXTRA_INSTALL_RESULT,
-                        PackageManager.INSTALL_FAILED_INVALID_URI);
-                setResult(RESULT_FIRST_USER, result);
+                Uri packageUri = intent.getData();
+
+                if (packageUri != null && (packageUri.getScheme().equals(ContentResolver.SCHEME_FILE)
+                        || packageUri.getScheme().equals(ContentResolver.SCHEME_CONTENT))) {
+                    // Copy file to prevent it from being changed underneath this process
+                    nextActivity.setClass(this, InstallStaging.class);
+                } else if (packageUri != null && packageUri.getScheme().equals(
+                        PackageInstallerActivity.SCHEME_PACKAGE)) {
+                    nextActivity.setClass(this, PackageInstallerActivity.class);
+                } else {
+                    Intent result = new Intent();
+                    result.putExtra(Intent.EXTRA_INSTALL_RESULT,
+                            PackageManager.INSTALL_FAILED_INVALID_URI);
+                    setResult(RESULT_FIRST_USER, result);
 
-                nextActivity = null;
+                    nextActivity = null;
+                }
             }
-        }
 
-        if (nextActivity != null) {
-            startActivity(nextActivity);
+            if (nextActivity != null) {
+                startActivity(nextActivity);
+            }
         }
+
         finish();
     }
 
@@ -247,4 +279,94 @@ public class InstallStart extends Activity {
     void injectIActivityManager(IActivityManager iActivityManager) {
         mIActivityManager = iActivityManager;
     }
+
+    private static String providerUri2Path(Context context, Uri uri) {
+        Log.i(LOG_TAG, "providerUri2Path, uri: " + uri.toString());
+
+        try {
+            List<PackageInfo> packs = context.getPackageManager()
+                    .getInstalledPackages(PackageManager.GET_PROVIDERS);
+            if (packs != null) {
+                for (PackageInfo pack : packs) {
+                    ProviderInfo[] providers = pack.providers;
+                    if (providers != null) {
+                        for (ProviderInfo provider : providers) {
+                            if (provider.authority.equals(uri.getAuthority())) {
+                                Class<FileProvider> fileProviderClass = FileProvider.class;
+                                try {
+                                    Method getPathStrategy = fileProviderClass.getDeclaredMethod(
+                                            "getPathStrategy", Context.class, String.class);
+                                    getPathStrategy.setAccessible(true);
+                                    Object invoke = getPathStrategy.invoke(null, context, uri.getAuthority());
+                                    if (invoke != null) {
+                                        String PathStrategyStringClass = FileProvider.class.getName() + "$PathStr
+                                        Class<?> PathStrategy = Class.forName(PathStrategyStringClass);
+                                        Method getFileForUri = PathStrategy.getDeclaredMethod("getFileForUri", Ur
+                                        getFileForUri.setAccessible(true);
+                                        Object invoke1 = getFileForUri.invoke(invoke, uri);
+                                        if (invoke1 instanceof File) {
+                                            return ((File) invoke1).getAbsolutePath();
+                                        }
+                                    } else {
+                                        Log.e(LOG_TAG, "providerUri2Path, invoke is null.");
+                                    }
+                                } catch (Exception e) {
+                                    Log.e(LOG_TAG, e.getMessage());
+                                }
+                                break;
+                            }
+                        }
+                    }
+                }
+            } else {
+                Log.w(LOG_TAG, "providerUri2Path, packs is null.");
+            }
+        } catch (Exception e) {
+            Log.e(LOG_TAG, e.getMessage());
+        }
+
+        return "";
+    }
+
+    private boolean isSilentInstall(Intent intent, String path) {
+        if (!TextUtils.isEmpty(path)) {
+            if (intent.getBooleanExtra(EXTRA_SILENT_INSTALL, false)) {
+                Log.i(LOG_TAG, "isSilentInstall, Intent include EXTRA_SILENT_INSTALL.");
+                return true;
+
+            } else {
+                String value = SystemProperties.get("ro.silentinstallapps", "");
+                if (!TextUtils.isEmpty(value)) {
+                    if (TextUtils.equals(value, "all")) {
+                        Log.i(LOG_TAG, "isSilentInstall, All.");
+                        return true;
+
+                    } else {
+                        File sourceFile = new File(path);
+                        PackageParser.Package parsed = PackageUtil.getPackageInfo(this, sourceFile);
+                        if (parsed != null) {
+                            PackageInfo pkgInfo = PackageParser.generatePackageInfo(parsed, null,
+                                    PackageManager.GET_PERMISSIONS, 0, 0, null,
+                                    new PackageUserState());
+                            if (pkgInfo != null) {
+                                if (TextUtils.equals(value, "system")) {
+                                    if (TextUtils.equals(pkgInfo.sharedUserId, "android.uid.system")) {
+                                        Log.i(LOG_TAG, "isSilentInstall, System.");
+                                         return true;
+                                     }
+
+                                } else {
+                                    String[] pkgNames = value.split(",");
+                                    if (pkgNames != null && pkgNames.length > 0) {
+                                        for (String pkgName : pkgNames) {
+                                            if (TextUtils.equals(pkgName, pkgInfo.packageName)) {
+                                                Log.i(LOG_TAG, "isSilentInstall, Included in the whitelist.");
+                                                return true;
+                                            }
+                                        }
+                                    }
+                                 }
+                            }
+                        }
+                    }
+                }
+            }
+        } else {
+            Log.w(LOG_TAG, "isSilentInstall, path is null.");
+        }
+
+        return false;
+    }
+
+    class PackageInstallObserver extends IPackageInstallObserver.Stub {
+
+        @Override
+        public void packageInstalled(String packageName, int returnCode) throws RemoteException {
+            Log.i(LOG_TAG, packageName + " silent installed.");
+        }
+    }
 }

配置指定包名走静默安装

支持通过属性配置需要静默安装的APP包名,只要是属性配置的包名就走静默安装,其它APP走默认安装。这个操作由系统端配置,APP端按Android标准API调应用安装即可。配置参考:

ro.silentinstallapps=com.ayst.sample1,com.ayst.sample1

注意 支持同时配置多个包名,包名之间用逗号隔开。

配置全部APP走静默安装

所有APP都走静默安装。

ro.silentinstallapps=all

配置系统APP走静默安装
仅系统uid的APP走静默安装,其它APP走默认安装。

ro.silentinstallapps=system

指定Intent参数走静默安装

通过Intent参数指定是否要静默安装。使用方法如下:文章来源地址https://www.toymoban.com/news/detail-497966.html

intent.putExtra("silent_install", true); // 静默安装
private static final String EXTRA_SILENT_INSTALL = "silent_install";

public static void install(Context context, String path) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
        installO(context, path);
    } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        installN(context, path);
    } else {
        installOther(context, path);
    }
}

/**
 * android1.x-6.x
 *
 * @param context Context
 * @param path    Package
 */
private static void installOther(Context context, String path) {
    Intent install = new Intent(Intent.ACTION_VIEW);
    install.setDataAndType(Uri.parse("file://" + path),
                           "application/vnd.android.package-archive");
    install.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    install.putExtra(EXTRA_SILENT_INSTALL, true); // 静默安装
    context.startActivity(install);
}

/**
 * android7.x
 *
 * @param context Context
 * @param path    Package
 */
private static void installN(Context context, String path) {
    Uri apkUri = FileProvider.getUriForFile(context, AUTHORITY, new File(path));
    Intent install = new Intent(Intent.ACTION_VIEW);
    install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    install.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
    install.setDataAndType(apkUri, "application/vnd.android.package-archive");
    install.putExtra(EXTRA_SILENT_INSTALL, true); // 静默安装
    context.startActivity(install);
}

/**
 * android8.x
 *
 * @param context Context
 * @param path    Package
 */
@RequiresApi(api = Build.VERSION_CODES.O)
private static void installO(Context context, String path) {
    boolean isGranted = context.getPackageManager().canRequestPackageInstalls();
    if (isGranted) {
        installN(context, path);
    } else {
        Dialog dialog = new AlertDialog.Builder(context.getApplicationContext())
            .setTitle("Unknown sources")
            .setPositiveButton("OK", new DialogInterface.OnClickListener() {
                public void onClick(DialogInterface d, int w) {
                    Intent intent = new Intent(Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES);
                    context.startActivity(intent);
                }
            }).create();
        dialog.setCancelable(false);
        dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
        dialog.show();
    }
}

2、Android支持usb打印机

Android 5.1
diff --git a/system/core/init/devices.cpp b/system/core/init/devices.cpp
index 754ac5f..3614f5f 100755
--- a/system/core/init/devices.cpp
+++ b/system/core/init/devices.cpp
static void handle_generic_device_event(struct uevent *uevent){
    make_dir("/dev/bus", 0755);
    make_dir("/dev/bus/usb", 0755);
    snprintf(devpath, sizeof(devpath), "/dev/bus/usb/%03d", bus_id);
    make_dir(devpath, 0755);
    snprintf(devpath, sizeof(devpath), "/dev/bus/usb/%03d/%03d", bus_id, device_id);
    }
- 	} 
+	} else if (!strncmp(uevent->subsystem, "usbmisc", 7)) {  
+			if (!strncmp(uevent->device_name, "usb/lp", 6)){
+			base = "/dev/usb/"; 
+			mkdir(base, 0755);
+		}
+	}
Android 8.1
diff --git a/system/core/init/devices.cpp b/system/core/init/devices.cpp
index 754ac5f..3614f5f 100755
--- a/system/core/init/devices.cpp
+++ b/system/core/init/devices.cpp
@@ -471,6 +471,10 @@ void DeviceHandler::HandleDeviceEvent(const Uevent& uevent) {
                 int device_id = uevent.minor % 128 + 1;
                 devpath = StringPrintf("/dev/bus/usb/%03d/%03d", bus_id, device_id);
             }
+        } else if (uevent.subsystem == "usbmisc") {
+            if (StartsWith(uevent.device_name, "usb/lp")) {
+                devpath = "/dev/" + uevent.device_name;
+            }
         } else {
             // ignore other USB events
             return;

到了这里,关于【干货】Android系统定制基础篇:第三部分(Android静默安装、Android支持usb打印机)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【干货】Android系统定制基础篇:第十五部分(Android支持鼠标右键返回、GPIO 控制方案、属性标识USB摄像头的VID与PID)

    1、修改 frameworks/native/services/inputflinger/InputReader.cpp 如下: 二、Android GPIO 控制方案 GPIO 功能在 Android Framework 中增加 GPIO 相关 API,让 APP 可以直接通过 JAVA API 操控 GPIO。支持 输入、输出、模拟按键 三种模式。做为输入时可以用于app获取外部设备的电平状态。做为输出时可以输

    2024年02月11日
    浏览(43)
  • 【干货】Android系统定制基础篇:第二部分(Launcher3支持键盘切换焦点、开发者模式密码确认、禁止非预装应用安装、配置时间)

    Android Launcher3 默认并不支持键盘操作,无法切换焦点,在一些需要支持键盘或遥控操作的设备中无法使用,因些对 Launcher3 做简单修改,使其支持键盘切换焦点。 在安全性要求比较高的产品中,一般会默认关闭『adb调试』,同时禁止用户打开『adb调试』功能。在Android8.1中默认

    2024年02月10日
    浏览(55)
  • 【干货】Android系统定制基础篇:第一部分(文件权限、增加信号强度、双路背光控制)

    当需要修改某文件或路径权限时,我们可以在init.rc开机启动某节点添加chmod命令进行修改。但是对于system分区,由于是ro权限,在init.rc使用chmod修改权限无效。需要在文件编译时,对权限进行修改。不同的Android版本改法一样,但是文件所在目录有差异,Android O主要修改文件是

    2024年02月09日
    浏览(53)
  • Android11.0系统中实现静默安装并启动App以及静默卸载

    本文描述Android11中通过修改系统PackageInstaller应用支持静默安装App,并启动安装的App。 PackageInstaller是系统中专门负责app安装的App,静默安装逻辑添加到此应用中,应用所在路径 /frameworks/base/packages/PackageInstaller/src/com/android/packageinstaller/ 添加SilenceInstallManager,路径为 /frameworks

    2024年02月03日
    浏览(72)
  • 【MySQL8入门到精通】基础篇- Linux系统静默安装MySQL,跨版本升级

    📢📢📢📣📣📣 哈喽!大家好,我是【IT邦德】,江湖人称jeames007,10年DBA工作经验 一位上进心十足的【大数据领域博主】!😜😜😜 中国DBA联盟(ACDU)成员,目前从事DBA及程序编程 擅长主流数据Oracle、MySQL、PG 运维开发,备份恢复,安装迁移,性能优化、故障应急处理等。

    2023年04月09日
    浏览(51)
  • C/C++网络编程基础知识超详细讲解第三部分(系统性学习day13)

                                                        懒大王感谢大家的关注和三连支持~    目录 前言 一、并发服务器 1.进程并发服务器 实例代码如下:  2.线程并发服务器 实例代码如下:  二、域通信 域通信TCP实例代码如下:  三、广播与组播(UDP)  1.广播 实例代码

    2024年02月05日
    浏览(47)
  • Android APK 签名打包原理分析(三)【静默安装的实现方案】

    小编目前从事的系统定制类工作,有客户提出了,需要后台“静默安装”他们的app,也就是悄无声息的安装,而且特别强调,不可以跳出任何安装引导页面,他们的app下载完成之后,后台调用公开的android install代码,系统就后台完成安装,安装完成之后,重新打开应用就可以

    2024年02月01日
    浏览(62)
  • Android Jetpack组件库(第三部分)---WorkManager

    Android Jetpack 是 Google 推出的一整套帮助 Android 应用程序开发的库、工具包和架构指南,旨在为 Android 应用程序提供更快,更轻松,更稳定的开发体验。自推出以来已经发展成了一个庞大的技术生态系统,包括了许多使用方便、功能强大的库,以下是其中一些新特性、新组件:

    2024年02月07日
    浏览(42)
  • 达梦8 在CentOS 系统下静默安装

    确认系统参数 创建用户组及用户,设置用户密码 上传安装文件 安装文件授权 静默安装配置文件 指定一下字符集,否则可能会出现不能解析配置文件的错误 执行安装,注意配置文件要用全路径 数据库安装后,自动启动,,查看数据库状态 问题一 不能解析安装文件 在运行安

    2024年02月10日
    浏览(33)
  • InnoSetup安装时先静默安装VS运行库并判断系统版本

    [Setup] ArchitecturesInstallIn64BitMode=x64 [Files] Source: \\\"vc_redist.x64.exe\\\"; DestDir: \\\"{app}\\\"; Check: NeedInstallVC9SP1 Source: \\\"vc_redist.x86.exe\\\"; DestDir: \\\"{app}\\\"; Check: NeedInstallVC9SP1 [Code] var   vc9SP1Missing: Boolean; function NeedInstallVC9SP1(): Boolean; begin   Result := vc9SP1Missing; end; function InitializeSetup(): Boolean; var    ve

    2024年02月02日
    浏览(38)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包