使用Runtime.getRuntime在Android上运行Shell命令

时间:2019-11-28 13:04:08

标签: java android shell adb

我正在使用设备管理应用程序(已由制造商签名)。我正在使用以下用于Android 9的adb shell命令安装其他应用程序:-

cat /sdcard/Download/myfolder/newapp.apk | pm install -S 1528293

我只是这样传递它:-

String command = "cat /sdcard/Download/myfolder/newapp.apk | pm install -S 1528293"
Runtime.getRuntime().exec(command);

但是我收到错误消息“猫未知选项S”。

当我从adb shell运行它时,相同的命令可以很好地工作。 不知道我在做什么错,可以寻求帮助。

编辑1:- 我尝试运行如下命令:-

String[] commandInstall = {
                "/system/bin/sh",
                "-c",
                "cat /sdcard/Download/myfolder/newapp.apk | pm install -S 1528293"
        };
Process process = Runtime.getRuntime().exec(commandInstall);

但是现在我得到了错误:-

ava.lang.SecurityException: Reverse mode only supported from shell
    at com.android.server.pm.PackageInstallerSession.doWriteInternal(PackageInstallerSession.java:679)
    at com.android.server.pm.PackageInstallerSession.write(PackageInstallerSession.java:612)
    at android.content.pm.PackageInstaller$Session.write(PackageInstaller.java:852)
    at com.android.server.pm.PackageManagerShellCommand.doWriteSplit(PackageManagerShellCommand.java:2447)
    at com.android.server.pm.PackageManagerShellCommand.runInstall(PackageManagerShellCommand.java:915)
    at com.android.server.pm.PackageManagerShellCommand.onCommand(PackageManagerShellCommand.java:158)
    at android.os.ShellCommand.exec(ShellCommand.java:103)
    at com.android.server.pm.PackageManagerService.onShellCommand(PackageManagerService.java:21330)
    at android.os.Binder.shellCommand(Binder.java:634)
    at android.os.Binder.onTransact(Binder.java:532)
    at android.content.pm.IPackageManager$Stub.onTransact(IPackageManager.java:2821)
    at com.android.server.pm.PackageManagerService.onTransact(PackageManagerService.java:3856)
    at android.os.Binder.execTransact(Binder.java:731)

编辑2:-在android 9之前,我可以执行以下安装应用程序的操作:-

Runtime.getRuntime().exec("pm install -r app.apk");

2 个答案:

答案 0 :(得分:4)

查看PackageInstallerSession的源代码,我发现它已更改为:-

switch (Binder.getCallingUid()) {
    case android.os.Process.SHELL_UID:
    case android.os.Process.ROOT_UID:
        break;
    default:
        throw new SecurityException("Reverse mode only supported from shell");
    }

Source 因此,即使它是系统应用程序,安装的shell命令也可能不起作用。从提交消息看来,这样做是为了让PackageInstaller完成这项工作。

但是似乎这又在某个时候改变了,但可能不包含在android 9中:-

    switch (Binder.getCallingUid()) {
                case android.os.Process.SHELL_UID:
                case android.os.Process.ROOT_UID:
                case android.os.Process.SYSTEM_UID:
                    break;
                default:
                    throw new SecurityException(
                            "Reverse mode only supported from shell or system");
            }

Source

因此,如果该应用是系统应用,则最好的方法是使用PackageInstaller。

答案 1 :(得分:1)

来自PackageInstallerSession.java:

switch (Binder.getCallingUid()) {
                    case android.os.Process.SHELL_UID:
                    case android.os.Process.ROOT_UID:
                    case android.os.Process.SYSTEM_UID:
                        break;
                    default:
                        throw new SecurityException(
                                "Reverse mode only supported from shell or system");
                }

如果您的应用程序位于系统/应用程序下,则可能必须将其放入系统/专用应用程序中。希望异常的源头能以某种方式帮助您。