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