当前位置: 首页 > article >正文

Python+Appium+Pytest+Allure自动化测试框架-代码篇

文章目录

  • 自动化测试框架工程目录
  • 示例测试代码
  • 示例结果查看
  • allure
  • pytest
    • 编写pytest测试样例的规则
    • pytest conftest.py向测试函数传参
  • appium
    • 启动appium服务
    • 代码端通过端口与appium服务通信对设备进行操作
    • 在pytest测试用例中调用appium
  • 更多功能

Python+Appium+Pytest+Allure自动化测试框架,本文主要讲述工程的搭建,测试用例的编写

自动化测试框架工程目录

pycharm创建python工程AutoTest,main.py做为主入口
设定目录结构如下
在这里插入图片描述

各目录用途说明

├─common		# 公共模块目录
├─datas         # 配置文件目录
├─outputs		# 测试输出目录
│  ├─logs		# 日志目录, 4723.log代表appium端口为4723服务的日志
│  ├─picture	# 截图存放目录
│  └─reports	# 测试报告存放目录
└─testcase		# 测试用例目录

示例测试代码

main.py调用pytest进行用例测试, 然后再调用allure生成报告

# _*_coding:utf-8 _*_

import os
import pytest


def run_case():
    pytest.main(["-s",
                 "-v",
                 "--alluredir", "./outputs/reports/data"])

    generate_report()


def generate_report():
    # 查看allure报告
    os.system("allure generate outputs/reports/data -o outputs/reports/html --clean")


if __name__ == "__main__":
    run_case()

写个打印Hello World的测试用例,里面加上allure添加报告信息,
pytest与allure的使用后面会讲到,这里只做个简单示例,
testcase/test_device_info.py

# _*_coding:utf-8 _*_
import os
import time

import allure


@allure.feature("设备信息模块")
class TestDeviceInfo:

    @allure.story("设备信息故事")
    @allure.description("设备信息描述")
    @allure.severity(allure.severity_level.BLOCKER)
    def test_get_device_info(self):
        print('test_get_device_info start')
        print('Hello World')
        print('test_get_device_info end')

运行main.py就会开始进行自动化测试。

示例结果查看

有没有错可以直接在pycharm的打印中看
在这里插入图片描述
执行完用例,会生成allure的测试结果报告在outputs/reports/html目录下
在这里插入图片描述
在命令行中输入以下指令查看html结果
D:\workspace>allure open .\AutoTest\outputs\reports\html
在这里插入图片描述

allure

Allure 是一个灵活且功能强大的测试报告框架,主要用于为自动化测试生成美观且信息丰富的测试报告。
在这里插入图片描述

pytest

pytest是一个功能强大的 Python 测试框架,用于编写简单且可扩展的测试。
这里面有个testcase目录是专门给pytest用的,测试用例都放在这个目录下

编写pytest测试样例的规则

可以在工程根目录下放置pytest.in指定pytest规则,
它会在指定的目录及其子目录中自动搜索符合命名规则的测试函数和测试类。不需要手动添加每个测试用例到测试套件中。
比如,如果你有一个测试模块test_device_info.py,只要运行pytest命令,它就会自动发现并执行其中以test_开头的函数。
在这里插入图片描述

  1. 测试文件以test_开头
  2. 测试类以Test开头,并且不能带有 init 方法
  3. 测试函数以test_开头
  4. 断言使用基本的assert即可

pytest conftest.py向测试函数传参

main.py与testcase下的测试用例是隔离的,比如在main.py中创建个单例模式对象,在pytest测试函数中再调用这个单例对象实际会新创建个实例,二者不是一个对象。
这时需要用到pytest conftest.py向测试函数传参

  1. pytest会默认读取conftest.py里面的所有fixture
  2. conftest.py 文件名称是固定的,不能改动
  3. conftest.py只对同一个package下的所有测试用例生效
  4. 不同目录可以有自己的conftest.py,一个项目中可以有多个conftest.py
  5. 测试用例文件中不需要手动import conftest.py,pytest会自动查找

在testcase目录下新建文件conftest.py里面

# _*_coding:utf-8 _*_
import time
import pytest


# pytest.addoption()钩子函数,注册自定义参数 cmdopt 添加到配置对象
def pytest_addoption(parser):
    parser.addoption("--cmdopt",
                     action="store",
                     default="device_info",  # 默认参数值
                     help=None)
                     
# 从配置对象中读取自定义参数的值
@pytest.fixture(scope='session')
def cmdopt(pytestconfig):
    # 两种写法
    return pytestconfig.getoption("--cmdopt")         

通过pytest.fixture, 把cmdopt传给测试函数。
修改main.py, 加了参数–cmdopt=Hi, 把Hi传过去。

# _*_coding:utf-8 _*_

import os
import pytest


def run_case():
    pytest.main(["-s",
                 "-v",
                 "--cmdopt=Hi",
                 "--alluredir", "outputs/reports/data"])

    generate_report()


def generate_report():
    # 生成allure报告
    os.system("allure generate outputs/reports/data -o outputs/reports/html --clean")


if __name__ == "__main__":
    run_case()

test_device_info.py中,函数添加cmdopt,把参数接收进来

# _*_coding:utf-8 _*_
import os
import time

import allure


@allure.feature("设备信息模块")
class TestDeviceInfo:

    @allure.story("设备信息故事")
    @allure.description("设备信息描述")
    @allure.severity(allure.severity_level.BLOCKER)
    def test_get_device_info(self, cmdopt):
        print('test_get_device_info start')
        print('get cmdopt:' + cmdopt)
        print('test_get_device_info end')

执行结果,可以看到打印了Hi
在这里插入图片描述

appium

Appium 是一个开源的自动化测试框架,主要用于移动应用(包括 iOS 和 Android)的自动化测试。
其使用步骤是

启动appium服务

在命令行中输入

start /b appium -a 127.0.0.1 -p 4723 -pa /wd/hub

注意新版本appium不支持-bp参数(bootstrap port),加了会报错,另外需要加上-pa /wd/hub的参数,否则会报错“No route found for /wb/hub/session”。
第一次启动要慢些,有时候要花十多秒,等待appium的打印出现才执行后面的操作
在这里插入图片描述

代码端通过端口与appium服务通信对设备进行操作

把调试设备(比如开adb功能的手机)连usb到电脑,
命令行中查看设备名称
adb devices
比如得到:X36405Y4041600056
我们把这个值填写到代码配置中,

from appium.options.android import UiAutomator2Options


def get_appium_driver():
    desired_caps = {
        "platformName": "Android",
        "platformVersion": "13", # 这个填设备的安卓版本,如果写错会报错的
        "deviceName": "X36405Y4041600056", # 这个填设备adb device获取的名称,写错会报错
        "noReset": True,
    }

    #driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', desired_caps)

    # 新版本appium需要对参数进行转化
    options = UiAutomator2Options()
    options.load_capabilities(desired_caps)

    driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', options=options)
    return driver

注意,新版本appium要用options参数,不能直接用desired_caps,否则会报错:

>           capabilities = options.to_capabilities()
E           AttributeError: 'NoneType' object has no attribute 'to_capabilities'

报错图如下:
在这里插入图片描述

在pytest测试用例中调用appium

这个例子我们调用appium获取设备的屏幕宽度并打印出来

@allure.feature("设备信息模块")
class TestDeviceInfo:

    @allure.story("设备信息故事")
    @allure.description("设备信息描述")
    @allure.severity(allure.severity_level.BLOCKER)
    def test_get_device_info(self, cmdopt):
        print('test_get_device_info start')
        print('get cmdopt:' + cmdopt)
        
        driver = get_appium_driver()
        width = driver.get_window_size()['width']
        print('screen width {}'.format(width))
        
        print('test_get_device_info end')

运行结果:
在这里插入图片描述
到此Python+Appium+Pytest+Allure框架就串连起来了。

更多功能

  1. yaml
    yaml也可以整合进来,比如把desired_caps存放到datas/caps.yml中,然后代码中去读这个配置,而不是写在代码里面。测试用例的步骤也可以用yaml表示,这里不再细说。
  2. 多进程/线程支持多设备同时进行测试
    这块可以单独拿一篇来说了,这里只说下思路,在代码中自动执行adb devices然后把所有adb设备都列举出来,每台设备开个线程或进程执行appium服务,每个服务端口配置成不一样的。
    不同设备根据不同端口访问appium服务进行设备操作。
    Python+Appium+Pytest+Allure自动化测试框架-安装篇
    作者:帅得不敢出门

http://www.kler.cn/a/374503.html

相关文章:

  • Canoe E2E校验自定义Checksum算法
  • 解读Makefile中,`=`、`:=`、`?=` 和 `+=`差异
  • CAN201 Introduction to Networking(计算机网络)Pt.1 导论和应用层
  • 算法题(13):异或变换
  • Go web 开发框架 Iris
  • LightGBM分类算法在医疗数据挖掘中的深度探索与应用创新(上)
  • springboot获取七牛云文件上传凭证token
  • 从二维图像到三维重建:由运动到结构(SfM)的完整流程推导【含数学原理及推导】
  • 基于STM32+华为云IOT设计的大棚育苗管理系统
  • Java 反射
  • 图论BFS
  • 微信小程序之流浪动物救助:爱与希望同行
  • 【SQL】 Navicate 17 连接sql server
  • 【小白学机器学习31】 大数定律,中心极限定理,标准正态分布与概率的使用
  • Vue2指令原理手写
  • 基于SSM+微信小程序的汽车预约维修管理系统(汽车3)
  • sublime text 常用快捷键
  • Chrome与夸克谁更节省系统资源
  • 宝塔使用clickhouse踩坑
  • 《AI产品经理手册》——解锁AI时代的商业密钥
  • 从新手到专家:7款电脑平面设计软件评测
  • WPF+MVVM案例实战(十五)- 实现一个下拉式菜单(上)
  • OpenCV视觉分析之目标跟踪(3)实现基于金字塔的 Lucas-Kanade 算法来进行稀疏光流计算的类SparsePyrLKOpticalFlow的使用
  • 《解锁 TDD 魔法:高效软件开发的利器》
  • 读写chrome.storage.local
  • 股票基础交易规则——涨跌幅限制、价格笼子?