Pytest-Bdd-Playwright 系列教程(12):步骤参数 parsers参数解析

embedded/2024/11/24 4:28:54/

Pytest-Bdd-Playwright 系列教程(12):步骤参数 & parsers参数解析

  • 前言
  • 一、什么是步骤参数?
  • 二、pytest-bdd 的步骤参数用法
    • 2.1 简单字符串解析
    • 2.2 自定义正则表达式解析
    • 2.3 参数类型转换
  • 三、案例:基于 pytest-bdd 实现计算器功能测试
    • 3.1. 编写 Feature 文件
    • 3.2 实现步骤定义
    • 3.3 执行测试
  • 四、最佳实践
  • 总结

前言

  • pytest-bdd 中,步骤参数是构建动态、灵活测试用例的核心功能之一,它允许通过占位符的形式将具体值插入步骤中,从而避免重复编写相似的场景;
  • 我们可以通过给步骤添加参数来重用步骤,实现单一实现和多重使用,从而使代码更简洁;
  • 本文将探讨如何通过简单字符串解析、自定义正则表达式解析以及参数类型转换等方法来灵活处理不同的参数需求,并通过一个基于计算器功能的完整案例进行说明。

一、什么是步骤参数?

在 Gherkin 描述中,步骤参数(Step Parameters)是动态定义测试数据的关键方式。它允许通过占位符的形式将具体值插入步骤中,从而避免重复编写相似的场景。例如:

Given 第一个数字是 5
And 第二个数字是 3
When 我按下 加号
Then 结果应该是 8

上述例子中的 538 是参数,通过步骤实现函数将它们传递到测试逻辑中进行处理。

pytestbdd__21">二、pytest-bdd 的步骤参数用法

pytest-bdd 中,步骤参数通过字符串匹配实现,配合 pytest-bdd.parsers 模块,可以实现灵活的参数解析。以下是几种常见的参数解析方式:

2.1 简单字符串解析

简单字符串解析是 pytest-bdd 的默认行为,可以直接使用占位符解析参数。例如:

Given 第一个数字是 <first_number>

对应的 Python 实现:

from pytest_bdd import given, parsers@given(parsers.parse("第一个数字是 {first_number:d}"))
def first_number(first_number):return first_number

解析规则通过 {} 语法实现,其中 :d 表示参数是整数类型。

2.2 自定义正则表达式解析

如果需要更复杂的匹配逻辑,可以使用自定义正则表达式。例如:

Then 计算器的宽度应该是 12.5

实现代码:

@then(parsers.re(r"计算器的宽度应该是 (?P<expected_width>\d+\.\d+)"))
def check_width(expected_width):assert float(expected_width) == 12.5

通过 parsers.re() 定义正则表达式,可以精确控制参数提取规则。

2.3 参数类型转换

pytest-bdd 支持内置的数据类型转换,例如:

  • {name:s}:字符串
  • {value:d}:整数
  • {value:f}:浮点数

在示例中,我们可以指定参数类型:

Given 第一个数字是 5

对应 Python 代码:

@given(parsers.parse("第一个数字是 {first_number:d}"))
def first_number(first_number):return first_number

测试框架会自动将参数 first_number 转换为整数。


pytestbdd__89">三、案例:基于 pytest-bdd 实现计算器功能测试

接下来,我们通过一个具体的案例,展示如何在 pytest-bdd 中使用步骤参数。

3.1. 编写 Feature 文件

首先,新增 featuress/calculator_example.feature文件,定义计算器的几个功能测试场景:

Feature: 计算器一个简单的计算器,用于执行基本的算术操作。Background:Given 我已经准备好计算器Scenario: 检查计算器的尺寸Then 计算器的宽度应该是 12.5And 计算器的高度应该是 20.0And 计算器的厚度应该是 0.5Scenario: 打开计算器Given 我按下电源按钮Then 屏幕应该亮起Scenario Outline: 两个数之间的计算Given 我检查按钮是否正常And 第一个数字是 <first_number>And 第二个数字是 <second_number>When 我按下 <operation>Then 结果应该是 <expected_result>Examples:| first_number | second_number | operation | expected_result || 5            | 3            | 加号       | 8               || 10           | 4            | 减号       | 6               || 2            | 6            | 乘号       | 12              || 8            | 2            | 除号       | 4               |

3.2 实现步骤定义

tests/test_calculator_example.py 文件中为上述场景定义步骤参数:

import pytest
from pytest_bdd import given, when, then, parsers, scenariosscenarios("calculator_example.feature")@given("我已经准备好计算器")
def _():print("计算器已准备好!")@given("我检查按钮是否正常")
def _():print("按钮已检查。")@given("我按下电源按钮")
def _():pass@then("屏幕应该亮起")
def _():pass@then(parsers.parse("计算器的宽度应该是 {expected_width:f}"))
def _(expected_width: float):print(f"宽度: {expected_width}")@then(parsers.parse("计算器的高度应该是 {expected_height:f}"))
def _(expected_height: float):print(f"高度: {expected_height}")@then(parsers.parse("计算器的厚度应该是 {expected_thickness:f}"))
def _(expected_thickness: float):print(f"厚度: {expected_thickness}")@given(parsers.parse("第一个数字是 {first_number:d}"), target_fixture="first_number")
def _(first_number):return first_number@given(parsers.parse("第二个数字是 {second_number:d}"), target_fixture="second_number")
def _(second_number):return second_number@when(parsers.parse("我按下 {operation}"), target_fixture="result")
def _(operation, first_number, second_number):if operation == "加号":return first_number + second_numberelif operation == "减号":return first_number - second_numberelif operation == "乘号":return first_number * second_numberelif operation == "除号":return first_number / second_numberelse:raise ValueError(f"不支持的操作: {operation}")@then(parsers.parse("结果应该是 {expected_result:d}"))
def _(result, expected_result):assert result == expected_result

3.3 执行测试

运行测试命令:

pytest .\tests\test_calculator_example.py

输出结果如下:

在这里插入图片描述

四、最佳实践

  1. 合理使用参数类型
    根据步骤的需求选择合适的数据类型解析器,确保输入值正确解析。

  2. 避免重复代码
    如果多个步骤有相似逻辑,可以使用 Python 的函数装饰器和参数化技术优化代码。

  3. 使用目标 Fixture
    利用 target_fixture 提取步骤中的数据,简化上下文传递。

  4. 测试数据分离
    将测试数据与逻辑分离(如使用 Examples 表),提高测试用例的可读性和可维护性。

总结

pytest-bdd 的步骤参数功能通过灵活的解析机制,为测试用例的动态数据支持提供了强大的工具。在本文中,我们通过对步骤参数的详尽讲解和计算器案例的实操演示,展示了如何高效使用这一功能。


http://www.ppmy.cn/embedded/140031.html

相关文章

从0开始的数据结构速过——番外(1)

目录 尝试 思考与架构设置 编写&#xff01; 一些额外知识的补充 可变参数模板 std::common_type std::forward 这是《数据结构从0开始》的一个番外。实际上是介绍一下一些现代C的写法。这里以快速构建std::array作为契机来说明一下一些现代C的语法。 尝试 我们在这里呢…

探索Python PDF处理的奥秘:pdfrw库揭秘

文章目录 探索Python PDF处理的奥秘&#xff1a;pdfrw库揭秘1. 背景&#xff1a;为何选择pdfrw&#xff1f;2. pdfrw是什么&#xff1f;3. 如何安装pdfrw&#xff1f;4. 五个简单的库函数使用方法4.1 读取PDF信息4.2 修改PDF元数据4.3 旋转PDF页面4.4 提取PDF中的图片4.5 合并P…

CUDA补充笔记

文章目录 一、不同核函数前缀二、指定kernel要执行的线程数量三、线程需要两个内置坐标变量来唯一标识线程四、不是blocksize越大越好&#xff0c;上限一般是1024个blocksize 一、不同核函数前缀 二、指定kernel要执行的线程数量 总共需要线程数是&#xff1a; 1 * N N个线程…

给机器装上“脑子”—— 一文带你玩转机器学习

目录 一、引言&#xff1a;AI浪潮中的明星——机器学习 二、机器学习的定义与概念 1. 机器学习与传统编程的区别 2. 机器学习的主要任务类型 3. 机器学习的重要组成部分 三、机器学习的工作原理&#xff1a;从数据到模型的魔法之旅 1. 数据收集与预处理——数据是机器的…

解决Windows + Chrome 使用Blob下载大文件时,部分情况下报错net:ERR_FAILED 200 (OK)的问题

背景&#xff1a; 部分线上用户反馈&#xff0c;下载文件会报错&#xff0c;但重启电脑又好了。测试无法复现。遂远程客户&#xff0c;发现在下载超过一定阈值大小的文件时&#xff0c;会报错。 但直接点击下载链接&#xff0c;可以正常下载 查阅代码&#xff0c;以前的写法是…

数据结构(链栈——c语言实现)

链式栈&#xff08;Linked Stack&#xff09;是一种基于链表数据结构实现的栈。它利用链表节点的指针来存储元素&#xff0c;并通过指针的链接关系来维护栈的后进先出&#xff08;LIFO, Last In First Out&#xff09;特性。 链式栈的优点 动态大小&#xff1a; 链式栈…

Oracle数据库安全扫描1158/3938端口出现弱SSL加密算法解决方法之一

问题复述 某国企项目现场反应安全扫描出部署某历史项目的Windows服务器上的1158及3938两个端口出现了弱SSL加密算法漏洞&#xff0c;要求整改。 经过核实&#xff0c;该Windows服务器上部署了tomcat与Oracle 11g数据库&#xff0c;其中1158和3938两个端口均为Oracle数据库所使…

使用Python和OpenCV连接并处理IP摄像头视频流

使用Python和OpenCV连接并处理IP摄像头视频流 随着智能设备的发展&#xff0c;越来越多的家庭和企业开始使用IP摄像头进行安全监控或远程查看。这些摄像头通常可以通过网络访问&#xff0c;提供了丰富的功能&#xff0c;如实时视频流、云台控制等。本文将详细介绍如何利用Pyth…