如何使用pytest组织自动化测试用例结构?

如何组织自动化测试工程的目录结构?这篇文章介绍了我是如何组织整个自动化工程目录结构的,本篇介绍下我是如何利用pytest框架组织一个测试用例文件的。

用例文件组织原则

整个testsuite目录下整体上按照特性模块划分目录,每个目录下可以只包含一个"test"开头的用例文件,也可以根据实际情况包含多个用例文件,尽量做到每个用例文件"test_xxx.py"只编写特性比较接近的用例,而且建议不要在一个文件中写过多的用例。

单个用例文件(test_xxx.py)如何组织用例结构

如果用例结构组织不合理,往往积累到一定量的用例时就会发现再增加用例时就会受制于各种因素的影响而举步维艰,往往为了规避影响需要写大量冗余代码,这样的惩罚可能还不足以让你记忆深刻,最痛苦的莫过于大量的自动化用例并发运行时一些意想不到的因素互相影响导致通过率始终无法达到预期的目标,调试、定位、修复用例消耗大量精力。解决这些问题需要有丰富的经验,开始编写用例时就精心组织、潜心思考、仔细推敲,就是付出成本最小的办法。

pytest框架可以很方便地组织我们的测试用例,默认配置下pytest会识别test开头的文件和test开头的方法,每一个方法就会被当做一个测试用例,就像下面这个示例文件包含了n个测试用例,拉起自动化任务时这些方法都会被运行。

# 用例文件名:test_feature01.py
# setup和teardown代码
...
def test_用例001:
  # 用例代码
  ...
  
def test_用例002:
  # 用例代码
  ..
  
def test_用例00n:
  # 用例代码
  ...

编写测试用例和开发写代码也是类似的,都有一些通用的原则,下面几个是针对测试用例代码比较重要的原则:

  • DRY原则:Don't Repeat Yourself,意思就是不要写重复代码,重复代码写得越多,你就会发现后面自动化维护就是个噩梦。实际判断的原则就是,如果你发现有些代码你需要再大于2个地方使用它,就要想办法把它封装为一个关键字。

  • 每个用例都要添加检查点,否则用例就没有任何意义了,pytest框架只需要使用assert进行检查就可以了。

  • 每个测试用例要能够独立执行,不能与其他用例有任何顺序依赖等关系。

每一种产品的自动化场景千差万别,大家公认的自动化用例的执行步骤包括环境初始化setup-->用例执行-->环境清理teardown三大部分,使用pytest的fixture固件可以很方便地实现模块级、方法级的setup和teardown(还支持会话级的),自动化测试用例编写人员的代码技能一般都不会太高,最好是能形成统一的用例模板和用例编写规范,测试人员只要按照模板级规范要求去写测试用例代码就可以了,下面是我使用的一种用例组织结构(项目中实际使用的自动化测试用例代码,摘取了关键部分),关键在注释中进行了一一说明,对其他的代码不需要做过多关注,只需要关注整个用例的执行流程即可。

# 文件名:test_远程扫描配置.py
import pytest

from logic.logic import *
from logic.mysql import *
from conf.settings import *
from utils.utils import *

""" 这是一个模块级setup及teardown示例
	1- yield之前的部分为setup,之后的部分为teardown
  2- 拉起自动化后该文件下的所有用例执行前只进行一次系统登录和打开菜单的操作
	(因为都是同一个菜单下的特性用例)
  3- 该文件下的所有用例执行完毕后仅执行一次系统关闭操作	
"""
@pytest.fixture(scope="module")
def csam():
   # 该特性下所有用例公共使用的初始化配置都放在yield之前编写
    csam = CSAM()
    log.info("======setup======")
    csam.login()
    csam.menu("远程扫描配置")		
    yield csam		
		# 该特性下所有用例公共使用的初始化配置清理放在yield之后编写
    log.info("======teardown======")
    csam.driver.quit()
		
""" 这是一个方法级setup及teardown示例
	1- yield之前的部分为setup,之后的部分为teardown
  2- 每个用例执行前都会执行一次yield之前的代码
  3- 每个用例执行后都会执行一次yield之后的代码	
"""
@pytest.fixture
def setup(csam):
		# 每个用例独立的初始化配置放在yield之前编写
    # ts是一个时间戳字符串,目的是为每个用例提供一个唯一的标识符供用例代码使用
    # 主要是为了避免创建的配置信息出现重名等问题
    ts = timestamp()
    yield csam, ts
    # 每个用例独立的配置清理放在yield之后编写
		# 其中清理部分的代码也使用ts这个时间戳,主要目的为了避免删除了其他用例的配置而导致影响
    mysql.run("delete from csam.rs_task where task_name like '%{0}%';".format(ts))
    mysql.run("delete from csam.scan_targets where name like '%{0}%';".format(ts))
    pgsql.run("delete from schedules where name like '%{0}%';".format(ts))
    pgsql.conn.commit()

def test_探测任务_扫描任务管理_新增扫描任务_资产扫描(setup):
    csam, ts = setup
    csam.button_click(name="新增")
    csam.dropdown_menu_select("资产扫描")
    asset_scan = {
        "name": "asset_{0}".format(ts),
        "desc": "资产扫描",
    }
    csam.scan_task_basic(asset_scan)
    csam.button_click(name="下一步")
    # 新增扫描目标
    csam.link_click(name="新增")
    target = {
        "name": "target_{0}".format(ts),
        "ip": "192.168.30.148",
        "gport": "TOP 10",
        "desc": "auto test",
    }
    csam.scan_target_new(target)
    csam.button_click(name="保 存")
    csam.button_click(name="下一步")
    csam.button_click(name="保存")
    # 结果检查
    assert csam.text(csam.table_cell_filter(asset_scan["name"], "任务名称", "任务名称")) == asset_scan["name"]
    assert csam.text(csam.table_cell_filter(asset_scan["name"], "任务名称", "任务类型")) == "资产扫描"
    assert csam.text(csam.table_cell_filter(asset_scan["name"], "任务名称", "扫描状态")) in ["未执行", "执行中", "执行完成"]
    assert csam.text(csam.table_cell_filter(asset_scan["name"], "任务名称", "扫描目标")) == target["name"]

def test_探测任务_扫描任务管理_新增扫描任务_漏洞扫描(setup):
    csam, ts = setup
    csam.button_click(name="新增")
    csam.dropdown_menu_select("漏洞扫描")
    vul_scan = {
        "name": "vul_{0}".format(ts),
        "desc": "漏洞扫描",
    }
    csam.scan_task_basic(vul_scan)
    csam.button_click(name="下一步")
    # 新增扫描目标
    csam.link_click(name="新增")
    target = {
        "name": "target_{0}".format(ts),
        "ip": "192.168.30.148",
        "gport": "All IANA assigned TCP",
        "desc": "扫描目标",
    }
    csam.scan_target_new(target)
    csam.button_click(name="保 存")
    csam.button_click(name="下一步")
    # 新增漏扫计划
    csam.link_click(name="新增")
    csam.input_set("plan_{0}".format(ts), label="计划名称")
    csam.datetime_set(datetime_shift(days=10), label="第一次执行时间")
    csam.datetime_set(datetime_shift(days=12), label="第一次结束时间")
    csam.select_dropdown("一次", label="重复频率")
    csam.textarea_set("漏扫计划", label="说明")
    csam.button_click(name="保 存")
    csam.button_click(name="保存")
    # 结果检查
    assert csam.text(csam.table_cell_filter(vul_scan["name"], "任务名称", "任务名称")) == vul_scan["name"]
    assert csam.text(csam.table_cell_filter(vul_scan["name"], "任务名称", "任务类型")) == "漏洞扫描"
    assert csam.text(csam.table_cell_filter(vul_scan["name"], "任务名称", "扫描器")) == "脚本库扫描器"
    assert csam.text(csam.table_cell_filter(vul_scan["name"], "任务名称", "扫描状态")) in ["未执行", "执行中", "执行完成"]
    assert csam.text(csam.table_cell_filter(vul_scan["name"], "任务名称", "扫描目标")) == target["name"]

# 后面的用例代码省略了...

针对示例,用例在执行期间的执行流程示意见下图

图片

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

软件测试面试文档

我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。

在这里插入图片描述

  • 24
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: pytest是一种灵活而强大的自动化测试框架,用于编写和执行测试用例。在pytest的面试中,可能会涉及以下问题: 1. 你是如何使用pytest进行测试的? 使用pytest进行测试很简单。首先,安装pytest包。然后,创建一个以test_为前缀的Python文件,并在其中定义测试用例。在测试用例中,可以使用pytest提供的装饰器进行设置和标记。最后,通过运行pytest命令来执行测试。 2. pytest与其他测试框架相比有什么优势? 相比其他测试框架,pytest具有以下优势: - 简单易用:通过使用合适的装饰器和命名约定,pytest能够自动发现和执行测试用例,无需任何额外配置。 - 强大灵活:pytest提供了许多强大且灵活的功能,如参数化测试、测试数据生成、测试维护工具等。 - 易于扩展:使用pytest插件机制,可以通过安装插件来扩展框架的功能,满足各种测试需求。 - 丰富的报告和日志:pytest可以生成易于阅读的测试报告和详细的日志,方便定位和分析问题。 - 完善的社区支持:pytest拥有庞大的用户社区,可以获取到丰富的文档、教程和示例。 3. pytest如何处理测试用例的失败? 当某个测试用例失败时,pytest会自动记录失败的原因,并在测试执行完成后生成一个详细的测试报告。报告中会列出所有失败的测试用例及其失败的原因,方便开发人员进行排查和修复。 4. 如何使用pytest进行参数化测试? 在pytest中,可以使用@pytest.mark.parametrize装饰器来进行参数化测试。通过传入不同的参数组合,可以运行同一个测试用例的多个变种。这样可以有效减少编写重复的测试用例的工作量,提高测试的覆盖率。 5. pytest中的夹具(fixture)是什么?如何使用它? 夹具是pytest中的一个重要概念,它用于在测试用例之前进行一些准备工作,并在测试用例运行完成后进行清理。可以使用@pytest.fixture装饰器来创建夹具。夹具可以在测试函数参数中进行声明,从而在测试用例中直接使用。夹具还可以通过作用域的设置,控制其生命周期。 这些是关于pytest自动化测试的一些面试题及其答案,希望能对你有所帮助。当然,在实际面试中,还可能涉及到更多的问题和深入的理解。 ### 回答2: pytest是一个流行的Python测试框架,用于自动化测试。下面是关于pytest自动化测试的面试题的回答: 1. 为什么要使用pytest进行自动化测试pytest提供了一种简单而强大的方式来编写自动化测试脚本。它支持丰富的断言,易于阅读的测试报告和灵活的测试组织结构,并且与其他流行的测试工具和框架(如Selenium)集成良好。 2. pytest是如何工作的? pytest基于Python的unittest模块,通过使用装饰器和命名约定来定义测试用例。它可以自动发现和执行测试,生成详细的测试报告,并且具有丰富的插件生态系统,可以扩展其功能。 3. pytest与unittest有什么区别? 相比unittest,pytest提供了更简洁、更直观的语法和更丰富的功能。pytest不需要测试类的继承,而是使用函数来定义测试用例pytest还提供了更多的断言方法和更灵活的测试组织结构,使得编写和维护测试脚本更加容易。 4. pytest如何处理测试用例的数据驱动? pytest支持使用参数化装饰器来实现数据驱动测试。通过在测试用例函数上添加参数化装饰器,可以传递多组参数进行多次测试,从而减少代码重复和维护成本。 5. pytest如何处理夹具(fixtures)? 夹具在pytest中是一个Python函数,用于创建测试环境或者提供测试数据。pytest的夹具功能非常强大,可以通过使用装饰器将夹具与测试函数关联起来,并可以在不同的测试用例之间共享夹具。 6. pytest如何生成测试报告? pytest默认生成详细而易于阅读的测试报告。可以通过使用pytest-html插件生成漂亮的HTML测试报告,或者使用pytest-xdist插件在分布式环境中运行测试。 总之,pytest是一个功能丰富、易于使用和扩展的自动化测试框架。它提供了简洁的语法、强大的断言和灵活的测试组织结构,可以帮助开发人员编写高效而可靠的自动化测试脚本。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值