# 技术面试没过,居然是没用这个测试框架

1、引言

我有一个朋友是做 Python 自动化测试的。前几天他告诉我去参加一个大厂面试被刷了。

我问他是有没有总结被刷下来的原因。他说面试官问了一些 pytest 单元测试框架相关的知识,包括什么插件系统和用力筛选。但是他所在的公司用的技术是基于 unittest 的,没有用过 pytest。

我跟他说你可以和技术面试官说明,在实际过程当中你没有使用过 pytest,但是你可以后面再学。这哥们说:我就是这样跟面试官说的,但是面试官告诉我 pytest 现在已经是行业里面的主流,还在坚持用 unittest 说明我的技术已经过时了,没有跟上现在测试领域的发展。

实际上他在面试之前已经了解过 pytest 的一些基础用法,但是网上的一些资料,都是停留在用法和一些知识点的讲解,没有深入到 pytest 内部运行和一些高级特性。所以被问到的时候,自己临时抱佛脚的一些知识都没有用上。

后面我给这位朋友补习了一些关于 Python 的高级特性。现在我连同基础部分的内容一起贴出来,希望对 Python 自动化测试的一些朋友有所帮助。

2、为什么用单元测试框架?

首先我要说明一下什么是单元测试框架?

unittest 和 pytest 都是单元测试框架。单元测试指的是在编程过程当中形成的对函数或者是类下面的方法进行测试的一个过程。

在不使用任何框架的前提下,其实也是可以进行单元测试的。比如我们可以通过 if 判断
、异常处理或者是其他的流程控制来表示测试是否通过。

 def add(a, b):
     return a + b
     
 def test_add():
     ret = add(3, 4)
     if ret == 7:
         print("add 函数的测试通过")
     else:
         print("add 函数的测试失败")

如果要运行这个用例,需要手工调用 test_add 这个函数:

 test_add()

接下来,使用 python 运行这个文件,就能得到测试结果:

 python test_add.py

虽然说上面我们通过 if 判断,对一个函数进行了测试,而且得到了测试结果,但是流程是比较复杂的:

这还只是在我们只测试了一个函数的情况下,当需要测试的函数或者类越来越多的时候,这个过程会越来越复杂。

而使用单元测试框架,可以极大的简化我们对单元测试的过程,使用单元测试框架以后,框架会帮我们自动去收集用例、运行用例、生成报告。

framework.png

3、pytest 的基础使用

上面的测试代码使用 pytest 编写,可以这样写。

 def add(a, b):
     return a + b
     
 def test_add():
     assert 7 == add(3,4)

写完测试用例以后,我们只需要在目录下输入 pytest 指令,就可以自动运行用例,而且呢结果会直接显示在命令行的下方。

image20201219192616608.png

上面讲的是单元测试过程,也就是说对某个函数或者是类下面的方法进行测试,有的人可能会不理解。在实际工作过程当中很少进行单元测试啊,测试人员做的更多的是接口测试,UI 测试,pytest 怎么用呢?

实际上不管是接口测试还是 UI 测试,都是可以使用 pytest。当你进行接口测试的时候,你只需要把访问接口的过程封装成一个 Python 函数。

 def visit_api():
    print("访问接口,得到结果...")
    return response
     
 def test_api():
    assert expected_response == visit_api()

当你进行 Web 测试的时候,你只需要操作浏览器的过程封装成一个函数?

 def browser_method():
    print("点点点")
    return ui_response
     
 def test_web():
    assert expected_response == browser_method()

在这种情况下。接口访问和 Web 操作都是以函数形式存在的,我们直接去测试这个 Python 函数,也是一个单元测试的过程。

因为 pytest 是一个第三方的框架,所以我们先要安装。安装方式非常简单,只需要通过 pip 这个包管理工具安装就可以了。

 pip install -U pytest

安装完成以后,我们可以向使用上面的那个例子一样:

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

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

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

image20201224195304229.png

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

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

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

fixture.jpg

现在我们来举一个夹具的例子,我们需要测试一个注册的函数。这个注册函数提供两个参数,第 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 的关系,也就是说,当你想测试某个功能场景的时候,你必须要去编写一个对应的测试函数。当测试的场景越来越多,测试数据越来越复杂的情况下,需要编写更多的测心率函数,而这些函数的逻辑基本上是重复的。

image20201221154754539.png

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

 import pytest
 
 cases = [
    (1, 2, 3),
    ("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 个元素代表预期结果。

image20201221160111259.png

@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 的文件,打开就可以查看测试报告了。

image20201221181320782.png

pytest 之所以成为主流,有很多的原因,其中最重要的一个原因是因为其强大的插件系统。任何一个程序员,只要遵循一些简单的规范,就可以轻易的编写插件。后面我们再跟大家深入去研究 pytest 当中的夹具系统,插件系统和钩子函数这些特性。

回帖
请输入回帖内容 ...