使用mock,可以將某個函數所依賴的對象或者變量mock掉,從而降低測試條件的負責度。如下所示:
創(chuàng)新互聯是一家成都做網站、成都網站建設、成都外貿網站建設,提供網頁設計,網站設計,網站制作,建網站,按需網站建設,網站開發(fā)公司,自2013年起是互聯行業(yè)建設者,服務者。以提升客戶品牌價值為核心業(yè)務,全程參與項目的網站策劃設計制作,前端開發(fā),后臺程序制作以及后期項目運營并提出專業(yè)建議和思路。
上述是mock對象的簡單使用方法,通過實例化一個Mock對象從而模擬掉原始函數的返回值,高級一些的用法就是通過mock.patch裝飾器,裝飾在類或者函數上進行模擬測試,如下在test.py文件中有兩個類:
測試用例設計如下:
以上測試用例說明,通過patch裝飾器模擬了 test.ProductionClass1 這個類,在 test_01 中使用 mock_class 模擬 test.ProductionClass1 。首先通過 mock_class.return_value 獲取類實例(如果模擬的是函數,則不需要這一步),然后通過 obj1.pro1_method.return_value 設置方法的返回值,并進行測試。測試結果說明無論是通過 mock_class 還是 test.ProductionClass1 還是 obj1 執(zhí)行方法,獲取到的結果都是設置的值,并且在另一個類中調用模擬類的方法,也能成功獲取到設置的 return_value 。
測試函數是用于自動化測試,使用python模塊中的unittest中的工具來測試
附上書中摘抄來的代碼:
#coding=utf-8import unittestfrom name_function import get_formatted_nameclass NamesTestCase(unittest.TestCase): def test_first_last_name(self): formatted_name=get_formatted_name('janis','joplin') self.assertEqual(formatted_name,'Janis Joplin') def test_first_last_middle_name(self): formatted_name=get_formatted_name('wolfgang','mozart','amadeus') self.assertEqual(formatted_name,'Wolfgang Amadeus Mozart')#注意下面這行代碼,不寫會報錯哦~~~書中沒有這行if __name__=="__main__": unittest.main()
①整數類型
簡稱整型,與數學中整數的概念一致。整型數據的表示方式有4種,分別是十進制、二進制(以0B或0b開頭)、八進制(以0o或0O開頭)和十六進制(以0X或0x開頭)。
使用Python的內置函數type()可以測試各種數據類型。
②浮點型
用于表示數學中的實數,是帶有小數的數據類型。例如:3.14、1.0都是浮點型。
浮點型可以用十進制或科學記數法表示。
③字符串類型
Python的字符串,是用單引號、雙引號和三引號括起來的字符序列。
例如: "python"
④列表類型
列表是一種數據集合,列表用中括號[]來表示,列表內容以逗號進行分隔。
例如:[1,2,3]
⑤元組類型
元組是由0個或多個元素組成的不可變序列類型。元組用小括號()來表示。
例如:(1,2,3)
元組與列表的區(qū)別在于:元組的元素不能修改
⑥字典類型
字典是Python中唯一內置的映射類型,可用來實現通過數據查找關聯數據的功能。
字典包括兩個部分:鍵和值,用花括號{}表示,元素之間用逗號分隔,鍵和值之間用冒號分隔。
例如:{"name":"sun","age":12}
⑦集合類型
集合由各種類型的元素組成,但元素之間沒有任何順序,并且元素都不重復。
例如:set([1,2,3])
⑧復數類型
用于表示數學中的復數。例如:1 5j
⑨布爾類型
布爾型數據只有兩個取值:True 和 False.
如果將布爾值進行數值運算,True會被當做整型1,False會被當做整型0。
目錄
pytest是Python的單元測試框架,同自帶的unittest框架類似,但pytest框架使用起來更簡潔,效率更高。
pytest特點
安裝
測試
在測試之前要做的準備
我的演示腳本處于這樣一個的目錄中:
踩坑:你創(chuàng)建的pytest腳本名稱中不允許含有 . ,比如 1.簡單上手.py ,這樣會報錯。當然,可以這么寫 1-簡單上手.py
demo1.py :
上例中,當我們在執(zhí)行(就像Python解釋器執(zhí)行普通的Python腳本一樣)測試用例的時候, pytest.main(["-s", "demo1.py"]) 中的傳參需要是一個元組或者列表(我的pytest是5.2.2版本),之前的版本可能需要這么調用 pytest.main("-s demo1.py") ,傳的參數是str的形式,至于你使用哪種,取決于報不報錯:
遇到上述報錯,就是參數需要一個列表或者元組的形式,而我們使用的是str形式。
上述代碼正確的執(zhí)行結果是這樣的:
大致的信息就是告訴我們:
pytest.main(["-s", "demo1.py"])參數說明
除了上述的函數這種寫法,也可以有用例類的寫法:
用法跟unittest差不多,類名要以 Test 開頭,并且其中的用例方法也要以 test 開頭,然后執(zhí)行也一樣。
執(zhí)行結果:
那么,你這個時候可能會問,我記得unittest中有setup和teardown的方法,難道pytest中沒有嘛?你怎么提都不提?穩(wěn)住,答案是有的。
接下來,我們來研究一下pytest中的setup和teardown的用法。
我們知道,在unittest中,setup和teardown可以在每個用例前后執(zhí)行,也可以在所有的用例集執(zhí)行前后執(zhí)行。那么在pytest中,有以下幾種情況:
來一一看看各自的用法。
模塊級別setup_module/teardown_module
執(zhí)行結果:
類級別的setup_class/teardown_class
執(zhí)行結果:
類中方法級別的setup_method/teardown_method
執(zhí)行結果:
函數級別的setup_function/teardown_function
執(zhí)行結果:
小結
該腳本有多種運行方式,如果處于PyCharm環(huán)境,可以使用右鍵或者點擊運行按鈕運行,也就是在pytest中的主函數中運行:
也可以在命令行中運行:
這種方式,跟使用Python解釋器執(zhí)行Python腳本沒有什么兩樣。也可以如下面這么執(zhí)行:
當然,還有一種是使用配置文件運行,來看看怎么用。
在項目的根目錄下,我們可以建立一個 pytest.ini 文件,在這個文件中,我們可以實現相關的配置:
那這個配置文件中的各項都是什么意思呢?
首先, pytest.ini 文件必須位于項目的根目錄,而且也必須叫做 pytest.ini 。
其他的參數:
OK,來個示例。
首先,(詳細目錄參考開頭的目錄結構)在 scripts/test_case_01.py 中:
在 scripts/test_case_dir1/test_case02.py 中:
那么,在不同的目錄或者文件中,共有5個用例將被執(zhí)行,而結果則是兩個失敗三個成功。來執(zhí)行驗證一下,因為有了配置文件,我們在終端中(前提是在項目的根目錄),直接輸入 pytest 即可。
由執(zhí)行結果可以發(fā)現, 2 failed, 3 passed ,跟我們的預期一致。
后續(xù)執(zhí)行相關配置都來自配置文件,如果更改,會有相應說明,終端都是直接使用 pytest 執(zhí)行。
我們知道在unittest中,跳過用例可以用 skip ,那么這同樣是適用于pytest。
來看怎么使用:
跳過用例,我們使用 @pytest.mark.skipif(condition, reason) :
然后將它裝飾在需要被跳過用例的的函數上面。
效果如下:
上例執(zhí)行結果相對詳細,因為我們在配置文件中為 addopts 增加了 -v ,之前的示例結果中,沒有加!
另外,此時,在輸出的控制臺中, 還無法打印出 reason 信息,如果需要打印,則可以在配置文件中的 addopts 參數的 -s 變?yōu)?-rs :
如果我們事先知道測試函數會執(zhí)行失敗,但又不想直接跳過,而是希望顯示的提示。
Pytest 使用 pytest.mark.xfail 實現預見錯誤功能::
需要掌握的必傳參數的是:
那么關于預期失敗的幾種情況需要了解一下:
結果如下:
pytest 使用 x 表示預見的失?。╔FAIL)。
如果預見的是失敗,但實際運行測試卻成功通過,pytest 使用 X 進行標記(XPASS)。
而在預期失敗的兩種情況中,我們不希望出現預期失敗,結果卻執(zhí)行成功了的情況出現,因為跟我們想的不一樣嘛,我預期這條用例失敗,那這條用例就應該執(zhí)行失敗才對,你雖然執(zhí)行成功了,但跟我想的不一樣,你照樣是失敗的!
所以,我們需要將預期失敗,結果卻執(zhí)行成功了的用例標記為執(zhí)行失敗,可以在 pytest.ini 文件中,加入:
這樣就就把上述的情況標記為執(zhí)行失敗了。
pytest身為強大的單元測試框架,那么同樣支持DDT數據驅動測試的概念。也就是當對一個測試函數進行測試時,通常會給函數傳遞多組參數。比如測試賬號登陸,我們需要模擬各種千奇百怪的賬號密碼。
當然,我們可以把這些參數寫在測試函數內部進行遍歷。不過雖然參數眾多,但仍然是一個測試,當某組參數導致斷言失敗,測試也就終止了。
通過異常捕獲,我們可以保證程所有參數完整執(zhí)行,但要分析測試結果就需要做不少額外的工作。
在 pytest 中,我們有更好的解決方法,就是參數化測試,即每組參數都獨立執(zhí)行一次測試。使用的工具就是 pytest.mark.parametrize(argnames, argvalues) 。
使用就是以裝飾器的形式使用。
只有一個參數的測試用例
來看(重要部分)結果::
可以看到,列表內的每個手機號,都是一條測試用例。
多個參數的測試用例
(重要部分)結果:
可以看到,每一個手機號與每一個驗證碼都組合一起執(zhí)行了,這樣就執(zhí)行了4次。那么如果有很多個組合的話,用例數將會更多。我們希望手機號與驗證碼一一對應組合,也就是只執(zhí)行兩次,怎么搞呢?
在多參數情況下,多個參數名是以 , 分割的字符串。參數值是列表嵌套的形式組成的。
固件(Fixture)是一些函數,pytest 會在執(zhí)行測試函數之前(或之后)加載運行它們,也稱測試夾具。
我們可以利用固件做任何事情,其中最常見的可能就是數據庫的初始連接和最后關閉操作。
Pytest 使用 pytest.fixture() 定義固件,下面是最簡單的固件,訪問主頁前必須先登錄:
結果:
在之前的示例中,你可能會覺得,這跟之前的setup和teardown的功能也類似呀,但是,fixture相對于setup和teardown來說更靈活。pytest通過 scope 參數來控制固件的使用范圍,也就是作用域。
比如之前的login固件,可以指定它的作用域:
很多時候需要在測試前進行預處理(如新建數據庫連接),并在測試完成進行清理(關閉數據庫連接)。
當有大量重復的這類操作,最佳實踐是使用固件來自動化所有預處理和后處理。
Pytest 使用 yield 關鍵詞將固件分為兩部分, yield 之前的代碼屬于預處理,會在測試前執(zhí)行; yield 之后的代碼屬于后處理,將在測試完成后執(zhí)行。
以下測試模擬數據庫查詢,使用固件來模擬數據庫的連接關閉:
結果:
可以看到在兩個測試用例執(zhí)行前后都有預處理和后處理。
pytest中還有非常多的插件供我們使用,我們來介紹幾個常用的。
先來看一個重要的,那就是生成測試用例報告。
想要生成測試報告,首先要有下載,才能使用。
下載
如果下載失敗,可以使用PyCharm下載,怎么用PyCharm下載這里無需多言了吧。
使用
在配置文件中,添加參數:
效果很不錯吧!
沒完,看我大招
Allure框架是一個靈活的輕量級多語言測試報告工具,它不僅以web的方式展示了簡潔的測試結果,而且允許參與開發(fā)過程的每個人從日常執(zhí)行的測試中最大限度的提取有用信息。
從開發(fā)人員(dev,developer)和質量保證人員(QA,Quality Assurance)的角度來看,Allure報告簡化了常見缺陷的統(tǒng)計:失敗的測試可以分為bug和被中斷的測試,還可以配置日志、步驟、fixture、附件、計時、執(zhí)行 歷史 以及與TMS和BUG管理系統(tǒng)集成,所以,通過以上配置,所有負責的開發(fā)人員和測試人員可以盡可能的掌握測試信息。
從管理者的角度來看,Allure提供了一個清晰的“大圖”,其中包括已覆蓋的特性、缺陷聚集的位置、執(zhí)行時間軸的外觀以及許多其他方便的事情。allure的模塊化和可擴展性保證了我們總是能夠對某些東西進行微調。
少扯點,來看看怎么使用。
Python的pytest中allure下載
但由于這個 allure-pytest 插件生成的測試報告不是 html 類型的,我們還需要使用allure工具再“加工”一下。所以說,我們還需要下載這個allure工具。
allure工具下載
在現在allure工具之前,它依賴Java環(huán)境,我們還需要先配置Java環(huán)境。
注意,如果你的電腦已經有了Java環(huán)境,就無需重新配置了。
配置完了Java環(huán)境,我們再來下載allure工具,我這里直接給出了百度云盤鏈接,你也可以去其他鏈接中自行下載:
下載并解壓好了allure工具包之后,還需要將allure包內的 bin 目錄添加到系統(tǒng)的環(huán)境變量中。
完事后打開你的終端測試:
返回了版本號說明安裝成功。
使用
一般使用allure要經歷幾個步驟:
來看配置 pytest.ini :
就是 --alluredir ./report/result 參數。
在終端中輸入 pytest 正常執(zhí)行測試用例即可:
執(zhí)行完畢后,在項目的根目下,會自動生成一個 report 目錄,這個目錄下有:
接下來需要使用allure工具來生成HTML報告。
此時我們在終端(如果是windows平臺,就是cmd),路徑是項目的根目錄,執(zhí)行下面的命令。
PS:我在pycharm中的terminal輸入allure提示'allure' 不是內部或外部命令,也不是可運行的程序或批處理文件。但windows的終端沒有問題。
命令的意思是,根據 reportresult 目錄中的數據(這些數據是運行pytest后產生的)。在 report 目錄下新建一個 allure_html 目錄,而這個目錄內有 index.html 才是最終的allure版本的HTML報告;如果你是重復執(zhí)行的話,使用 --clean 清除之前的報告。
結果很漂亮:
allure open
默認的,allure報告需要HTTP服務器來打開,一般我們可以通過pycharm來完成,另外一種情況就是通過allure自帶的open命令來完成。
allure的其他用法
當然,故事還是沒有完!在使用allure生成報告的時候,在編寫用例階段,還可以有一些參數可以使用:
allure.title與allure.description
feature和story
由上圖可以看到,不同的用例被分為不同的功能中。
allure.severity
allure.severity 用來標識測試用例或者測試類的級別,分為blocker,critical,normal,minor,trivial5個級別。
severity的默認級別是normal,所以上面的用例5可以不添加裝飾器了。
allure.dynamic
在之前,用例的執(zhí)行順序是從上到下依次執(zhí)行:
正如上例的執(zhí)行順序是 3 1 2 。
現在,來看看我們如何手動控制多個用例的執(zhí)行順序,這里也依賴一個插件。
下載
使用
手動控制用例執(zhí)行順序的方法是在給各用例添加一個裝飾器:
那么, 現在的執(zhí)行順序是 2 1 3 ,按照order指定的排序執(zhí)行的。
如果有人較勁傳個0或者負數啥的,那么它們的排序關系應該是這樣的:
失敗重試意思是指定某個用例執(zhí)行失敗可以重新運行。
下載
使用
需要在 pytest.ini 文件中, 配置:
給 addopts 字段新增(其他原有保持不變) --reruns=3 字段,這樣如果有用例執(zhí)行失敗,則再次執(zhí)行,嘗試3次。
來看示例:
結果:
我們也可以從用例報告中看出重試的結果:
上面演示了用例失敗了,然后重新執(zhí)行多少次都沒有成功,這是一種情況。
接下來,來看另一種情況,那就是用例執(zhí)行失敗,重新執(zhí)行次數內通過了,那么剩余的重新執(zhí)行的次數將不再執(zhí)行。
通過 random 模塊幫助我們演示出在某次執(zhí)行中出現失敗的情況,而在重新執(zhí)行的時候,會出現成功的情況,看結果:
可以看到,用例 02 重新執(zhí)行了一次就成功了,剩余的兩次執(zhí)行就終止了。
一條一條用例的執(zhí)行,肯定會很慢,來看如何并發(fā)的執(zhí)行測試用例,當然這需要相應的插件。
下載
使用
在配置文件中添加:
就是這個 -n=auto :
并發(fā)的配置可以寫在配置文件中,然后其他正常的執(zhí)行用例腳本即可。另外一種就是在終端中指定,先來看示例:
結果:
pytest-sugar 改變了 pytest 的默認外觀,添加了一個進度條,并立即顯示失敗的測試。它不需要配置,只需 下載插件即可,用 pytest 運行測試,來享受更漂亮、更有用的輸出。
下載
其他照舊執(zhí)行用例即可。
pytest-cov 在 pytest 中增加了覆蓋率支持,來顯示哪些代碼行已經測試過,哪些還沒有。它還將包括項目的測試覆蓋率。
下載
使用
在配置文件中:
也就是配置 --cov=./scripts ,這樣,它就會統(tǒng)計所有 scripts 目錄下所有符合規(guī)則的腳本的測試覆蓋率。
執(zhí)行的話,就照常執(zhí)行就行。
結果:
更多插件參考:
有的時候,在 pytest.ini 中配置了 pytest-html 和 allure 插件之后,執(zhí)行后報錯:
出現了這個報錯,檢查你配置的解釋器中是否存在 pytest-html 和 allure-pytest 這兩個模塊。如果是使用的pycharm ide,那么你除了檢查settings中的解釋器配置之外,還需要保證運行腳本的編輯器配置是否跟settings中配置一致。