1. 前言

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

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

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

2. 准备

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

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

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

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

3. 实战

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

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

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

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

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

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

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

2、获取所有在线的设备

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

  1. # 所有设备ID
  2. devices = []
  3. def get_online_devices(self):
  4. """
  5. 获取所有在线的设备
  6. :return:
  7. """
  8. global devices
  9. try:
  10. for device_serias_name in exec_cmd("adb devices"):
  11. # 过滤掉第一条数据及不在线的设备
  12. if "device" in device_serias_name:
  13. devices.append(device_serias_name.split("\t")[0])
  14. devices = devices[1:]
  15. except Exception as e:
  16. print(e)
  17. # 连上的所有设备及数量
  18. return devices

3、群控打开目标应用

遍历设备列表,使用 adb -s 设备ID shell am start -W 命令分别打开目标应用

  1. def start_app(self):
  2. """
  3. 打开App
  4. :return:
  5. """
  6. for device in devices:
  7. os.popen("adb -s " + device + " shell am start -W {}/{}".format(self.packageName, self.home_activity))
  8. print('等待加载完成...')
  9. sleep(10)

4、封装执行步骤

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

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

  1. # steps_adb.yaml
  2. # 包名和Activity
  3. package_name: com.taobao.idlefish
  4. home_activity: com.taobao.fleamarket.home.activity.InitActivity
  5. # 执行步骤
  6. steps:
  7. - save_ui_tree_to_local:
  8. method: save_ui_tree_to_local
  9. args:
  10. - find_element_and_click:
  11. id: com.taobao.idlefish:id/tx_id
  12. - save_ui_tree_to_local:
  13. method: save_ui_tree_to_local
  14. - input_content:
  15. content: Python
  16. - find_element_and_click:
  17. id: com.taobao.idlefish:id/search_button

需要指出的是,为了提高群控的适配性,控件的实际坐标需要通过下面的步骤去获取:

  • 导出界面的控件树

  • 解析控件树 XML 文件,利用正则表达式得到目标控件的坐标值

  • 计算出控件的中心点坐标

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

  1. def get_element_position(element_id, uidump_name):
  2. """
  3. 通过元素的id,使用ElementTree,解析元素控件树,查找元素的坐标中心点
  4. :param element_id: 元素id,比如:
  5. :return: 元素坐标
  6. """
  7. # 解析XML
  8. tree = ET.parse('./../%s.xml' % uidump_name)
  9. root = tree.getroot()
  10. # 待查找的元素
  11. result_element = None
  12. # print('查找数目', len(root.findall('.//node')))
  13. # 遍历查找node元素
  14. # 通过元素id
  15. for node_element in root.findall('.//node'):
  16. if node_element.attrib['resource-id'] == element_id:
  17. result_element = node_element
  18. break
  19. # 如果找不到元素,直接返回空
  20. if result_element is None:
  21. print('抱歉!找不到元素!')
  22. return None
  23. # 解析数据
  24. coord = re.compile(r"\d+").findall(result_element.attrib['bounds'])
  25. # 中心点坐标
  26. position_center = int((int(coord[0]) + int(coord[2])) / 2), int((int(coord[1]) + int(coord[3])) / 2)
  27. return position_center

5、区分设备

为了保证群控脚本执行不会产生干扰,在每个步骤执行之前,都应该将设备 ID 作为参数进行区分

比如:将控件的界面控件树按照设备保存为不同的名称、点击界面和输入的命令传相应设备 ID 作为入参

  1. def save_ui_tree_to_local(dName):
  2. """
  3. 获取当前Activity控件树,保存到本地
  4. 文件名固定为:uidump.xml
  5. :param dName: 设备id
  6. :return:
  7. """
  8. exec_cmd("adb -s %s shell uiautomator dump /data/local/tmp/%s.xml" % (dName, dName))
  9. sleep(2)
  10. exec_cmd("adb -s %s pull /data/local/tmp/%s.xml ./../" % (dName, dName))

6、执行步骤

从 YAML 文件中读取执行步骤,遍历步骤集合,内部遍历设备列表,以保证每一个步骤,分别执行到每台设备上

  1. # 执行步骤
  2. for step in self.steps:
  3. # 设备
  4. for device in devices:
  5. pass

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

  1. ​# 操作名称
  2. step_name = list(step)[0]
  3. if step_name == 'save_ui_tree_to_local':
  4. # 保存UI数到本地
  5. method = step.get(step_name).get('method')
  6. save_ui_tree_to_local(device)
  7. elif step_name == 'find_element_and_click':
  8. element_id = step.get(step_name).get('id')
  9. # 获取元素的坐标
  10. bound_search_input = get_element_position(element_id, device)
  11. # 点击元素
  12. exec_cmd('adb -s %s shell input tap %s %s' % (device, bound_search_input[0], bound_search_input[1]))
  13. elif step_name == 'input_content':
  14. input_content = step.get(step_name).get('content')
  15. # 模拟输入
  16. exec_cmd('adb -s %s shell input text %s' % (device, input_content))
  17. else:
  18. print('其他操作步骤')

7、关闭应用

当所有的操作完成之后,同样是遍历设备,利用 adb 命令去关闭 App 即可

  1. def stop_all(self):
  2. """
  3. 关闭应用
  4. :return:
  5. """
  6. for device in devices:
  7. 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. Windows电脑多个SSH Key管理.md

    笔者偏在阿里云,Github,开源中国上均存放一些私有项目代码,因此需要再Windows电脑上配置多个SSH Key 环境 操作系统:windows 7 Git 提示:Git 安装后就可以使用 Git ...

  2. .net core 使用 Serilog 作为日志提供者

    nuget引入 Serilog.AspNetCore Startup构造函数: public Startup(IConfiguration configuration) { Configuration ...

  3. React Native 中使用Redux

    参考https://jspang.com/detailed?id=48和印度同事的代码简单整理一下在RN中使用Redux的步骤 1. 首先我们应该先了解Redux是什么,什么情况下需要用到它 在Red ...

  4. 使用brew services管理服务

    简介 官网: https://github.com/Homebrew/homebrew-services macOS使用launchctl命令加载开机自动运行的服务,brew service可以简化l ...

  5. 基于Centos7安装Docker-registry2.0

    我们可能希望构建和存储包含不想公开的信息或数据的镜像,因为Docker公司的团队开源了docker-registry的代码,这样我们就可以基于此代码在内部运行自己的registry. 服务端1.拉去仓 ...

  6. dp入门例题(1)

    按摩师问题 https://leetcode-cn.com/problems/the-masseuse-lcci/ (找好状态转移方程) 今天只和昨天的状态相关,依然是分类讨论: 今天不接受预约:或者 ...

  7. PHP date_timestamp_get() 函数

    实例 返回今天的日期和时间的 Unix 时间戳: <?php$date=date_create();echo date_timestamp_get($date);?> 运行实例 » 定义和 ...

  8. 原生JS 实现点击按钮创建元素

    要求: 点击按钮,随机生成一个20px-100px宽度正方形,随机颜色,随机位置,不能出可视区域外 思路:(1)创建按钮,为按钮添加事件侦听 (2)触发事件,创建一个元素 (3)设置元素样式,包括大小 ...

  9. C/C++编程笔记:《C语言》—— 数组知识详解,学编程建议收藏!

    作者:龙跃十二链接:https://www.imooc.com/article/300814 ,微信公众号:龙跃十二 数组的基本概念 我们把一组数据的集合称为数组(Array),它所包含的每一个数据叫 ...

  10. ElasticSearch 基本概念 and 索引操作 and 文档操作 and 批量操作 and 结构化查询 and 过滤查询

    基本概念 索引: 类似于MySQL的表.索引的结构为全文搜索作准备,不存储原始的数据. 索引可以做分布式.每一个索引有一个或者多个分片 shard.每一个分片可以有多个副本 replica. 文档: ...