Pytest系列(4) - fixture的詳細使用


如果你還想從頭學起Pytest,可以看看這個系列的文章哦!

https://www.cnblogs.com/poloyy/category/1690628.html

 

前言

  • 前面一篇講了setup、teardown可以實現在執行用例前或結束后加入一些操作,但這種都是針對整個腳本全局生效的
  • 如果有以下場景:用例 1 需要先登錄,用例 2 不需要登錄,用例 3 需要先登錄。很顯然無法用 setup 和 teardown 來實現了
  • fixture可以讓我們自定義測試用例的前置條件

 

fixture的優勢

  • 命名方式靈活,不局限於 setup 和teardown 這幾個命名
  • conftest.py 配置里可以實現數據共享,不需要 import 就能自動找到fixture
  • scope="module" 可以實現多個.py 跨文件共享前置
  • scope="session" 以實現多個.py 跨文件使用一個 session 來完成多個用例
 

fixture參數列表

@pytest.fixture(scope="function", params=None, autouse=False, ids=None, name=None)
def test():
    print("fixture初始化的參數列表")

參數列表

  • scope:可以理解成fixture的作用域,默認:function,還有class、module、package、session四個【常用】
  • autouse:默認:False,需要用例手動調用該fixture;如果是True,所有作用域內的測試用例都會自動調用該fixture
  • name:默認:裝飾器的名稱,同一模塊的fixture相互調用建議寫個不同的name

注意

session的作用域:是整個測試會話,即開始執行pytest到結束測試

 

測試用例如何調用fixture

  1. 將fixture名稱作為測試用例函數的輸入參數
  2. 測試用例加上裝飾器:@pytest.mark.usefixtures(fixture_name)
  3. fixture設置autouse=True
#!/usr/bin/env python
# -*- coding: utf-8 -*-

"""
__title__  =
__Time__   = 2020-04-06 15:50
__Author__ = 小菠蘿測試筆記
__Blog__   = https://www.cnblogs.com/poloyy/
"""
import pytest

# 調用方式一
@pytest.fixture
def login():
    print("輸入賬號,密碼先登錄")


def test_s1(login):
    print("用例 1:登錄之后其它動作 111")


def test_s2():  # 不傳 login
    print("用例 2:不需要登錄,操作 222")


# 調用方式二
@pytest.fixture
def login2():
    print("please輸入賬號,密碼先登錄")


@pytest.mark.usefixtures("login2", "login")
def test_s11():
    print("用例 11:登錄之后其它動作 111")


# 調用方式三
@pytest.fixture(autouse=True)
def login3():
    print("====auto===")


# 不是test開頭,加了裝飾器也不會執行fixture
@pytest.mark.usefixtures("login2")
def loginss():
    print(123)

執行結果

知識點

  • 在類聲明上面加 @pytest.mark.usefixtures() ,代表這個類里面所有測試用例都會調用該fixture
  • 可以疊加多個 @pytest.mark.usefixtures() ,先執行的放底層,后執行的放上層
  • 可以傳多個fixture參數,先執行的放前面,后執行的放后面
  • 如果fixture有返回值,用 @pytest.mark.usefixtures() 是無法獲取到返回值的,必須用傳參的方式(方式一)

 

fixture的實例化順序

  • 較高 scope 范圍的fixture(session在較低 scope 范圍的fixture( function 、 class 之前實例化【session > package > module > class > function】
  • 具有相同作用域的fixture遵循測試函數中聲明的順序,並遵循fixture之間的依賴關系【在fixture_A里面依賴的fixture_B優先實例化,然后到fixture_A實例化】
  • 自動使用(autouse=True)的fixture將在顯式使用(傳參或裝飾器)的fixture之前實例化
#!/usr/bin/env python
# -*- coding: utf-8 -*-

"""
__title__  =
__Time__   = 2020-04-06 16:14
__Author__ = 小菠蘿測試筆記
__Blog__   = https://www.cnblogs.com/poloyy/
"""
import pytest

order = []

@pytest.fixture(scope="session")
def s1():
    order.append("s1")


@pytest.fixture(scope="module")
def m1():
    order.append("m1")


@pytest.fixture
def f1(f3, a1):
    # 先實例化f3, 再實例化a1, 最后實例化f1
    order.append("f1")
    assert f3 == 123


@pytest.fixture
def f3():
    order.append("f3")
    a = 123
    yield a


@pytest.fixture
def a1():
    order.append("a1")


@pytest.fixture
def f2():
    order.append("f2")


def test_order(f1, m1, f2, s1):
    # m1、s1在f1后,但因為scope范圍大,所以會優先實例化
    assert order == ["s1", "m1", "f3", "a1", "f1", "f2"]

執行結果

斷言成功

 

關於fixture的注意點

添加了 @pytest.fixture ,如果fixture還想依賴其他fixture,需要用函數傳參的方式,不能用 @pytest.mark.usefixtures() 的方式,否則會不生效

@pytest.fixture(scope="session")
def open():
    print("===打開瀏覽器===")

@pytest.fixture
# @pytest.mark.usefixtures("open") 不可取!!!不生效!!!
def login(open):
    # 方法級別前置操作setup
    print(f"輸入賬號,密碼先登錄{open}")

 

前面講的,其實都是setup的操作,那么現在就來講下teardown是怎么實現的

 

fixture之yield實現teardown

用fixture實現teardown並不是一個獨立的函數,而是用 yield 關鍵字來開啟teardown操作

#!/usr/bin/env python
# -*- coding: utf-8 -*-

"""
__title__  =
__Time__   = 2020-04-06 15:50
__Author__ = 小菠蘿測試筆記
__Blog__   = https://www.cnblogs.com/poloyy/
"""

import pytest


@pytest.fixture(scope="session")
def open():
    # 會話前置操作setup
    print("===打開瀏覽器===")
    test = "測試變量是否返回"
    yield test
    # 會話后置操作teardown
    print("==關閉瀏覽器==")


@pytest.fixture
def login(open):
    # 方法級別前置操作setup
    print(f"輸入賬號,密碼先登錄{open}")
    name = "==我是賬號=="
    pwd = "==我是密碼=="
    age = "==我是年齡=="
    # 返回變量
    yield name, pwd, age
    # 方法級別后置操作teardown
    print("登錄成功")


def test_s1(login):
    print("==用例1==")
    # 返回的是一個元組
    print(login)
    # 分別賦值給不同變量
    name, pwd, age = login
    print(name, pwd, age)
    assert "賬號" in name
    assert "密碼" in pwd
    assert "年齡" in age


def test_s2(login):
    print("==用例2==")
    print(login)

yield注意事項

  • 如果yield前面的代碼,即setup部分已經拋出異常了,則不會執行yield后面的teardown內容
  • 如果測試用例拋出異常,yield后面的teardown內容還是會正常執行

 

yield+with的結合

# 官方例子
@pytest.fixture(scope="module")
def smtp_connection():
    with smtplib.SMTP("smtp.gmail.com", 587, timeout=5) as smtp_connection:
        yield smtp_connection  # provide the fixture value

該 smtp_connection 連接將測試完成執行后已經關閉,因為 smtp_connection 對象自動關閉時, with 語句結束。

 

addfinalizer 終結函數

@pytest.fixture(scope="module")
def test_addfinalizer(request):
    # 前置操作setup
    print("==再次打開瀏覽器==")
    test = "test_addfinalizer"

    def fin():
        # 后置操作teardown
        print("==再次關閉瀏覽器==")

    request.addfinalizer(fin)
    # 返回前置操作的變量
    return test


def test_anthor(test_addfinalizer):
    print("==最新用例==", test_addfinalizer)

注意事項

  • 如果 request.addfinalizer() 前面的代碼,即setup部分已經拋出異常了,則不會執行 request.addfinalizer() 的teardown內容(和yield相似,應該是最近新版本改成一致了)
  • 可以聲明多個終結函數並調用

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM