Airtest+Poco多设备并发自动化游戏测试框架(遇到的问题)

  •  📢专注于分享软件测试干货内容,欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!
  • 📢交流讨论:欢迎加入我们一起学习!
  • 📢资源分享:耗时200+小时精选的「软件测试」资料包
  • 📢 软件测试学习教程推荐:火遍全网的《软件测试》教程


图片

项目:基于unity3d项目 涉及相关模块:关卡,战斗,商城,装备,强化,天赋等

记录:Airtest+poco+pycharm自动化游戏测试过程中发现的问题以及解决的方法

目标:自动化打包安装,多机器并发测试生成报告

其他:相关敏感的字眼会修改为XXX。目前能力有限,有很多大佬的框架都是非常厉害,鉴于自己的能力不足,只能一边学习一边摸索游戏测试
 

测试用例1:
# -*- encoding=utf8 -*-
__author__ = "Administrator"
#天赋测试需要跳过新手
import unittest
from airtest.core.api import *
from airtest.core.android.adb import *
sys.path.append(r"E:\test\xxxx")
using("test_function")
from lasthero_report import LastHeroReport
import lasthero_gm as gm
from poco.drivers.unity3d import UnityPococlass Talent(unittest.TestCase):@classmethoddef setUpClass(cls):global poco, lhrpoco = UnityPoco()lhr = LastHeroReport(__file__, os.path.basename(__file__))def setUp(self):pass# 天赋9级没有开启def test_1_level(self):'''等级9级不能打开天赋'''a = TalentSet(9)a.setlevel()# 天赋10级开启def test_2_level(self):'''等级10级正常打开天赋'''a = TalentSet(10)a.setlevel()# 天赋11级正常开启def test_3_level(self):'''等级11级正常打开天赋'''a = TalentSet(11)a.setlevel()print("天赋11")def tearDown(self):poco("xxx").click()@classmethoddef tearDownClass(cls):lhr.creat_report()class TalentSet:def __init__(self, x):self.x = xpoco = UnityPoco()self.gm = gm.GM(poco)def setlevel(self):self.gm.gm_open()self.gm.xxx(str(self.x))self.gm.gm_close()sleep(1)poco("xxx").click()if self.x == 9:assert_equal(poco("xxx").exists(), False, "天赋9级没开启")elif self.x == 10:assert_equal(poco("xxx").exists(), True, "天赋10级打开")elif self.x == 11:assert_equal(poco("xxx").exists(), True, "天赋11级打开")
测试用例2:
import unittest
from poco.drivers.unity3d import UnityPoco
from airtest.core.api import *
from airtest.core.android.adb import *
sys.path.append(r"E:\test\xxx")
using("test_function")
from lasthero_report import LastHeroReportclass Shop(unittest.TestCase):# 类属性video_num = 5@classmethoddef setUpClass(cls) -> None:global poco,lhrpoco = UnityPoco()lhr = LastHeroReport(__file__, os.path.basename(__file__))def setUp(self) -> None:passdef test_1_video_diamond_num(self):"""看视频兑换钻石次数检查"""if poco("xxx").exists():passelse:poco("xxx").click()Shop.video_num = int(poco("xxx").get_text())assert_equal(Shop.video_num,4,"今日看视频兑换次数为5")print(Shop.video_num)def test_2_accelerate(self):"""检查打开加速提示文字是否正常"""try:poco("xxx").swipe([0, -4], duration=1)sleep(3)poco("xxx").click()sleep(1)assert_equal(poco("xxx").exists(),True,"正常显示加速提示文字")sleep(1)poco("xxx").click()except Exception as e:log(e)def tearDown(self) -> None:pass@classmethoddef tearDownClass(cls) -> None:lhr.creat_report()if __name__ == "__main__":unittest.main()

  

生成报告封装类:
class LastHeroReport:def __init__(self,filepath,filename):self.file_name_list = filename.split(".")self.log_dir = os.path.dirname(os.getcwd()) + "/log/" + self.file_name_list[0]self.atr = os.path.dirname(os.getcwd()) + "/airtest_report/"if not os.path.exists(self.log_dir):os.makedirs(self.log_dir)auto_setup(logdir=self.log_dir)self.filepath = filepathdef creat_report(self):simple_report(self.filepath, logpath=self.log_dir, logfile='log.txt', output=self.atr + self.file_name_list[0] +'_log.html')
main:
# 用例顺序并发
from multiprocessing import Pool,Manager
import unittest
from airtest.core.android.adb import *
from airtest.core.api import *
sys.path.append(r"E:\test\xxxx")cash_path = "E:/test/xxxx"def g1_run(tag, q):t = q.get()connect_device("Android://127.0.0.1:5037/"+t)a = []suit = addsuit()for s in suit:a.append(s)runner = unittest.TextTestRunner()runner.run(a[tag])q.put(t)def addsuit():discover = unittest.defaultTestLoader.discover(cash_path,pattern="test*.py",top_level_dir=None)return discoverdef getdevices():devices = []adb = ADB()devices_liset = adb.devices()for i in devices_liset:devices.append(i[0])print(devices)return devicesif __name__ == "__main__":q = Manager().Queue()a = []dev = []dev = getdevices()pool = Pool(len(dev))for i in dev:q.put(i)suit = addsuit()for i in suit:a.append(i)i = len(a)for tag in range(i):pool.apply_async(g1_run, (tag, q,))pool.close()pool.join()

 

问题汇总:

1.如何导入其他air文件方法

from airtest.core.api import usingusing("actionset.air")from actionset import sign_board,authorize,callten

2.在python中运行airtest脚本

os.system('airtest run C:/Users/Administrator/AppData/Local/Temp/AirtestIDE/scripts/main.air --device Android://127.0.0.1:5037/HKL48N6K --log log/')

 

–device Android://127.0.0.1:5037/HKL48N6K 中的–device是标识,不能直接去掉,这个标识是安卓手机

3.在python中运行完脚本后,输出和airtest中一样的报告

os.system('airtest report C:/Users/Administrator/AppData/Local/Temp/AirtestIDE/scripts/main.air --log_root D:/pythonwork/venv/log --outfile D:/pythonwork/venv/log/log.html --lang zh')

4.在airtest中直接调用adb

(1)获取应用列表
adb shell dumpsys package > package.txt
(2)在package.txt查找自己需要启动的包名,及main activity,如
com.androlua.compass/com.androlua.Main
(3)在airtest中连接adb并且调用app
dev = connect_device("Android://127.0.0.1:5037/xxxxxx")
dev.shell("am start -n com.xxxx.xxxx/com.unity3d.xxxx.xxxx")

 

5.在airtest内导入py文件 在airtest路径下新建一个文件夹或者新建py文件,直接from 文件夹.文件 import 方法即可

6.airtest在start_app后再进行poco初始化

ConnectionResetError: [WinError 10054] 远程主机强迫关闭了一个现有的连接
所以必须启动后再进行初始化,如下:
start_app('com.xxxx.test')
assert_exists(APP启动后的图片, "是否启动APP")
poco = UnityPoco()

7.使用poco().click()出现了偏移时解决方法(只适用于控件框范围内) 官方说明:点击默认点在 anchorPoint 上,每个UI都会有一个 anchorPoint ,也就是检视器(Inspector)中UI包围盒的那个红点,大部分情况下 anchorPoint 都在UI包围盒的正中央。如果想指定其他的点击位置,可以传一个参数到 click 方法中,这个参数是一个用list或tuple表示的2维向量,其 [x, y] 值分别表示相对于包围盒左上角的偏移量,左上角为 [0, 0] ,右下角为 [1, 1]
在这里插入图片描述
在这里插入图片描述

8.poco判断界面是否存在或者控件是否存在该界面

第一种情况
assert poco("控件name").exists(),"天赋10级打开"
第二种情况
assert poco("控件name").get_text() == "金币购买","金币不足跳转兑换界面"

8.poco判断界面是否存在或者控件是否存在该界面

第一种情况
assert poco("控件name").exists(),"天赋10级打开"
第二种情况
assert poco("控件name").get_text() == "金币购买","金币不足跳转兑换界面"

 

9.poco查找控件的时候,一开始用的是poco(“控件name”),此时如果界面上相同name的时候可以用poco(text=“控件text内容”)

10.由于有时候会出现项目覆盖,导致出现ConnectionResetError: [WinError 10054] 远程主机强迫关闭了一个现有的连接 覆盖后需要重新在maincamera中添加pocomanaget

11.使用poco语句的断言导致在报告中没有显是断言,暂时使用airtest封装的断言,后续有方法在更新上来
 

1.assert poco("lg_talentInstructions").exists(),"天赋11级打开"
2.assert_equal(poco("lg_talentInstructions").exists(), True, "天赋11级打开")

12.转移airtest上得用例,让项目规范化

 

新建pycharm项目时,导入了用例后需要进行pip安装airtest以及pocoui包 如果当前测试用例中有airtest的找图的方法时,转移到pycharm中要把图片也移动过去

test_case:存放测试用例

test_funcion:写了一些公共得方法

test_image:用例中需要用到得一些图片

13.迁移项目得时候进行调试,发现了一些问题,所以想在airtestide中进行调试会出现如下图的报错,但是看了一下却是没问题的,直接在pycharm运行也不会报错,后面请教了一些大佬,airtestide中编写的任务代码会隐式的封装成unittest类函数,所以unittest里嵌套了unittest导致报错

在这里插入图片描述
14.在调试的时候误以为自己本机是有使用venv环境,发现运行脚本的时候不是使用venv路径的下的adb,所以开始看是什么原因导致。

<airtest.core.android.adb> C:\Users\Administrator\AppData\Local\Programs\Python\Python37\lib\site-packages\airtest\core\android\static\adb\windows\adb.exe devices

一开始通过网上查找,可能是配置解析器没有弄好,所以一开始再弄解析器,当时遇到几个问题:

1.由于自己本地的pycharm进行了汉化,所以导致图1图2一开始就显示异常,图1汉化的时候直接显示报错的一些文字信息,无法判断是对新项目进行的设置,第二个是再配置好解析器(实际配置的是新项目)后进行运行,此处发现一直运行不了,继续查资料,说是运行配置没有选择python解析器,当时找到配置解析器时发现左侧文件一开始选中,然后选择其他文件后无法点击回去,并且右侧显示空白,所以删除了本地汉化后,重新打开运行配置的时候就正常显示,并且发现一开始设置的解析器也是针对新项目,所以此时也修改回来,后续如果有同事电脑上的环境有不同版本的python再进行venv的配置
 

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

 15.使用pycharm调用到adb的时候出现如下报错:

报错:adb server version (40) doesn't match this client (41); killing...
需要替换的adb版本路径:C:\Users\Administrator\AppData\Local\Programs\Python\Python37\lib\site-packages\airtest\core\android\static\adb\windows\

这个错是由于电脑上有2个adb版本导致的,接着开始替换电脑上的adb版本,从环境变量中查看了一下配置了adb的环境变量都替换掉,之后再运行依旧还是有报错,后面发现airtest调用当前路径的adb,所以只要找到airtest第三方库的路径下的adb替换即可,因为没有使用venv环境,所以包还是下载到本地的python路径

16.目前先使用unittest框架,后续再了解下pytest框架,先做出简单的用例并发

17.进程池通信的小坑

#以下会导致多进程不会执行
q = Queue()
pool = Pool(2)
pool.apply_async(g1_run, (q,))#进程池内通信需要使用Manager().Queue()
q = Manager().Queue()
pool = Pool(2)
pool.apply_async(g1_run, (q,))

 18.版本发完后有空进行一下调试,增加每个脚本前启动APP和结束APP,接着在调试的过程发现并发突然不好使了,启动APP是同事写的,这里创建了一个Android对象,使用了check_app检查手机上的app是否存在和启动app。后面经过排查,使用了airtest的start_app却是可以正常并发运行,查看了源码发现在connect_device的时候调用过init_device,获取到了当前的设备,而我们在创建command = Android()的时候并没有指定设备,当设备为空的时候就去找默认的设备,所以在并发的时候都是针对同一个手机进行了检查和启动,而没有指定设备,所以添加个设备参数即可

错误的源码:
def startapps():command = Android()if not command.check_app(packgename):print("当前未安装App!!!正在安装所需的App")install(apkpath)command.start_app(packgename)sleep(5)if pocoo(text="xx申请以下权限").wait_for_appearance(30):pocoo(text="允许").click()
修改:
def startapps(dev):command = Android(dev)

19.运行脚本比较久的时候,突然提示

[pocoservice.apk] stdout: b'INSTRUMENTATION_RESULT: shortMsg=Process crashed.\r\nINSTRUMENTATION_CODE: 0\r\n'

后面查询到相关信息,开启手机相关的权限参考:https://airtest.doc.io.netease.com/IDEdocs/device_connection/2_android_faq/#_6 

我使用的是vivo手机,所以在设置了相关功能后依旧会提示,所以我将开发者模式等等的开启,接着把自己测试的apk权限全部开启,并且开启了自启动,后续调试就正常了,如果有报这个错误的同学,可以尝试下把所有的权限都开启

20.pycharm运行pytest一些问题

一开始进行学习pytest调试时,发现运行时就报错:
TypeError: `args` parameter expected to be a list of strings, got: '-q test_g1.py' (type: <class 'str'>)if __name__ == '__main__':pytest.main('-s test_g1.py')之前的版本使用的是pytest.main('-s test_g1.py'),传的是一个str,我本地的pytest6.0.1版本,所以需要传的是List[str],所以修改如下:if __name__ == '__main__':pytest.main(["-s", "test_g1.py"])

 最后我邀请你进入我们的软件测试学习交流群:785128166, 大家可以一起探讨交流软件测试,共同学习软件测试技术、面试等软件测试方方面面,还会有免费直播课,收获更多测试技巧,我们一起进阶Python自动化测试/测试开发,走向高薪之路

感谢每一个认真阅读我文章的人,看着粉丝一路的上涨和关注,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于从事【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/307465.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Google Play上架:2023年度总结报告

今天是2023年的最后一个工作日&#xff0c;今天用来总结一下2023年关于谷歌商店上架的相关政策改动和对应的拒审解决方法。 目录 政策更新与改动2023 年 2 月 22 日2023 年 4 月5 日2023 年 7 月 12 日2023 年 10 月 25 日 开发者计划政策拒审邮件内容和解决办法 政策更新与改…

Shell脚本-bin/bash: 解释器错误: 没有那个文件或目录-完整路径执行-“/”引发的脑裂

引起该不适的一种可能以及解决方案&#xff0c;网上较多&#xff0c;比如&#xff1a; 但按以上方式操作&#xff0c;并经过查看&#xff0c;发现仍然未能解决问题。 因为两种方式执行&#xff0c;有一种能成功&#xff0c;有一种不能&#xff0c;刚开始未怀疑是文件问题&…

进程

进程 进程的概念、组成与特征进程的状态与转换进程控制进程通信 进程的概念、组成与特征 程序是静态的&#xff0c;是一个存放在磁盘里的可执行文件&#xff0c;是一系列的指令集合。 进程是动态的&#xff0c;是程序的一次执行过程&#xff0c;同一个程序多次执行会对应多个进…

自定义View绘制基础之Canvas

画布Canvas Canvas&#xff08;画布&#xff09;是一个用于绘制图形的重要类&#xff0c;它提供了一组绘图操作的方法&#xff0c;允许你在屏幕上绘制各种形状、图像和文本。下面是Canvas类的主要函数&#xff1a; 目录 一、绘制图形和图像 1.drawColor 2.drawCircle 3.dr…

方案:智能分析网关V4区域人数超员AI算法模型的应用场景介绍

视频AI智能分析技术已经深入到人类生活的各个角落&#xff0c;与社会发展的方方面面紧密相连。从日常生活中的各种场景&#xff0c;如人脸识别、车牌识别&#xff0c;到工业生产中的安全监控&#xff0c;如工厂园区的翻越围栏识别、入侵识别、工地的安全帽识别、车间流水线产品…

一篇文章掌握 NestJS 所有的生命周期以及生命周期的执行时机

前言 NestJS 是一个用于构建高效、可扩展的 Node.js 服务器端应用程序的框架&#xff0c;它使用 TypeScript 作为开发语言&#xff0c;也支持原生的 JavaScript。在 NestJS 中&#xff0c;生命周期事件是一个重要的概念。在我们构建和管理应用程序时&#xff0c;有时需要在特定…

【数据结构】C语言实现双链表的基本操作

双链表及其基本操作的实现 导言一、单链表与双链表二、双链表类型的创建三、双链表的初始化四、双链表的创建五、双链表的遍历六、双链表的查找七、双链表的插入八、双链表的删除结语 导言 大家好&#xff0c;很高兴又和大家见面啦&#xff01;&#xff01;&#xff01; 经过…

全方面了解vcruntime140_1.dll的解决方法,多种vcruntime140_1.dll丢失的方法

在日常使用电脑时&#xff0c;我们常常遇到各种各样的问题。其中之一就是丢失vcruntime140_1.dll文件&#xff0c;这是一个重要的系统文件&#xff0c;会影响到电脑的正常运行。今天小编就来给大家详细的说说这一方面的咨询&#xff0c;教会大家多种的丢失vcruntime140_1.dll的…

文件销毁 硬盘销毁 数据销毁:护航数据安全的最后一公里

希望与业界各位志同道合的伙伴交流切磋最新的网络、服务器行业动态信息&#xff0c;同时分享腾讯在网络与服务器领域&#xff0c;规划、运营、研发、服务等层面的实战干货&#xff0c;期待与您的共同成长。 网络平台部以构建敏捷、弹性、低成本的业界领先海量互联网云计算服务…

PBR纹理贴图类型详解

在线工具推荐&#xff1a; 3D数字孪生场景编辑器 - GLTF/GLB材质纹理编辑器 - 3D模型在线转换 - Three.js AI自动纹理开发包 - YOLO 虚幻合成数据生成器 - 三维模型预览图生成器 - 3D模型语义搜索引擎 PBR 纹理是一种帮助 3D 艺术家使他们的 3D 渲染看起来更逼真的技术。…

Live800:客服售前、售中,售后需要做哪些服务?有哪些区别?

客服在售前、售中和售后阶段扮演着重要的角色&#xff0c;他们为顾客提供全方位的服务&#xff0c;确保顾客的满意度和忠诚度。尽管这三个阶段都涉及到与顾客的互动&#xff0c;但每个阶段的服务内容和重点有所不同。 在售前阶段&#xff0c;客服的主要任务是提供产品或服务的信…

腾讯云服务器买1年送3个月,怎么送?免费吗?

腾讯云服务器买1年送3个月怎么送&#xff1f;在买赠活动页面 https://curl.qcloud.com/oRMoSucP 打开后&#xff0c;腾讯云百科以2核4G5M轻量应用服务器为例&#xff0c;点击立即购买&#xff0c;然后在“赠送福利”中选择“免费赠送时长”或“免费再领一台2核4G5M轻量”&#…