小伙伴们在使用appium测试时, 应该会遇到在一个模块中需要调用start_activity方法启动另一个APP进行交互测试的场景, 可是某些App在首次打开时, 会弹出应用权限请求, 这个时候会出现虽然可以使用UI查看器看到元素的ID, 但是却在测试时发现无法定位到元素而报错, 以下是我解决该问题思路, 有用得到的小伙伴仅供参考

以GPSTest这款APP为例, 为大家分析从安装到批准权限的执行流程, 从而找到问提所在:

初始化环境:

为了模拟在模块中启动程序, 我们在初始化时不设置启动程序的初始选项

from appium import webdriver  # 载入驱动
from time import sleep  # 载入延时库
from subprocess import run  # 载入命令执行库

class Automatic:
    def __init__(self):
        run('adb start-server')
        desired_caps = {
            'platformName': 'Android',  # 被测手机是安卓
            'platformVersion': '11',  # 手机安卓版本
            'deviceName': 'motorola',  # 设置设备名称
            'noReset': True,  # 不要重置App
            'newCommandTimeout': 10000,
            'automationName': 'UiAutomator2'
        }
        self.driver = webdriver.Remote('http://localhost:4723/wd/hub', desired_caps)
        self.work_dir = os.getcwd()

安装GPSTest:

检测是否已安装gpstest, 如果否, 则执行adb install 指令安装该程序:

def gpstest_install(self):
        if self.driver.is_app_installed('com.android.gpstest'):
            print('检测到GPSTest已安装')
        else:
            run('adb install data/gpstest.apk')
            print('GPSTest已经安装完成')

启动测试程序:

首先通过命令获取GPSTest包名和启动界面:

adb shell dumpsys activity | findstr "mResumedActivity"

然后用start_activity方法启动程序:

def gpstest_start(self):
	self.driver.start_activity('com.android.gpstest', '.GpsTestActivity')

程序启动后的界面如下图:

Android无法通过uiautomator2定位元素 appium无法定位元素_软件测试

到这里, 小伙伴们就要开始定位弹窗元素, 获取到允许选项的id, 然后一路下一步进行测试了:

Android无法通过uiautomator2定位元素 appium无法定位元素_python_02

获取到的ID为:com.android.permissioncontroller:id/permission_allow_foreground_only_button

def gpstest_start(self):
	self.driver.start_activity('com.android.gpstest', '.GpsTestActivity')
	self.driver.find_element_by_id('com.android.permissioncontroller:id/permission_allow_foreground_only_button').click()

可是问提就出在这里, 当脚本运行到这里时, 却因为找不到元素而报错了

错误分析:

当出现权限请求窗时, 我们再次使用查询界面名指令查看界面:

adb shell dumpsys activity | findstr "mResumedActivity"

我们会发现返回的包名和界面名并不是gpstest这个app的数据com.android.gpstest', '.GpsTestActivity', 而是:

com.google.android.permissioncontroller/
com.android.permissioncontroller.permission.ui.GrantPermissionsActivity t65}

也就是说当脚本执行到启动GPSTest这个app的指令时, 在程序启动瞬间, 界面被权限请求的界面替代了, 导致并没有返回给启动指令正确的回执,;
所以脚本一直卡在self.driver.start_activity('com.android.gpstest', '.GpsTestActivity'), 直到超时报错.
知道问提所在就很简单了, 咱们把启动程序的代码替换为:

try:
	self.driver.start_activity('com.android.gpstest', '.GpsTestActivity')
except:
	pass

问提轻松解决.

最后教大家一个一劳永逸的方法, 在安装程序的时候, 给install指令加一个-g参数, 即可赋予应用所有请求权限, 在启动时再也没有烦人的权限弹窗啦:

adb install -g filename.apk