Python+Excel+Unittest+HTMLTestRunner實現數據驅動介面自動化測試(二)

来源:https://www.cnblogs.com/Sandy-1128/archive/2019/03/16/appium-python-sandy-0316-1.html
-Advertisement-
Play Games

因為小白,這2天研究了好久才算是搞好。先附上一個測試完成後郵件的截圖: 上一篇有提到: unittest中實際運行了一個介面的很多條用例,而報告中只會有一條記錄。這是因為unittest test case中我只寫了一個test方法,迴圈測試數據。這樣寫用例更簡單,目前在想是從報告中處理,還是測試用 ...


因為小白,這2天研究了好久才算是搞好。先附上一個測試完成後郵件的截圖:

 

 

上一篇有提到:

unittest中實際運行了一個介面的很多條用例,而報告中只會有一條記錄。這是因為unittest test case中我只寫了一個test方法,迴圈測試數據。這樣寫用例更簡單,目前在想是從報告中處理,還是測試用例讀取部分進行處理。

而且這個問題會導致後續寫test case出錯幾率大,而且耗時較多。經過幾天的思考,寫了個簡易的解決辦法。思路:輕微修改excel讀取通過excel名/sheet名/用例id名來返回數據,那我們就可以在一個unittest case中固化test_開頭的測試方法,如10條(我們約定excel中每個sheet最多寫10條用例)。 這裡我們在excel中添加一個“是否執行用例”欄位,unittest中啟動skipIf方法來確認是否跳過該用例,即可完美解決上一篇的問題。

excel用例數據截圖:(增加runselect欄位 YES代表執行用例,No代表不執行)

 

 

 那我們的OperateExcel方法也做輕微修改:

def Readexcelrow_of_sheet(filename,sheet_name,row):
    '''返回某個sheet中某一行數據的列表'''
    folder = "C:\\Users\\LENOVO\\PycharmProjects\\Apiautotest-master\\testdata\\"
    fullfilename = folder + filename
    rb = xlrd.open_workbook(fullfilename)
    sheet = rb.sheet_by_name(sheet_name)
    datalist = []
    for i in range(10):
        if sheet.cell_value(row,9) =="YES":
            try:
                data = sheet.cell_value(row,i)
                datalist.append(data)
            except TypeError as f:
                break
        #print (datalist)
    return datalist

def Readexcelalldata_of_sheet(filename,sheet_name):
    '''返回整個sheet的內容的列表'''
    datalistlist =[]
    for i in range(1,100):
        try:
            data = Readexcelrow_of_sheet(filename,sheet_name,i)
            datalistlist.append(data)
        except IndexError as e:
            break
    #print(datalistlist)
    return  datalistlist

 

這裡我們增加一個數據處理DataProcess方法,把excel讀取的整個sheet數據(是一個列表,它的元素也是列表),我們用這個方法,根據行號返回一個json數據,也就是我們每條用例所要測試請求的數據。

import json

def DataProcess(datalist,myindex):
    '''datalist是一個列表,列表的元素也是列表,是從excel讀到的指定sheet全部數據,根據index返回一個標準化的dict'''
    data = datalist[myindex]
    caseid =       data[0]
    casename =     data[1]
    method =       data[2]
    url =          data[3]
    payload =      data[4].encode('utf-8')
    expectedcode = data[5]
    expectedmsg =  data[6]
    expecteddata = data[7]
    headers =      json.loads(data[8])
    runselect =    data[9]

    newdict = {"runselect":runselect,
               "method":method,
               "caseid":caseid,
               "url":url,
               "data":payload,
               "headers":headers,
               "expectedcode":expectedcode,
               "expectedmsg":expectedmsg,
               "expecteddata":expecteddata}

    return newdict

  

我們的test case樣子如下:每個test方法除了一個數據處理的入參不一樣,其他完全一致,無需更改。 而且我們可以做到所有的測試用例都按這個模板來撰寫。

class LoginTest(unittest.TestCase):
    ''' '''

    def setUp(self):
        self.logname = logname
        self.mylog = AddLog.AddLog(self.logname).getlog()
        self.testcaseinfo = OperateExcel.Readexcelalldata_of_sheet(Excelname,Sheetname)
        self.result = {}
        global caseid
        caseid += 1

    def tearDown(self):
        self.mylog.info("case:%d completed" % caseid)

    @unittest.skipIf((max_testcaseid < 1) or (0 in indexlist),"用例最大id<1或用例標記為'不執行'")
    def test_caseid_1(self):
        testdata = DataProcess.DataProcess(self.testcaseinfo,0)
        headers= {"Content-Type": "application/json"}
        self.result = RunRequest.RunRequests(testdata["method"],
                                             testdata["url"],
                                             testdata["data"],
                                             testdata['headers']).run()
        self.myassert = AssertResult.AssertResult(self.logname, testdata, self.result)
        self.myassert.AssertEqualResult()

    @unittest.skipIf((max_testcaseid < 2) or (1 in indexlist),"用例最大id<2或用例標記為'不執行'")
    def test_caseid_2(self):
        testdata = DataProcess.DataProcess(self.testcaseinfo, 1)
        self.result = RunRequest.RunRequests(testdata["method"],
                                             testdata["url"],
                                             testdata["data"],
                                             testdata['headers']).run()
        self.myassert = AssertResult.AssertResult(self.logname,testdata,self.result)
        self.myassert.AssertEqualResult()

    @unittest.skipIf((max_testcaseid < 3) or (2 in indexlist),"用例最大id<3或用例標記為'不執行'")
    def test_caseid_3(self):
        testdata = DataProcess.DataProcess(self.testcaseinfo, 2)
        self.result = RunRequest.RunRequests(testdata["method"],
                                             testdata["url"],
                                             testdata["data"],
                                             testdata['headers']).run()
        self.myassert = AssertResult.AssertResult(self.logname,testdata,self.result)
        self.myassert.AssertEqualResult()

    @unittest.skipIf((max_testcaseid < 4) or (3 in indexlist), "用例最大id<4或用例標記為'不執行'")
    def test_caseid_4(self):
        testdata = DataProcess.DataProcess(self.testcaseinfo, 3)
        self.result = RunRequest.RunRequests(testdata["method"],
                                             testdata["url"],
                                             testdata["data"],
                                             testdata['headers']).run()
        self.myassert = AssertResult.AssertResult(self.logname,testdata,self.result)
        self.myassert.AssertEqualResult()

  因為要用到不同請求方式,我們把他封裝在一起了。

#coding = utf-8
import requests
import json

class RunRequests(object):

    def __init__(self,method,url,data,headers):

        self.url = url

        self.method = method

        self.data = data

        self.headers = headers

        self.result = ""


    def run(self):

        if self.method == "POST":

            if self.headers != "No":

                self.result = requests.post(self.url, data = self.data, headers = self.headers)

            else:
                self.result = requests.post(self.url, data = self.data)


        elif self.method == "GET":

            if self.headers != "No":

                self.result = requests.get(self.url, data=self.data)

                print("暫未處理headers")

            else:
                self.result = requests.get(self.url,data = self.data)

        elif self.method == "PUT":

            self.result = 1

        elif self.method == "DELETE":

            self.result = 1

        elif self.method == "HEAD":

            self.result = 1

        else:

            self.result = 1

        return  self.result.json()

  

這些都完成後,想到測試完成後需要日誌和郵件功能。也添加進來。

寫完後整個框架就如下這個樣子:

 

 如果有朋友看到,覺得可以學習練練手,有需要源碼可以留言,看到會及時回覆。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在《Java編程思想》中關於泛型的講解中,提到了自限定類型: 作者說道: 這就像兩面鏡子彼此照向對方所引起的目眩效果一樣,是一種無限反射。 類接受泛型參數 ,而 由一個邊界限定,這個邊界就是擁有T作為其參數的 接下來,作者用了近3頁的紙來進行解釋這個問題,且語言極其晦澀難懂(可能是翻譯的問題),但是 ...
  • 第一問:TCP與UDP的區別 參考答案: 1.基於連接與無連接 2.TCP要求系統資源較多,UDP較少; 3.UDP程式結構較簡單 4.流模式(TCP)與數據報模式(UDP); 5.TCP保證數據正確性,UDP可能丟包 6.TCP保證數據順序,UDP不保證 考點:聽說騰訊必考TCP,反正TCP的三次 ...
  • 1. 模板字元串簡介: 顧名思義,模板字元串是用來定義一個模板是使用的,就像Vue,React中的template語法。 首先,先來瞭解一下template string的基本用法: 在ES5中,我們大多都有過拼串的經歷吧。 模板字元串的語法是反引號(`` --> 鍵盤左上角),利用反引號將字元串封 ...
  • 題目鏈接:http://acm.hdu.edu.cn/showproblem.php?pid=6467 看到這題,簡單數學???對不起我給數學老師丟臉了! 這裡解釋一下第二步到第三步:假設n=3,第二步{1*C(1,1)+1*C(1,2)+1*C(1,3)+2*C(2,2)+2*C(2,3)+3*C ...
  • java基礎--提示對話框的使用 2019-03-17-00:35:50 雲林原創 一、顯示信息對話框:使用“JOptionPane.showMessageDialog”顯示: 使用實例: 1、顯示錯誤類型對話框: 視圖: 2、傳達信息類型對話框: 視圖: 3、警告對話框: 視圖: 4、提問對話框: ...
  • java基礎--常用函數總結 2019-3-16-23:28:01 雲林原創 1、split()字元串分割函數 將一個字元串分割為子字元串,然後將結果作為字元串數組返回。 2、Math.floor( )舍掉小數取整數 3、Math.rint( )四捨五入取整數 4、Math.ceil( )進位取整數 ...
  • 下午在刷題過程中,忽然想寫2048了,以彌補以前寫的那個千多行的,所以簡單思考了一下準備採取的數據結構就開始了,本以為一個小時能搞定,結果後面改bug還是多花了些時間。因為在醫院,所以聲音不敢太大,如果看,建議耳機+聲音最大,可以考慮倍速。個人感覺用C寫這些東西的意義在於,你去掉了一些花里胡哨的東西 ...
  • 文章大綱 一、加密與驗簽介紹二、介面驗簽實操三、項目源碼下載 一、加密與驗簽介紹 大多數公共網路是不安全的,一切基於HTTP協議的請求/響應(Request or Response)都是可以被截獲的、篡改、重放(重發)的。因此我們需要考慮以下幾點內容: 防偽裝攻擊(案例:在公共網路環境中,第三方 有 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...