1. 前言

群控,相信大部分人都不会陌生!印象里是一台电脑控制多台设备完成一系列的操作,更多的人喜欢把它和灰产绑定在一起!

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

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

2. 准备

在本机安装 Android 开发环境,保证 adb 被添加到环境变量

将准备好的多台设备,使用数据线( 或者通过 Hub )连接到电脑上

通过 adb devices 命令查看已经连接的所有设备

# 下面显示连接了3台设备
xag:Test xingag$ adb devices
List of devices attached
822QEDTL225T7 device
ca2b3455 device
DE45d9323SE96 device

3. 实战

自动化群控以闲鱼 App 的一次关键字搜索为例,步骤包含:打开应用、点击到搜索界面、输入内容、点击搜索按钮​

下面通过7步来完成这一操作

1、获取目标应用的包名及初始化 Activity

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

这里推荐使用 adb 命令这种方式

# 获取当前运行应用的包名及初始Activity
adb shell dumpsys activity | grep -i run

打开闲鱼 App,在命令终端输入上面的命令,终端会将包名及 Activity 名称显示出来

2、获取所有在线的设备

通过 adb devices 命令,通过输出内容,进行一次过滤,得到所有连接到 PC 端的设备

# 所有设备ID
devices = [] 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

# 包名和Activity
package_name: com.taobao.idlefish
home_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 」即可获得全部源码

如果你觉得文章还不错,请大家 点赞、分享、留言下,因为这将是我持续输出更多优质文章的最强动力!

推荐阅读

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

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

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

带你用 Python 实现自动化群控设备的更多相关文章

  1. Python+selenium 自动化-启用带插件的chrome浏览器,调用浏览器带插件,浏览器加载配置信息。

    Python+selenium 自动化-启用带插件的chrome浏览器,调用浏览器带插件,浏览器加载配置信息.   本文链接:https://blog.csdn.net/qq_38161040/art ...

  2. Appium + Python App自动化第一个脚本

    今天跟大家讲解一个Appium和Python App自动化的脚本.[1]打开你的夜神模拟器(或者连接你的手机) [2]打开桌面的Appium [3]下载你要测的App的apk文件,放到桌面[4]拖动你 ...

  3. python接口自动化(十)--post请求四种传送正文方式(详解)

    简介 post请求我在python接口自动化(八)--发送post请求的接口(详解)已经讲过一部分了,主要是发送一些较长的数据,还有就是数据比较安全等.我们要知道post请求四种传送正文方式首先需要先 ...

  4. python接口自动化1-发送get请求

    前言 requests模块,也就是老污龟,为啥叫它老污龟呢,因为这个官网上的logo就是这只污龟,接下来就是学习它了. 一.环境安装 1.用pip安装requests模块 >>pip in ...

  5. python接口自动化10-token登录

    前言 有些登录不是用cookie来验证的,是用token参数来判断是否登录. token传参有两种一种是放在请求头里,本质上是跟cookie是一样的,只是换个单词而已:另外一种是在url请求参数里,这 ...

  6. python接口自动化22-签名(signature)鉴权(authentication)之加密(HEX、MD5、HMAC-SHA256)

    前言 开放的接口为了避免被别人乱调用,浪费服务器资源,这就涉及到签名(Signature)加密了 API 使用签名方法(Signature)对接口进行鉴权(Authentication).每一次请求都 ...

  7. python接口自动化18-multipart/form-data上传多个附件

    前言 reuqests上传一张图片到服务器,前面已经介绍过了,那么如何在提交BUG的时候,上传附件呢? 上传附件的时候,文件的name参数名称是一样的,python里面key是不可以重复的,又如何处理 ...

  8. 强大的 Python 任务自动化工具!invoke 十分钟入门指南

    接着前面的<tox 教程>,以及刚翻译好的<nox文档>,我们继续聊聊 Python 任务自动化的话题. nox 的作者在去年的 Pycon US 上,做了一场题为<Br ...

  9. Python UI自动化

    Python3--Uiautomator2--Pytest--Alure使用 官方源码GitHub地址:https://github.com/openatx/uiautomator2 介绍 uiaut ...

随机推荐

  1. 【几何+模拟】二次元变换 计蒜客 - T3213

    题目 aslky 有一个 n×n 的矩形,每个位置上都有一个数,有 q 次操作,每次他会让你上下翻转 (UD),左右反转 (LR),顺时针旋转 90∘(SZ),逆时针旋转 90∘(NZ),请你输出最后 ...

  2. java基础知识--数据类型

    计算机时识别不了我们编写的代码语言,计算机中的数据全部采用二进制表示,即0和1表示的数字,每一个0或者1就是一个位,一个位叫做一个bit(比特).(实际上计算机只能识别高低电平,而不是0和1.) 字节 ...

  3. vue项目打包踩坑记

    基于webpack+vue-cli下的vue项目打包命令是 npm run build ,等待打包完成后在根目录生成dist文件夹,里面包含了所有项目相关的内容. 注意:需要完整版的vue-cli项目 ...

  4. 【转载】基于dom的一些前端漏洞

    最直接的xss --dom xss function trackSearch(query) { document.write('<img src="/resources/images/ ...

  5. 2n皇后问题-------递归 暴力求解题与分布讨论题

    问题描述 给定一个n*n的棋盘,棋盘中有一些位置不能放皇后.现在要向棋盘中放入n个黑皇后和n个白皇后,使任意的两个黑皇后都不在同一行.同一列或同一条对角线上,任意的两个白皇后都不在同一行.同一列或同一 ...

  6. Thymeleaf从入门到精通

    什么是Thymeleaf 大家好,我是bigsai,今天我们来学习Thymeleaf,如果你对Thymeleaf比较陌生也不要紧,它很容易学习与理解,并有着自己鲜明的特色. 开始之前,我们依旧问一个问 ...

  7. matplotlib 显示中文问题

    import matplotlib.pyplot as plt plt.rcParams['font.sans-serif']=['SimHei'] #用来正常显示中文标签plt.rcParams[' ...

  8. PHP array_fill() 函数

    ------------恢复内容开始------------ 实例 用给定的键值填充数组: <?php$a1=array_fill(3,4,"blue");print_r($ ...

  9. Python os.fsync() 方法

    概述 os.fsync() 方法强制将文件描述符为fd的文件写入硬盘.在Unix, 将调用fsync()函数;在Windows, 调用 _commit()函数.高佣联盟 www.cgewang.com ...

  10. Python os.fchdir() 方法

    概述 os.fchdir() 方法通过文件描述符改变当前工作目录.高佣联盟 www.cgewang.com Unix, Windows 上可用. 语法 fchdir()方法语法格式如下: os.fch ...