Pytest框架之fixture詳解(一)
相關(guān)文章
我們?cè)诰帉憸y(cè)試用例,都會(huì)涉及到用例執(zhí)行之前的環(huán)境準(zhǔn)備工作,和用例執(zhí)行之后的環(huán)境清理工作。
代碼版的測(cè)試用例也不例外。在自動(dòng)化測(cè)試框架當(dāng)中,我們也需要編寫:
用例執(zhí)行之前的環(huán)境準(zhǔn)備工作代碼(前置工作代碼)
用例執(zhí)行之后的環(huán)境清理工作(后置工作代碼)
通常,在自動(dòng)化測(cè)試框架當(dāng)中,都叫做fixture。
pytest作為python語(yǔ)言的測(cè)試框架,它的fixture有2種實(shí)現(xiàn)方式。
- 一種是xunit-style,跟unittest框架的機(jī)制非常相似,即setup/teardown系列
- 一種是它自己的fixture機(jī)制,以@pytest.fixture裝飾器來(lái)申明。
pytest的fixture實(shí)現(xiàn)方式一:xunit-style
pytest的xunit-style有三個(gè)級(jí)別的fixture:測(cè)試模塊、測(cè)試類、測(cè)試函數(shù)。
1、測(cè)試函數(shù)/方法級(jí)別:每一個(gè)測(cè)試函數(shù)都會(huì)執(zhí)行的前置和后置。
測(cè)試類內(nèi)部的測(cè)試方法:
- 前置函數(shù)名稱:setup_method
- 后置函數(shù)名稱:teardown_method
模塊下的測(cè)試函數(shù):
- 前置函數(shù)名稱:setup_function
- 后置函數(shù)名稱:teardown_function
2、測(cè)試類級(jí)別:一個(gè)測(cè)試類只執(zhí)行一次前置和后置。
- 前置函數(shù)名稱:setup_class
- 后置函數(shù)名稱:teardown_class
注意:用@classmethod裝飾
3、測(cè)試模塊級(jí)別:一個(gè)測(cè)試模塊只執(zhí)行一次前置和后置。
- 前置函數(shù)名稱:setup_module
- 后置函數(shù)名稱:teardown_module
from selenium import webdriver from time import sleep from random import randint ? ? def setup_module(): print("==== 模塊級(jí)的 setup 操作 ====") ? ? def teardown_module(): print("==== 模塊級(jí)的 teardown 操作 ====") ? ? def test_random(): assert randint(1, 5) == 3 ? ? class TestWeb: ? @classmethod def setup_class(cls): print("==== 測(cè)試類級(jí)的 setup 操作 ====") ? @classmethod def teardown_class(cls): print("==== 測(cè)試類級(jí)的 teardown 操作 ====") ? def setup_method(self): print("==== 測(cè)試用例級(jí)的 setup 操作 ====") self.driver = webdriver.Chrome() ? def teardown_method(self): print("==== 測(cè)試用例級(jí)的 teardown 操作 ====") self.driver.quit() ? def test_search(self): self.driver.get("https://www.baidu.com/") self.driver.find_element_by_id("kw").send_keys("檸檬班") self.driver.find_element_by_id("su").click() sleep(1)
pytest也支持運(yùn)行unittest的測(cè)試用例。支持unittest以下特性:
- @unittest.skip
- setUp/tearDown;
- setUpClass/tearDownClass;
- setUpModule/tearDownModule;
pytest的fixture實(shí)現(xiàn)方式二:fixture機(jī)制
通過(guò)@pytest.fixture裝飾器來(lái)定義fixture。一個(gè)函數(shù)被@pytest.fixture裝飾,那么這個(gè)函數(shù)就是fixture。
使用fixture時(shí),分為二個(gè)部分:fixture定義、fixture調(diào)用。
除此之外,還有fixture的共享機(jī)制,嵌套調(diào)用機(jī)制。
1、定義fixture。
1)fixture通過(guò)函數(shù)實(shí)現(xiàn)。
2)使用@pytest.fixture進(jìn)行裝飾
import pytest ? @pytest.fixture def init(): pass
3)前置準(zhǔn)備工作代碼和后置清理工作代碼,都寫在一個(gè)函數(shù)里面。
4)通過(guò)yeild關(guān)鍵字,區(qū)分前置代碼和后置代碼 。yeild之前的代碼為前置代碼,yeild之后的代碼為后置代碼
在實(shí)際應(yīng)用場(chǎng)景當(dāng)中,可以只有前置準(zhǔn)備工作代碼,也可以只有后置清理工作代碼。
import pytest ? ? @pytest.fixture def init(): print("用例執(zhí)行之前,執(zhí)行的代碼") # 前置代碼 yield print("用例執(zhí)行之后,執(zhí)行的代碼") # 后置代碼 ? @pytest.fixture def init2(): print("用例執(zhí)行之前,執(zhí)行的代碼") # 只有用例執(zhí)行之前的前置準(zhǔn)備代碼 ? @pytest.fixture def init3(): yield print("用例執(zhí)行之后,執(zhí)行的代碼") # 只有用例執(zhí)行之后的后置清理代碼
5)fixture有4個(gè)作用域:測(cè)試會(huì)話(session)、測(cè)試模塊(module)、測(cè)試類(class)、測(cè)試用例(function)
測(cè)試會(huì)話:pytest執(zhí)行測(cè)試用例的整個(gè)過(guò)程,稱為會(huì)話。
比如pytest收集到了100條用例并執(zhí)行完成,這個(gè)過(guò)程稱為測(cè)試會(huì)話。
設(shè)置fixture的作用域:通過(guò)@pytest.fixture(scope=作用域)來(lái)設(shè)置。默認(rèn)情況下,scope=function
import pytest ? ? # 沒(méi)有設(shè)置scope,默認(rèn)為測(cè)試函數(shù)級(jí)別。即調(diào)用此fixture的測(cè)試類/模塊/函數(shù)下,每個(gè)測(cè)試函數(shù)都會(huì)執(zhí)行一次這個(gè)fixture @pytest.fixture def init(): print("用例執(zhí)行之前,執(zhí)行的代碼") # 前置代碼 yield print("用例執(zhí)行之后,執(zhí)行的代碼") # 后置代碼 ? ? # 設(shè)置scope為class。調(diào)用此fixture的測(cè)試類下,只執(zhí)行一次這個(gè)fixture. @pytest.fixture(scope="class") def init2(): print("用例執(zhí)行之前,執(zhí)行的代碼") # 只有用例執(zhí)行之前的前置準(zhǔn)備代碼 ? ? # 設(shè)置scope為session。autouse表示自動(dòng)使用。 # 那么在pytest收集用例后,開(kāi)始執(zhí)行用例之前會(huì)自動(dòng)化執(zhí)行這個(gè)fixture當(dāng)中的前置代碼, # 當(dāng)所有用例執(zhí)行完成之后,自動(dòng)化執(zhí)行這個(gè)fixture的后置代碼。 @pytest.fixture(scope="session",autouse=True) def init3(): yield print("用例執(zhí)行之后,執(zhí)行的代碼") # 只有用例執(zhí)行之后的后置清理代
6)fixture的返回值設(shè)置:yeild 返回值
當(dāng)測(cè)試用例當(dāng)中,要使用fixture里生成的數(shù)據(jù)時(shí),則需要fixture返回?cái)?shù)據(jù)。
若有數(shù)據(jù)返回則:yeild 返回值
import pytest from selenium import webdriver from time import sleep ? ? # 設(shè)置scope為class。調(diào)用此fixture的測(cè)試類下,只執(zhí)行一次這個(gè)fixture. @pytest.fixture(scope="class") def init2(): print("==== 測(cè)試類下,執(zhí)行所有用例之前,執(zhí)行的代碼 ====") driver = webdriver.Chrome() yield driver # 返回driver對(duì)象 print("==== 測(cè)試類下,執(zhí)行所有用例之后,執(zhí)行的代碼 ====") driver.quit()
2、調(diào)用fixture
在fixture定義好之后,可以明確:
- 1)fixture處理了哪些前置準(zhǔn)備工作、哪些后置清理工作
- 2)fixture作用在哪個(gè)范圍(是測(cè)試函數(shù)?還是測(cè)試類?還是測(cè)試會(huì)話?還是測(cè)試模塊?)
在以上2點(diǎn)都定下來(lái)了之后,接下來(lái)就是,在測(cè)試用例當(dāng)中,根據(jù)需要調(diào)用不同的fixture。
調(diào)用方法有2種:
- 1、在測(cè)試用例/測(cè)試類上面加上:@pytest.mark.usefixture("fixture的函數(shù)名字")
- 2、將fixture函數(shù)名,作為測(cè)試用例函數(shù)的參數(shù)。
第2種用法,主要是用參數(shù)來(lái)接收f(shuō)ixture的返回值,以便在測(cè)試用例中使用。
第一種方式案例如下:
第二種方式案例如下:
3、conftest.py共享機(jī)制
在某些大的業(yè)務(wù)場(chǎng)景下,很多用例當(dāng)中,會(huì)使用相同的前置準(zhǔn)備工作,和后置清理工作。
如果在每個(gè)測(cè)試模塊下,都把前置準(zhǔn)備工作,和后置清理工作寫一遍,在維護(hù)上和優(yōu)化上講不夠好。
pytest框架提供了一個(gè)fixture共享的機(jī)制 ,可以讓不同的用例模塊,使用同一個(gè)fixture。這就是conftest.py文件。
3.1 conftest.py共享實(shí)現(xiàn)
1)在項(xiàng)目根目錄下,創(chuàng)建一個(gè)conftest.py文件。
2)文件名必須是conftest.py,大小寫敏感,不可改名字。
3)conftest.py當(dāng)中,可以編寫多個(gè)fixture
4)在測(cè)試用例文件當(dāng)中,不需要引入conftest.py文件。直接調(diào)用fixture的函數(shù)名,會(huì)自動(dòng)去conftest.py當(dāng)中查找的。
3.2 conftest.py層級(jí)作用域
conftest.py在項(xiàng)目根目錄下,則項(xiàng)目下的所有測(cè)試用例,均可使用conftest.py中定義的fixture。即項(xiàng)目根目錄下的conftest.py,作用域是整個(gè)項(xiàng)目。
那,如果,conftest.py當(dāng)中的fixture,只想在某個(gè)python包內(nèi)可用呢?
conftest.py實(shí)現(xiàn)了層級(jí)作用域。
簡(jiǎn)單來(lái)說(shuō)就是:conftest.py 在哪個(gè)目錄下,此目錄下(包含子目錄)的所有用例可使用其中的fixture。
如下圖:
- 根目錄下的conftest.py里的fixture,無(wú)論項(xiàng)目下的哪個(gè)用例,都可以使用。
- 子目錄moduleA下的conftest.py里的fixture,只有moduleA下的用例可以使用。
- 子目錄moduleB下的conftest.py里的fixture,只有moduleB下的用例可以使用。
moduleB下的用例文件test_module_b.py中的用例,即可以使用根目錄下的conftest.py中的fixuture,又可以使用自己目錄下的conftest.py的fixture:
那么有個(gè)問(wèn)題,如果出現(xiàn)了同名fixture怎么辦呢?
這里涉及到了,測(cè)試用例在執(zhí)行時(shí),調(diào)用fixture的順序。一般來(lái)講,按就近原則調(diào)用。
測(cè)試用例文件中的fixture > 當(dāng)前目錄中的fixture > 上級(jí)目錄中的fixture > 根目錄中的fixture
4、fixture嵌套
fixture不但支持共享 ,還支持嵌套使用。
嵌套使用即:一個(gè)fixture,可以做另外一個(gè)fixture的參數(shù)。
如下圖所示:名為init2的fixture,可以作為init的參數(shù)。
并且,init當(dāng)中,將init2的返回值,同樣返回。
當(dāng)在用例當(dāng)中,調(diào)用init時(shí),init會(huì)自動(dòng)去調(diào)用init2。
下圖案例中,init2為class級(jí)作用域,init為function級(jí)作用域。
fixture的執(zhí)行順序如下:
- init2的后置代碼
- init的后置代碼
- init的前置代碼
- init2的前置代碼
到此這篇關(guān)于Pytest框架之fixture的文章就介紹到這了。希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
python控制windows剪貼板,向剪貼板中寫入圖片的實(shí)例
今天小編就為大家分享一篇python控制windows剪貼板,向剪貼板中寫入圖片的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-05-05python基于pdfminer庫(kù)提取pdf文字代碼實(shí)例
這篇文章主要介紹了python 提取pdf文字代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08Python?pyecharts?Boxplot箱線圖的實(shí)現(xiàn)
本文主要介紹了Python?pyecharts?Boxplot箱線圖的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05python爬蟲(chóng)爬取網(wǎng)頁(yè)表格數(shù)據(jù)
這篇文章主要為大家詳細(xì)介紹了python爬蟲(chóng)爬取網(wǎng)頁(yè)表格數(shù)據(jù),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-03-03Python基礎(chǔ)之語(yǔ)法錯(cuò)誤和異常詳解
Python有兩種錯(cuò)誤很容易辨認(rèn):語(yǔ)法錯(cuò)誤和異常.本文就給大家詳細(xì)介紹一下Python錯(cuò)誤和異常,對(duì)正在學(xué)習(xí)python的小伙伴們很有幫助哦,需要的朋友可以參考下2021-05-05python爬蟲(chóng)實(shí)戰(zhàn)之爬取京東商城實(shí)例教程
這篇文章主要介紹了python爬取京東商城的相關(guān)資料,文中通過(guò)爬取一個(gè)實(shí)例頁(yè)面進(jìn)行了講解,通過(guò)示例代碼和圖文介紹的非常詳細(xì),相信對(duì)大家具有一定的參考價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2017-04-04