以编程方式在应用生成中执行 adb 外壳 dpm 命令 无法运行程序 "adb":错误 = 13,权限被拒绝



我正在尝试执行以下命令:

Process process = Runtime.getRuntime().exec("adb shell dpm set-device-owner com.example.package/.DeviceAdmin", null,null);

并得到以下异常

W/System.err:java.io.IOException: Cannot run program "adb": error=13, Permission denied
W/System.err: at java.lang.ProcessBuilder.start(ProcessBuilder.java:1048)
W/System.err: at java.lang.Runtime.exec(Runtime.java:692)
W/System.err: at java.lang.Runtime.exec(Runtime.java:525)

我正在尝试通过以编程方式执行"adb shell dpm set-device-owner com.example.package/"来将我的设备设置为所有者。设备管理员"。

我一直在参考以下SO链接,但是我似乎无法摆脱此错误。

https://stackoverflow.com/a/27909315/5521089

https://stackoverflow.com/a/44164984/5521089

注意:我尝试运行不带 adb 外壳前缀的命令,但它返回 null 并且没有应用任何更改。

以下代码执行我的命令。

try {
StringBuffer output = new StringBuffer();
Process process = Runtime.getRuntime().exec("dpm set-device-owner com.example.package/.DeviceAdmin", null,null);
process.waitFor();
BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream()));
String line = "";
while ((line = reader.readLine()) != null) {
output.append(line + "n");
Log.d("OUTPUT = ", output.toString());
}
} catch (Exception e) {
Log.e("LOGINACTIVITY ", "device owner not set");
Log.e("LOGINACTIVITY ", e.toString());
e.printStackTrace();
}

下面是我的 DeviceAdminReceiver 子类

public class DeviceAdmin extends DeviceAdminReceiver {
public ComponentName getComponentName(Context context){
return new ComponentName(context.getApplicationContext(), DeviceAdmin.class);
}
void showToast(Context context, String msg) {
String status = msg;
Toast.makeText(context, status, Toast.LENGTH_SHORT).show();
}
@Override
public void onEnabled(Context context, Intent intent) {
showToast(context, "Enabled");
}

@Override
public void onDisabled(Context context, Intent intent) {
showToast(context,"Disabled");
}
}

我的清单注册我的接收者

<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<receiver
android:name=".DeviceAdmin"
android:permission="android.permission.BIND_DEVICE_ADMIN">
<meta-data
android:name="android.app.device_admin"
android:resource="@xml/policies"/>
<intent-filter>
<action android:name="android.app.action.DEVICE_ADMIN_ENABLED"/>
</intent-filter>
</receiver>

</application>

检查设置和设备没有帐户(这是一个没有任何谷歌播放服务的安卓设备,没有植根。该应用程序的minSdk版本为21/targetSdk版本为27。

最终,我希望将设备设置为所有者(没有NFC(,其唯一目的是在不征得用户许可的情况下固定屏幕。(这是一个 POS 应用程序(。如何执行命令,以便我可以将设备设置为所有者而不会出现任何异常?

简短的回答是:你的应用无法运行 ADB 命令。对于Android来说,允许这样做将是一个巨大的安全漏洞。

如果要以编程方式将设备设置为设备管理员,请查看文档中的示例

Intent intent = new Intent(DevicePolicyManager.ACTION_ADD_DEVICE_ADMIN);
intent.putExtra(DevicePolicyManager.EXTRA_DEVICE_ADMIN, mDeviceAdminSample);
intent.putExtra(DevicePolicyManager.EXTRA_ADD_EXPLANATION, getString(R.string.add_admin_extra_app_text));
startActivityForResult(intent, REQUEST_CODE_ENABLE_ADMIN);

我正在尝试以编程方式将我的设备设置为所有者

只有两种合法的方式可以将应用程序设置为设备所有者。

a( 设备预配

拆开设备包装后,您可以使用专门准备的 NFC 标签轻触它。由存储在 NFC 标签上的指令指定的应用现在设置为设备所有者。

b( 亚行

只要满足以下条件,您就可以使用问题中提到的命令:

  • 未设置设备所有者
  • 设备上没有注册帐户

dpm set-device-owner命令只能从用户打开的 shell(即从开发计算机的 ADB(运行。它不适用于Runtime.exec()打开的外壳.而且,显然,即使您可以将ADB从设备本身转移到自身,它也将受到相同的限制。

源:

  • https://github.com/aosp-mirror/platform_frameworks_base/blob/2d82493ded10d88cb44e84b90a1b485d3a895d75/services/devicepolicy/java/com/android/server/devicepolicy/DevicePolicyManagerService.java#L7172
  • https://github.com/aosp-mirror/platform_frameworks_base/blob/2d82493ded10d88cb44e84b90a1b485d3a895d75/services/devicepolicy/java/com/android/server/devicepolicy/DevicePolicyManagerService.java#L12184

最终,我希望将设备设置为所有者(没有NFC(,其唯一目的是在不征得用户许可的情况下固定屏幕

您可以使用隐藏的系统 API(android.app.StatusBarManager( 自行隐藏状态栏和导航按钮,并使用设备制造商的平台签名对应用进行签名。您需要签名才能获得android.permission.STATUS_BAR权限。

最新更新