个人技术分享

        在 pytest 测试框架中,setup 和 teardown是用于在每个测试函数之前和之后执行设置和清理的动作,而pytest 实际上并没有内置的 setup 和 teardown 函数,而是使用了一些装饰器或钩子函数来实现类似的功能。

学习目录

钩子函数(Hook Functions)

Fixture方式

    1 在测试函数执行前执行setup和teardown的动作 4

    2 在测试类执行前执行setup和teardown的动作 5

    3 同时实现测试类和测试函数的setup和teardown 6

     

钩子函数(Hook Functions)

以下钩子函数运行的优先级是模块级 > 类级 > 方法级

函数名

说明

setup_module

在python文件执行中的第1条用例执行前,先执行setup_module函数

teardown_module

在python文件执行中的最后1条用例执行结束后(不管成功还是失败),执行teardown_module函数

setup_class

在测试类中定义该函数,在测试类的第1条用例执行前,先执行setup_class函数

teardown_class

在测试类中定义该函数,在测试类的最后1条用例执行结束后(不管成功还是失败),执行teardown_class函数

setup_function

在python文件执行中的每一条用例执行前执行setup_function函数(适用于类外面和类里面的函数)

teardown_function

在python文件执行中的每一条用例执行结束后(不管成功还是失败),执行teardown_function函数(适用于类外面和类里面的函数)

举例:定义test_case1.py文件,设置函数用例test_1,测试类TestCase中两条用例test_2和test3。

定义setup_module和teardown_module、setup_class和teardown_class、setup_function和teardown_function函数。

import pytest
def setup_module():
   print("\nSetting up module... 模块开始前执行")
def teardown_module():
   print("\nTearing down module...模块结束后执行")

def setup_function(function):
   print(f"\nSetting up for {function.__name__}... 函数开始前执行")
def teardown_function(function):
   print(f"\nTearing down for {function.__name__}...函数结束后执行")
#函数1
def test_1():
   assert 1!=1

class TestCase():
   def setup_class():
      print(f"\nSetting up for class... 类开始前执行")
   def teardown_class():
      print(f"\nTearing down for class...类结束后执行")
   
   @staticmethod
   def test_2():        #函数2
     assert 1 == 1
   @staticmethod
   def test_3():         #函数3
      assert 1 == 1

运行该测试文件,执行结果如下:

按照测试用例的顺序,第一个函数test_1在类外面,执行函数的setup和teardown,而到了test_2和test_3函数因为在类里面,所以按照优先级先执行类的setup和teardown,再执行函数级setup和teardown

Fixture方式

fixture是pytest中最强大和推荐的方式,用于设置和清理测试环境。我们可以定义一个fixture,并在测试函数中使用它,该fixture将在测试函数之前执行,并在测试函数之后进行清理

  • 使用@pytest.fixture装饰器定义fixture。
  • 在测试函数中,通过函数参数的方式使用fixture。

1 在测试函数执行前执行setup和teardown的动作

设置一个函数,使用@pytest.fixture进行装饰

import pytest
@pytest.fixture
def my_fixture():
   print("\nSetting up fixture... 函数执行前执行该步骤")
   # 这里可以放置设置代码
   yield
   print("\nTearing down fixture... 函数执行后执行该步骤")
   # 这里可以放置清理代码
 
def test_1(my_fixture):
    print("\nRunning testcase1...")

     yield 关键字的使用,它允许在 fixture 函数中定义两部分:设置代码(在 yield 之前)和清理代码(在 yield 之后)。将my_fixture函数作为参数传入测试函数test_1中。

用例执行后结果如下 同样实现了函数级的setup和teardown的功能

2 在测试类执行前执行setup和teardown的动作

定义一个fixture函数class_setup_teardown,在该函数使用 @pytest.fixture(scope='class', autouse=True) 装饰器(应用于类时scope参数定义为class,而上面的例子中应用于函数时可保持默认值function),并将 这个fixture 作为类中测试方法的参数,以此来实现类中的setup和teardown功能。

@pytest.fixture(scope='class', autouse=True)
def class_setup_teardown():
   print("\nClass setup (equal to setup_class)")
   # 设置代码
   yield
   print("\nClass teardown (equal to teardown_class)")
   # 清理代码

第一种方式:在测试函数中传入定义的fixture函数作为参数

class Testcase:
   def test_one(self, class_setup_teardown): # 直接将fixture作为参数
       print("\nRunning test_one")
       # 测试代码
   def test_two(self, class_setup_teardown): # 同样地,直接作为参数
       print("\nRunning test_two")
       # 测试代码
  • 第二种方式:在测试类中使用装饰器注明使用fixture,测试函数中不传参数
@pytest.mark.usefixtures("class_setup_teardown")
class Testcase:
   def test_one(self): 
      print("\nRunning test_one")
     # 测试代码
   def test_two(self):
      print("\nRunning test_two")
      # 测试代码

用例执行后结果如下:两种方式都实现了测试类setup和teardown的功能

3 同时实现测试类和测试函数的setup和teardown

将以上两种方式结合在一起就可以了

@pytest.fixture
def my_fixture():
   print("\nSetting up fixture... 函数执行前执行该步骤")
   # 这里可以放置设置代码
   yield
   print("\nTearing down fixture... 函数执行后执行该步骤")
   # 这里可以放置清理代码

@pytest.fixture(scope='class', autouse=True)
def class_setup_teardown():
   print("\nClass setup (equal to setup_class)")
   # 设置代码
   yield
   print("\nClass teardown (equal to teardown_class)")
   # 清理代码
  • 第1种方式: 在测试类中注明使用类fixture,在测试函数传入函数级fixture
@pytest.mark.usefixtures("class_setup_teardown")
class Testcase:
    def test_one(self,my_fixture): # 直接将fixture作为参数
        print("\nRunning test_one")
        # 测试逻辑
    def test_two(self,my_fixture): # 同样地,直接作为参数
        print("\nRunning test_two")
        # 测试逻辑
  • 第二种方式:在测试函数同时传入类级fixture和函数级fixture
class Testcase:
    def test_one(self,class_setup_teardown,my_fixture): # 直接将fixture作为参数
        print("\nRunning test_one")
        # 测试逻辑
    def test_two(self,class_setup_teardown,my_fixture): # 同样地,直接作为参数
        print("\nRunning test_two")
       # 测试逻辑

用例执行后结果如下:两种方式都实现了测试类setup和teardown的功能

当然Pytest的fixture功能远不止这些,后续我们再做其他功能的介绍。

共勉: 东汉·班固《汉书·枚乘传》:“泰山之管穿石,单极之绠断干。水非石之钻,索非木之锯,渐靡使之然也。”

-----指水滴不断地滴,可以滴穿石头;

-----比喻坚持不懈,集细微的力量也能成就难能的功劳。

----感谢读者的阅读和学习,谢谢大家