让Android自动化辅助APP成为设备所有者(一)

简介: 我们之所以需要将Android自动化测试的辅助APP设置成设备所有者是为了更好的控制系统的一些行为从而让整个测试过程更稳定。

DeviceOwner简介


DeviceOwner 是指在设备上以管理员身份运行的应用程序,该应用程序可以使用 DevicePolicyManager 类中的 API 来控制设备的一些行为,例如:重启设备、设置锁屏方式、设置密码、强制清除密码、设置状态栏、设置系统更新策略等。


Android 提供了三种设备管理方案:DeviceAdmin(设备管理员)、ProfileOwner(配置文件所有者) 和 DeviceOwner(设备所有者),这三种设备管理方案的权限大小分别为:DeviceAdmin < ProfileOwner < DeviceOwner。应用需要最大的授权才能成为DeviceOwner,DeviceOwner具有设备的最高权限。


创建DeviceOwner


基本配置

首先在res/xml目录下新建device_admin.xml文件,如下:

<?xml version="1.0" encoding="utf-8"?>
<device-admin xmlns:android="http://schemas.android.com/apk/res/android" >
    <uses-policies>
        <!-- 设置密码规则 -->
        <limit-password />
        <!-- 监视屏幕解锁尝试次数 -->
        <watch-login />
        <!-- 更改解锁密码 -->
        <reset-password />
        <!-- 锁定屏幕 -->
        <force-lock />
        <!-- 清除数据,恢复出厂模式,在不发出警告的情况下 -->
        <wipe-data />
        <!-- 锁屏密码有效期 -->
        <expire-password />
        <!-- 对存储的应用数据加密 -->
        <encrypted-storage />
        <!-- 禁用锁屏信息 -->
        <disable-keyguard-features/>
     <!-- 禁用摄像头 -->
        <disable-camera />
    </uses-policies>
</device-admin>

注册一个自定义广播接收器继承自DeviceAdminReceiver

代码如下:

package com.android.jarvis.receivers
import android.app.admin.DeviceAdminReceiver
import android.app.admin.DevicePolicyManager
import android.content.ComponentName
import android.content.Context
import android.content.Intent
import android.os.Build
import android.util.Log
import androidx.annotation.RequiresApi
/**
 * adb shell dpm set-device-owner com.android.jarvis/.receivers.JarvisAdminReceiver
 */
class JarvisAdminReceiver : DeviceAdminReceiver() {
    @RequiresApi(Build.VERSION_CODES.LOLLIPOP)
    override fun onEnabled(context: Context, intent: Intent) {
        Log.d("JarvisAdminReceiver", "onEnabled")
        val devicePolicyManager =
            context.getSystemService(Context.DEVICE_POLICY_SERVICE) as DevicePolicyManager
        //设置应用不可卸载
        devicePolicyManager.setUninstallBlocked(
            getComponentName(context),
            context.packageName,
            true
        )
        super.onEnabled(context, intent)
    }
    /**
     * 获取ComponentName,DevicePolicyManager的大多数方法都会用到
     */
    private fun getComponentName(context: Context): ComponentName {
        return ComponentName(
            context.applicationContext,
            JarvisAdminReceiver::class.java
        )
    }
}

在AndroidManifest.xml中注册广播

代码如下:

  <receiver
            android:name=".receivers.JarvisAdminReceiver"
            android:permission="android.permission.BIND_DEVICE_ADMIN">
            <meta-data
                android:name="android.app.device_admin"
                android:resource="@xml/device_policies" />
            <intent-filter>
                <action android:name="android.app.action.DEVICE_ADMIN_ENABLED" />
                <action android:name="android.app.action.DEVICE_ADMIN_DISABLE_REQUESTED" />
                <action android:name="android.app.action.DEVICE_ADMIN_DISABLED" />
            </intent-filter>
        </receiver>

激活DeviceOwner



先安装应用,然后在命令行中执行:

adb shell dpm set-device-owner com.android.jarvis/.receivers.JarvisAdminReceiver

移除DeviceOwner


当一个APP成为DeviceOwner后,这个APP是不能被卸载的,也无法在设置中关闭其权限,要想卸载这个APP就必须移除DeviceOwner权限,首先需要在AndroidManifest.xml文件中的<application/>节点添加android:testOnly="true",然后可以通过如下命令移除:

adb shell dpm remove-active-admin com.android.jarvis/.receivers.JarvisAdminReceiver

但是在有些机型上即使设置了testOnly=true也是无法移除,会报以下错误:

java.lang.SecurityException: Attempt to remove non-test admin ComponentInfo{....AppAdminReceiver} 0

这个时候就需要通过代码的方式来移除了,我们可以新建一个广播接收器:

package com.android.jarvis.receivers
import android.app.admin.DevicePolicyManager
import android.content.BroadcastReceiver
import android.content.Context
import android.content.Intent
import android.os.Build
import androidx.annotation.RequiresApi
/**
 *adb shell am broadcast -a com.android.jarvis.action.DEVICE_ADMIN_DISABLED
 */
class JarvisDeviceReceiver(
    private val ACTION_DEVICE_ADMIN_DISABLED: String = "com.android.jarvis.action.DEVICE_ADMIN_DISABLED"
) :
    BroadcastReceiver() {
    @RequiresApi(Build.VERSION_CODES.LOLLIPOP)
    override fun onReceive(context: Context, intent: Intent) {
        val action = intent.action
        if (action == ACTION_DEVICE_ADMIN_DISABLED) {
            val devicePolicyManager =
                context.getSystemService(Context.DEVICE_POLICY_SERVICE) as DevicePolicyManager
            devicePolicyManager.clearDeviceOwnerApp(context.packageName)
        }
    }
}

然后在AndroidManifest.xml文件中注册:

        <receiver android:name=".receivers.JarvisDeviceReceiver">
            <intent-filter>
                <action android:name="com.android.jarvis.action.DEVICE_ADMIN_DISABLED" />
            </intent-filter>
        </receiver>

最后在命令行中执行:

adb shell am broadcast -a com.android.jarvis.action.DEVICE_ADMIN_DISABLED

这样就可以成功移除DeviceOwner权限了。

相关文章
|
2天前
|
安全 Java 数据挖掘
当 App 有了系统权限,真的可以为所欲为? Android Performance Systrace
当 App 有了系统权限,真的可以为所欲为? Android Performance Systrace 转载自: https://androidperformance.com/2023/05/14/bad-android-app-with-system-permissions/#/0-Dex-%E6%96%87%E4%BB%B6%E4%BF%A1%E6%81%AF
18 0
|
6天前
|
监控 数据安全/隐私保护 Android开发
智能家电设备连接下载安装APP
智能家电设备连接下载安装APP
18 7
|
29天前
|
存储 人工智能 自动驾驶
自动化物料搬运设备
自动化物料搬运设备
16 2
|
1月前
|
JavaScript Java 开发工具
Python+Appium2.0的APP自动化环境搭建
Python+Appium2.0的APP自动化环境搭建
29 0
|
1月前
|
Android开发 Python
Python封装ADB获取Android设备wifi地址的方法
Python封装ADB获取Android设备wifi地址的方法
23 0
|
1月前
|
Web App开发 Android开发
App自动化查看webview的元素定位信息
App自动化查看webview的元素定位信息
20 0
|
1月前
|
开发工具 Android开发
Android平台RTMP推送|轻量级RTSP服务|GB28181设备接入模块之实时快照保存JPG还是PNG?
Android平台RTMP推送|轻量级RTSP服务|GB28181设备接入模块之实时快照保存JPG还是PNG?
|
1月前
|
JavaScript Android开发
Cordova 后台运行 Android APP
Cordova 后台运行 Android APP
|
2月前
|
物联网 Android开发
Android Ble蓝牙App(七)扫描过滤
Android Ble蓝牙App(七)扫描过滤
|
2月前
|
传感器 监控 物联网
Android Ble蓝牙App(三)特性和属性
Android Ble蓝牙App(三)特性和属性

热门文章

最新文章

相关产品

  • 云迁移中心