pytest入門基礎(chǔ)

參考學(xué)習(xí)官網(wǎng)文檔:https://docs.pytest.org/en/latest/getting-started.html

一嫌褪、pytest簡(jiǎn)介

1. 前言:測(cè)試框架與測(cè)試工具

  • 測(cè)試框架、測(cè)試工具這兩個(gè)概念經(jīng)常容易搞混了医咨,測(cè)試框架是諸如 Unittest、Pytest、TestNG 這類烛亦,而測(cè)試工具指的則是 Selenium射富、Appium解虱、Jmeter 這類费变。
  • 測(cè)試框架的作用是未玻,幫助管理測(cè)試用例、執(zhí)行測(cè)試用例胡控、參數(shù)化、斷言旁趟、生成測(cè)試報(bào)告等基礎(chǔ)性工作昼激,讓我們將精力用在測(cè)試用例的編寫(xiě)上。好的測(cè)試框架應(yīng)該具有很高的擴(kuò)展性锡搜,支持二次開(kāi)發(fā)橙困,并能夠支持多種類型的自動(dòng)化測(cè)試。
  • 測(cè)試工具的作用是為了完成某一類型的測(cè)試耕餐,比如 Selenium 用于對(duì) WEB UI 進(jìn)行自動(dòng)化測(cè)試凡傅,Appium 用來(lái)對(duì) APP 進(jìn)行自動(dòng)化測(cè)試,Jmeter 可以用來(lái)進(jìn)行 API 自動(dòng)化測(cè)試和性能測(cè)試肠缔。另外夏跷,Java 語(yǔ)言中 OkHttp 庫(kù)哼转,Python 語(yǔ)言中的 requests 庫(kù),這些 HTTP 的 client 也可以看做是一種 API 測(cè)試工具槽华。

2. 單元測(cè)試框架

  • 1)什么是單元測(cè)試框架壹蔓?
    單元測(cè)試是指軟件開(kāi)發(fā)中,針對(duì)軟件的最小單位(函數(shù)猫态、方法)進(jìn)行正確性的檢查測(cè)試佣蓉,Java:junit、testing
    Python:unittest亲雪、pytest
  • 2)單元測(cè)試框架主要做什么勇凭?
    ① 測(cè)試發(fā)現(xiàn):從多個(gè)文件里面去找我們得測(cè)試用例
    ② 測(cè)試執(zhí)行:按照一定的順序和規(guī)則去執(zhí)行,并生成結(jié)果
    ③ 測(cè)試判斷:通過(guò)斷言判斷預(yù)期結(jié)果和實(shí)際結(jié)果的差異
    ④ 測(cè)試報(bào)告:統(tǒng)計(jì)測(cè)試進(jìn)度义辕、耗時(shí)虾标、通過(guò)率、生成測(cè)試報(bào)告

3. pytest簡(jiǎn)介

  • Pytest 是 Python 的一種易用终息、高效和靈活的單元測(cè)試框架夺巩,可以支持單元測(cè)試和功能測(cè)試
  • pytest可以和selenium,requests周崭,appium結(jié)合實(shí)現(xiàn)web自動(dòng)化柳譬、接口自動(dòng)化、app自動(dòng)化
  • pytest可以實(shí)現(xiàn)測(cè)試用例的跳過(guò)以及reruns(失敗用例重試)
  • pytest可以和allure生成非常美觀的測(cè)試報(bào)告
  • pytest可以和jenkins持續(xù)集成
  • pytest有很多非常強(qiáng)大的插件续镇,并且這些插件能夠?qū)崿F(xiàn)很多實(shí)用的操作
    ① pytest-xdist:測(cè)試用例分布式執(zhí)行美澳,多cpu分發(fā)
    ② pytest-ordering:用于改變測(cè)試用例的執(zhí)行順序
    ③ pytest-rerunfailures:用例失敗后重跑
    ④ pytest-html:生成html格式的自動(dòng)化測(cè)試報(bào)告
    ⑤ allure-pytest:用于生成美觀測(cè)試報(bào)告


二、安裝pytest

pytest需要:Python 3.7+ 或 PyPy3

  1. 在命令行中運(yùn)行以下命令
pip install -U pytest

其他插件安裝方法也是一樣pip安裝

  1. 檢查是否安裝成功
    pytest


三摸航、pytest使用規(guī)則

  1. 模塊名必須以test_開(kāi)頭或test_結(jié)尾
  2. 測(cè)試類必須以Test開(kāi)頭制跟,且不能有init方法
  3. 測(cè)試方法必須以test開(kāi)頭
    使用規(guī)則


四、測(cè)試用例執(zhí)行

pytest 用例的執(zhí)行主要有以下三種方式:

  • 通過(guò)命令模式執(zhí)行
  • 通過(guò)主函數(shù)執(zhí)行
  • 通過(guò)讀取 pytest.ini 配置文件執(zhí)行

4.1 通過(guò)命令模式執(zhí)行

測(cè)試文件test_demo.py:Pytest 使用 Python 的 assert 進(jìn)行條件判斷酱虎,簡(jiǎn)單的測(cè)試函數(shù)如:

def test_product_case001():
    print("模擬測(cè)試模塊1的功能")
    assert 1 + 1 == 3


def test_product_case002():
    print("模擬測(cè)試模塊2的功能")
    assert 1 == 1
1)執(zhí)行所有用例

pytest 命令執(zhí)行當(dāng)前目錄下所有的測(cè)試用例雨膨,所以要想執(zhí)行項(xiàng)目中所有的用例,只需要將執(zhí)行目錄切換到根目錄即可:

image.png

可以加上-v參數(shù)查看詳情pytest -v

image.png

2)指定執(zhí)行

執(zhí)行指定目錄:
執(zhí)行指定目錄

執(zhí)行指定文件:
執(zhí)行指定文件

執(zhí)行指定用例:
執(zhí)行指定用例


4.2 通過(guò)主函數(shù)執(zhí)行

1)主函數(shù)運(yùn)行

test_demo.py添加主函數(shù):

import pytest

class Testdemo:
    def test_case_passed(self):
        print("模擬執(zhí)行成功場(chǎng)景的case")
        assert (1, 2, 3) == (1, 2, 3)

def test_case_failed():
    print("模擬執(zhí)行失敗場(chǎng)景的case")
    assert 3 == 1 + 1

if __name__ == '__main__':
    pytest.main(["-v"])    # 執(zhí)行測(cè)試用例读串,-v參數(shù)打印詳細(xì)信息

然后直接運(yùn)行:
執(zhí)行結(jié)果

從終端的打印可以看到聊记,用例文件中的用例方法 test_case_failed 以及測(cè)試類中的測(cè)試方法 test_case_passed 都被執(zhí)行了。

2)批量執(zhí)行

只需要在對(duì)應(yīng)的用例文件中調(diào)用主函數(shù)恢暖,就可以自動(dòng)執(zhí)行文件中的所有用例排监。
但是在實(shí)際的使用中,一般會(huì)將用例分類寫(xiě)在不同目錄下的不同用例文件里杰捂,且執(zhí)行時(shí)要批量去執(zhí)行所有的用例舆床,所以往往會(huì)將主函數(shù)寫(xiě)在一個(gè)與用例文件夾平級(jí)的 py 文件中,pytest 內(nèi)部設(shè)有對(duì)應(yīng)的機(jī)制,會(huì)自動(dòng)掃描全局挨队,執(zhí)行所有的用例文件谷暮。

主函數(shù)文件所在目錄:
主函數(shù)文件

運(yùn)行主函數(shù)文件main.py,輸出結(jié)果可以看到所有用例均被執(zhí)行了:

批量執(zhí)行

3)指定執(zhí)行

pytest.main()可以加上命令參數(shù)運(yùn)行瞒瘸,如下

指定目錄
指定目錄

指定文件
指定文件

指定用例
指定用例


4.3 通過(guò) pytest.ini 配置文件執(zhí)行

在項(xiàng)目中坷备,通常通過(guò)配置全局的配置文件來(lái)執(zhí)行測(cè)試用例的。其余兩種執(zhí)行方式在編寫(xiě)和調(diào)試用例時(shí)比較方便情臭。對(duì)于配置文件省撑,有以下幾點(diǎn)要求:

  • 配置文件的名字必須為 pytest.ini
  • 配置文件一般放置在項(xiàng)目的根目錄
  • 配置文件必須時(shí) ANSI 編碼
  • 配置了pytest.ini文件后,不管是主函數(shù)模式運(yùn)行用例俯在,還是命令行模式運(yùn)行用例竟秫,都會(huì)去讀取pytest.ini配置文件中的內(nèi)容。
  • CMD命令行中執(zhí)行pytest -h命令跷乐,可以查看pytest.ini的設(shè)置選項(xiàng)肥败。

pytest.ini 是 pytest 框架的核心配置文件,它可以改變 pytest 的默認(rèn)行為愕提,不管是通過(guò)主函數(shù)還是命令模式去執(zhí)行用例馒稍,都會(huì)先去讀取配置文件。

修改完配置文件浅侨,再去執(zhí)行用例時(shí)就會(huì)按照配置文件的要求去執(zhí)行:
配置文件


五纽谒、執(zhí)行參數(shù)

我們?cè)趫?zhí)行用例時(shí),可以通過(guò)添加相關(guān)的參數(shù)實(shí)現(xiàn)不同的執(zhí)行效果如输,提升測(cè)試效率鼓黔。

1)-s:用于輸出調(diào)試信息,包括打印輸出的信息

# 通過(guò)main函數(shù)執(zhí)行
pytest.main(["-s", "./product_cases/test_product.py"])
# 通過(guò)命令模式執(zhí)行
pytest -s ./product_cases/test_product.py
-s

2)-v:顯示更詳細(xì)的信息

# 通過(guò)main函數(shù)執(zhí)行
pytest.main(["-v", "./product_cases/test_product.py"])
# 通過(guò)命令模式執(zhí)行
pytest -v ./product_cases/test_product.py
-v

3)-vs:將 -v 和 -s 參數(shù)效果結(jié)合不见,推薦使用
-vs

4)-n:支持多線程或者分布式運(yùn)行測(cè)試用例

n是進(jìn)程數(shù)澳化,也就是cpu個(gè)數(shù),最大值為當(dāng)前機(jī)器cpu個(gè)數(shù)稳吮,也可以設(shè)置為auto缎谷,自動(dòng)識(shí)別cpu個(gè)數(shù)。

# 通過(guò)main函數(shù)執(zhí)行
# 開(kāi)兩個(gè)線程去執(zhí)行用戶操作目錄下所有的用例
pytest.main(["-vs", "./product_cases", "-n=2"])

# 通過(guò)命令模式執(zhí)行
pytest -vs ./product_cases -n 2

為了方便看出測(cè)試效果灶似,給test_product.py文件中的每個(gè)測(cè)試用例都增加sleep(2)

import pytest,time


def test_product_case001():
    time.sleep(2)
    print("模擬測(cè)試模塊1的功能")
    assert 1 + 1 == 2


def test_product_case002():
    time.sleep(2)
    print("模擬測(cè)試模塊2的功能")
    assert 1 == 1

正常執(zhí)行結(jié)果:用時(shí)4.05s
單線程執(zhí)行

開(kāi)啟2個(gè)線程執(zhí)行結(jié)果:用時(shí)2.60s
多線程執(zhí)行

5)--reruns:失敗用例重跑

# 通過(guò)main函數(shù)執(zhí)行
# 如果用例執(zhí)行失敗慎陵,該用例再執(zhí)行2次
pytest.main(["-vs", "./product_cases", "--reruns=2"])

# 通過(guò)命令模式執(zhí)行
pytest -vs ./product_cases --reruns 2

修改test_product.py文件

def test_product_case001():
    print("模擬測(cè)試模塊1的功能")
    assert 1 + 1 == 3


def test_product_case002():
    print("模擬測(cè)試模塊2的功能")
    assert 1 == 1
失敗重跑

6)-x:只要有一個(gè)用例失敗,就停止執(zhí)行

# 通過(guò)main函數(shù)執(zhí)行
pytest.main(["-vs", "./product_cases", "-x"])

# 通過(guò)命令模式執(zhí)行
pytest -vs ./product_cases -x

test_product_case001失敗喻奥,停止執(zhí)行,test_product_case002未執(zhí)行

失敗停止執(zhí)行g(shù)

7)--maxfail:失敗的用例數(shù)達(dá)到規(guī)定值時(shí)停止測(cè)試

# 失敗的用例數(shù)達(dá)到2個(gè)時(shí)停止測(cè)試
# 通過(guò)main函數(shù)執(zhí)行
pytest.main(["-vs", "./product_cases", "--maxfail=2"])

# 通過(guò)命令模式執(zhí)行
pytest -vs ./product_cases --maxfail 2
失敗停止測(cè)試

8)-k:關(guān)鍵字過(guò)濾執(zhí)行

-k會(huì)匹配文件名捏悬、類名撞蚕、方法名匹配表達(dá)式的用例,如果指定的表達(dá)式是被文件名/測(cè)試類名包含的过牙,那么文件/測(cè)試類中所有的測(cè)試方法都會(huì)被執(zhí)行甥厦。

# 執(zhí)行文件名/類名/方法名中包含01的用例
# 通過(guò)main函數(shù)執(zhí)行
pytest.main(["-vs", "./product_cases", "-k=01"])

# 通過(guò)命令模式執(zhí)行
pytest -vs ./product_cases -k="01"

指定執(zhí)行測(cè)試用例名稱包含01

過(guò)濾執(zhí)行

指定執(zhí)行測(cè)試用例名稱包含login

過(guò)濾執(zhí)行

9)--html:用例執(zhí)行完成后生成 html 格式的測(cè)試報(bào)告

pytest -vs --html ./report/result.html --html+報(bào)告存放路徑參數(shù)

生成html報(bào)告

result.html


六纺铭、用例執(zhí)行順序

修改 test_product.py 中測(cè)試方法:

def test_product_case():
    print("模擬測(cè)試產(chǎn)品相關(guān)的操作")

def test_case_001():
    print("測(cè)試方法_001")

def test_case_002():
    print("測(cè)試方法_002")

def test_case_003():
    print("測(cè)試方法_003")

def test_case_004():
    print("測(cè)試方法_004")

def test_case_005():
    print("測(cè)試方法_005")

1)默認(rèn)執(zhí)行順序

pytest 對(duì)于用例的執(zhí)行順序是按照從上到下執(zhí)行的

從上往下執(zhí)行

2) 改變用例的執(zhí)行順序

通過(guò)裝飾器方法@pytest.mark.run()可以改變用例的執(zhí)行順序,參數(shù)order=n指定排在第n個(gè)執(zhí)行

# product_cases/test_product.py
import pytest

def test_product_case():
    print("模擬測(cè)試產(chǎn)品相關(guān)的操作")

# 通過(guò)裝飾器方法 @pytest.mark.run() 可以改變用例的執(zhí)行順序
# order參數(shù)表示該用例要排在第幾個(gè)執(zhí)行
@pytest.mark.run(order=1)
def test_case_002():
    print("測(cè)試方法_002")

@pytest.mark.run(order=2)
def test_case_003():
    print("測(cè)試方法_003")

def test_case_004():
    print("測(cè)試方法_004")

def test_case_005():
    print("測(cè)試方法_005")
指定執(zhí)行順序

從終端輸出的結(jié)果可以看到刀疙,被裝飾器裝飾的測(cè)試用例都按照了指定的順序執(zhí)行舶赔,沒(méi)有被裝飾到的用例還是按照從上到下的順序執(zhí)行


七、標(biāo)記函數(shù)谦秧,分組執(zhí)行用例

1)裝飾器@pytest.mark.name裝飾用例

第一步竟纳,使用裝飾器@pytest.mark.name去裝飾要執(zhí)行的用例,其中 name 為自定義的名稱

# product_cases/test_product.py
import pytest

# 通過(guò)裝飾器方法 @pytest.mark.finished標(biāo)記已完成的用例
@pytest.mark.finished
def test_product_case():
    print("模擬測(cè)試產(chǎn)品相關(guān)的操作")

# 通過(guò)裝飾器方法 @pytest.mark.unfinished標(biāo)記未完成功能的用例
@pytest.mark.unfinished
def test_case_002():
    print("測(cè)試方法_002")

@pytest.mark.unfinished
def test_case_003():
    print("測(cè)試方法_003")

def test_case_004():
    print("測(cè)試方法_004")

def test_case_005():
    print("測(cè)試方法_005")

2)pytest.ini 添加分組配置

第二步疚鲤,去 pytest.ini 中添加分組配置锥累,配置名必須和裝飾器中的 name 值一致
配置文件pytest.ini

3)參數(shù) -m 指定執(zhí)行用例

第三步,執(zhí)行時(shí)通過(guò)參數(shù) -m 指定要執(zhí)行的用例
執(zhí)行標(biāo)記用例


八集歇、跳過(guò)用例

對(duì)于某些不想執(zhí)行的用例桶略,可以使用 pytest 提供的裝飾器方法@pytest.mark.skip()跳過(guò)該用例不去執(zhí)行

1)直接跳過(guò)

裝飾器語(yǔ)法:@pytest.mark.skip(reason="") ,reason 參數(shù)可選诲宇,表示跳過(guò)該測(cè)試方法的原因

@pytest.mark.skip

2)條件跳過(guò)

裝飾器語(yǔ)法:@pytest.mark.skipif(判斷條件, reason="") 际歼,為測(cè)試函數(shù)指定被忽略的條件,reason 參數(shù)可選

# product_cases/test_product.py
import pytest

version = 1

# V2版本才支持姑蓝,版本小于V2則跳過(guò)不執(zhí)行
@pytest.mark.skipif(version < 2, reason='not supported until v2')
def test_product_case():
    print("模擬測(cè)試產(chǎn)品相關(guān)的操作")

def test_case_001():
    print("測(cè)試方法_001")

def test_case_002():
    print("測(cè)試方法_002")
  • version=1滿足條件鹅心,跳過(guò)不執(zhí)行
    滿足條件跳過(guò)
  • version=2不滿足條件,不跳過(guò)執(zhí)行
    不滿足條件不跳過(guò)

文章轉(zhuǎn)載參考:http://www.chendacheng.com/content/?category=Pytest%E4%B8%93%E9%A2%98&article=Pytest_01.html

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末它掂,一起剝皮案震驚了整個(gè)濱河市巴帮,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌虐秋,老刑警劉巖榕茧,帶你破解...
    沈念sama閱讀 222,000評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異客给,居然都是意外死亡用押,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,745評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門靶剑,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)铃绒,“玉大人,你說(shuō)我怎么就攤上這事往扔【保” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,561評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵坑匠,是天一觀的道長(zhǎng)血崭。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么夹纫? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,782評(píng)論 1 298
  • 正文 為了忘掉前任咽瓷,我火速辦了婚禮,結(jié)果婚禮上舰讹,老公的妹妹穿的比我還像新娘茅姜。我一直安慰自己,他們只是感情好月匣,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,798評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布钻洒。 她就那樣靜靜地躺著,像睡著了一般桶错。 火紅的嫁衣襯著肌膚如雪航唆。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 52,394評(píng)論 1 310
  • 那天院刁,我揣著相機(jī)與錄音糯钙,去河邊找鬼。 笑死退腥,一個(gè)胖子當(dāng)著我的面吹牛任岸,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播狡刘,決...
    沈念sama閱讀 40,952評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼享潜,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了嗅蔬?” 一聲冷哼從身側(cè)響起剑按,我...
    開(kāi)封第一講書(shū)人閱讀 39,852評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎澜术,沒(méi)想到半個(gè)月后艺蝴,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,409評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡鸟废,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,483評(píng)論 3 341
  • 正文 我和宋清朗相戀三年猜敢,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片盒延。...
    茶點(diǎn)故事閱讀 40,615評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡缩擂,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出添寺,到底是詐尸還是另有隱情胯盯,我是刑警寧澤,帶...
    沈念sama閱讀 36,303評(píng)論 5 350
  • 正文 年R本政府宣布计露,位于F島的核電站陨闹,受9級(jí)特大地震影響楞捂,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜趋厉,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,979評(píng)論 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望胶坠。 院中可真熱鬧君账,春花似錦、人聲如沸沈善。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,470評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)闻牡。三九已至净赴,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間罩润,已是汗流浹背玖翅。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,571評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留割以,地道東北人金度。 一個(gè)月前我還...
    沈念sama閱讀 49,041評(píng)論 3 377
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像严沥,于是被迫代替她去往敵國(guó)和親猜极。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,630評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容