学习python自动化——pytest单元测试框架

news2024/11/26 14:59:14

一、什么是pytest

单元测试框架,unittest(python自带的),pytest(第三方库)。

用于编写测试用例、收集用例、执行用例、生成测试结果文件(html、xml)

1.1、安装pytest

pip install pytest

1.2、导入

import pytest

二、步骤

  2.1、TestCase(测试用例)

    2.1.1、创建测试类      

      1)类名要以Test开头

class TestData:
    2.1.2、测试用例函数      

      1)可以直接用函数、类的形式
      2)函数名必须是test_开头
      3)执行顺序,按照从上到下的代码顺序执行

def test_data():        #测试函数
    print("test_data")

class TestData:
    def test_hello(self):    #测试函数
        print("test_hello")
    2.1.3、前置后置(fixture)
      前置:环境准备、数据准备
      后置:清理工作、数据清理

    pytest的fixture有2种实现方式:

      2.1.3.1、xunit-style

        跟unittest框架的机制非常相似,即setup/teardown系列

        1、测试函数/方法级别:每一个测试函数都会执行的前置和后置。

          1)测试类内部的测试方法:

            前置函数名称:setup_method

            后置函数名称:teardown_method

          2)模块下的测试函数:

            前置函数名称:setup_function

            后置函数名称:teardown_function

        2、测试类级别:一个测试类只执行一次前置和后置。

          前置函数名称:setup_class

          后置函数名称:teardown_class

          注意:用@classmethod装饰

        3、测试模块级别:一个测试模块只执行一次前置和后置。

          前置函数名称:setup_module

          后置函数名称:teardown_module

from selenium import webdriver
from time import sleep
from random import randint

def setup_module():
    print("====  模块级的 setup 操作  ====")

def teardown_module():
    print("====  模块级的 teardown 操作  ====")

def test_random():
    assert randint(1, 5) == 3

class TestWeb:

    @classmethod
    def setup_class(cls):
        print("====  测试类级的 setup 操作  ====")

    @classmethod
    def teardown_class(cls):
        print("====  测试类级的 teardown 操作  ====")

    def setup_method(self):
        print("====  测试用例级的 setup 操作  ====")
        self.driver = webdriver.Chrome()

    def teardown_method(self):
        print("====  测试用例级的 teardown 操作  ====")
        self.driver.quit()

    def test_search(self):
        self.driver.get("https://www.baidu.com/")
        self.driver.find_element_by_id("kw").send_keys("柠檬班")
        self.driver.find_element_by_id("su").click()
        sleep(1)
      2.1.3.2、fixture机制

        通过@pytest.fixture装饰器来定义fixture

         1、fixture定义

          1)fixture通过函数实现

          2)使用@pytest.fixture进行装饰

          3)前置和后置都放在同一个函数当中,通过yeild关键字区分前置和后置,yeild前面的为前置,yeild后面的为后置

import pytest

@pytest.fixture
def init():
    print("用例执行之前,执行的代码")  # 前置代码 
    yield
    print("用例执行之后,执行的代码")  # 后置代码

@pytest.fixture
def init2():
    print("用例执行之前,执行的代码")  # 只有用例执行之前的前置准备代码

@pytest.fixture
def init3():
    yield
    print("用例执行之后,执行的代码")  # 只有用例执行之后的后置清理代码

          4)fixture有4个作用域:测试会话(session)、测试模块(module)、测试类(class)、测试用例(function)

           设置fixture的作用域,通过@pytest.fixture(scope=作用域)来设置。默认情况下,scope=function

import pytest

# 没有设置scope,默认为测试函数级别。即调用此fixture的测试类/模块/函数下,每个测试函数都会执行一次这个fixture
@pytest.fixture
def init():
    print("用例执行之前,执行的代码")  # 前置代码
    yield
    print("用例执行之后,执行的代码")  # 后置代码

# 设置scope为class。调用此fixture的测试类下,只执行一次这个fixture.
@pytest.fixture(scope="class")
def init2():
    print("用例执行之前,执行的代码")  # 只有用例执行之前的前置准备代码

# 设置scope为session。autouse表示自动使用。
# 那么在pytest收集用例后,开始执行用例之前会自动化执行这个fixture当中的前置代码,
# 当所有用例执行完成之后,自动化执行这个fixture的后置代码。
@pytest.fixture(scope="session",autouse=True)
def init3():
    yield
    print("用例执行之后,执行的代码")  # 只有用例执行之后的后置清理代

          5)fixture的返回值设置:yeild 返回值

            如果前置准备工作当中,有变量要传递给测试用例去使用的话,那么就需要返回这个变量,可以有多个返回值,返回值是个元组。

            yeild 返回值,返回值2,返回值3    

yield mysql_obj,True,123

        2、fixture调用

           2.1、自动调用:定义fixture有个参数:autouse,默认autouse=False

              自动调用时,所有的测试函数都会调用一次,所以只有session作用域时才适用。

import pytest

@pytest.fixture(autouse=True)           #自动调用
def init():
    print("===我是前置准备工作===")
    yield
    print("***我是后置清理工作***")

@pytest.fixture(scope="class")
def init_class():
    print("===我是测试类  前置准备工作===")
    yield
    print("***我是测试类  后置清理工作***")

def test_hello():
    print("hello pytest!!")

class Test_demo:
    def test_bey(self):
        print("bey pytest!!")

           2.2、主动调用:测试函数/测试类上加上:@pytest.mark.usefixtures("fixture的函数名称"),可以调用多个。

             当调用两个同级别的fixture函数时,按就近原则调用

            1)无需接收fixture返回值

import pytest
from random import randint

#测试类级别
@pytest.fixture(scope="class")
def init_class():
    print("===我是 测试类  前置准备工作===")
    mysql_obj = "fixture要返回出去的数据库对象"
    yield mysql_obj,True,123  # 返回一个变量/对象
    print("***我是 测试类  后置清理工作***")

# 测试函数级别
@pytest.fixture()
def init():
    print("===我是前置准备工作===")
    yield
    print("***我是后置清理工作***")

# 测试函数级别
@pytest.fixture()
def init2():
    print("===我是前置准备工作2===")
    yield
    print("***我是后置清理工作2***")

@pytest.mark.usefixtures("init")        #去执行名为init的fixture函数
@pytest.mark.usefixtures("init2")       #当调用两个同级别的fixture函数时,就近原则,先调用init2,再调用init
def test_hello():
    print("hello pytest!!")

def test_demo():                        #无前置后置
    print("hello world")

@pytest.mark.usefixtures("init_class")      #执行名为init_class测试类级别的前置后置
@pytest.mark.usefixtures("init")            #当调用两个不同级别的fixture函数时,先调用init_class,再每个函数调用init
class TestDemo:

    def test_random(self):
        assert randint(3, 3) == 3

    def test_abc(self):
        assert True
        
    @pytest.mark.usefixtures("init2")           #当类内部函数有调用时,就近原则,先调用init2,再调用init
    def test_bey(self):
        print("bey pytest!!")

             2)要接收fixture返回值:fixture的函数名作为用例的参数,相当于fixture函数名 = fixture调用的返回值

              当fixture函数名作为测试用例参数时,在执行用例时,会自动寻找该函数的fixture,并执行。

              ps:pytest当中的测试用例,如果有参数,分为2种:

                1、fixture的名字

def test_random(self,init_class)
  pass

                2、参数化要传的参数

@pytest.mark.parametrize("case",datas)
def test_read_datas(case):
    pass
 
import pytest
from random import randint

# 测试类级别
@pytest.fixture(scope="class")
def init_class():
    print("===我是 测试类  前置准备工作===")
    mysql_obj = "fixture要返回出去的数据库对象"
    yield mysql_obj,True,123  # 返回一个变量/对象
    print("***我是 测试类  后置清理工作***")

#@pytest.mark.usefixtures("init_class")  # 当fixture函数名作为测试用例参数时,在执行用例时,会自动寻找该函数名并执行,所以这个可以不写
class TestDemo2:

    def test_random(self,init_class): # fixture的名字作为用例的参数,相当于fixture名字 = fixture调用的返回值
        mysql_obj, bool_obj, int_obj = init_class
        print("从fixture当中接收的变量为:{}".format(init_class))
        assert randint(1, 5) == 3

    def test_abc(self):
        assert True

      2.1.3.3、conftest.py共享机制

       1、共享实现

        1)在项目根目录下,创建一个conftest.py文件。

        2)文件名必须是conftest.py,大小写敏感,不可改名字。

        3)conftest.py当中,可以编写多个fixture

        4)在测试用例文件当中,不需要引入conftest.py文件。直接调用fixture的函数名,会自动去conftest.py当中查找的。

        2、conftest.py层级作用域

         conftest.py在项目根目录下,则项目下的所有测试用例,均可使用conftest.py中定义的fixture。即项目根目录下的conftest.py,作用域是整个项目。

         根目录下的conftest.py里的fixture,无论项目下的哪个用例,都可以使用。

         子目录day3-login下的conftest.py里的fixture,只有day3-login下的用例可以使用。

         子目录day3-register下的conftest.py里的fixture,只有day3-register下的用例可以使用。

           测试用例在执行时,调用fixture的顺序,按 就近原则 调用。

           测试用例文件中的fixture > 当前目录中的fixture > 上级目录中的fixture > 根目录中的fixture

       2.1.3.4、fixture嵌套

        嵌套使用即:一个fixture,可以做另外一个fixture的参数,自动调用参数的fixture。

        注意:由于参数里的fixture要先执行,所以参数里的fixture的作用域 >= fixture的作用域。

@pytest.fixture
def init_class():
    print(" %%%% 我是 day3-login包下test_random模块的测试类  前置准备工作 %%%% ")
    yield
    print(" %%%% 我是 day3-login包下test_random模块的测试类  后置清理工作 %%%% ")

@pytest.fixture
def init_case(init_class):      #init_class作为init_case的参数,当用例调用init_case时,init_case会自动调用init_class
    print("测试用例级别 的前置")
    print("=================================")
    yield
    print("测试用例级别 的后置")
    print("===============用例  完结==================")

      2.1.3.5、session 和 module 级别的 fixture

       1)session 级别的 fixture

         从 pytest 开始收集用例,到执行用例完成的整个过程,为一次会话,可以设置 autouse=True 

import pytest

@pytest.fixture(scope="session",autouse=True)
def fix_session():
    print("===  session级别的前置 ,只执行1次 ============")
    yield True
    print("===  session级别的后置 ,只执行1次 ============")

                

       2)module 级别的 fixture

         module 是指测试 py 文件,夹具夹的是整个 test_*.py 文件里的测试用例。

         在 test_*.py 里,哪一行代码调用了 module 级别的 fixture,那么夹的就是那行代码之后的所有测试用例。

import pytest

@pytest.fixture(scope="module")
def fix_module():
    print("===  module级别的 前置 ======")
    yield
    print("===  module级别的 后置 ======")

    2.1.4、断言

      将预期结果与实际结果进行对比
      assert 表达式(True表示断言为真,False表示断言失败)

assert a==b

  2.2、用例收集:自动收集

    1)pytest自动收集用例规则:在当前文件所在的目录下,收集文件名为test_*.py或*_test.py的测试文件中,以test_开头的测试用例或Test开头的测试类。

    2)pytest.main():以当前文件所在的目录下,收集用例并执行。

pytest.main(["-s","-v"])

    3)用例分类执行

pytest.main(["day1/test_case1.py"])      #只执行day1目录下test_case1.py文件的测试用例

  2.3、TestRunner(用例执行)

    1)需要配置执行器:File>>>Settings>>>Tools>>>Python Integrated Tools>>>Default test runner:pytest
    2)执行顺序:测试文件顺序按ASCII码,文件内部测试用例按代码顺序执行

    3)若py文件执行过一次后,会保留上一次执行的方式,需要清除上一次执行记录。右上角——Edit Configurations——Remove掉该py文件。

   2.4、生成测试报告

    2.4.1、生成JunitXML 格式的测试报告

      命令:--junitxml=path     

    2.4.2、生成result log格式的测试报告

      命令:--resultlog=report\log.txt     

    2.4.3、生成Html格式的测试报告 

      1)安装pytest-html插件

pip install pytest-html

      2)在pytest.main()的参数当中添加:--html=报告路径

pytest.main(["-s","-v","--html=report.html"])      #生成html格式的测试报告
    2.4.4、生成allure的测试文件 

      2.4.4.1、allure是依据测试结果文件,呈现更好看的更多维度的测试报告。
      注意:它不执行用例也不生成测试结果文件。所以,要使用allure的时候,一定要先生成测试结果文件。allure需要依赖测试框架,执行测试用例后,根据测试结果再生成测试报告。

      2.4.4.2、allure的使用

        1、安装allure命令行工具
          1)下载allure安装包。下载地址:https://repo.maven.apache.org/maven2/io/qameta/allure/allure-commandline/

          2)下载完成之后,解压到任意目录
          3)配置环境变量:allure的bin目录。比如D:\allure-2.12.1\bin

          4)在cmd当中,执行allure命令,不报错即allure安装成功。

        2、pytest在执行用例之后,要生成allure可以识别的测试结果文件。
          1)安装插件

pip install allure-pytest

          2)在pytest.main()的参数当中添加:--alluredir=allure生成的文件目录(手动创建一个)

pytest.main(["-s","-v","--alluredir=allure_file"])

pytest.main(["--alluredir=allure_file","--clean-alluredir"])      #每次运行前清除allure_file中的文件,为防止数据重复

        3、使用allure去解析2中的测试结果文件,再呈现最终的测试报告
          1)在cmd当中首先跳转到rootdir下面

          2)在cmd当中执行命令:allure serve 2中的测试结果文件目录

   2.5、参数化(数据驱动)

    1、@pytest.mark.parametrize("参数名",列表数据)

datas={"api":"login","url":"http://api.lemoban.com/futureloan/login",
       "api":"register","url":"http://api.lemoban.com/futureloan/register",
       "api":"recharge","url":"http://api.lemoban.com/futureloan/recharge",
       "api":"withdraw","url":"http://api.lemoban.com/futureloan/withdraw"}

@pytest.mark.parametrize("case",datas)
def test_read_datas(case):
    pass

    2、组合参数化:多组参数,依次组合。
      使用多个@pytest.mark.parametrize

#示例:用例有4个:0,2/0,3/1,2/1,3 迪卡尔积
@pytest.mark.parametrize("x", [0, 1])
@pytest.mark.parametrize("y", [2, 3])
def test_foo(x, y):
    pass

   2.6、重试机制(rerun)

1)安装插件:pytest-rerunfailures

pip install pytest-rerunfailures

2)命令

pytest --reruns 次数

pytest --reruns-delay  重运行次数之间的间隔时间(以秒为单位)

pytest.main(["--reruns","2","--reruns-delay","5"])

   2.7、筛选用例,mark一下

当我们想只运行诸多py文件当中的部分用例,需要过滤用例时,可以使用pytest.mark一下。

1)注册标签名

通过pytest.ini配置文件注册。在pytest.ini文件当中:

[pytest]   # 固定的section名

markers= # 固定的option名称

  标签名1: 标签名的说明内容。

  标签名2

  标签名N

 2)在测试用例/测试类中给用例打标记(只能使用已注册的标记名)

 在测试用例的前面加上:@pytest.mark.已注册标签名

 3)运行时,根据用例标签过滤(-m 标签名)

调用pytest.main()函数,将运行时的参数以列表传进去,同样也可以达到命令行运行的效果。可以使用逻辑运算符。

pytest.main(["-m","demo"])      #执行标签为demo的测试用例
pytest.main(["-m not demo"])    #执行标签不是demo的测试用例
pytest.main(["-m demo and test"])  #执行标签是demo又是test的测试用例

  【已更新】B站讲的最详细的Python接口自动化测试实战教程全集(实战最新版)
》,以及配套的接口文档/项目实战【网盘资源】,需要的朋友可以下方视频的置顶评论获取。肯定会给你带来帮助和方向。

【已更新】B站讲的最详细的Python接口自动化测试实战教程全集(实战最新版)

 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【ARM Linux驱动开发】嵌入式ARM Linux驱动开发基本步骤

【ARM Linux驱动开发】嵌入式ARM Linux驱动开发基本步骤 文章目录 开发环境驱动开发(以字符设备为例)安装驱动应用程序开发附录:压缩字符串、大小端格式转换压缩字符串浮点数压缩Packed-ASCII字符串 开发环境 首先需要交叉编译器和Linux环境…

豆包PixelDance指南:字节跳动推出的AI视频生成大模型,突破多主体互动难关

豆包PixelDance是由字节跳动旗下火山引擎发布的AI视频生成大模型。它是业界首个突破多主体互动难关的视频生成模型,支持多风格多比例的一致性多镜头生成。PixelDance基于DiT架构,具备高效的DiT融合计算单元,能够实现复杂的多主体运动交互和多…

【高等数学】 一元函数积分学

1. 不定积分的计算 1.1. 基本积分表 知识点 例题 1.2. 凑微分(第一类换元法) 知识点 本质:利用复合函数求导法则的逆运算 第一步,识别或者凑出来复合函数的导函数 如果被积函数具备以下特点: 1.它由两项相乘来表…

《案例》—— OpenCV 实现2B铅笔填涂的答题卡答案识别

文章目录 一、案例介绍二、代码解析 一、案例介绍 下面是一张使用2B铅笔填涂选项后的答题卡 使用OpenCV 中的各种方法进行真确答案识别,最终将正确填涂的答案用绿色圈出,错误的答案不圈出,用红色圈出错误题目的正确答案最终统计正确的题目数…

PCL用KDtree,给搜索到的邻近点上色

用KDtree&#xff0c;给搜索到的邻近点上色。 #include <pcl/io/pcd_io.h> #include <pcl/point_types.h>#include <pcl/search/kdtree.h> // 包含kdtree头文件 #include <pcl/visualization/pcl_visualizer.h> #include <boost/thread/thread.hpp&…

【MATLAB】基于RSSI的蓝牙定位程序,4个锚点、二维平面

目录 ​编辑 商品描述 主要功能 技术细节 适用场景 下载链接 商品描述 这款基于接收信号强度指示&#xff08;RSSI&#xff09;原理的蓝牙定位程序&#xff0c;专为需要高效、可靠定位解决方案的开发者和研究人员设计。它能够在二维平面内&#xff0c;通过4个锚点实现对未…

ESNP静态路由进阶

0x01 前言 前一篇文章说了一个简单的ESNP静态路由实验&#xff0c;那么后面给了一个进阶拓扑&#xff1a;https://blog.csdn.net/Moxin1044/article/details/142850946 现在我们来解决一下 0x02 拓扑图 在上一个拓扑图基础上&#xff0c;规划了一个新的7.0网段&#xff0c;那…

DGX平台

NVIDIA DGX 平台 文章目录 前言一、优势经过验证的企业 AI 标准体验 DGX 平台的优势1. 先进的 AI 开发平台2. 融入了 NVIDIA AI 专业技术和服务3. 性能卓越、成本可预测二、用例DGX 助力 AI 走向各行各业1. 壳牌推动能源领域不断取得突破2. 宝马借助 AI 重新定义工厂物流3. 索尼…

【Linux】段错误(核心已转储)

原因:linux在安装docker 安装完之后再添加用户就报错了。。。。 各种查原因: 内存问题:系统可能存在内存损坏或不足的问题。磁盘空间不足:系统分区可能没有足够的空间来创建新用户。文件系统错误:文件系统可能存在错误。SELinux或AppArmor:安全模块可能阻止了 useradd 命…

el-table表头加红色星标

代码&#xff1a; <el-table-column prop"name" label"姓名" width"auto"><template #header><span style"color: red; margin-right: 4px">*</span><span>姓名</span></template></el…

Java idea中项目在maven执行install命令后出现Java heap space错误

Java idea中项目在maven执行install命令后出现Java heap space错误 这代表着你的maven需要足够的内存来编译和打包你的项目&#xff0c;而你分配的堆内存却不够用。 解决办法&#xff1a; idea左上角File —> Settings Build —> Build Tools —> Maven —> Run…

【GPT提问技巧】如何高效提问GPT:掌握核心技巧,获取高质量回答的终极指南!

人工智能&#xff08;AI&#xff09;已经成为我们获取知识和信息的重要工具。尤其是GPT&#xff08;Generative Pre-trained Transformer&#xff09;等语言模型&#xff0c;凭借其强大的自然语言处理能力&#xff0c;能够为我们提供丰富的回答。然而&#xff0c;如何向GPT提问…

【论文分享】ShEF: Shielded Enclaves for Cloud FPGAs 22‘ASPLOS

目录 ABSTRACTINTRODUCTIONBACKGROUND AND MOTIVATIONTrusted Execution Environments (TEEs)Conventional FPGA Security MechanismsRemote FPGAs-as-a-ServiceChallenges for Secure and Customized ComputingA lack of asymmetric keys.Presence of an untrusted ShellLack …

【Vue3】 h()函数的用法

目录 介绍 参数 使用案例 1.创建虚拟 DOM 元素 2. 组件的动态渲染 3. 创建功能组件 4.渲染动态属性 5. 使用插槽 6. 创建动态标签 介绍 h() 函数用于辅助创建虚拟 DOM 节点&#xff0c;它是 hypescript 的简称——能生成 HTML (超文本标记语言) 的 JavaScript&#x…

开发实时美颜系统:视频美颜SDK与直播平台的集成方案详解

本文将详细介绍如何开发一个实时美颜系统&#xff0c;并探讨视频美颜SDK与直播平台的集成方案&#xff0c;帮助开发者实现流畅的美颜功能。 一、视频美颜SDK的核心功能 视频美颜SDK主要提供了一系列实时处理视频图像的算法&#xff0c;常见的功能包括&#xff1a; 1.美白、磨…

Vue3教程 - 3 HelloWorld

更好的阅读体验&#xff1a;点这里 &#xff08; www.foooor.com &#xff09; 3 HelloWorld 下面来正式进入 Vue3 的学习。先写一个 HelloWorld。 3.1 关于Vite Vite 是一个新型前端构建工具。 在 Vue2 中&#xff0c;我们主要使用 vue-cli 来构建我们的前端项目&#x…

进程的优先级详解(1)(在Linux中观察进程优先级,优先级的概念,优先级范围,进程的切换和切换的过程和理解)

文章目录 进程的优先级什么是进程的优先级&#xff1f;进程的优先级的划分1. 优先级的级别划分2. 动态与静态优先级 那么为什么要有优先级呢&#xff1f;区分 那么是怎么确定优先级的呢&#xff1f; Linux中的优先级Linux中的进程优先级补充 Linux优先级的范围进程的切换切换过…

四款pdf转图片在线转换免费工具推荐:

大家好&#xff01;今天我来给大家推荐几款PDF转图片的在线转换工具&#xff0c;让你轻松将PDF文件转换成图片&#xff0c;无论是工作还是学习&#xff0c;都能派上大用场。下面&#xff0c;让我们来看看这几款工具吧&#xff01; 一、福昕转换器 直通车&#xff08;粘贴到浏览…

多线程(二):创建线程关键属性终止线程

目录 1、run & start 2、Thread类常见的属性和方法 2.1 构造方法 2.2 属性 3、后台进程 & 前台进程 4、后台线程的判断和设定——isDaemon & setDaemon 5、线程是否存活——isAlive 6、终止一个线程 6.1 lambda变量捕获 6.2 currentThread & isInterr…

atomic physics

​​​​​​​ https://www.youtube.com/watch?v6F8Wjblb0vE&listPLX2gX-ftPVXVqAS_q3OfJDmPn8-EQld_r&index18