Python接口測試實戰3(下)- unittest測試框架

若有任何學習問題,能夠添加做者微信:lockingfreehtml

課程目錄

Python接口測試實戰1(上)- 接口測試理論 Python接口測試實戰1(下)- 接口測試工具的使用 Python接口測試實戰2 - 使用Python發送請求 Python接口測試實戰3(上)- Python操做數據庫 Python接口測試實戰3(下)- unittest測試框架 Python接口測試實戰4(上) - 接口測試框架實戰 Python接口測試實戰4(下) - 框架完善:用例基類,用例標籤,從新運行上次失敗用例 Python接口測試實戰5(上) - Git及Jenkins持續集成 Python接口測試實戰5(下) - RESTful、Web Service及Mock Serverpython

更多學習資料請加QQ羣: 822601020獲取git

本節內容

  • unittest簡介
  • 用例編寫
  • 用例組織及運行
  • 生成測試報告

unitttest簡介

參考:unittest官方文檔github

爲何要使用unittest? 在編寫接口自動化用例時,咱們通常針對一個接口創建一個.py文件,一條測試用例封裝爲一個函數(方法),可是在批量執行的過程當中,若是其中一條出錯,後面的用例就沒法執行。使用測試框架能夠互不影響的用例執行及更靈活的執行控制數據庫

unittest特色json

  • python自帶的單元測試框架,無需安裝
  • 用例執行互不干擾
  • 提供不一樣範圍的setUp(測試準備)和tearDown(測試清理)方法
  • 提供豐富的斷言方法
  • 能夠經過discover批量執行全部模塊的用例
  • 能夠經過TestSuite(測試集)靈活的組織用例

unittest幾大組成部分api

  • TestCase: 用例對象,編寫測試用例時要繼承該類,以具備TestCase的屬性和方法
  • TestSuite: 測試集或測試套件,測試用例的集合,用來組織用例,支持嵌套
  • TestLoader: 用例加載器,用於向TestSuite中添加用例
  • TextTestRunner: 用例執行器(輸出文本結果),通常以TestSuite爲單位執行用例
  • TestResult: 測試結果

用例編寫

  1. 新建一個test_開頭(必須)的.py文件,如test_user_login.py
  2. 導入unittest
  3. 編寫一個Test開頭(必須)的類,並繼承unittest.TestCase,作爲測試類
  4. 在類中編寫一個test_開頭(必須)的方法,做爲用例

test_user_login.py # 文件必須test_開頭瀏覽器

import unittest  # 導入unittest
import requests

class TestUserLogin(unittest.TestCase):  # 類必須Test開頭,繼承TestCase才能識別爲用例類
    url = 'http://115.28.108.130:5000/api/user/login/'
    
    def test_user_login_normal(self):   # 一條測試用例,必須test_開頭
        data = {"name": "張三", "password": "123456"}
        res = requests.post(url=self.url, data=data)
        self.assertIn('登陸成功', res.text)  # 斷言
        
    def test_user_login_password_wrong(self):
        data = {"name": "張三", "password": "1234567"}
        res = requests.post(url=self.url, data=data)
        self.assertIn('登陸失敗', res.text)  # 斷言


if __name__ == '__main__':  # 若是是直接從當前模塊執行(非別的模塊調用本模塊)
	unittest.main(verbosity=2)    # 運行本測試類全部用例,verbosity爲結果顯示級別

用例執行順序:並不是按書寫順序執行,而是按用例名ascii碼前後順序執行微信

用例斷言 unittest提供了豐富的斷言方法,經常使用爲如下幾種:網絡

  • 判斷相等
    • assertEqual(a,b)/assertNotEqual(a,b): 斷言值是否相等
    • assertIs(a,b)/assertIsNot(a,b): 斷言是否同一對象(內存地址同樣)
    • assertListEqual(list1, list2)/assertItemNotEqual(list1, list2): 斷言列表是否相等
    • assertDictEqual(dict1, dict2)/assertDictNotEqual(dict1, dict2): 斷言字典是否相等
  • 是否爲空
    • assertIsNone(a)/assertIsNotNone(a)
  • 判斷真假
    • assertTrue(a)/assertFalse(a)
  • 是否包含
    • assertIn(a,b)/assertNotIn(a,b) # b中是否包含a
  • 大小判斷
    • assertGreater(a,b)/assertLess(a,b) : 斷言a>b / 斷言a<b
    • assertGreaterEqual(a,b)/assertLessEqual: 斷言a>=b / 斷言a<=b
  • 類型判斷
    • assertIsInstance(a,dict)/assertNotIsInstance(a,list) # 斷言a爲字典 / 斷言a非列表

示例:

import unittest

case = unittest.TestCase()
case.assertEqual(1,2.0/2) # 經過1=2.0/2
case.assertEqual(1, True) # 經過
case.assertIs(1.0, 2.0/2) # 失敗,不是同一對象
case.assertListEqual([1,2],[1,2]) # 經過(順序要一致)
case.assertDictEqual({"a":1,"b":2},{"b":2,"a":1}) # 經過,字典本無序
case.assertIsNone({}) # 失敗
case.assertFalse({}) # 經過,空字典爲False
case.assertIn("h","hello") # 經過
case.assertGreater(3,2) # 經過,3>2
case.assertIsInstance({"a":1}, dict) # 經過

斷言是unittest.TestCase的一種方法,經過斷言判斷用例是否經過(Pass/Fail)

Test Fixtures(用例包裹方法) Test Fixtures即setUp(用例準備)及tearDown(測試清理)方法,用於分別在測試前及測試後執行

按照不一樣的做用範圍分爲:

  • setUp()/tearDown(): 每一個用例執行前/後執行一次
  • setUpClass()/tearDownClass(): 每一個測試類加載時/結束時執行一次
  • setUpMoudle()/tearDownMoudle(): 每一個測試模塊(一個py文件爲一個模塊)加載/結束時執行一次
import unittest

def setUpModule():    # 當前模塊執行前只執行一次
    print('=== setUpModule ===')

def tearDownModule(): # 當前模塊執行後只執行一次
    print('=== tearDownModule ===')

class TestClass1(unittest.TestCase):
    @classmethod          # 聲明爲類方法(必須)
    def setUpClass(cls):  # 類方法,注意後面是cls,整個類只執行一次
        print('--- setUpClass ---')

    @classmethod
    def tearDownClass(cls):  
        print('--- tearDownClass ---')
        
    def setUp(self):  # 該類中每一個測試用例執行一次
        print('... setUp ...')
        
    def tearDown(self):
        print('... tearDown ...')
        
    def test_a(self):  # 測試用例
        print("a")
        
    def test_B(self): # 大寫B的ascii比小寫a靠前,會比test_a先執行
        print("B")

class TestClass2(unittest.TestCase):  # 該模塊另外一個測試類
    def test_A(self):
        print("A")

if __name__ == '__main__':
    unittest.main()

執行結果:

=== setUpModule ===
--- setUpClass ---
... setUp ...
B
... tearDown ...
... setUp ...
a
... tearDown ...
--- tearDownClass ---
A
=== tearDownModule ===
...
----------------------------------------------------------------------
Ran 3 tests in 0.001s

OK

完整的接口測試用例 一條完整的測試接口用例須要包含:

  1. 數據準備:準備測試數據,可手工準備,也可以使用代碼準備(一般涉及數據庫操做)
  2. 環境檢查:若是手工準備的數據,鏈接數據庫進行環境檢查會使用例更健壯
  3. 發送請求:發送接口請求
  4. 響應斷言/數據庫斷言:響應斷言後,還須要進行數據庫斷言,以確保接口數據庫操做的正確性
  5. 數據清理:若是接口有更數據庫操做,斷言結束後須要還原更改

test_user_reg.py

import unittest
import requests
from db import *   # 導入db.py文件,源碼見上篇

# 數據準備
NOT_EXIST_USER = '范冰冰'
EXIST_USER = '張三'


class TestUserReg(unittest.TestCase):
    url = 'http://115.28.108.130:5000/api/user/reg/'
    
    def test_user_reg_normal(self):
        # 環境檢查
        if check_user(NOT_EXIST_USER):
            del_user(NOT_EXIST_USER)
        
        # 發送請求
        data = {'name': NOT_EXIST_USER, 'password': '123456'}
        res = requests.post(url=self.url, json=data)
        
        # 指望響應結果,注意字典格式和json格式的區別(若是有true/false/null要轉化爲字典格式)
        except_res = {
                        "code": "100000",
                        "msg": "成功",
                        "data": {
                                    "name": NOT_EXIST_USER,
                                    "password": "e10adc3949ba59abbe56e057f20f883e"
                                }
                      }
        
        # 響應斷言(總體斷言)
        self.assertDictEqual(res.json(), except_res)
        
        # 數據庫斷言
        self.assertTrue(check_user(NOT_EXIST_USER))
        
        # 環境清理(因爲註冊接口向數據庫寫入了用戶信息)
        del_user(NOT_EXIST_USER)
        
    def test_user_reg_exist(self):
         # 環境檢查
        if not check_user(EXIST_USER):
            add_user(EXIST_USER)
        
        # 發送請求
        data = {'name': EXIST_USER, 'password': '123456'}
        res = requests.post(url=self.url, json=data)
        
        # 指望響應結果,注意字典格式和json格式的區別(若是有true/false/null要轉化爲字典格式)
        except_res = {
                        "code": "100001",
                        "msg": "失敗,用戶已存在",
                        "data": {
                                    "name": EXIST_USER,
                                    "password": "e10adc3949ba59abbe56e057f20f883e"
                                }
                      }
        
        # 響應斷言(總體斷言)
        self.assertDictEqual(res.json(), except_res)
        
        # 數據庫斷言(沒有註冊成功,數據庫沒有添加新用戶)
        
        # 環境清理(無需清理)
        
if __name__ == '__main__':
    unittest.main(verbosity=2)   # 運行全部用例

用例組織及運行

除了使用unittest.main()運行整個測試類以外,咱們還能夠經過TestSuite來靈活的組織要運行的測試集

  1. 新建TestSuite並添加測試用例
import unittest
from test_user_login import TestUserLogin  
from test_user_reg import TestUserReg # 從上面兩個例子裏導入測試類

suite = unittest.TestSuite()
suite.addTest(TestUserLogin('test_user_login_normal')) # 添加單個用例
suite.addTests([TestUserReg('test_user_reg_normal'),TestUserReg('test_user_reg_exist')]) # 添加多個用例

# 運行測試集
unittest.TextTestRunner(verbosity=2).run(suite)  # verbosity顯示級別,運行順序爲添加到suite中的順序
  1. 使用makeSuite來製做用例集
import unittest
from test_user_login import TestUserLogin

suite1 = unittest.makeSuite(TestUserLogin, 'test_user_login_normal') # 使用測試類的單條用例製做測試集
suite2 = unittest.makeSuite(TestUserLogin) # 使用整個測試類製做測試集合(包含該測試類全部用例)

unittest.TextTestRunner(verbosity=2).run(suite1)
  1. 使用TestLoader(用例加載器)生成測試集
improt unittest
from test_user_login import TestUserLogin

suite = unittest.TestLoader().loadTestsFromTestCase(TestUserLogin) # 加載該測試類全部用例並生成測試集

unittest.TextTestRunner(verbosity=2).run(suite)
  1. 使用discover(用例發現)遍歷全部的用例
import unittest

suite = unittest.defaultTestLoader.discover("./")  # 遍歷當前目錄及子包中全部test_*.py中全部unittest用例
unittest.TextTestRunner(verbosity=2).run(suite)

注意:

  • 子目錄中須要包含__init__.py文件,及應爲的Python包
  • 全部用例由於test_*.py,包含測試類應以Test開頭,並繼承unittest.TestCase, 用例應以test_開頭
  1. 測試集嵌套
import unittest
from test_user_login import TestUserLogin

suite1 = unittest.TestSuite()
suite1.addTest(TestUserLogin('test_user_login_normal'))

suite2 = makeSuite(TestUserLogin, 'test_user_login_password_wrong')

suite = unittest.TestSuite([suite1, suite2])  # 將兩個測試集組合爲一個

unittest.TextTestRunner(verbosity=2).run(suite)

生成測試報告

生成文本報告

import unittest

suite = unittest.defaultTestLoader.discover("./")

# 輸出測試結果到文本文件
with open("result.txt","w") as f:
    unittest.TextTestRunner(stream=f,verbosity=2).run(suite) # 將輸出流stream輸出到文件

生成HTML報告

  1. 下載HTMLTestRunnerCN
  2. 解壓並將解壓包中python3x文件夾下的HTMLTestRunnerCN.py拷貝到項目目錄
  3. 在目錄下新建腳本run_all.py
import unittest
from HTMLTestReportCN import HTMLTestRunner

suite = unittest.defaultTestLoader.discover("./")

f = open("report.html", 'wb') # 二進制寫格式打開要生成的報告文件
HTMLTestRunner(stream=f,title="Api Test",description="測試描述",runner="卡卡").run(suite)
f.close()
  1. 運行腳本,會在當前文件夾下生成report.html,用瀏覽器打開便可

項目預覽

測試報告

源碼下載連接:https://pan.baidu.com/s/1gXwYOCL6BfRW4XVgxkCpIA 密碼:jdgo

此爲北京龍騰育才 Python高級自動化(接口測試部分)授課筆記 課程介紹 想要參加現場(北京)/網絡課程的能夠聯繫做者微信:lockingfree

  1. 高效學習,快速掌握Python自動化全部領域技能
  2. 同步快速解決各類問題
  3. 配套實戰項目練習
相關文章
相關標籤/搜索