1. 前言

群控,置信大部分人都不会生疏!印象里是一台电脑管制多台设施实现一系列的操作,更多的人喜爱把它和灰产绑定在一起!

事实上,群控在自动化测试中也被宽泛应用!接下来的几篇文章,我将带大家聊聊企业级自动化中,群控正确的应用姿态!

本篇先从根底篇开始,聊聊应用「 Python + adb 」命令如何编写一套群控脚本

2. 筹备

在本机装置 Android 开发环境,保障 adb 被增加到环境变量

将筹备好的多台设施,应用数据线( 或者通过 Hub )连贯到电脑上

通过 adb devices 命令查看曾经连贯的所有设施

# 上面显示连贯了3台设施xag:Test xingag$ adb devicesList of devices attached822QEDTL225T7    deviceca2b3455        deviceDE45d9323SE96   device

3. 实战

自动化群控以闲鱼 App 的一次关键字搜寻为例,步骤蕴含:关上利用、点击到搜寻界面、输出内容、点击搜寻按钮

上面通过7步来实现这一操作

1、获取指标利用的包名及初始化 Activity

获取形式有很多种,支流形式蕴含:adb 命令、解析 APK、第三方 APK、无障碍服务

这里举荐应用 adb 命令这种形式

# 获取以后运行利用的包名及初始Activityadb shell dumpsys activity | grep -i run

关上闲鱼 App,在命令终端输出下面的命令,终端会将包名及 Activity 名称显示进去

2、获取所有在线的设施

通过 adb devices 命令,通过输入内容,进行一次过滤,失去所有连贯到 PC 端的设施

# 所有设施IDdevices = []def get_online_devices(self):    """    获取所有在线的设施    :return:    """    global devices    try:        for device_serias_name in exec_cmd("adb devices"):           # 过滤掉第一条数据及不在线的设施           if "device" in device_serias_name:              devices.append(device_serias_name.split("\t")[0])           devices = devices[1:]    except Exception as e:            print(e)    # 连上的所有设施及数量    return devices

3、群控关上指标利用

遍历设施列表,应用 adb -s 设施ID shell am start -W 命令别离关上指标利用

def start_app(self):    """    关上App    :return:     """    for device in devices:        os.popen("adb -s " + device + " shell am start -W {}/{}".format(self.packageName, self.home_activity))    print('期待加载实现...')    sleep(10)

4、封装执行步骤

为了方便管理设施,将每一步的操作写入到 YAML 文件中,能够通过 ID 查找元素并执行点击操作、在输入框中输出内容、调用本地办法及输出参数

这里别离对应:保留 UI 树控件、查找输入框元素并执行点击操作、保留 UI 树控件(界面变动了)、输出文本内容、查看搜寻按钮元素并执行点击操作

# steps_adb.yaml# 包名和Activitypackage_name:  com.taobao.idlefishhome_activity:  com.taobao.fleamarket.home.activity.InitActivity# 执行步骤steps:  - save_ui_tree_to_local:      method:  save_ui_tree_to_local      args:  - find_element_and_click:      id:  com.taobao.idlefish:id/tx_id  - save_ui_tree_to_local:      method:  save_ui_tree_to_local  - input_content:      content:  Python  - find_element_and_click:      id:  com.taobao.idlefish:id/search_button

须要指出的是,为了进步群控的适配性,控件的理论坐标须要通过上面的步骤去获取:

  • 导出界面的控件树
  • 解析控件树 XML 文件,利用正则表达式失去指标控件的坐标值
  • 计算出控件的中心点坐标

利用控件 ID 获取元素中心点坐标的实现代码如下:

def get_element_position(element_id, uidump_name):    """    通过元素的id,应用ElementTree,解析元素控件树,查找元素的坐标中心点    :param element_id: 元素id,比方:    :return: 元素坐标    """    # 解析XML    tree = ET.parse('./../%s.xml' % uidump_name)    root = tree.getroot()    # 待查找的元素    result_element = None    # print('查找数目', len(root.findall('.//node')))    # 遍历查找node元素    # 通过元素id    for node_element in root.findall('.//node'):        if node_element.attrib['resource-id'] == element_id:            result_element = node_element            break    # 如果找不到元素,间接返回空    if result_element is None:        print('道歉!找不到元素!')        return None    # 解析数据    coord = re.compile(r"\d+").findall(result_element.attrib['bounds'])    # 中心点坐标    position_center = int((int(coord[0]) + int(coord[2])) / 2), int((int(coord[1]) + int(coord[3])) / 2)    return position_center

5、辨别设施

为了保障群控脚本执行不会产生烦扰,在每个步骤执行之前,都应该将设施 ID 作为参数进行辨别

比方:将控件的界面控件树依照设施保留为不同的名称、点击界面和输出的命令传相应设施 ID 作为入参

def save_ui_tree_to_local(dName):    """    获取以后Activity控件树,保留到本地    文件名固定为:uidump.xml    :param dName: 设施id    :return:    """    exec_cmd("adb  -s %s shell uiautomator dump /data/local/tmp/%s.xml" % (dName, dName))    sleep(2)    exec_cmd("adb -s %s pull /data/local/tmp/%s.xml ./../" % (dName, dName))

6、执行步骤

从 YAML 文件中读取执行步骤,遍历步骤汇合,外部遍历设施列表,以保障每一个步骤,别离执行到每台设施上

# 执行步骤for step in self.steps:    # 设施    for device in devices:         pass

接着,通过步骤名称匹配不同的操作,即可操作设施了

# 操作名称step_name = list(step)[0]if step_name == 'save_ui_tree_to_local':    # 保留UI数到本地    method = step.get(step_name).get('method')    save_ui_tree_to_local(device)elif step_name == 'find_element_and_click':    element_id = step.get(step_name).get('id')    # 获取元素的坐标    bound_search_input = get_element_position(element_id, device)    # 点击元素    exec_cmd('adb -s %s shell input tap %s %s' % (device, bound_search_input[0], bound_search_input[1]))elif step_name == 'input_content':    input_content = step.get(step_name).get('content')    # 模仿输出    exec_cmd('adb -s %s shell input text %s' % (device, input_content))else:    print('其余操作步骤')

7、敞开利用

当所有的操作实现之后,同样是遍历设施,利用 adb 命令去敞开 App 即可

def stop_all(self):   """   敞开利用   :return:   """   for device in devices:       os.popen("adb -s " + device + " shell am force-stop  %s" % self.packageName)

4. 最初

本篇仅仅是 Python 自动化群控最简略的实现形式,前面将和大家探讨更加简单的实现形式。

我曾经将文中全副源码上传到后盾,关注公众号「 AirPython 」后回复「 qk 」即可取得全副源码

如果你感觉文章还不错,请大家 点赞、分享、留言下,因为这将是我继续输入更多优质文章的最强能源!

举荐浏览

教你如何批量运行自动化脚本,高效工作!

自动化篇 | 朋友圈被折叠?会自动化不存在的

自动化篇 | 再也不必放心老人们用智能机了