Appuim并发多进程基于pytest测试框架进行兼容性测试

前言

在实际工作中,如果要用appium实现多设备的兼容性测试,大家想到的也许是“多线程”,但由于python中GIL的影响,多线程并不能做到"多机并行",这时候可以考虑使用多进程的方式。

为什么基于pytest

我们知道,pytest中的conftest.py可以定义不同的fixture,测试用例方法可以调用这些fixture,来实现数据共享。以前的框架的思路是:Common目录下的base_driver.py定义生成driver的方法-->conftest.py中调用前者生成driver-->TestCases下的测试用例调用fixture,来实现driver共享 。但是现在不同了,我们有多个设备,这些设备的信息如果只是单纯的写在yml中,我们并行去取的时候似乎也不方便,那可以写在哪里?conftest.py似乎也不是写设备信息的好地方,最后只剩下了main.py,而且将main.py作为多进程的入口再合适不过了。

但问题又来了,如果我们想启动多个appium服务,需要考虑以下几点:

1、appium通过什么方式启动?

2、设备信息如何传递给base_driver方法来生成driver

第一点很明确,客户端启动appium server的方式似乎有点不合时宜了,如果你要同时测5个手机,难道要一个个启动客户端吗?最好的方式是启动命令行,因为命令行启动更方便更快。

再说第二点前,先整理一下思路:main.py定义多个设备信息-->base_driver方法调用,生成多个driver-->TestCases下的测试用例调用fixture,但是设备信息怎么传递给base_driver方法呢?这时候pytest中的pytestconfig就派上用场了。
 

使用pytestconfig

内置的pytestconfig可以通过命令行参数、选项、配置文件、插件、运行目录等方式来控制pytest。pytestconfig是request.config的快捷方式,它在pytest文档里有时候被称为"pytest配置对象"。

要理解pytestconfig是如何工作的,可以查看如何添加一个自定义的命令行选项,然后在测试用例中读取该选项。你可以直接从pytestconfig里读取自定义的命令行选项,但是,为了让pytest能够解析它,还需要使用hook函数pytest_addoption。

下面使用pytest的hook函数pytest_addoption添加几个命令行选项:

pytestconfig/conftest.py
  def pytest_addoption(parser):
      parser.addoption("--myopt", action="store_true", help="some boolean option")
      parser.addoption("--foo", action="store", default="bar", help="foo: bar or baz")

接下来就可以在测试用例中使用这些选项了:

pytest/test_config.py
  import pytest
  def test_option(pytestconfig):
      print("'foo' set to:", pytestconfig.getoption('foo'))
      print("'myopt' set to:", pytestconfig.getoption('myopt'))

让我们看看它是如何工作的:

E:\virtual_workshop\pytest-demo\test_demo7\pytestconfig>pytest -s -q test_config.py::test_config
  'foo' set to: bar
  'myopt' set to: False
  .
  1 passed in 0.02s
  E:\virtual_workshop\pytest-demo\test_demo7\pytestconfig>pytest -s -q --myopt test_config.py::test_config
  'foo' set to: bar
  'myopt' set to: True
  .
  1 passed in 0.01s
  E:\virtual_workshop\pytest-demo\test_demo7\pytestconfig>pytest -s -q --myopt --foo baz test_config.py::test_config
  'foo' set to: baz
  'myopt' set to: True
  .
  1 passed in 0.01s

因为pytestconfig是一个fixture,所以它也可以被其他的fixture使用。如果你喜欢,也可以为这些选项创建fixture。

@pytest.fixture()
  def foo(pytestconfig):
      return pytestconfig.option.foo
     
  @pytest.fixture()
  def myopt(pytestconfig):
      return pytestconfig.option.myopt
     
  def test_fixtures_for_options(foo, myopt):
      print("'foo' set to: ", foo)
      print("'myopt' set to: ", myopt)

具体实现
 

定义main.py

既然可以使用pytest命令行参数了,那只需要在pytest.main中加上参数--cmdopt即可,main.py类似这样:

import pytest, os
  from multiprocessing import Pool
  device_infos = [{"platform_version": "5.1.1", "server_port": 4723, "device_port": 62001, "system_port": 8200},
                  {"platform_version": "7.1.2", "server_port": 4725, "device_port": 62025, "system_port": 8201}]
  def run_parallel(device_info):
      pytest.main([f"--cmdopt={device_info}",
                   "--alluredir", "Reports"])
      os.system("allure generate Reports -o Reports/html --clean")
  if __name__ == "__main__":
      with Pool(2) as pool:
          pool.map(run_parallel, device_infos)
          pool.close()
          pool.join()

为什么设备信息我只写了四个?platform_version、server_port、device_port、system_port。其他的类似于appPackage、appActivity、platformName等去哪了?当然你也可以写在这儿,其他的应该都是多个设备相同的,我写在yml的配置信息中了

·值得注意的是,这里的server_port多个设备不能重复,这是appium server启动的端口号,如果多个设备server_port都重复,那只能启动一个服务了,所以要不同

· system_port又是什么?这个是为了防止"互争互抢"现象的发生。多进程多设备并行时,如果多个设备同时使用同一个appium remote port(如8200)。对多个设备而言,它们并不知道相互使用同一port,因此就会出现多个设备发出的Request和接收的Action衔接不上而造成的测试混乱,可能会出现"Original error:Could not proxy command to remote server"的报错

定义Caps下的caps.yml

这里基本上定义的是多设备相同的desired_caps的公共部分:

platformName: Android
  appPackage: com.xxzb.fenwoo
  appActivity: com.xxzb.fenwoo.activity.addition.WelcomeActivity
  newCommonTimeout: 500
  noReset: False

定义Common下的base_driver.py

这里有几点需要注意下:

·多进程在调用BaseDriver类的base_driver方法时,实例化时应该先通过命令行的方式启动appium server,设想一下,如果启动appium server放在get_base_driver中,会出现什么样的场景?conftest中每调用一次get_base_driver方法,就会打开一个cmd窗口,试图去启动appium server

· yaml.load方法注意新的写法,加上参数 Loader=yaml.FullLoader,这样据说更安全

from appium import webdriver
  from .conf_dir import caps_dir
  import yaml
  import os
  class BaseDriver:
      def __init__(self, device_info):
          self.device_info = device_info
          cmd = "start appium -p {0} -bp {1} -U 127.0.0.1:{2}".format(self.device_info["server_port"], self.device_info["server_port"] + 1, self.device_info["device_port"])
          os.system(cmd)
      def base_driver(self, automationName="appium"):
          fs = open(f"{caps_dir}//caps.yml")
          #平台名称、包名、Activity名称、超时时间、是否重置、server_ip、
          desired_caps = yaml.load(fs, Loader=yaml.FullLoader)
          #版本信息
          desired_caps["platform_version"] = self.device_info["platform_version"]
          #设备名称
          desired_caps["deviceName"] = f"127.0.0.1:{self.device_info['device_port']}"
          #系统端口号
          desired_caps["systemPort"] = self.device_info["system_port"]
          if automationName != "appium":
              desired_caps["automationName"] = automationName
          driver = webdriver.Remote(f"http://127.0.0.1:{self.device_info['server_port']}/wd/hub", desired_capabilities=desired_caps)
          return driver

定义conftest.py

关键点是pytest_addoption和request.config.getoption这两个函数的使用,一个添加命令行,一个解析命令行,但仍有需要注意的:

eval(cmdopt):之所以使用eval将cmdopt转为字典,是因为cmdopt本身是字符串,类似这样的:"{'platform_version': '7.1.2', 'server_port': 4725, 'device_port': 62025, 'system_port': 8201}",这样取值多不方便。

此外,还需要解决一个问题,如果有多个fixture,必须保证第一个测试用例用到的fixture实现BaseDriver的实例化,并且将这一实例化的结果base_driver作为全局变量,供所有的fixture共用,否则就会出现启动多个cmd窗口,启动多个appium server的问题。

from common.base_driver import BaseDriver
  import pytest
  driver = None
  def pytest_addoption(parser):
      parser.addoption("--cmdopt", action="store", default="device_info", help=None)
  @pytest.fixture
  def cmdopt(pytestconfig):
      #两种写法
      return pytestconfig.getoption("--cmdopt")
      #return pytestconfig.option.cmdopt
  #定义公共的fixture
  @pytest.fixture
  def common_driver(cmdopt):
      global driver
      base_driver = BaseDriver(eval(cmdopt))
      driver = base_driver.base_driver()
      yield driver
      driver.close_app()
      driver.quit()

因为pytestconfig是request.config的快捷方式,所以cmdopt也可以写作

@pytest.fixture
  def cmdopt(request):
      return request.config.getoption("--cmdopt")

多进程运行

运行main.py,展示多进程运行的截图:

遗留问题

多进程兼容性测试也会带来一些问题:

  •   ·测试报告如何更好的区分多台设备。
  •   · 对于分辨率不同的机型,要保证一些操作方法的健壮性和稳定性。如A手机屏幕大,确定按钮就在屏幕可见位置,B手机屏幕小,需要多次滑动才能看到按钮,这就要求定义方法时足够健壮。
  •   · 业务逻辑问题。如果并行的去操作(调用同一个接口),会不会有业务逻辑上的限制,比如要抢一个免单券,一天同一个ip,同一个设备只能抢一件,这时候应该只会有一个成功,另一个无疑会失败。这就需要要么调整限制,要么调整方法。

最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

 

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!有需要的小伙伴可以点击下方小卡片领取    

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


学习编程是顺着互联网的发展潮流,是一件好事。新手如何学习编程?其实不难,不过在学习编程之前你得先了解你的目的是什么?这个很重要,因为目的决定你的发展方向、决定你的发展速度。
IT行业是什么工作做什么?IT行业的工作有:产品策划类、页面设计类、前端与移动、开发与测试、营销推广类、数据运营类、运营维护类、游戏相关类等,根据不同的分类下面有细分了不同的岗位。
女生学Java好就业吗?女生适合学Java编程吗?目前有不少女生学习Java开发,但要结合自身的情况,先了解自己适不适合去学习Java,不要盲目的选择不适合自己的Java培训班进行学习。只要肯下功夫钻研,多看、多想、多练
Can’t connect to local MySQL server through socket \'/var/lib/mysql/mysql.sock问题 1.进入mysql路径
oracle基本命令 一、登录操作 1.管理员登录 # 管理员登录 sqlplus / as sysdba 2.普通用户登录
一、背景 因为项目中需要通北京网络,所以需要连vpn,但是服务器有时候会断掉,所以写个shell脚本每五分钟去判断是否连接,于是就有下面的shell脚本。
BETWEEN 操作符选取介于两个值之间的数据范围内的值。这些值可以是数值、文本或者日期。
假如你已经使用过苹果开发者中心上架app,你肯定知道在苹果开发者中心的web界面,无法直接提交ipa文件,而是需要使用第三方工具,将ipa文件上传到构建版本,开...
下面的 SQL 语句指定了两个别名,一个是 name 列的别名,一个是 country 列的别名。**提示:**如果列名称包含空格,要求使用双引号或方括号:
在使用H5混合开发的app打包后,需要将ipa文件上传到appstore进行发布,就需要去苹果开发者中心进行发布。​
+----+--------------+---------------------------+-------+---------+
数组的声明并不是声明一个个单独的变量,比如 number0、number1、...、number99,而是声明一个数组变量,比如 numbers,然后使用 nu...
第一步:到appuploader官网下载辅助工具和iCloud驱动,使用前面创建的AppID登录。
如需删除表中的列,请使用下面的语法(请注意,某些数据库系统不允许这种在数据库表中删除列的方式):
前不久在制作win11pe,制作了一版,1.26GB,太大了,不满意,想再裁剪下,发现这次dism mount正常,commit或discard巨慢,以前都很快...
赛门铁克各个版本概览:https://knowledge.broadcom.com/external/article?legacyId=tech163829
实测Python 3.6.6用pip 21.3.1,再高就报错了,Python 3.10.7用pip 22.3.1是可以的
Broadcom Corporation (博通公司,股票代号AVGO)是全球领先的有线和无线通信半导体公司。其产品实现向家庭、 办公室和移动环境以及在这些环境...
发现个问题,server2016上安装了c4d这些版本,低版本的正常显示窗格,但红色圈出的高版本c4d打开后不显示窗格,
TAT:https://cloud.tencent.com/document/product/1340