1. 程式人生 > >Android採用pm實現靜默安裝(降級安裝)的解決方案

Android採用pm實現靜默安裝(降級安裝)的解決方案

最近在做一個apk分析器,裡面可以解析系統中所有安裝app的資訊,並提供組內開發的apk檔案下載、靜默安裝(包括降級安裝),其中在降級安裝中難度較大,在Android4.4與Android 8的解決方案不同,其他版本沒有做測試。在此之前,打算聊聊adb的安裝方式
目前暫時支援已經簽過系統簽名檔案的apk,非系統簽名暫時不支援。

一、adb安裝apk

adb安裝apk常用命令如下:

adb install G:\demo.apk

即install後面接包在電腦上的路徑,這裡要確保已經通過adb連線到裝置,常用以下命令連線,確保電腦與裝置處於同一個區域網:

adb connect
裝置ip

如果需要替換原來的應用,上面的安裝命令是行不通的,需要加上“-r”,即替換原來的應用:

adb install -r G:\demo.apk

那如果是降級安裝呢?再加“-d”:

adb install -r -d G:\demo.apk

這裡的“r”指的是“replace”,替換原來的應用;“-d”指的是“downgrade”,降級安裝
這不是成了嗎?不對,這是通過adb命令,在Android應用中無法使用該命令,那麼該如何解決呢?這裡要引出另一個概念“pm”

二、pm安裝apk

“pm” 是指 “packageManager”,Android自帶的PackageInstaller是通過pm來執行具體的安裝工作,具體流程這裡就不做分析了。我們來看如何直接通過pm來安裝apk,首先進入shell模式,然後就可以使用pm命令:

adb shell
pm install /data/data/demo.apk

這裡的apk路徑是在裝置中的路徑,同理如果要實現降級安裝:

pm install -r -d /data/data/demo.apk

哈哈,是不是感覺在Android應用端實現該命令就很簡單了?如下:

String cmd = "pm install -r -d /data/data/demo.apk"
Runtime run = Runtime.getRuntime();
Process process = run.exec(cmd);

然後就failure了~~
該命令不是每一個應用都可以執行的,需要系統簽名,將應用宣告為系統應用。在Androidmanifest中配置:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="xxx.xxx.xxxx"   
    android:sharedUserId="android.uid.system">
    <uses-permission android:name="android.permission.INSTALL_PACKAGES" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
</manifest>

在run一次,ok,在Android4.4以及之前的版本沒問題,但是在更高的版本,例如Android 7 就不行了,這裡需要稍微修改一下:

pm install -r -d -i packageName --user 0  /data/data/demo.apk

這裡的packageName是指呼叫這行命令的應用的包名

需要注意的是,runtime執行命令列會阻塞執行緒,因此需要在子執行緒中執行

那麼,我們應該如何判斷是否安裝成功呢?很簡單了,通過runtime執行返回的process就可以拿到輸出的結果,完整程式碼如下:

   public void install(File apkFile) {
        String cmd = "";
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH) {
            cmd = "pm install -r -d " + apkFile.getAbsolutePath();
        } else {
            cmd = "pm install -r -d -i packageName --user 0 " + apkFile.getAbsolutePath();
        }
        Runtime runtime = Runtime.getRuntime();
        try {
            Process process = runtime.exec("");
            InputStream errorInput = process.getErrorStream();
            InputStream inputStream = process.getInputStream();
            BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(inputStream));
            String error = "";
            String result = "";
            String line = "";
            while ((line = bufferedReader.readLine()) != null) {
                result += line;
            }
            bufferedReader = new BufferedReader(new InputStreamReader(errorInput));
            while ((line = bufferedReader.readLine()) != null) {
                error += line;
            }
            if(result.equals("Success")){
                Log.i(TAG, "install: Success");
            }else{
                Log.i(TAG, "install: error"+error);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

三、總結

該方案需要獲取系統許可權,進行系統簽名。當然,大家也可以在模擬器上試一試,拿到模擬器系統版本對應的原始碼,找到這3個檔案

SignApk.jar 目錄:/out/host/linux-x86/framework/signapk.jar
platform.x509.pem 目錄:/build/target/product/security/platform.x509.pem
platform.pk8 目錄:/build/target/product/security/platform.pk8

將這三個檔案copy到同一個目錄下,在該目錄下執行:

java -jar SignApk.jar platform.x509.pem  platform.pk8 舊的apk.apk 生成的apk.apk  

即可以得到系統簽名檔案