pytest自动生成测试类 demo

news2025/1/23 7:25:23

一、 pytest自动生成测试类 demo

# -*- coding:utf-8 -*-
# @Author: 喵酱
# @time: 2023 - 08 -15
# @File: test4.py
# desc:
import pytest
import unittest

# 动态生成测试类

def create_test_class(class_name:str, test_cases:list) -> type:
    """
    生成测试类
    :param class_name: 测试类的类名,是一个字符串
    :param test_cases: 是一个可迭代的对象(list),每一个元素case是一个字典
    在pytest测试类中,每一个方法,就是一条case,所以这里将多条case数据,生成测试类的多个方法
    :return: 测试类,包含多个方法(测试case)
    """
    # 使用 type() 函数动态创建一个继承自 unittest.TestCase 的子类
    test_class = type(class_name, (unittest.TestCase,), {})
    for case in test_cases:
        test_method = generate_test_method(case)
        # 动态为测试类添加测试方法
        setattr(test_class, test_method.__name__, test_method)
    return test_class

# 生成测试方法
def generate_test_method(case: dict) -> callable:
    """
    生成单个测试方法。

    Args:
        case (dict): 测试用例字典,包含输入和输出的信息。

    Returns:
        callable: 生成的测试方法。

    """
    def test_method(self):
        """
        实际执行测试的方法。

        """
        assert case["input"] == case["output"]

    # 设置测试方法的名称
    test_method.__name__ = f"test_{case['name']}"

    return test_method


# 定义测试数据
@pytest.fixture(params=[
    {"name": "case1", "input": 1, "output": 1},
    {"name": "case2", "input": 3, "output": 3},
])
def test_case(request):
    return request.param

# 测试生成的测试类
@pytest.mark.usefixtures("test_case")
def test_generator(test_case):
    test_class = create_test_class("DynamicTestClass", [test_case])
    # 加载测试类,并创建测试套件
    suite = unittest.TestLoader().loadTestsFromTestCase(test_class)
    # 运行测试套件
    result = unittest.TextTestRunner().run(suite)
    # 断言测试结果是否成功
    assert result.wasSuccessful()

执行 test_generator 方法

运行顺序:

1、手动触发运行  test_generator 方法

2、在执行test_generator 方法 方法之前,会先执行test_case 方法。

3、test_case 方法,数据有2条数据,则test_case 方法会执行两次

4、test_generator 同样也会执行两次。

最终效果,就是生成了两次名为 DynamicTestClass 的测试类,每个测试类,只生成了一个方法(case)

 二、代码详解

2.1 整体运行顺序

1、定义了一个函数 create_test_class,用于动态生成测试类。
2、定义了一个函数 generate_test_method,用于生成单个测试方法。
3、使用了 @pytest.fixture 装饰器定义了一个夹具(fixture) test_case,用于提供测试数据。
4、使用了 @pytest.mark.usefixtures("test_case") 装饰器标记了一个测试函数 test_generator,表示在执行该测试函数之前需要先执行 test_case 夹具。
5、在函数 test_generator 中,调用了 create_test_class 函数,传入一个测试数据。然后加载并运行生成的测试类。
6、执行测试时,先执行夹具函数 test_case 来获取测试数据。
7、根据测试数据,动态生成一个测试类 DynamicTestClass,并为该类添加一个测试方法。
8、创建测试套件,并运行测试套件中的测试用例。
9、使用 unittest.TextTestRunner 运行测试套件,并返回测试结果。
10、最后,根据测试结果判断是否成功,并进行断言。


总结起来,整个代码的执行流程是:先执行夹具函数获取测试数据,然后动态生成测试类并创建测试套件,最后运行测试套件中的测试用例,输出测试结果,并进行断言判断。

2.1 定义测试数据

# 定义测试数据
@pytest.fixture(params=[
    {"name": "case1", "input": 1, "output": 1},
    {"name": "case2", "input": 3, "output": 3},
])
def test_case(request):
    return request.param

1、@pytest.fixture 是 pytest 框架提供的装饰器,用于定义夹具(fixture)。夹具是在测试函数或测试类执行前后进行特定操作、提供数据等的函数。

2、在这段代码中,test_case 是一个夹具函数。通过 @pytest.fixture 装饰器对其进行标记,表示它是一个夹具。

3、params 参数是用来定义多个测试用例数据的列表。每个测试用例数据都是一个字典,包含了三个键值对:name、input 和 output。name 是用来描述测试用例的名称,input 是输入数据,output 是预期输出结果。

4、当执行测试函数时,pytest 会自动调用夹具函数,并根据参数列表中的每个字典生成不同的测试数据。每个测试用例都会独立执行一次测试函数,并且将对应的测试数据作为参数传递给测试函数。

在这段代码中,夹具函数 test_case 返回 request.param,request.param 表示当前测试用例的数据。当执行 test_generator 函数时,会先调用夹具函数 test_case 获取测试数据,然后将该测试数据作为参数传递给 test_generator 函数。

2.2 运行生成的测试类 test_generator

# 测试生成的测试类
@pytest.mark.usefixtures("test_case")
def test_generator(test_case):
    test_class = create_test_class("DynamicTestClass", [test_case])
    # 加载测试类,并创建测试套件
    suite = unittest.TestLoader().loadTestsFromTestCase(test_class)
    # 运行测试套件
    result = unittest.TextTestRunner().run(suite)
    # 断言测试结果是否成功
    assert result.wasSuccessful()

1、装饰器: @pytest.mark.usefixtures("test_case")

@pytest.mark.usefixtures("test_case") 是 Pytest 测试框架中一个装饰器标记(decorator marker),用于指定在运行测试用例之前需要先执行的 fixture(夹具)。

fixture 是一种在测试运行之前准备测试环境、数据或者资源的机制。它可以用来为测试用例提供必要的前置条件,例如设置数据库连接、创建临时文件、初始化测试数据等。

在给定的代码中,@pytest.mark.usefixtures("test_case") 将 "test_case" 作为 fixture 的名称进行标记。这意味着在运行使用了该标记的测试用例之前,Pytest 将先执行名为 "test_case" 的 fixture 方法。

使用 @pytest.mark.usefixtures 标记可以方便地应用 fixture,而无需在每个测试用例函数中显式调用 fixture。通过将标记应用于测试用例函数,测试框架将自动处理 fixture 的执行和销毁,并将其提供给测试用例函数作为参数使用。

总之,@pytest.mark.usefixtures("test_case") 允许你在测试运行之前自动执行名为 "test_case" 的 fixture 方法,为测试用例提供必要的准备工作。

2.3 generate_test_method

generate_test_method 函数中的 case 参数是一个字典类型。在示例代码中,test_case 是一个包含测试用例信息的字典,其中包括字段 "name""input""output",用于表示测试名称、输入和期望输出。

传递给 generate_test_method 函数的 case 参数应该具有与示例中定义的测试用例字典相同的结构。例如:

case = {"name": "example", "input": 1, "output": 2}
test_method = generate_test_method(case)

在这个示例中,case 是一个字典,表示一个测试用例,包含 "name""input""output" 字段。generate_test_method 函数将根据这些字段生成一个测试方法,并为该方法设置名称。

2、create_test_class

在函数 create_test_class 中,test_cases 参数的类型可以是任何可迭代对象,比如列表或元组。

示例代码中使用了迭代循环来遍历 test_cases,假设 test_cases 是一个列表,每个元素都是一个测试用例字典。所以 test_cases 的结构可以类似以下的形式:

test_cases = [
    {"name": "case1", "input": 1, "output": 2},
    {"name": "case2", "input": 3, "output": 3},
]

其中,每个 test_cases 列表的元素都是一个包含测试用例信息的字典。

所以,在调用 create_test_class 函数时,你可以传递一个包含测试用例字典的列表作为 test_cases 参数,动态生成相应的测试类。

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

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

相关文章

未能为数据库对象分配空间,因为文件组primary已满

今天正式环境突发需要这种情况 原因:数据库磁盘已满 解决方式:清理掉占用磁盘空间大的表,清理无效的历史备份文件。

Git分布式版本控制系统基础概念

前言 我们在大学毕业写论文的时候碰到过如下的现象&#xff1a; <<毕业论文第一版.doc>> <<毕业论文第二版.doc>> <<毕业论文第三版.doc>> <<毕业论文最终版.doc>> <<毕业论文完结版.doc>> 你的论文会不断地修改…

LeetCode150道面试经典题-- 有效的括号(简单)

1.题目 给定一个只包括 (&#xff0c;)&#xff0c;{&#xff0c;}&#xff0c;[&#xff0c;] 的字符串 s &#xff0c;判断字符串是否有效。 有效字符串需满足&#xff1a; 左括号必须用相同类型的右括号闭合。左括号必须以正确的顺序闭合。每个右括号都有一个对应的相同类…

easyx图形库基础:3实现弹球小游戏

实现弹球小游戏 一.实现弹球小游戏:1.初始化布&#xff1a;2.初始化一个球的信息&#xff1a;3.球的移动和碰撞反弹4.底边挡板的绘制和移动碰撞重置数据。 二.整体代码&#xff1a; 一.实现弹球小游戏: 1.初始化布&#xff1a; int main() {initgraph(800, 600);setorigin(40…

相互之间差异较大的15种颜色、35种颜色 | 颜色 色卡 色盘 RGB HEX十六进制

任意两个颜色之间&#xff0c;RGB的欧氏距离大于120 1: (211, 44, 31), #d32c1f 2: (205, 140, 149), #CD8C95 3: (67, 107, 173), #436bad 4: (205, 173, 0), #CDAD00 5: (4, 244, 137), #04f489 6: (254, 1, 154), #fe019a 7: (6, 71, 12), #06470c 8: (97, 222, 42), #61de…

PHP 从 URL(链接) 字符串中获取参数

PHP 从 URL&#xff08;链接&#xff09; 字符串中获取参数 //URL(链接)字符串 $url https://www.baidu.com/?name小洪帽i&sex男&age999; //parse_url 函数从一个 URL 字符串中获取参数 $urlparse_url($url); //输出获取到的内容 echo "<pre>"; pri…

微信小程序:函数节流与函数防抖

目录 问题引入&#xff1a; 定义 解决方案&#xff1a;函数节流 一、案例举例 1.页面展示 2.search.wxml标签展示 3.search.js展示 4.结果展示 二、函数节流解决问题 1.函数 2.实例应用 三、函数防抖解决问题 1.函数 2.原理 3.应用场景 4.应用实例 总结 问题引入…

创建用户账户

题目&#xff1a; 创建下列用户、用户祖&#xff0c;并按要求完成设置&#xff1a; 组名为 sysmgrs natasha 用户的附属组是 sysmgrs harry 用户的附属组是 sysmgrs john 用户的 shell 是非交互式 shell&#xff0c;且不是 sysmgrs 组的成员 natasha、harry、john 的密码是…

前端代理配置

dev: {env: require(./dev.env),port: process.env.PORT || 8080,autoOpenBrowser: true,assetsSubDirectory: static,assetsPublicPath: /,proxyTable: {// 以 /party/fundamental/ 开头的请求&#xff0c;全部转发到 target 设置的地址/party/fundamental/: {// target: http…

vim键盘图

国外&#xff1a;http://www.viemu.com/a_vi_vim_graphical_cheat_sheet_tutorial.html&#xff0c;原创&#xff0c;有SVG图&#xff0c;有分步骤的图。 国内翻译&#xff1a;[https://blog.csdn.net/qq_41052753/article/details/101031847 有几个配色&#xff0c;很高清&…

深入了解电脑硬件以及多线程编程

文章目录 认识计算机硬件与多核CPU的工作原理单核CPU多核CPU并发与并行 深入了解进程、线程及其优先级进程与线程线程的创建与命名线程的优先级与控制线程的休眠与等待 线程安全与锁机制同步与异步线程安全问题与锁可重入锁解决线程安全问题 多线程间的通信与线程池的使用线程通…

人脸检测 - mtcnn

文章目录 1. 人脸检测2. mtcnn2.1 概述2.2 网络结构2.2.1 构建图像金字塔2.2.2 P - Net2.2.3 R-Net&#xff08;Refine Network&#xff09;&#xff1a;2.2.4 O-Net&#xff08;Output Network&#xff09;: 3. 总结4. 代码示例4.1 mtcnn.py4.2 detect.py4.3 utils.py 1. 人脸…

架构演进及常用架构

1架构演进及常用架构 1.1单体分层架构 1.2 多应用微服务架构 1.3 分布式集群部署 部署 CDN 节点&#xff1a; 用户访问量的增加意味着用户地域的分散请求&#xff0c;如果所有请求都直接发送中心服务器的话&#xff0c;距离越远&#xff0c;响应速度越差&#xff0c;这时就需…

spring-boot-admin笔记

spring-boot-admin笔记 本篇教程是基于springboot 2.3.8.RELEASE版本 和spring-boot-admin-dependencies 2.3.0版本搭建spring-boot-admin的server端app pom配置 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.ap…

Java课题笔记~ 数据提交的方式

前四种数据注入的方式&#xff0c;会自动进行类型转换。但无法自动转换日期类型。 &#xff08;1&#xff09;单个数据&#xff08;基本数据类型&#xff09;注入 在方法中声明一个和表单提交的参数名称相同的参数&#xff0c;由框架按照名称直接注入。 &#xff08;2&#x…

GD32F103的DAC1

GD32F103的DAC1采用定时器1触发&#xff0c;DMA传输&#xff0c;将数据转换为电压输出到PA5引脚。 GD32F103的DAC为12位DA转换器,可以将12位的数据转换为电压,从外部引脚上输出; DAC_OUT0映射到PA4,DAC_OUT1映射到PA5; DACx引脚上的模拟输出电压: DACoutput DAC_DO * Vref/4…

Git命令详解

1 常用命令 1&#xff09;初始化本地仓库 git init <directory> 是可选的&#xff0c;如果不指定&#xff0c;将使用当前目录。 2&#xff09;克隆一个远程仓库 git clone <url> 3&#xff09;添加文件到暂存区 git add <file> 要添加当前目录中的所…

概念解析 | 长尾分布:从无处不在的‘少数派’中挖掘价值

注1:本文系“概念解析”系列之一,致力于简洁清晰地解释、辨析复杂而专业的概念。本次辨析的概念是:长尾分布(Long-Tail Distribution)。 揭秘长尾分布:从无处不在的‘少数派’中挖掘价值 What is a Long Tail Distribution? (Definition & Example) - Statology 一、背…

Neo4j的使用场景_以及Windows版安装_欺诈检测_推荐_知识图谱---Neo4j图数据库工作笔记0003

可以看到使用场景,比如欺诈检测, 要建立图谱,才能进行,欺诈人员检测 可以看到图谱的各种应用场景 然后推荐引擎也需要,可以看到 在金融,旅行,求职招聘,保健,服务,媒体娱乐,都可以进行推荐 然后还有知识图谱 身份访问管理,这里,可以进行安全管理,可以挖掘出潜在关系,分析, 某…

Android学习之路(4) UI控件之Button (按钮)与 ImageButton (图像按钮)

本节引言&#xff1a; 今天给大家介绍的Android基本控件中的两个按钮控件&#xff0c;Button普通按钮和ImageButton图像按钮&#xff1b; 其实ImageButton和Button的用法基本类似&#xff0c;至于与图片相关的则和后面ImageView相同&#xff0c;所以本节 只对Button进行讲解&am…