技术面试没过,居然是没有用pytest测试框架 (2)

第2步:使用assert断言,assert 后面可以跟任意的 Python 条件表达式。

assert 4 < 5
assert "yuze" in "yuze wang"
assert isinstance(6, int)

测试用例函数有 2 个注意事项:

函数名称以test_开头;

保存测试用例的文件以test_*.py 的形式或者 *_test.py 的形式。

例行用了以后呢,在命令行当中会显示4个部分的内容:

第1个部分,测试用例和通过的结果,

第2个部分,失败用例回溯信息,

第3个部分,输出捕获信息,

第4个部分,总结信息。

在拍test当中通过的测试用例,不会有特别详细的结果,但是这是失败的测试用例默认会有非常详细的结果,而且会帮你捕获错误原因。

4、测试夹具(Fixture)是什么?

在测试过程当中,有时你需要提前给你的测试用例去准备一个运行环境。这个测试环境通常来说被称为测试夹具(Fixture),又被称为固定装置、测试固件等。

技术面试没过,居然是没有用pytest测试框架

当你要测试一个电器的时候,你需要提供不同的输入电压电流的环境,

当你测试一台电脑网络的时候,必须要提供网络环境,

当你要测试一个手机游戏能否被安装时,你需要提供一台手机环境,

当你要测试一个软件能否登录的时候,你需要准备用户名和密码这样的用户环境,

当你要测试一个数据库能否操作的时候,需要提供数据库的连接环境。

现在我们来举一个夹具的例子,我们需要测试一个注册的函数。这个注册函数提供两个参数,第1个参数是手机号,第2个参数是密码。注册函数的逻辑就是对手机号码和密码进行校验,如果通过校验表示注册成功,如果不通过表示注册失败。

def is_mobile(number: str):
    if re.search(r"^1[3-9][0-9]{9}$", number):
        return True
    return False

def register(mobile, password):
    if is_mobile(mobile) and len(password) >= 6:
        return "success"
    return "fail"

def test_register():
    assert "success" == register("13177778888""123456")

这个测试用例并没有什么问题,但是它存在优化的空间。一个优化的空间是每个手机号码都是我们手工生成的,当需要编写多组数据测试时,会有一点费时间。现在我们可以编写一个函数,自动生成一个手机号码,当我有多组数据需要测试的时候,我只需要重复调用生成手机号码的函数,就可以获取测试数据。这个生成手机号码的函数呢,就是一个夹具。

def gen_a_mobile():
    """随机生成 13 开头的手机号码。"""
    random_num = "".join([str(random.randint(1,9)) for i in range(9)])
    return "".join(["13", random_num])

pytest 提供了一种叫做依赖注入的机制,当一个函数被声明为夹具的时候,可以在测试函数中传入这个夹具的名称,pytest会自动调用它。

import random
import pytest

@pytest.fixture
def gen_a_mobile():
    """随机生成 13 开头的手机号码。"""
    random_num = "".join([str(random.randint(1,9)) for i in range(9)])
    return "".join(["13", random_num])

def test_register(gen_a_mobile):
    assert "success" == register(gen_a_mobile, "123456")

pytest 当中的夹具系统非常非常的灵活,后面如果有时间的我专门写文章跟大家讲解夹具系统。

5、数据驱动和参数化

现在我们编写的函数和测试用例是1对1的关系,也就是说,当你想测试某个功能场景的时候,你必须要去编写一个对应的测试函数。当测试的场景越来越多,测试数据越来越复杂的情况下,需要编写更多的测心率函数,而这些函数的逻辑基本上是重复的。

技术面试没过,居然是没有用pytest测试框架

在 pytest 当中可以使用参数化这种测试手段,简化编写用例函数的过程。我们并不需要为每一组测试数据单独去编写一个测试函数,而是采取多种数据共用一个函数的方式。如果测试操作几乎一致,可以重复使用这一个函数进行测试。

import pytest

cases = [
    (123),
    ("hello""world""hello world"),
    (1"world""1world")
]


@pytest.mark.parametrize("a,b,expected", cases)
def test_add(a, b, expected):
    assert expected == add(a, b)

在这个例子当中,cases这个变量存储了三组测试用例的数据,每一组测试数据用一个元组表示,元组的第1个元素代表a,第2个元素代表B,第3个元素代表预期结果。

技术面试没过,居然是没有用pytest测试框架

@pytest.mark.parametrize("a,b,expected", cases) 这一行代码的意思是说,每一次从cases变量当中取出一组测试数据。分别用a、b 、expected 三个变量接收,然后我们把这三个变量名作为函数的参数传递到 test_add 当中,就实现了参数化的过程。

当使用这一种参数化的手段进行测试的时候,测试数据和测试函数是多对一的关系,对于多组测试数据,我们只需要去编写一个测试函数,极大的提升了代码编写效率。

6、测试报告和插件

最后我们来说一下测试报告。pytest 当中的测试报告,通常是以插件的形式生成的,如果你想生成一个html格式的测试报告,可以先安装 pytest-html 这个插件。

pip install pytest-html

接下来你需要在运行用例的时候,在 pytest 命令后面加上 --html=<测试报告名称>.html

pytest --html=report.html

当运行完用例以后,你可以在当前目录下找到一个 report.html 的文件,打开就可以查看测试报告了。

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/wsppgj.html